1 // SPDX-License-Identifier: GPL-2.0+
5 * Copyright 2021 Google LLC
19 * report_bootflow_err() - Report where a bootflow failed
21 * When a bootflow does not make it to the 'loaded' state, something went wrong.
22 * Print a helpful message if there is an error
24 * @bflow: Bootflow to process
25 * @err: Error code (0 if none)
27 static void report_bootflow_err(struct bootflow *bflow, int err)
32 /* Indent out to 'Method' */
35 switch (bflow->state) {
37 printf("No media/partition found");
39 case BOOTFLOWST_MEDIA:
40 printf("No partition found");
43 printf("No filesystem found");
46 printf("File not found");
49 printf("File cannot be loaded");
51 case BOOTFLOWST_READY:
54 case BOOTFLOWST_COUNT:
58 printf(", err=%d\n", err);
62 * show_bootflow() - Show the status of a bootflow
64 * @seq: Bootflow index
65 * @bflow: Bootflow to show
66 * @errors: True to show the error received, if any
68 static void show_bootflow(int index, struct bootflow *bflow, bool errors)
70 printf("%3x %-11s %-6s %-9.9s %4x %-25.25s %s\n", index,
71 bflow->method->name, bootflow_state_get_name(bflow->state),
72 bflow->dev ? dev_get_uclass_name(dev_get_parent(bflow->dev)) :
73 "(none)", bflow->part, bflow->name, bflow->fname);
75 report_bootflow_err(bflow, bflow->err);
78 static void show_header(void)
80 printf("Seq Method State Uclass Part Name Filename\n");
81 printf("--- ----------- ------ -------- ---- ------------------------ ----------------\n");
84 static void show_footer(int count, int num_valid)
86 printf("--- ----------- ------ -------- ---- ------------------------ ----------------\n");
87 printf("(%d bootflow%s, %d valid)\n", count, count != 1 ? "s" : "",
91 static int do_bootflow_scan(struct cmd_tbl *cmdtp, int flag, int argc,
94 struct bootstd_priv *std;
95 struct bootflow_iter iter;
96 struct udevice *dev = NULL;
97 struct bootflow bflow;
98 bool all = false, boot = false, errors = false, no_global = false;
99 bool list = false, no_hunter = false;
101 const char *label = NULL;
106 ret = bootstd_get_priv(&std);
108 return CMD_RET_FAILURE;
110 has_args = argc > 1 && *argv[1] == '-';
111 if (IS_ENABLED(CONFIG_CMD_BOOTFLOW_FULL)) {
113 all = strchr(argv[1], 'a');
114 boot = strchr(argv[1], 'b');
115 errors = strchr(argv[1], 'e');
116 no_global = strchr(argv[1], 'G');
117 list = strchr(argv[1], 'l');
118 no_hunter = strchr(argv[1], 'H');
125 dev = std->cur_bootdev;
128 printf("Flags not supported: enable CONFIG_BOOTFLOW_FULL\n");
129 return CMD_RET_USAGE;
134 std->cur_bootflow = NULL;
138 flags |= BOOTFLOWIF_SHOW;
140 flags |= BOOTFLOWIF_ALL;
142 flags |= BOOTFLOWIF_SKIP_GLOBAL;
144 flags |= BOOTFLOWIF_HUNT;
147 * If we have a device, just scan for bootflows attached to that device
150 printf("Scanning for bootflows ");
152 printf("in bootdev '%s'\n", dev->name);
154 printf("with label '%s'\n", label);
156 printf("in all bootdevs\n");
160 bootdev_clear_bootflows(dev);
162 bootstd_clear_glob();
164 ret = bootflow_scan_first(dev, label, &iter, flags, &bflow);
165 i < 1000 && ret != -ENODEV;
166 i++, ret = bootflow_scan_next(&iter, &bflow)) {
170 ret = bootdev_add_bootflow(&bflow);
172 printf("Out of memory\n");
173 return CMD_RET_FAILURE;
176 show_bootflow(i, &bflow, errors);
177 if (boot && !bflow.err)
178 bootflow_run_boot(&iter, &bflow);
180 bootflow_iter_uninit(&iter);
182 show_footer(i, num_valid);
187 #ifdef CONFIG_CMD_BOOTFLOW_FULL
188 static int do_bootflow_list(struct cmd_tbl *cmdtp, int flag, int argc,
191 struct bootstd_priv *std;
193 struct bootflow *bflow;
198 if (argc > 1 && *argv[1] == '-')
199 errors = strchr(argv[1], 'e');
201 ret = bootstd_get_priv(&std);
203 return CMD_RET_FAILURE;
204 dev = std->cur_bootdev;
206 /* If we have a device, just list bootflows attached to that device */
208 printf("Showing bootflows for bootdev '%s'\n", dev->name);
210 for (ret = bootdev_first_bootflow(dev, &bflow), i = 0;
212 ret = bootdev_next_bootflow(&bflow), i++) {
213 num_valid += bflow->state == BOOTFLOWST_READY;
214 show_bootflow(i, bflow, errors);
217 printf("Showing all bootflows\n");
219 for (ret = bootflow_first_glob(&bflow), i = 0;
221 ret = bootflow_next_glob(&bflow), i++) {
222 num_valid += bflow->state == BOOTFLOWST_READY;
223 show_bootflow(i, bflow, errors);
226 show_footer(i, num_valid);
231 static int do_bootflow_select(struct cmd_tbl *cmdtp, int flag, int argc,
234 struct bootstd_priv *std;
235 struct bootflow *bflow, *found;
242 ret = bootstd_get_priv(&std);
244 return CMD_RET_FAILURE;
247 std->cur_bootflow = NULL;
250 dev = std->cur_bootdev;
253 seq = simple_strtol(name, &endp, 16);
257 * If we have a bootdev device, only allow selection of bootflows
258 * attached to that device
261 for (ret = bootdev_first_bootflow(dev, &bflow), i = 0;
263 ret = bootdev_next_bootflow(&bflow), i++) {
264 if (*endp ? !strcmp(bflow->name, name) : i == seq) {
270 for (ret = bootflow_first_glob(&bflow), i = 0;
272 ret = bootflow_next_glob(&bflow), i++) {
273 if (*endp ? !strcmp(bflow->name, name) : i == seq) {
281 printf("Cannot find bootflow '%s' ", name);
283 printf("in bootdev '%s' ", dev->name);
284 printf("(err=%d)\n", ret);
285 return CMD_RET_FAILURE;
287 std->cur_bootflow = found;
292 static int do_bootflow_info(struct cmd_tbl *cmdtp, int flag, int argc,
295 struct bootstd_priv *std;
296 struct bootflow *bflow;
300 if (argc > 1 && *argv[1] == '-')
301 dump = strchr(argv[1], 'd');
303 ret = bootstd_get_priv(&std);
305 return CMD_RET_FAILURE;
307 if (!std->cur_bootflow) {
308 printf("No bootflow selected\n");
309 return CMD_RET_FAILURE;
311 bflow = std->cur_bootflow;
313 printf("Name: %s\n", bflow->name);
314 printf("Device: %s\n", bflow->dev->name);
315 printf("Block dev: %s\n", bflow->blk ? bflow->blk->name : "(none)");
316 printf("Method: %s\n", bflow->method->name);
317 printf("State: %s\n", bootflow_state_get_name(bflow->state));
318 printf("Partition: %d\n", bflow->part);
319 printf("Subdir: %s\n", bflow->subdir ? bflow->subdir : "(none)");
320 printf("Filename: %s\n", bflow->fname);
321 printf("Buffer: %lx\n", (ulong)map_to_sysmem(bflow->buf));
322 printf("Size: %x (%d bytes)\n", bflow->size, bflow->size);
323 printf("OS: %s\n", bflow->os_name ? bflow->os_name : "(none)");
324 printf("Logo: %s\n", bflow->logo ?
325 simple_xtoa((ulong)map_to_sysmem(bflow->logo)) : "(none)");
327 printf("Logo size: %x (%d bytes)\n", bflow->logo_size,
330 printf("FDT: %s\n", bflow->fdt_fname);
331 if (bflow->fdt_fname) {
332 printf("FDT size: %x (%d bytes)\n", bflow->fdt_size,
334 printf("FDT addr: %lx\n", bflow->fdt_addr);
336 printf("Error: %d\n", bflow->err);
337 if (dump && bflow->buf) {
338 /* Set some sort of maximum on the size */
339 int size = min(bflow->size, 10 << 10);
342 printf("Contents:\n\n");
343 for (i = 0; i < size; i++) {
345 if (!(i % 128) && ctrlc()) {
346 printf("...interrupted\n");
355 static int do_bootflow_boot(struct cmd_tbl *cmdtp, int flag, int argc,
358 struct bootstd_priv *std;
359 struct bootflow *bflow;
362 ret = bootstd_get_priv(&std);
364 return CMD_RET_FAILURE;
367 * Require a current bootflow. Users can use 'bootflow scan -b' to
368 * automatically scan and boot, if needed.
370 if (!std->cur_bootflow) {
371 printf("No bootflow selected\n");
372 return CMD_RET_FAILURE;
374 bflow = std->cur_bootflow;
375 ret = bootflow_run_boot(NULL, bflow);
377 return CMD_RET_FAILURE;
382 static int do_bootflow_menu(struct cmd_tbl *cmdtp, int flag, int argc,
385 struct bootstd_priv *std;
386 struct bootflow *bflow;
387 bool text_mode = false;
390 if (argc > 1 && *argv[1] == '-')
391 text_mode = strchr(argv[1], 't');
393 ret = bootstd_get_priv(&std);
395 return CMD_RET_FAILURE;
397 if (IS_ENABLED(CONFIG_EXPO)) {
398 ret = bootflow_menu_run(std, text_mode, &bflow);
401 printf("Nothing chosen\n");
403 printf("Menu failed (err=%d)\n", ret);
406 printf("Menu not supported\n");
410 return CMD_RET_FAILURE;
412 printf("Selected: %s\n", bflow->os_name ? bflow->os_name : bflow->name);
413 std->cur_bootflow = bflow;
417 #endif /* CONFIG_CMD_BOOTFLOW_FULL */
419 #ifdef CONFIG_SYS_LONGHELP
420 static char bootflow_help_text[] =
421 #ifdef CONFIG_CMD_BOOTFLOW_FULL
422 "scan [-abeGl] [bdev] - scan for valid bootflows (-l list, -a all, -e errors, -b boot, -G no global)\n"
423 "bootflow list [-e] - list scanned bootflows (-e errors)\n"
424 "bootflow select [<num>|<name>] - select a bootflow\n"
425 "bootflow info [-d] - show info on current bootflow (-d dump bootflow)\n"
426 "bootflow boot - boot current bootflow (or first available if none selected)\n"
427 "bootflow menu [-t] - show a menu of available bootflows";
429 "scan - boot first available bootflow\n";
431 #endif /* CONFIG_SYS_LONGHELP */
433 U_BOOT_CMD_WITH_SUBCMDS(bootflow, "Boot flows", bootflow_help_text,
434 U_BOOT_SUBCMD_MKENT(scan, 3, 1, do_bootflow_scan),
435 #ifdef CONFIG_CMD_BOOTFLOW_FULL
436 U_BOOT_SUBCMD_MKENT(list, 2, 1, do_bootflow_list),
437 U_BOOT_SUBCMD_MKENT(select, 2, 1, do_bootflow_select),
438 U_BOOT_SUBCMD_MKENT(info, 2, 1, do_bootflow_info),
439 U_BOOT_SUBCMD_MKENT(boot, 1, 1, do_bootflow_boot),
440 U_BOOT_SUBCMD_MKENT(menu, 2, 1, do_bootflow_menu),