在configs/rk3506_defconfig
文件中可以配置启动延时选项
common/main.c –> main_loop:bootdelay_process
处理delay延时值解析,按键捕获由autoboot_command
负责
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 void main_loop (void ) { const char *s; bootstage_mark_name(BOOTSTAGE_ID_MAIN_LOOP, "main_loop" ); #ifdef CONFIG_VERSION_VARIABLE env_set("ver" , version_string); #endif cli_init(); run_preboot_environment_command(); #if defined(CONFIG_UPDATE_TFTP) update_tftp(0UL , NULL , NULL ); #endif s = bootdelay_process(); if (cli_process_fdt(&s)) cli_secure_boot_cmd(s); autoboot_command(s); cli_loop(); panic("No CLI available" ); }
common/autoboot.c –> bootdelay_process:要么从环境变量bootdelay
里边获取,要么就从CONFIG_BOOTDELAY
里边获取,最后更新stored_bootdelay
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 const char *bootdelay_process (void ) { char *s; int bootdelay; #ifdef CONFIG_BOOTCOUNT_LIMIT unsigned long bootcount = 0 ; unsigned long bootlimit = 0 ; #endif #ifdef CONFIG_BOOTCOUNT_LIMIT bootcount = bootcount_load(); bootcount++; bootcount_store(bootcount); env_set_ulong("bootcount" , bootcount); bootlimit = env_get_ulong("bootlimit" , 10 , 0 ); #endif s = env_get("bootdelay" ); bootdelay = s ? (int )simple_strtol(s, NULL , 10 ) : CONFIG_BOOTDELAY; #ifdef CONFIG_OF_CONTROL bootdelay = fdtdec_get_config_int(gd->fdt_blob, "bootdelay" , bootdelay); #endif debug("### main_loop entered: bootdelay=%d\n\n" , bootdelay); #if defined(CONFIG_MENU_SHOW) bootdelay = menu_show(bootdelay); #endif bootretry_init_cmd_timeout(); #ifdef CONFIG_POST if (gd->flags & GD_FLG_POSTFAIL) { s = env_get("failbootcmd" ); } else #endif #ifdef CONFIG_BOOTCOUNT_LIMIT if (bootlimit && (bootcount > bootlimit)) { printf ("Warning: Bootlimit (%u) exceeded. Using altbootcmd.\n" , (unsigned )bootlimit); s = env_get("altbootcmd" ); } else #endif s = env_get("bootcmd" ); process_fdt_options(gd->fdt_blob); stored_bootdelay = bootdelay; return s; }
common/autoboot.c –> autoboot_command:在执行启动命令run_command_list
前先检查stored_bootdelay
值
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 void autoboot_command (const char *s) { debug("### main_loop: bootcmd=\"%s\"\n" , s ? s : "<UNDEFINED>" ); if (stored_bootdelay != -1 && s && !abortboot(stored_bootdelay)) { #if defined(CONFIG_AUTOBOOT_KEYED) && !defined(CONFIG_AUTOBOOT_KEYED_CTRLC) int prev = disable_ctrlc(1 ); #endif run_command_list(s, -1 , 0 ); autoboot_command_fail_handle(); #if defined(CONFIG_AUTOBOOT_KEYED) && !defined(CONFIG_AUTOBOOT_KEYED_CTRLC) disable_ctrlc(prev); #endif } #ifdef CONFIG_MENUKEY if (menukey == CONFIG_MENUKEY) { s = env_get("menucmd" ); if (s) run_command_list(s, -1 , 0 ); } #endif }
common/autoboot.c –> abortboot:真正的功能实现在__abortboot
1 2 3 4 5 6 7 8 9 10 11 12 13 14 static int abortboot (int bootdelay) { int abort = 0 ; if (bootdelay >= 0 ) abort = __abortboot(bootdelay); #ifdef CONFIG_SILENT_CONSOLE if (abort ) gd->flags &= ~GD_FLG_SILENT; #endif return abort ; }
common/autoboot.c –> __abortboot:bootdelay
延时计数 + 等待按键ctrl+c
输入
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 static int __abortboot(int bootdelay){ int abort = 0 ; unsigned long ts; #ifdef CONFIG_MENUPROMPT printf (CONFIG_MENUPROMPT); #else printf ("Hit key to stop autoboot('CTRL+C'): %2d " , bootdelay); #endif #ifdef CONFIG_ARCH_ROCKCHIP if (!IS_ENABLED(CONFIG_CONSOLE_DISABLE_CLI) && ctrlc()) { #else if (tstc()) { #endif (void ) getc(); puts ("\b\b\b 0" ); abort = 1 ; } while ((bootdelay > 0 ) && (!abort )) { --bootdelay; ts = get_timer(0 ); do { if (ctrlc()) { abort = 1 ; bootdelay = 0 ; # ifdef CONFIG_MENUKEY menukey = 0x03 ; # endif break ; } udelay(10000 ); } while (!abort && get_timer(ts) < 1000 ); printf ("\b\b\b%2d " , bootdelay); } putc('\n' ); return abort ; }
如果不小心把启动延时设置为0,需要手动设置bootdelay的值才可以进入uboot配置菜单