xref: /openbmc/u-boot/arch/sandbox/cpu/start.c (revision d94604d5)
183d290c5STom Rini // SPDX-License-Identifier: GPL-2.0+
2bace3d00SSimon Glass /*
36fb62078SSimon Glass  * Copyright (c) 2011-2012 The Chromium OS Authors.
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) {
1801c5a81d8SSimon Glass 		printf("Failed to read RAM buffer '%s': %d\n", arg, err);
1815c2859cdSSimon Glass 		return err;
1825c2859cdSSimon Glass 	}
183*a65d1a06SSimon Glass 	state->ram_buf_read = true;
1845c2859cdSSimon Glass 
1855c2859cdSSimon Glass 	return 0;
1865c2859cdSSimon Glass }
1875c2859cdSSimon Glass SANDBOX_CMDLINE_OPT_SHORT(memory, 'm', 1,
1885c2859cdSSimon Glass 			  "Read/write ram_buf memory contents from file");
1895c2859cdSSimon Glass 
sandbox_cmdline_cb_rm_memory(struct sandbox_state * state,const char * arg)190ab839dc3SSimon Glass static int sandbox_cmdline_cb_rm_memory(struct sandbox_state *state,
191ab839dc3SSimon Glass 					const char *arg)
192ab839dc3SSimon Glass {
193ab839dc3SSimon Glass 	state->ram_buf_rm = true;
194ab839dc3SSimon Glass 
195ab839dc3SSimon Glass 	return 0;
196ab839dc3SSimon Glass }
197ab839dc3SSimon Glass SANDBOX_CMDLINE_OPT(rm_memory, 0, "Remove memory file after reading");
198ab839dc3SSimon Glass 
sandbox_cmdline_cb_state(struct sandbox_state * state,const char * arg)1991209e272SSimon Glass static int sandbox_cmdline_cb_state(struct sandbox_state *state,
2001209e272SSimon Glass 				    const char *arg)
2011209e272SSimon Glass {
2021209e272SSimon Glass 	state->state_fname = arg;
2031209e272SSimon Glass 	return 0;
2041209e272SSimon Glass }
2051209e272SSimon Glass SANDBOX_CMDLINE_OPT_SHORT(state, 's', 1, "Specify the sandbox state FDT");
2061209e272SSimon Glass 
sandbox_cmdline_cb_read(struct sandbox_state * state,const char * arg)2071209e272SSimon Glass static int sandbox_cmdline_cb_read(struct sandbox_state *state,
2081209e272SSimon Glass 				   const char *arg)
2091209e272SSimon Glass {
2101209e272SSimon Glass 	state->read_state = true;
2111209e272SSimon Glass 	return 0;
2121209e272SSimon Glass }
2131209e272SSimon Glass SANDBOX_CMDLINE_OPT_SHORT(read, 'r', 0, "Read the state FDT on startup");
2141209e272SSimon Glass 
sandbox_cmdline_cb_write(struct sandbox_state * state,const char * arg)2151209e272SSimon Glass static int sandbox_cmdline_cb_write(struct sandbox_state *state,
2161209e272SSimon Glass 				    const char *arg)
2171209e272SSimon Glass {
2181209e272SSimon Glass 	state->write_state = true;
2191209e272SSimon Glass 	return 0;
2201209e272SSimon Glass }
2211209e272SSimon Glass SANDBOX_CMDLINE_OPT_SHORT(write, 'w', 0, "Write state FDT on exit");
2221209e272SSimon Glass 
sandbox_cmdline_cb_ignore_missing(struct sandbox_state * state,const char * arg)2231209e272SSimon Glass static int sandbox_cmdline_cb_ignore_missing(struct sandbox_state *state,
2241209e272SSimon Glass 					     const char *arg)
2251209e272SSimon Glass {
2261209e272SSimon Glass 	state->ignore_missing_state_on_read = true;
2271209e272SSimon Glass 	return 0;
2281209e272SSimon Glass }
2291209e272SSimon Glass SANDBOX_CMDLINE_OPT_SHORT(ignore_missing, 'n', 0,
2301209e272SSimon Glass 			  "Ignore missing state on read");
2311209e272SSimon Glass 
sandbox_cmdline_cb_show_lcd(struct sandbox_state * state,const char * arg)2327d95f2a3SSimon Glass static int sandbox_cmdline_cb_show_lcd(struct sandbox_state *state,
2337d95f2a3SSimon Glass 				       const char *arg)
2347d95f2a3SSimon Glass {
2357d95f2a3SSimon Glass 	state->show_lcd = true;
2367d95f2a3SSimon Glass 	return 0;
2377d95f2a3SSimon Glass }
2387d95f2a3SSimon Glass SANDBOX_CMDLINE_OPT_SHORT(show_lcd, 'l', 0,
2397d95f2a3SSimon Glass 			  "Show the sandbox LCD display");
2407d95f2a3SSimon Glass 
241ffb87905SSimon Glass static const char *term_args[STATE_TERM_COUNT] = {
242ffb87905SSimon Glass 	"raw-with-sigs",
243ffb87905SSimon Glass 	"raw",
244ffb87905SSimon Glass 	"cooked",
245ffb87905SSimon Glass };
246ffb87905SSimon Glass 
sandbox_cmdline_cb_terminal(struct sandbox_state * state,const char * arg)247ffb87905SSimon Glass static int sandbox_cmdline_cb_terminal(struct sandbox_state *state,
248ffb87905SSimon Glass 				       const char *arg)
249ffb87905SSimon Glass {
250ffb87905SSimon Glass 	int i;
251ffb87905SSimon Glass 
252ffb87905SSimon Glass 	for (i = 0; i < STATE_TERM_COUNT; i++) {
253ffb87905SSimon Glass 		if (!strcmp(arg, term_args[i])) {
254ffb87905SSimon Glass 			state->term_raw = i;
255ffb87905SSimon Glass 			return 0;
256ffb87905SSimon Glass 		}
257ffb87905SSimon Glass 	}
258ffb87905SSimon Glass 
259ffb87905SSimon Glass 	printf("Unknown terminal setting '%s' (", arg);
260ffb87905SSimon Glass 	for (i = 0; i < STATE_TERM_COUNT; i++)
261ffb87905SSimon Glass 		printf("%s%s", i ? ", " : "", term_args[i]);
262ffb87905SSimon Glass 	puts(")\n");
263ffb87905SSimon Glass 
264ffb87905SSimon Glass 	return 1;
265ffb87905SSimon Glass }
266ffb87905SSimon Glass SANDBOX_CMDLINE_OPT_SHORT(terminal, 't', 1,
267ffb87905SSimon Glass 			  "Set terminal to raw/cooked mode");
268ffb87905SSimon Glass 
sandbox_cmdline_cb_verbose(struct sandbox_state * state,const char * arg)2699ce8b402SSimon Glass static int sandbox_cmdline_cb_verbose(struct sandbox_state *state,
2709ce8b402SSimon Glass 				      const char *arg)
2719ce8b402SSimon Glass {
2729ce8b402SSimon Glass 	state->show_test_output = true;
2739ce8b402SSimon Glass 	return 0;
2749ce8b402SSimon Glass }
2759ce8b402SSimon Glass SANDBOX_CMDLINE_OPT_SHORT(verbose, 'v', 0, "Show test output");
2769ce8b402SSimon Glass 
sandbox_cmdline_cb_log_level(struct sandbox_state * state,const char * arg)2772b1dc29aSSimon Glass static int sandbox_cmdline_cb_log_level(struct sandbox_state *state,
2782b1dc29aSSimon Glass 					const char *arg)
2792b1dc29aSSimon Glass {
2802b1dc29aSSimon Glass 	state->default_log_level = simple_strtol(arg, NULL, 10);
2812b1dc29aSSimon Glass 
2822b1dc29aSSimon Glass 	return 0;
2832b1dc29aSSimon Glass }
2842b1dc29aSSimon Glass SANDBOX_CMDLINE_OPT_SHORT(log_level, 'L', 1,
2852b1dc29aSSimon Glass 			  "Set log level (0=panic, 7=debug)");
2862b1dc29aSSimon Glass 
sandbox_cmdline_cb_show_of_platdata(struct sandbox_state * state,const char * arg)2871ca910beSSimon Glass static int sandbox_cmdline_cb_show_of_platdata(struct sandbox_state *state,
2881ca910beSSimon Glass 					       const char *arg)
2891ca910beSSimon Glass {
2901ca910beSSimon Glass 	state->show_of_platdata = true;
2911ca910beSSimon Glass 
2921ca910beSSimon Glass 	return 0;
2931ca910beSSimon Glass }
2941ca910beSSimon Glass SANDBOX_CMDLINE_OPT(show_of_platdata, 0, "Show of-platdata in SPL");
2951ca910beSSimon Glass 
board_run_command(const char * cmdline)2962b6793deSSimon Glass int board_run_command(const char *cmdline)
2972b6793deSSimon Glass {
2982b6793deSSimon Glass 	printf("## Commands are disabled. Please enable CONFIG_CMDLINE.\n");
2992b6793deSSimon Glass 
3002b6793deSSimon Glass 	return 1;
3012b6793deSSimon Glass }
3022b6793deSSimon Glass 
setup_ram_buf(struct sandbox_state * state)303bb967240SSimon Glass static void setup_ram_buf(struct sandbox_state *state)
304bb967240SSimon Glass {
305*a65d1a06SSimon Glass 	/* Zero the RAM buffer if we didn't read it, to keep valgrind happy */
306*a65d1a06SSimon Glass 	if (!state->ram_buf_read) {
307*a65d1a06SSimon Glass 		memset(state->ram_buf, '\0', state->ram_size);
308*a65d1a06SSimon Glass 		printf("clear %p %x\n", state->ram_buf, state->ram_size);
309*a65d1a06SSimon Glass 	}
310*a65d1a06SSimon Glass 
311bb967240SSimon Glass 	gd->arch.ram_buf = state->ram_buf;
312bb967240SSimon Glass 	gd->ram_size = state->ram_size;
313bb967240SSimon Glass }
314bb967240SSimon Glass 
state_show(struct sandbox_state * state)315d66ddafaSSimon Glass void state_show(struct sandbox_state *state)
316d66ddafaSSimon Glass {
317d66ddafaSSimon Glass 	char **p;
318d66ddafaSSimon Glass 
319d66ddafaSSimon Glass 	printf("Arguments:\n");
320d66ddafaSSimon Glass 	for (p = state->argv; *p; p++)
321d66ddafaSSimon Glass 		printf("%s ", *p);
322d66ddafaSSimon Glass 	printf("\n");
323d66ddafaSSimon Glass }
324d66ddafaSSimon Glass 
main(int argc,char * argv[])325bace3d00SSimon Glass int main(int argc, char *argv[])
326bace3d00SSimon Glass {
32770db4212SSimon Glass 	struct sandbox_state *state;
3284d94dfa0SSimon Glass 	gd_t data;
3291209e272SSimon Glass 	int ret;
3306fb62078SSimon Glass 
3311209e272SSimon Glass 	ret = state_init();
3321209e272SSimon Glass 	if (ret)
3331209e272SSimon Glass 		goto err;
3346fb62078SSimon Glass 
33570db4212SSimon Glass 	state = state_get_current();
33670db4212SSimon Glass 	if (os_parse_args(state, argc, argv))
33770db4212SSimon Glass 		return 1;
33870db4212SSimon Glass 
3391209e272SSimon Glass 	ret = sandbox_read_state(state, state->state_fname);
3401209e272SSimon Glass 	if (ret)
3411209e272SSimon Glass 		goto err;
3421209e272SSimon Glass 
3434d94dfa0SSimon Glass 	memset(&data, '\0', sizeof(data));
3444d94dfa0SSimon Glass 	gd = &data;
3451fc50d72SAndy Yan #if CONFIG_VAL(SYS_MALLOC_F_LEN)
34629afe9e6SSimon Glass 	gd->malloc_base = CONFIG_MALLOC_F_ADDR;
34729afe9e6SSimon Glass #endif
3482b1dc29aSSimon Glass #if CONFIG_IS_ENABLED(LOG)
3492b1dc29aSSimon Glass 	gd->default_log_level = state->default_log_level;
3502b1dc29aSSimon Glass #endif
351bb967240SSimon Glass 	setup_ram_buf(state);
3524d94dfa0SSimon Glass 
353808434cdSSimon Glass 	/* Do pre- and post-relocation init */
354bace3d00SSimon Glass 	board_init_f(0);
3558ec21bbeSAllen Martin 
356808434cdSSimon Glass 	board_init_r(gd->new_gd, 0);
357808434cdSSimon Glass 
358808434cdSSimon Glass 	/* NOTREACHED - board_init_r() does not return */
3598ec21bbeSAllen Martin 	return 0;
3601209e272SSimon Glass 
3611209e272SSimon Glass err:
3621209e272SSimon Glass 	printf("Error %d\n", ret);
3631209e272SSimon Glass 	return 1;
364bace3d00SSimon Glass }
365