xref: /openbmc/u-boot/common/autoboot.c (revision 83d290c56fab2d38cd1ab4c4cc7099559c1d5046)
1*83d290c5STom Rini // SPDX-License-Identifier: GPL-2.0+
266ded17dSSimon Glass /*
366ded17dSSimon Glass  * (C) Copyright 2000
466ded17dSSimon Glass  * Wolfgang Denk, DENX Software Engineering, wd@denx.de.
566ded17dSSimon Glass  */
666ded17dSSimon Glass 
766ded17dSSimon Glass #include <common.h>
839e1230eSJeroen Hofstee #include <autoboot.h>
90098e179SSimon Glass #include <bootretry.h>
1066ded17dSSimon Glass #include <cli.h>
1124b852a7SSimon Glass #include <console.h>
1266ded17dSSimon Glass #include <fdtdec.h>
1366ded17dSSimon Glass #include <menu.h>
1466ded17dSSimon Glass #include <post.h>
158f0b1e24SStefan Roese #include <u-boot/sha256.h>
1666ded17dSSimon Glass 
1766ded17dSSimon Glass DECLARE_GLOBAL_DATA_PTR;
1866ded17dSSimon Glass 
1966ded17dSSimon Glass #define MAX_DELAY_STOP_STR 32
2066ded17dSSimon Glass 
2166ded17dSSimon Glass #ifndef DEBUG_BOOTKEYS
2266ded17dSSimon Glass #define DEBUG_BOOTKEYS 0
2366ded17dSSimon Glass #endif
2466ded17dSSimon Glass #define debug_bootkeys(fmt, args...)		\
2566ded17dSSimon Glass 	debug_cond(DEBUG_BOOTKEYS, fmt, ##args)
2666ded17dSSimon Glass 
27affb2156SSimon Glass /* Stored value of bootdelay, used by autoboot_command() */
28affb2156SSimon Glass static int stored_bootdelay;
29affb2156SSimon Glass 
3066ded17dSSimon Glass #if defined(CONFIG_AUTOBOOT_KEYED)
318f0b1e24SStefan Roese #if defined(CONFIG_AUTOBOOT_STOP_STR_SHA256)
328f0b1e24SStefan Roese 
338f0b1e24SStefan Roese /*
348f0b1e24SStefan Roese  * Use a "constant-length" time compare function for this
358f0b1e24SStefan Roese  * hash compare:
368f0b1e24SStefan Roese  *
378f0b1e24SStefan Roese  * https://crackstation.net/hashing-security.htm
388f0b1e24SStefan Roese  */
398f0b1e24SStefan Roese static int slow_equals(u8 *a, u8 *b, int len)
408f0b1e24SStefan Roese {
418f0b1e24SStefan Roese 	int diff = 0;
428f0b1e24SStefan Roese 	int i;
438f0b1e24SStefan Roese 
448f0b1e24SStefan Roese 	for (i = 0; i < len; i++)
458f0b1e24SStefan Roese 		diff |= a[i] ^ b[i];
468f0b1e24SStefan Roese 
478f0b1e24SStefan Roese 	return diff == 0;
488f0b1e24SStefan Roese }
498f0b1e24SStefan Roese 
508f0b1e24SStefan Roese static int passwd_abort(uint64_t etime)
518f0b1e24SStefan Roese {
5200caae6dSSimon Glass 	const char *sha_env_str = env_get("bootstopkeysha256");
538f0b1e24SStefan Roese 	u8 sha_env[SHA256_SUM_LEN];
548f0b1e24SStefan Roese 	u8 sha[SHA256_SUM_LEN];
558f0b1e24SStefan Roese 	char presskey[MAX_DELAY_STOP_STR];
568f0b1e24SStefan Roese 	const char *algo_name = "sha256";
578f0b1e24SStefan Roese 	u_int presskey_len = 0;
588f0b1e24SStefan Roese 	int abort = 0;
592d06fd83SMartin Etnestad 	int size = sizeof(sha);
608f0b1e24SStefan Roese 	int ret;
618f0b1e24SStefan Roese 
628f0b1e24SStefan Roese 	if (sha_env_str == NULL)
638f0b1e24SStefan Roese 		sha_env_str = CONFIG_AUTOBOOT_STOP_STR_SHA256;
648f0b1e24SStefan Roese 
658f0b1e24SStefan Roese 	/*
668f0b1e24SStefan Roese 	 * Generate the binary value from the environment hash value
678f0b1e24SStefan Roese 	 * so that we can compare this value with the computed hash
688f0b1e24SStefan Roese 	 * from the user input
698f0b1e24SStefan Roese 	 */
708f0b1e24SStefan Roese 	ret = hash_parse_string(algo_name, sha_env_str, sha_env);
718f0b1e24SStefan Roese 	if (ret) {
728f0b1e24SStefan Roese 		printf("Hash %s not supported!\n", algo_name);
738f0b1e24SStefan Roese 		return 0;
748f0b1e24SStefan Roese 	}
758f0b1e24SStefan Roese 
768f0b1e24SStefan Roese 	/*
778f0b1e24SStefan Roese 	 * We don't know how long the stop-string is, so we need to
788f0b1e24SStefan Roese 	 * generate the sha256 hash upon each input character and
798f0b1e24SStefan Roese 	 * compare the value with the one saved in the environment
808f0b1e24SStefan Roese 	 */
818f0b1e24SStefan Roese 	do {
828f0b1e24SStefan Roese 		if (tstc()) {
838f0b1e24SStefan Roese 			/* Check for input string overflow */
848f0b1e24SStefan Roese 			if (presskey_len >= MAX_DELAY_STOP_STR)
858f0b1e24SStefan Roese 				return 0;
868f0b1e24SStefan Roese 
878f0b1e24SStefan Roese 			presskey[presskey_len++] = getc();
888f0b1e24SStefan Roese 
898f0b1e24SStefan Roese 			/* Calculate sha256 upon each new char */
908f0b1e24SStefan Roese 			hash_block(algo_name, (const void *)presskey,
918f0b1e24SStefan Roese 				   presskey_len, sha, &size);
928f0b1e24SStefan Roese 
938f0b1e24SStefan Roese 			/* And check if sha matches saved value in env */
948f0b1e24SStefan Roese 			if (slow_equals(sha, sha_env, SHA256_SUM_LEN))
958f0b1e24SStefan Roese 				abort = 1;
968f0b1e24SStefan Roese 		}
978f0b1e24SStefan Roese 	} while (!abort && get_ticks() <= etime);
988f0b1e24SStefan Roese 
998f0b1e24SStefan Roese 	return abort;
1008f0b1e24SStefan Roese }
1018f0b1e24SStefan Roese #else
1028f0b1e24SStefan Roese static int passwd_abort(uint64_t etime)
10366ded17dSSimon Glass {
10466ded17dSSimon Glass 	int abort = 0;
10566ded17dSSimon Glass 	struct {
10666ded17dSSimon Glass 		char *str;
10766ded17dSSimon Glass 		u_int len;
10866ded17dSSimon Glass 		int retry;
10966ded17dSSimon Glass 	}
11066ded17dSSimon Glass 	delaykey[] = {
11100caae6dSSimon Glass 		{ .str = env_get("bootdelaykey"),  .retry = 1 },
11200caae6dSSimon Glass 		{ .str = env_get("bootstopkey"),   .retry = 0 },
11366ded17dSSimon Glass 	};
11466ded17dSSimon Glass 
11566ded17dSSimon Glass 	char presskey[MAX_DELAY_STOP_STR];
11666ded17dSSimon Glass 	u_int presskey_len = 0;
11766ded17dSSimon Glass 	u_int presskey_max = 0;
11866ded17dSSimon Glass 	u_int i;
11966ded17dSSimon Glass 
12066ded17dSSimon Glass #  ifdef CONFIG_AUTOBOOT_DELAY_STR
12166ded17dSSimon Glass 	if (delaykey[0].str == NULL)
12266ded17dSSimon Glass 		delaykey[0].str = CONFIG_AUTOBOOT_DELAY_STR;
12366ded17dSSimon Glass #  endif
12466ded17dSSimon Glass #  ifdef CONFIG_AUTOBOOT_STOP_STR
1252d908fa0SStefan Roese 	if (delaykey[1].str == NULL)
1262d908fa0SStefan Roese 		delaykey[1].str = CONFIG_AUTOBOOT_STOP_STR;
12766ded17dSSimon Glass #  endif
12866ded17dSSimon Glass 
12966ded17dSSimon Glass 	for (i = 0; i < sizeof(delaykey) / sizeof(delaykey[0]); i++) {
13066ded17dSSimon Glass 		delaykey[i].len = delaykey[i].str == NULL ?
13166ded17dSSimon Glass 				    0 : strlen(delaykey[i].str);
13266ded17dSSimon Glass 		delaykey[i].len = delaykey[i].len > MAX_DELAY_STOP_STR ?
13366ded17dSSimon Glass 				    MAX_DELAY_STOP_STR : delaykey[i].len;
13466ded17dSSimon Glass 
13566ded17dSSimon Glass 		presskey_max = presskey_max > delaykey[i].len ?
13666ded17dSSimon Glass 				    presskey_max : delaykey[i].len;
13766ded17dSSimon Glass 
13866ded17dSSimon Glass 		debug_bootkeys("%s key:<%s>\n",
13966ded17dSSimon Glass 			       delaykey[i].retry ? "delay" : "stop",
14066ded17dSSimon Glass 			       delaykey[i].str ? delaykey[i].str : "NULL");
14166ded17dSSimon Glass 	}
14266ded17dSSimon Glass 
14366ded17dSSimon Glass 	/* In order to keep up with incoming data, check timeout only
14466ded17dSSimon Glass 	 * when catch up.
14566ded17dSSimon Glass 	 */
14666ded17dSSimon Glass 	do {
14766ded17dSSimon Glass 		if (tstc()) {
14866ded17dSSimon Glass 			if (presskey_len < presskey_max) {
14966ded17dSSimon Glass 				presskey[presskey_len++] = getc();
15066ded17dSSimon Glass 			} else {
15166ded17dSSimon Glass 				for (i = 0; i < presskey_max - 1; i++)
15266ded17dSSimon Glass 					presskey[i] = presskey[i + 1];
15366ded17dSSimon Glass 
15466ded17dSSimon Glass 				presskey[i] = getc();
15566ded17dSSimon Glass 			}
15666ded17dSSimon Glass 		}
15766ded17dSSimon Glass 
15866ded17dSSimon Glass 		for (i = 0; i < sizeof(delaykey) / sizeof(delaykey[0]); i++) {
15966ded17dSSimon Glass 			if (delaykey[i].len > 0 &&
16066ded17dSSimon Glass 			    presskey_len >= delaykey[i].len &&
16166ded17dSSimon Glass 				memcmp(presskey + presskey_len -
16266ded17dSSimon Glass 					delaykey[i].len, delaykey[i].str,
16366ded17dSSimon Glass 					delaykey[i].len) == 0) {
16466ded17dSSimon Glass 					debug_bootkeys("got %skey\n",
16566ded17dSSimon Glass 						delaykey[i].retry ? "delay" :
16666ded17dSSimon Glass 						"stop");
16766ded17dSSimon Glass 
16866ded17dSSimon Glass 				/* don't retry auto boot */
16966ded17dSSimon Glass 				if (!delaykey[i].retry)
17066ded17dSSimon Glass 					bootretry_dont_retry();
17166ded17dSSimon Glass 				abort = 1;
17266ded17dSSimon Glass 			}
17366ded17dSSimon Glass 		}
17466ded17dSSimon Glass 	} while (!abort && get_ticks() <= etime);
17566ded17dSSimon Glass 
1768f0b1e24SStefan Roese 	return abort;
1778f0b1e24SStefan Roese }
1788f0b1e24SStefan Roese #endif
1798f0b1e24SStefan Roese 
1808f0b1e24SStefan Roese /***************************************************************************
1818f0b1e24SStefan Roese  * Watch for 'delay' seconds for autoboot stop or autoboot delay string.
1828f0b1e24SStefan Roese  * returns: 0 -  no key string, allow autoboot 1 - got key string, abort
1838f0b1e24SStefan Roese  */
184d8da8298SMasahiro Yamada static int __abortboot(int bootdelay)
1858f0b1e24SStefan Roese {
1868f0b1e24SStefan Roese 	int abort;
1878f0b1e24SStefan Roese 	uint64_t etime = endtick(bootdelay);
1888f0b1e24SStefan Roese 
1898f0b1e24SStefan Roese #  ifdef CONFIG_AUTOBOOT_PROMPT
1908f0b1e24SStefan Roese 	/*
1918f0b1e24SStefan Roese 	 * CONFIG_AUTOBOOT_PROMPT includes the %d for all boards.
1928f0b1e24SStefan Roese 	 * To print the bootdelay value upon bootup.
1938f0b1e24SStefan Roese 	 */
1948f0b1e24SStefan Roese 	printf(CONFIG_AUTOBOOT_PROMPT, bootdelay);
1958f0b1e24SStefan Roese #  endif
1968f0b1e24SStefan Roese 
1978f0b1e24SStefan Roese 	abort = passwd_abort(etime);
19866ded17dSSimon Glass 	if (!abort)
19966ded17dSSimon Glass 		debug_bootkeys("key timeout\n");
20066ded17dSSimon Glass 
20166ded17dSSimon Glass 	return abort;
20266ded17dSSimon Glass }
20366ded17dSSimon Glass 
20466ded17dSSimon Glass # else	/* !defined(CONFIG_AUTOBOOT_KEYED) */
20566ded17dSSimon Glass 
20666ded17dSSimon Glass #ifdef CONFIG_MENUKEY
20766ded17dSSimon Glass static int menukey;
20866ded17dSSimon Glass #endif
20966ded17dSSimon Glass 
210d8da8298SMasahiro Yamada static int __abortboot(int bootdelay)
21166ded17dSSimon Glass {
21266ded17dSSimon Glass 	int abort = 0;
21366ded17dSSimon Glass 	unsigned long ts;
21466ded17dSSimon Glass 
21566ded17dSSimon Glass #ifdef CONFIG_MENUPROMPT
21666ded17dSSimon Glass 	printf(CONFIG_MENUPROMPT);
21766ded17dSSimon Glass #else
21866ded17dSSimon Glass 	printf("Hit any key to stop autoboot: %2d ", bootdelay);
21966ded17dSSimon Glass #endif
22066ded17dSSimon Glass 
22166ded17dSSimon Glass 	/*
22266ded17dSSimon Glass 	 * Check if key already pressed
22366ded17dSSimon Glass 	 */
22466ded17dSSimon Glass 	if (tstc()) {	/* we got a key press	*/
22566ded17dSSimon Glass 		(void) getc();  /* consume input	*/
22666ded17dSSimon Glass 		puts("\b\b\b 0");
22766ded17dSSimon Glass 		abort = 1;	/* don't auto boot	*/
22866ded17dSSimon Glass 	}
22966ded17dSSimon Glass 
23066ded17dSSimon Glass 	while ((bootdelay > 0) && (!abort)) {
23166ded17dSSimon Glass 		--bootdelay;
23266ded17dSSimon Glass 		/* delay 1000 ms */
23366ded17dSSimon Glass 		ts = get_timer(0);
23466ded17dSSimon Glass 		do {
23566ded17dSSimon Glass 			if (tstc()) {	/* we got a key press	*/
23666ded17dSSimon Glass 				abort  = 1;	/* don't auto boot	*/
23766ded17dSSimon Glass 				bootdelay = 0;	/* no more delay	*/
23866ded17dSSimon Glass # ifdef CONFIG_MENUKEY
23966ded17dSSimon Glass 				menukey = getc();
24066ded17dSSimon Glass # else
24166ded17dSSimon Glass 				(void) getc();  /* consume input	*/
24266ded17dSSimon Glass # endif
24366ded17dSSimon Glass 				break;
24466ded17dSSimon Glass 			}
24566ded17dSSimon Glass 			udelay(10000);
24666ded17dSSimon Glass 		} while (!abort && get_timer(ts) < 1000);
24766ded17dSSimon Glass 
24866ded17dSSimon Glass 		printf("\b\b\b%2d ", bootdelay);
24966ded17dSSimon Glass 	}
25066ded17dSSimon Glass 
25166ded17dSSimon Glass 	putc('\n');
25266ded17dSSimon Glass 
25366ded17dSSimon Glass 	return abort;
25466ded17dSSimon Glass }
25566ded17dSSimon Glass # endif	/* CONFIG_AUTOBOOT_KEYED */
25666ded17dSSimon Glass 
25766ded17dSSimon Glass static int abortboot(int bootdelay)
25866ded17dSSimon Glass {
25946327392SMasahiro Yamada 	int abort = 0;
26009b9d9e5SMasahiro Yamada 
26146327392SMasahiro Yamada 	if (bootdelay >= 0)
26209b9d9e5SMasahiro Yamada 		abort = __abortboot(bootdelay);
26309b9d9e5SMasahiro Yamada 
26409b9d9e5SMasahiro Yamada #ifdef CONFIG_SILENT_CONSOLE
26509b9d9e5SMasahiro Yamada 	if (abort)
26609b9d9e5SMasahiro Yamada 		gd->flags &= ~GD_FLG_SILENT;
26709b9d9e5SMasahiro Yamada #endif
26809b9d9e5SMasahiro Yamada 
26909b9d9e5SMasahiro Yamada 	return abort;
27066ded17dSSimon Glass }
27166ded17dSSimon Glass 
27266ded17dSSimon Glass static void process_fdt_options(const void *blob)
27366ded17dSSimon Glass {
2749f73690cSStefan Roese #if defined(CONFIG_OF_CONTROL) && defined(CONFIG_SYS_TEXT_BASE)
27566ded17dSSimon Glass 	ulong addr;
27666ded17dSSimon Glass 
27766ded17dSSimon Glass 	/* Add an env variable to point to a kernel payload, if available */
27866ded17dSSimon Glass 	addr = fdtdec_get_config_int(gd->fdt_blob, "kernel-offset", 0);
27966ded17dSSimon Glass 	if (addr)
280018f5303SSimon Glass 		env_set_addr("kernaddr", (void *)(CONFIG_SYS_TEXT_BASE + addr));
28166ded17dSSimon Glass 
28266ded17dSSimon Glass 	/* Add an env variable to point to a root disk, if available */
28366ded17dSSimon Glass 	addr = fdtdec_get_config_int(gd->fdt_blob, "rootdisk-offset", 0);
28466ded17dSSimon Glass 	if (addr)
285018f5303SSimon Glass 		env_set_addr("rootaddr", (void *)(CONFIG_SYS_TEXT_BASE + addr));
2869f73690cSStefan Roese #endif /* CONFIG_OF_CONTROL && CONFIG_SYS_TEXT_BASE */
287affb2156SSimon Glass }
28866ded17dSSimon Glass 
289affb2156SSimon Glass const char *bootdelay_process(void)
29066ded17dSSimon Glass {
29166ded17dSSimon Glass 	char *s;
29266ded17dSSimon Glass 	int bootdelay;
29366ded17dSSimon Glass #ifdef CONFIG_BOOTCOUNT_LIMIT
29466ded17dSSimon Glass 	unsigned long bootcount = 0;
29566ded17dSSimon Glass 	unsigned long bootlimit = 0;
29666ded17dSSimon Glass #endif /* CONFIG_BOOTCOUNT_LIMIT */
29766ded17dSSimon Glass 
29866ded17dSSimon Glass #ifdef CONFIG_BOOTCOUNT_LIMIT
29966ded17dSSimon Glass 	bootcount = bootcount_load();
30066ded17dSSimon Glass 	bootcount++;
30166ded17dSSimon Glass 	bootcount_store(bootcount);
302018f5303SSimon Glass 	env_set_ulong("bootcount", bootcount);
303bfebc8c9SSimon Glass 	bootlimit = env_get_ulong("bootlimit", 10, 0);
30466ded17dSSimon Glass #endif /* CONFIG_BOOTCOUNT_LIMIT */
30566ded17dSSimon Glass 
30600caae6dSSimon Glass 	s = env_get("bootdelay");
30766ded17dSSimon Glass 	bootdelay = s ? (int)simple_strtol(s, NULL, 10) : CONFIG_BOOTDELAY;
30866ded17dSSimon Glass 
30966ded17dSSimon Glass #ifdef CONFIG_OF_CONTROL
31066ded17dSSimon Glass 	bootdelay = fdtdec_get_config_int(gd->fdt_blob, "bootdelay",
31166ded17dSSimon Glass 			bootdelay);
31266ded17dSSimon Glass #endif
31366ded17dSSimon Glass 
31466ded17dSSimon Glass 	debug("### main_loop entered: bootdelay=%d\n\n", bootdelay);
31566ded17dSSimon Glass 
31666ded17dSSimon Glass #if defined(CONFIG_MENU_SHOW)
31766ded17dSSimon Glass 	bootdelay = menu_show(bootdelay);
31866ded17dSSimon Glass #endif
319b26440f1SSimon Glass 	bootretry_init_cmd_timeout();
32066ded17dSSimon Glass 
32166ded17dSSimon Glass #ifdef CONFIG_POST
32266ded17dSSimon Glass 	if (gd->flags & GD_FLG_POSTFAIL) {
32300caae6dSSimon Glass 		s = env_get("failbootcmd");
32466ded17dSSimon Glass 	} else
32566ded17dSSimon Glass #endif /* CONFIG_POST */
32666ded17dSSimon Glass #ifdef CONFIG_BOOTCOUNT_LIMIT
32766ded17dSSimon Glass 	if (bootlimit && (bootcount > bootlimit)) {
32866ded17dSSimon Glass 		printf("Warning: Bootlimit (%u) exceeded. Using altbootcmd.\n",
32966ded17dSSimon Glass 		       (unsigned)bootlimit);
33000caae6dSSimon Glass 		s = env_get("altbootcmd");
33166ded17dSSimon Glass 	} else
33266ded17dSSimon Glass #endif /* CONFIG_BOOTCOUNT_LIMIT */
33300caae6dSSimon Glass 		s = env_get("bootcmd");
33466ded17dSSimon Glass 
33566ded17dSSimon Glass 	process_fdt_options(gd->fdt_blob);
336affb2156SSimon Glass 	stored_bootdelay = bootdelay;
33766ded17dSSimon Glass 
338affb2156SSimon Glass 	return s;
339affb2156SSimon Glass }
34066ded17dSSimon Glass 
341affb2156SSimon Glass void autoboot_command(const char *s)
342affb2156SSimon Glass {
34366ded17dSSimon Glass 	debug("### main_loop: bootcmd=\"%s\"\n", s ? s : "<UNDEFINED>");
34466ded17dSSimon Glass 
345affb2156SSimon Glass 	if (stored_bootdelay != -1 && s && !abortboot(stored_bootdelay)) {
34666ded17dSSimon Glass #if defined(CONFIG_AUTOBOOT_KEYED) && !defined(CONFIG_AUTOBOOT_KEYED_CTRLC)
34766ded17dSSimon Glass 		int prev = disable_ctrlc(1);	/* disable Control C checking */
34866ded17dSSimon Glass #endif
34966ded17dSSimon Glass 
35066ded17dSSimon Glass 		run_command_list(s, -1, 0);
35166ded17dSSimon Glass 
35266ded17dSSimon Glass #if defined(CONFIG_AUTOBOOT_KEYED) && !defined(CONFIG_AUTOBOOT_KEYED_CTRLC)
35366ded17dSSimon Glass 		disable_ctrlc(prev);	/* restore Control C checking */
35466ded17dSSimon Glass #endif
35566ded17dSSimon Glass 	}
35666ded17dSSimon Glass 
35766ded17dSSimon Glass #ifdef CONFIG_MENUKEY
35866ded17dSSimon Glass 	if (menukey == CONFIG_MENUKEY) {
35900caae6dSSimon Glass 		s = env_get("menucmd");
36066ded17dSSimon Glass 		if (s)
36166ded17dSSimon Glass 			run_command_list(s, -1, 0);
36266ded17dSSimon Glass 	}
36366ded17dSSimon Glass #endif /* CONFIG_MENUKEY */
36466ded17dSSimon Glass }
365