blob: 8a060780cab7adf7dd050a37a6cc87d56c328a9c [file] [log] [blame]
Patrice Chotard7b81c052019-11-25 09:07:38 +01001// SPDX-License-Identifier: GPL-2.0+
2
Patrice Chotard7b81c052019-11-25 09:07:38 +01003#include <command.h>
4#include <env.h>
5#include <fs.h>
Simon Glass0666fcb2021-10-14 12:48:00 -06006#include <pxe_utils.h>
Tom Rinidec7ea02024-05-20 13:35:03 -06007#include <vsprintf.h>
Patrice Chotard7b81c052019-11-25 09:07:38 +01008
Simon Glass086b0802021-10-14 12:48:09 -06009/**
10 * struct sysboot_info - useful information for sysboot helpers
11 *
12 * @fstype: Filesystem type (FS_TYPE_...)
13 * @ifname: Interface name (e.g. "ide", "scsi")
14 * @dev_part_str is in the format:
15 * <dev>.<hw_part>:<part> where <dev> is the device number,
16 * <hw_part> is the optional hardware partition number and
17 * <part> is the partition number
18 */
19struct sysboot_info {
20 int fstype;
21 const char *ifname;
22 const char *dev_part_str;
23};
Patrice Chotard7b81c052019-11-25 09:07:38 +010024
Simon Glass086b0802021-10-14 12:48:09 -060025static int sysboot_read_file(struct pxe_context *ctx, const char *file_path,
26 char *file_addr, ulong *sizep)
Patrice Chotard7b81c052019-11-25 09:07:38 +010027{
Simon Glass086b0802021-10-14 12:48:09 -060028 struct sysboot_info *info = ctx->userdata;
29 loff_t len_read;
30 ulong addr;
Simon Glassa9401b92021-10-14 12:48:08 -060031 int ret;
32
Simon Glass086b0802021-10-14 12:48:09 -060033 addr = simple_strtoul(file_addr, NULL, 16);
34 ret = fs_set_blk_dev(info->ifname, info->dev_part_str, info->fstype);
Simon Glassa9401b92021-10-14 12:48:08 -060035 if (ret)
Simon Glass086b0802021-10-14 12:48:09 -060036 return ret;
37 ret = fs_read(file_path, addr, 0, 0, &len_read);
Simon Glassa9401b92021-10-14 12:48:08 -060038 if (ret)
Simon Glass086b0802021-10-14 12:48:09 -060039 return ret;
40 *sizep = len_read;
Patrice Chotard7b81c052019-11-25 09:07:38 +010041
Simon Glass086b0802021-10-14 12:48:09 -060042 return 0;
Patrice Chotard7b81c052019-11-25 09:07:38 +010043}
44
45/*
46 * Boots a system using a local disk syslinux/extlinux file
47 *
48 * Returns 0 on success, 1 on error.
49 */
Simon Glassed38aef2020-05-10 11:40:03 -060050static int do_sysboot(struct cmd_tbl *cmdtp, int flag, int argc,
51 char *const argv[])
Patrice Chotard7b81c052019-11-25 09:07:38 +010052{
53 unsigned long pxefile_addr_r;
Simon Glassb0d08db2021-10-14 12:47:56 -060054 struct pxe_context ctx;
Patrice Chotard7b81c052019-11-25 09:07:38 +010055 char *pxefile_addr_str;
Simon Glass086b0802021-10-14 12:48:09 -060056 struct sysboot_info info;
Patrice Chotard7b81c052019-11-25 09:07:38 +010057 char *filename;
58 int prompt = 0;
Simon Glass791bbfe2021-10-14 12:48:03 -060059 int ret;
Patrice Chotard7b81c052019-11-25 09:07:38 +010060
Patrice Chotard7b81c052019-11-25 09:07:38 +010061 if (argc > 1 && strstr(argv[1], "-p")) {
62 prompt = 1;
63 argc--;
64 argv++;
65 }
66
67 if (argc < 4)
68 return cmd_usage(cmdtp);
69
70 if (argc < 5) {
71 pxefile_addr_str = from_env("pxefile_addr_r");
72 if (!pxefile_addr_str)
73 return 1;
74 } else {
75 pxefile_addr_str = argv[4];
76 }
77
Patrice Chotard1918e732019-11-25 09:07:40 +010078 if (argc < 6) {
Patrice Chotard7b81c052019-11-25 09:07:38 +010079 filename = env_get("bootfile");
Caleb Connollya1198332024-03-18 23:16:36 +000080 if (!filename) {
81 printf("Specify a filename or set the ${bootfile} environment variable\n");
82 return 1;
83 }
Patrice Chotard1918e732019-11-25 09:07:40 +010084 } else {
Patrice Chotard7b81c052019-11-25 09:07:38 +010085 filename = argv[5];
86 env_set("bootfile", filename);
87 }
88
Patrice Chotard1918e732019-11-25 09:07:40 +010089 if (strstr(argv[3], "ext2")) {
Simon Glass086b0802021-10-14 12:48:09 -060090 info.fstype = FS_TYPE_EXT;
Patrice Chotard1918e732019-11-25 09:07:40 +010091 } else if (strstr(argv[3], "fat")) {
Simon Glass086b0802021-10-14 12:48:09 -060092 info.fstype = FS_TYPE_FAT;
Patrice Chotard1918e732019-11-25 09:07:40 +010093 } else if (strstr(argv[3], "any")) {
Simon Glass086b0802021-10-14 12:48:09 -060094 info.fstype = FS_TYPE_ANY;
Patrice Chotard1918e732019-11-25 09:07:40 +010095 } else {
Patrice Chotard7b81c052019-11-25 09:07:38 +010096 printf("Invalid filesystem: %s\n", argv[3]);
97 return 1;
98 }
Simon Glass086b0802021-10-14 12:48:09 -060099 info.ifname = argv[1];
100 info.dev_part_str = argv[2];
Patrice Chotard7b81c052019-11-25 09:07:38 +0100101
102 if (strict_strtoul(pxefile_addr_str, 16, &pxefile_addr_r) < 0) {
103 printf("Invalid pxefile address: %s\n", pxefile_addr_str);
104 return 1;
105 }
106
Simon Glass086b0802021-10-14 12:48:09 -0600107 if (pxe_setup_ctx(&ctx, cmdtp, sysboot_read_file, &info, true,
Martyn Welch2c47aac2024-10-09 14:15:39 +0100108 filename, false, false)) {
Simon Glasse719fe02021-10-14 12:48:04 -0600109 printf("Out of memory\n");
110 return CMD_RET_FAILURE;
111 }
112
Simon Glassb0d08db2021-10-14 12:47:56 -0600113 if (get_pxe_file(&ctx, filename, pxefile_addr_r) < 0) {
Patrice Chotard7b81c052019-11-25 09:07:38 +0100114 printf("Error reading config file\n");
Simon Glasse719fe02021-10-14 12:48:04 -0600115 pxe_destroy_ctx(&ctx);
Patrice Chotard7b81c052019-11-25 09:07:38 +0100116 return 1;
117 }
118
Simon Glass791bbfe2021-10-14 12:48:03 -0600119 ret = pxe_process(&ctx, pxefile_addr_r, prompt);
Simon Glasse719fe02021-10-14 12:48:04 -0600120 pxe_destroy_ctx(&ctx);
Simon Glass791bbfe2021-10-14 12:48:03 -0600121 if (ret)
122 return CMD_RET_FAILURE;
Patrice Chotard7b81c052019-11-25 09:07:38 +0100123
124 return 0;
125}
126
Patrice Chotard1918e732019-11-25 09:07:40 +0100127U_BOOT_CMD(sysboot, 7, 1, do_sysboot,
128 "command to get and boot from syslinux files",
129 "[-p] <interface> <dev[:part]> <ext2|fat|any> [addr] [filename]\n"
130 " - load and parse syslinux menu file 'filename' from ext2, fat\n"
131 " or any filesystem on 'dev' on 'interface' to address 'addr'"
Patrice Chotard7b81c052019-11-25 09:07:38 +0100132);