1bace3d00SSimon Glass /*
26fb62078SSimon Glass * Copyright (c) 2011-2012 The Chromium OS Authors.
31a459660SWolfgang Denk * SPDX-License-Identifier: GPL-2.0+
4bace3d00SSimon Glass */
5bace3d00SSimon Glass
6bace3d00SSimon Glass #include <common.h>
738068820SSimon Glass #include <errno.h>
85c2859cdSSimon Glass #include <os.h>
97dbcb76eSRabin Vincent #include <cli.h>
101f32ae95SSimon Glass #include <malloc.h>
1170db4212SSimon Glass #include <asm/getopt.h>
124d94dfa0SSimon Glass #include <asm/io.h>
1370db4212SSimon Glass #include <asm/sections.h>
146fb62078SSimon Glass #include <asm/state.h>
15bace3d00SSimon Glass
16808434cdSSimon Glass DECLARE_GLOBAL_DATA_PTR;
17808434cdSSimon Glass
sandbox_early_getopt_check(void)1870db4212SSimon Glass int sandbox_early_getopt_check(void)
1970db4212SSimon Glass {
2070db4212SSimon Glass struct sandbox_state *state = state_get_current();
217b3efc66SSimon Glass struct sandbox_cmdline_option **sb_opt = __u_boot_sandbox_option_start;
2270db4212SSimon Glass size_t num_options = __u_boot_sandbox_option_count();
2370db4212SSimon Glass size_t i;
2470db4212SSimon Glass int max_arg_len, max_noarg_len;
2570db4212SSimon Glass
2670db4212SSimon Glass /* parse_err will be a string of the faulting option */
2770db4212SSimon Glass if (!state->parse_err)
2870db4212SSimon Glass return 0;
2970db4212SSimon Glass
3070db4212SSimon Glass if (strcmp(state->parse_err, "help")) {
3170db4212SSimon Glass printf("u-boot: error: failed while parsing option: %s\n"
3270db4212SSimon Glass "\ttry running with --help for more information.\n",
3370db4212SSimon Glass state->parse_err);
3470db4212SSimon Glass os_exit(1);
3570db4212SSimon Glass }
3670db4212SSimon Glass
3770db4212SSimon Glass printf(
3870db4212SSimon Glass "u-boot, a command line test interface to U-Boot\n\n"
3970db4212SSimon Glass "Usage: u-boot [options]\n"
4070db4212SSimon Glass "Options:\n");
4170db4212SSimon Glass
4270db4212SSimon Glass max_arg_len = 0;
4370db4212SSimon Glass for (i = 0; i < num_options; ++i)
44b4141195SMasahiro Yamada max_arg_len = max((int)strlen(sb_opt[i]->flag), max_arg_len);
4570db4212SSimon Glass max_noarg_len = max_arg_len + 7;
4670db4212SSimon Glass
4770db4212SSimon Glass for (i = 0; i < num_options; ++i) {
487b3efc66SSimon Glass struct sandbox_cmdline_option *opt = sb_opt[i];
4970db4212SSimon Glass
5070db4212SSimon Glass /* first output the short flag if it has one */
5170db4212SSimon Glass if (opt->flag_short >= 0x100)
5270db4212SSimon Glass printf(" ");
5370db4212SSimon Glass else
5470db4212SSimon Glass printf(" -%c, ", opt->flag_short);
5570db4212SSimon Glass
5670db4212SSimon Glass /* then the long flag */
5770db4212SSimon Glass if (opt->has_arg)
5870db4212SSimon Glass printf("--%-*s <arg> ", max_arg_len, opt->flag);
596ebcab8dSSimon Glass else
606ebcab8dSSimon Glass printf("--%-*s", max_noarg_len, opt->flag);
6170db4212SSimon Glass
6270db4212SSimon Glass /* finally the help text */
6370db4212SSimon Glass printf(" %s\n", opt->help);
6470db4212SSimon Glass }
6570db4212SSimon Glass
6670db4212SSimon Glass os_exit(0);
6770db4212SSimon Glass }
6870db4212SSimon Glass
misc_init_f(void)6968969778SSimon Glass int misc_init_f(void)
7068969778SSimon Glass {
7168969778SSimon Glass return sandbox_early_getopt_check();
7268969778SSimon Glass }
7368969778SSimon Glass
sandbox_cmdline_cb_help(struct sandbox_state * state,const char * arg)747b3efc66SSimon Glass static int sandbox_cmdline_cb_help(struct sandbox_state *state, const char *arg)
7570db4212SSimon Glass {
7670db4212SSimon Glass /* just flag to sandbox_early_getopt_check to show usage */
7770db4212SSimon Glass return 1;
7870db4212SSimon Glass }
797b3efc66SSimon Glass SANDBOX_CMDLINE_OPT_SHORT(help, 'h', 0, "Display help");
8070db4212SSimon Glass
81d0d0746eSSimon Glass #ifndef CONFIG_SPL_BUILD
sandbox_main_loop_init(void)82ab4e07ebSSimon Glass int sandbox_main_loop_init(void)
83ab4e07ebSSimon Glass {
8470db4212SSimon Glass struct sandbox_state *state = state_get_current();
8570db4212SSimon Glass
8670db4212SSimon Glass /* Execute command if required */
87ebaa832eSSjoerd Simons if (state->cmd || state->run_distro_boot) {
88ebaa832eSSjoerd Simons int retval = 0;
8988539e44SJoe Hershberger
907dbcb76eSRabin Vincent cli_init();
917dbcb76eSRabin Vincent
922b6793deSSimon Glass #ifdef CONFIG_CMDLINE
93ebaa832eSSjoerd Simons if (state->cmd)
9488539e44SJoe Hershberger retval = run_command_list(state->cmd, -1, 0);
95ebaa832eSSjoerd Simons
96ebaa832eSSjoerd Simons if (state->run_distro_boot)
97ebaa832eSSjoerd Simons retval = cli_simple_run_command("run distro_bootcmd",
98ebaa832eSSjoerd Simons 0);
992b6793deSSimon Glass #endif
100c5a62d4aSSimon Glass if (!state->interactive)
10188539e44SJoe Hershberger os_exit(retval);
10270db4212SSimon Glass }
10370db4212SSimon Glass
104ab4e07ebSSimon Glass return 0;
105ab4e07ebSSimon Glass }
106d0d0746eSSimon Glass #endif
107ab4e07ebSSimon Glass
sandbox_cmdline_cb_boot(struct sandbox_state * state,const char * arg)108ebaa832eSSjoerd Simons static int sandbox_cmdline_cb_boot(struct sandbox_state *state,
109ebaa832eSSjoerd Simons const char *arg)
110ebaa832eSSjoerd Simons {
111ebaa832eSSjoerd Simons state->run_distro_boot = true;
112ebaa832eSSjoerd Simons return 0;
113ebaa832eSSjoerd Simons }
114ebaa832eSSjoerd Simons SANDBOX_CMDLINE_OPT_SHORT(boot, 'b', 0, "Run distro boot commands");
115ebaa832eSSjoerd Simons
sandbox_cmdline_cb_command(struct sandbox_state * state,const char * arg)1167b3efc66SSimon Glass static int sandbox_cmdline_cb_command(struct sandbox_state *state,
1177b3efc66SSimon Glass const char *arg)
11870db4212SSimon Glass {
11970db4212SSimon Glass state->cmd = arg;
12070db4212SSimon Glass return 0;
12170db4212SSimon Glass }
1227b3efc66SSimon Glass SANDBOX_CMDLINE_OPT_SHORT(command, 'c', 1, "Execute U-Boot command");
12370db4212SSimon Glass
sandbox_cmdline_cb_fdt(struct sandbox_state * state,const char * arg)1247b3efc66SSimon Glass static int sandbox_cmdline_cb_fdt(struct sandbox_state *state, const char *arg)
125f828bf25SSimon Glass {
126f828bf25SSimon Glass state->fdt_fname = arg;
127f828bf25SSimon Glass return 0;
128f828bf25SSimon Glass }
1297b3efc66SSimon Glass SANDBOX_CMDLINE_OPT_SHORT(fdt, 'd', 1, "Specify U-Boot's control FDT");
130f828bf25SSimon Glass
sandbox_cmdline_cb_default_fdt(struct sandbox_state * state,const char * arg)1311f32ae95SSimon Glass static int sandbox_cmdline_cb_default_fdt(struct sandbox_state *state,
1321f32ae95SSimon Glass const char *arg)
1331f32ae95SSimon Glass {
1341f32ae95SSimon Glass const char *fmt = "%s.dtb";
1351f32ae95SSimon Glass char *fname;
1361f32ae95SSimon Glass int len;
1371f32ae95SSimon Glass
1381f32ae95SSimon Glass len = strlen(state->argv[0]) + strlen(fmt) + 1;
1391f32ae95SSimon Glass fname = os_malloc(len);
1401f32ae95SSimon Glass if (!fname)
1411f32ae95SSimon Glass return -ENOMEM;
1421f32ae95SSimon Glass snprintf(fname, len, fmt, state->argv[0]);
1431f32ae95SSimon Glass state->fdt_fname = fname;
1441f32ae95SSimon Glass
1451f32ae95SSimon Glass return 0;
1461f32ae95SSimon Glass }
1471f32ae95SSimon Glass SANDBOX_CMDLINE_OPT_SHORT(default_fdt, 'D', 0,
1481f32ae95SSimon Glass "Use the default u-boot.dtb control FDT in U-Boot directory");
1491f32ae95SSimon Glass
sandbox_cmdline_cb_interactive(struct sandbox_state * state,const char * arg)150c5a62d4aSSimon Glass static int sandbox_cmdline_cb_interactive(struct sandbox_state *state,
151c5a62d4aSSimon Glass const char *arg)
152c5a62d4aSSimon Glass {
153c5a62d4aSSimon Glass state->interactive = true;
154c5a62d4aSSimon Glass return 0;
155c5a62d4aSSimon Glass }
156c5a62d4aSSimon Glass
157c5a62d4aSSimon Glass SANDBOX_CMDLINE_OPT_SHORT(interactive, 'i', 0, "Enter interactive mode");
158c5a62d4aSSimon Glass
sandbox_cmdline_cb_jump(struct sandbox_state * state,const char * arg)159bda7773fSSimon Glass static int sandbox_cmdline_cb_jump(struct sandbox_state *state,
160bda7773fSSimon Glass const char *arg)
161bda7773fSSimon Glass {
162ab839dc3SSimon Glass /* Remember to delete this U-Boot image later */
163ab839dc3SSimon Glass state->jumped_fname = arg;
164bda7773fSSimon Glass
165bda7773fSSimon Glass return 0;
166bda7773fSSimon Glass }
167bda7773fSSimon Glass SANDBOX_CMDLINE_OPT_SHORT(jump, 'j', 1, "Jumped from previous U-Boot");
168bda7773fSSimon Glass
sandbox_cmdline_cb_memory(struct sandbox_state * state,const char * arg)1695c2859cdSSimon Glass static int sandbox_cmdline_cb_memory(struct sandbox_state *state,
1705c2859cdSSimon Glass const char *arg)
1715c2859cdSSimon Glass {
1725c2859cdSSimon Glass int err;
1735c2859cdSSimon Glass
1745c2859cdSSimon Glass /* For now assume we always want to write it */
1755c2859cdSSimon Glass state->write_ram_buf = true;
1765c2859cdSSimon Glass state->ram_buf_fname = arg;
1775c2859cdSSimon Glass
178f80a8bbeSSimon Glass err = os_read_ram_buf(arg);
179f80a8bbeSSimon Glass if (err) {
1805c2859cdSSimon Glass printf("Failed to read RAM buffer\n");
1815c2859cdSSimon Glass return err;
1825c2859cdSSimon Glass }
1835c2859cdSSimon Glass
1845c2859cdSSimon Glass return 0;
1855c2859cdSSimon Glass }
1865c2859cdSSimon Glass SANDBOX_CMDLINE_OPT_SHORT(memory, 'm', 1,
1875c2859cdSSimon Glass "Read/write ram_buf memory contents from file");
1885c2859cdSSimon Glass
sandbox_cmdline_cb_rm_memory(struct sandbox_state * state,const char * arg)189ab839dc3SSimon Glass static int sandbox_cmdline_cb_rm_memory(struct sandbox_state *state,
190ab839dc3SSimon Glass const char *arg)
191ab839dc3SSimon Glass {
192ab839dc3SSimon Glass state->ram_buf_rm = true;
193ab839dc3SSimon Glass
194ab839dc3SSimon Glass return 0;
195ab839dc3SSimon Glass }
196ab839dc3SSimon Glass SANDBOX_CMDLINE_OPT(rm_memory, 0, "Remove memory file after reading");
197ab839dc3SSimon Glass
sandbox_cmdline_cb_state(struct sandbox_state * state,const char * arg)1981209e272SSimon Glass static int sandbox_cmdline_cb_state(struct sandbox_state *state,
1991209e272SSimon Glass const char *arg)
2001209e272SSimon Glass {
2011209e272SSimon Glass state->state_fname = arg;
2021209e272SSimon Glass return 0;
2031209e272SSimon Glass }
2041209e272SSimon Glass SANDBOX_CMDLINE_OPT_SHORT(state, 's', 1, "Specify the sandbox state FDT");
2051209e272SSimon Glass
sandbox_cmdline_cb_read(struct sandbox_state * state,const char * arg)2061209e272SSimon Glass static int sandbox_cmdline_cb_read(struct sandbox_state *state,
2071209e272SSimon Glass const char *arg)
2081209e272SSimon Glass {
2091209e272SSimon Glass state->read_state = true;
2101209e272SSimon Glass return 0;
2111209e272SSimon Glass }
2121209e272SSimon Glass SANDBOX_CMDLINE_OPT_SHORT(read, 'r', 0, "Read the state FDT on startup");
2131209e272SSimon Glass
sandbox_cmdline_cb_write(struct sandbox_state * state,const char * arg)2141209e272SSimon Glass static int sandbox_cmdline_cb_write(struct sandbox_state *state,
2151209e272SSimon Glass const char *arg)
2161209e272SSimon Glass {
2171209e272SSimon Glass state->write_state = true;
2181209e272SSimon Glass return 0;
2191209e272SSimon Glass }
2201209e272SSimon Glass SANDBOX_CMDLINE_OPT_SHORT(write, 'w', 0, "Write state FDT on exit");
2211209e272SSimon Glass
sandbox_cmdline_cb_ignore_missing(struct sandbox_state * state,const char * arg)2221209e272SSimon Glass static int sandbox_cmdline_cb_ignore_missing(struct sandbox_state *state,
2231209e272SSimon Glass const char *arg)
2241209e272SSimon Glass {
2251209e272SSimon Glass state->ignore_missing_state_on_read = true;
2261209e272SSimon Glass return 0;
2271209e272SSimon Glass }
2281209e272SSimon Glass SANDBOX_CMDLINE_OPT_SHORT(ignore_missing, 'n', 0,
2291209e272SSimon Glass "Ignore missing state on read");
2301209e272SSimon Glass
sandbox_cmdline_cb_show_lcd(struct sandbox_state * state,const char * arg)2317d95f2a3SSimon Glass static int sandbox_cmdline_cb_show_lcd(struct sandbox_state *state,
2327d95f2a3SSimon Glass const char *arg)
2337d95f2a3SSimon Glass {
2347d95f2a3SSimon Glass state->show_lcd = true;
2357d95f2a3SSimon Glass return 0;
2367d95f2a3SSimon Glass }
2377d95f2a3SSimon Glass SANDBOX_CMDLINE_OPT_SHORT(show_lcd, 'l', 0,
2387d95f2a3SSimon Glass "Show the sandbox LCD display");
2397d95f2a3SSimon Glass
240ffb87905SSimon Glass static const char *term_args[STATE_TERM_COUNT] = {
241ffb87905SSimon Glass "raw-with-sigs",
242ffb87905SSimon Glass "raw",
243ffb87905SSimon Glass "cooked",
244ffb87905SSimon Glass };
245ffb87905SSimon Glass
sandbox_cmdline_cb_terminal(struct sandbox_state * state,const char * arg)246ffb87905SSimon Glass static int sandbox_cmdline_cb_terminal(struct sandbox_state *state,
247ffb87905SSimon Glass const char *arg)
248ffb87905SSimon Glass {
249ffb87905SSimon Glass int i;
250ffb87905SSimon Glass
251ffb87905SSimon Glass for (i = 0; i < STATE_TERM_COUNT; i++) {
252ffb87905SSimon Glass if (!strcmp(arg, term_args[i])) {
253ffb87905SSimon Glass state->term_raw = i;
254ffb87905SSimon Glass return 0;
255ffb87905SSimon Glass }
256ffb87905SSimon Glass }
257ffb87905SSimon Glass
258ffb87905SSimon Glass printf("Unknown terminal setting '%s' (", arg);
259ffb87905SSimon Glass for (i = 0; i < STATE_TERM_COUNT; i++)
260ffb87905SSimon Glass printf("%s%s", i ? ", " : "", term_args[i]);
261ffb87905SSimon Glass puts(")\n");
262ffb87905SSimon Glass
263ffb87905SSimon Glass return 1;
264ffb87905SSimon Glass }
265ffb87905SSimon Glass SANDBOX_CMDLINE_OPT_SHORT(terminal, 't', 1,
266ffb87905SSimon Glass "Set terminal to raw/cooked mode");
267ffb87905SSimon Glass
sandbox_cmdline_cb_verbose(struct sandbox_state * state,const char * arg)2689ce8b402SSimon Glass static int sandbox_cmdline_cb_verbose(struct sandbox_state *state,
2699ce8b402SSimon Glass const char *arg)
2709ce8b402SSimon Glass {
2719ce8b402SSimon Glass state->show_test_output = true;
2729ce8b402SSimon Glass return 0;
2739ce8b402SSimon Glass }
2749ce8b402SSimon Glass SANDBOX_CMDLINE_OPT_SHORT(verbose, 'v', 0, "Show test output");
2759ce8b402SSimon Glass
board_run_command(const char * cmdline)2762b6793deSSimon Glass int board_run_command(const char *cmdline)
2772b6793deSSimon Glass {
2782b6793deSSimon Glass printf("## Commands are disabled. Please enable CONFIG_CMDLINE.\n");
2792b6793deSSimon Glass
2802b6793deSSimon Glass return 1;
2812b6793deSSimon Glass }
2822b6793deSSimon Glass
setup_ram_buf(struct sandbox_state * state)283bb967240SSimon Glass static void setup_ram_buf(struct sandbox_state *state)
284bb967240SSimon Glass {
285bb967240SSimon Glass gd->arch.ram_buf = state->ram_buf;
286bb967240SSimon Glass gd->ram_size = state->ram_size;
287bb967240SSimon Glass }
288bb967240SSimon Glass
main(int argc,char * argv[])289bace3d00SSimon Glass int main(int argc, char *argv[])
290bace3d00SSimon Glass {
29170db4212SSimon Glass struct sandbox_state *state;
2924d94dfa0SSimon Glass gd_t data;
2931209e272SSimon Glass int ret;
2946fb62078SSimon Glass
2951209e272SSimon Glass ret = state_init();
2961209e272SSimon Glass if (ret)
2971209e272SSimon Glass goto err;
2986fb62078SSimon Glass
29970db4212SSimon Glass state = state_get_current();
30070db4212SSimon Glass if (os_parse_args(state, argc, argv))
30170db4212SSimon Glass return 1;
30270db4212SSimon Glass
3031209e272SSimon Glass ret = sandbox_read_state(state, state->state_fname);
3041209e272SSimon Glass if (ret)
3051209e272SSimon Glass goto err;
3061209e272SSimon Glass
307ab839dc3SSimon Glass /* Remove old memory file if required */
308ab839dc3SSimon Glass if (state->ram_buf_rm && state->ram_buf_fname)
309ab839dc3SSimon Glass os_unlink(state->ram_buf_fname);
310ab839dc3SSimon Glass
3114d94dfa0SSimon Glass memset(&data, '\0', sizeof(data));
3124d94dfa0SSimon Glass gd = &data;
313*1fc50d72SAndy Yan #if CONFIG_VAL(SYS_MALLOC_F_LEN)
31429afe9e6SSimon Glass gd->malloc_base = CONFIG_MALLOC_F_ADDR;
31529afe9e6SSimon Glass #endif
316bb967240SSimon Glass setup_ram_buf(state);
3174d94dfa0SSimon Glass
318808434cdSSimon Glass /* Do pre- and post-relocation init */
319bace3d00SSimon Glass board_init_f(0);
3208ec21bbeSAllen Martin
321808434cdSSimon Glass board_init_r(gd->new_gd, 0);
322808434cdSSimon Glass
323808434cdSSimon Glass /* NOTREACHED - board_init_r() does not return */
3248ec21bbeSAllen Martin return 0;
3251209e272SSimon Glass
3261209e272SSimon Glass err:
3271209e272SSimon Glass printf("Error %d\n", ret);
3281209e272SSimon Glass return 1;
329bace3d00SSimon Glass }
330