11da177e4SLinus Torvalds /* 21da177e4SLinus Torvalds * linux/init/main.c 31da177e4SLinus Torvalds * 41da177e4SLinus Torvalds * Copyright (C) 1991, 1992 Linus Torvalds 51da177e4SLinus Torvalds * 61da177e4SLinus Torvalds * GK 2/5/95 - Changed to support mounting root fs via NFS 71da177e4SLinus Torvalds * Added initrd & change_root: Werner Almesberger & Hans Lermen, Feb '96 81da177e4SLinus Torvalds * Moan early if gcc is old, avoiding bogus kernels - Paul Gortmaker, May '96 91da177e4SLinus Torvalds * Simplified starting of init: Michael A. Griffith <grif@acm.org> 101da177e4SLinus Torvalds */ 111da177e4SLinus Torvalds 121da177e4SLinus Torvalds #include <linux/types.h> 131da177e4SLinus Torvalds #include <linux/module.h> 141da177e4SLinus Torvalds #include <linux/proc_fs.h> 151da177e4SLinus Torvalds #include <linux/kernel.h> 161da177e4SLinus Torvalds #include <linux/syscalls.h> 179b5609fdSIngo Molnar #include <linux/stackprotector.h> 181da177e4SLinus Torvalds #include <linux/string.h> 191da177e4SLinus Torvalds #include <linux/ctype.h> 201da177e4SLinus Torvalds #include <linux/delay.h> 211da177e4SLinus Torvalds #include <linux/ioport.h> 221da177e4SLinus Torvalds #include <linux/init.h> 231da177e4SLinus Torvalds #include <linux/initrd.h> 241da177e4SLinus Torvalds #include <linux/bootmem.h> 254a7a16dcSLen Brown #include <linux/acpi.h> 261da177e4SLinus Torvalds #include <linux/tty.h> 271da177e4SLinus Torvalds #include <linux/percpu.h> 281da177e4SLinus Torvalds #include <linux/kmod.h> 29db64fe02SNick Piggin #include <linux/vmalloc.h> 301da177e4SLinus Torvalds #include <linux/kernel_stat.h> 31d7cd5611SRusty Russell #include <linux/start_kernel.h> 321da177e4SLinus Torvalds #include <linux/security.h> 333d442233SJens Axboe #include <linux/smp.h> 341da177e4SLinus Torvalds #include <linux/profile.h> 351da177e4SLinus Torvalds #include <linux/rcupdate.h> 361da177e4SLinus Torvalds #include <linux/moduleparam.h> 371da177e4SLinus Torvalds #include <linux/kallsyms.h> 381da177e4SLinus Torvalds #include <linux/writeback.h> 391da177e4SLinus Torvalds #include <linux/cpu.h> 401da177e4SLinus Torvalds #include <linux/cpuset.h> 41ddbcc7e8SPaul Menage #include <linux/cgroup.h> 421da177e4SLinus Torvalds #include <linux/efi.h> 43906568c9SThomas Gleixner #include <linux/tick.h> 446168a702SAndrew Morton #include <linux/interrupt.h> 45c757249aSShailabh Nagar #include <linux/taskstats_kern.h> 46ca74e92bSShailabh Nagar #include <linux/delayacct.h> 471da177e4SLinus Torvalds #include <linux/unistd.h> 481da177e4SLinus Torvalds #include <linux/rmap.h> 491da177e4SLinus Torvalds #include <linux/mempolicy.h> 501da177e4SLinus Torvalds #include <linux/key.h> 51b6cd0b77SAdrian Bunk #include <linux/buffer_head.h> 5294b6da5aSKAMEZAWA Hiroyuki #include <linux/page_cgroup.h> 539a11b49aSIngo Molnar #include <linux/debug_locks.h> 543ac7fe5aSThomas Gleixner #include <linux/debugobjects.h> 55fbb9ce95SIngo Molnar #include <linux/lockdep.h> 563c7b4e6bSCatalin Marinas #include <linux/kmemleak.h> 5784d73786SSukadev Bhattiprolu #include <linux/pid_namespace.h> 581f21782eSAdrian Bunk #include <linux/device.h> 5973c27992SEric W. Biederman #include <linux/kthread.h> 60e6fe6649SAdrian Bunk #include <linux/sched.h> 61a1c9eea9SAdrian Bunk #include <linux/signal.h> 62199f0ca5SAkinobu Mita #include <linux/idr.h> 630b4b3827SJason Wessel #include <linux/kgdb.h> 6468bf21aaSSteven Rostedt #include <linux/ftrace.h> 6522a9d645SArjan van de Ven #include <linux/async.h> 66dfec072eSVegard Nossum #include <linux/kmemcheck.h> 676ae6996aSFeng Tang #include <linux/sfi.h> 682b2af54aSKay Sievers #include <linux/shmem_fs.h> 695a0e3ad6STejun Heo #include <linux/slab.h> 7024a24bb6SPeter Zijlstra #include <linux/perf_event.h> 711da177e4SLinus Torvalds 721da177e4SLinus Torvalds #include <asm/io.h> 731da177e4SLinus Torvalds #include <asm/bugs.h> 741da177e4SLinus Torvalds #include <asm/setup.h> 75a940199fSAndi Kleen #include <asm/sections.h> 7637b73c82SArjan van de Ven #include <asm/cacheflush.h> 771da177e4SLinus Torvalds 781da177e4SLinus Torvalds #ifdef CONFIG_X86_LOCAL_APIC 791da177e4SLinus Torvalds #include <asm/smp.h> 801da177e4SLinus Torvalds #endif 811da177e4SLinus Torvalds 82aae5f662SSam Ravnborg static int kernel_init(void *); 831da177e4SLinus Torvalds 841da177e4SLinus Torvalds extern void init_IRQ(void); 851da177e4SLinus Torvalds extern void fork_init(unsigned long); 861da177e4SLinus Torvalds extern void mca_init(void); 871da177e4SLinus Torvalds extern void sbus_init(void); 881da177e4SLinus Torvalds extern void prio_tree_init(void); 891da177e4SLinus Torvalds extern void radix_tree_init(void); 901da177e4SLinus Torvalds extern void free_initmem(void); 9137b73c82SArjan van de Ven #ifndef CONFIG_DEBUG_RODATA 9237b73c82SArjan van de Ven static inline void mark_rodata_ro(void) { } 9337b73c82SArjan van de Ven #endif 941da177e4SLinus Torvalds 951da177e4SLinus Torvalds #ifdef CONFIG_TC 961da177e4SLinus Torvalds extern void tc_init(void); 971da177e4SLinus Torvalds #endif 981da177e4SLinus Torvalds 992ce802f6STejun Heo /* 1002ce802f6STejun Heo * Debug helper: via this flag we know that we are in 'early bootup code' 1012ce802f6STejun Heo * where only the boot processor is running with IRQ disabled. This means 1022ce802f6STejun Heo * two things - IRQ must not be enabled before the flag is cleared and some 1032ce802f6STejun Heo * operations which are not allowed with IRQ disabled are allowed while the 1042ce802f6STejun Heo * flag is set. 1052ce802f6STejun Heo */ 1062ce802f6STejun Heo bool early_boot_irqs_disabled __read_mostly; 1072ce802f6STejun Heo 108a6826048SPaul E. McKenney enum system_states system_state __read_mostly; 1091da177e4SLinus Torvalds EXPORT_SYMBOL(system_state); 1101da177e4SLinus Torvalds 1111da177e4SLinus Torvalds /* 1121da177e4SLinus Torvalds * Boot command-line arguments 1131da177e4SLinus Torvalds */ 1141da177e4SLinus Torvalds #define MAX_INIT_ARGS CONFIG_INIT_ENV_ARG_LIMIT 1151da177e4SLinus Torvalds #define MAX_INIT_ENVS CONFIG_INIT_ENV_ARG_LIMIT 1161da177e4SLinus Torvalds 1171da177e4SLinus Torvalds extern void time_init(void); 1181da177e4SLinus Torvalds /* Default late time init is NULL. archs can override this later. */ 119d2e3192bSJan Beulich void (*__initdata late_time_init)(void); 1201da177e4SLinus Torvalds extern void softirq_init(void); 1211da177e4SLinus Torvalds 12230d7e0d4SAlon Bar-Lev /* Untouched command line saved by arch-specific code. */ 12330d7e0d4SAlon Bar-Lev char __initdata boot_command_line[COMMAND_LINE_SIZE]; 12430d7e0d4SAlon Bar-Lev /* Untouched saved command line (eg. for /proc) */ 12530d7e0d4SAlon Bar-Lev char *saved_command_line; 12630d7e0d4SAlon Bar-Lev /* Command line for parameter parsing */ 12730d7e0d4SAlon Bar-Lev static char *static_command_line; 1281da177e4SLinus Torvalds 1291da177e4SLinus Torvalds static char *execute_command; 130ffdfc409SOlof Johansson static char *ramdisk_execute_command; 1311da177e4SLinus Torvalds 1328b3b2955SJan Beulich /* 1338b3b2955SJan Beulich * If set, this is an indication to the drivers that reset the underlying 1348b3b2955SJan Beulich * device before going ahead with the initialization otherwise driver might 1358b3b2955SJan Beulich * rely on the BIOS and skip the reset operation. 1368b3b2955SJan Beulich * 1378b3b2955SJan Beulich * This is useful if kernel is booting in an unreliable environment. 1388b3b2955SJan Beulich * For ex. kdump situaiton where previous kernel has crashed, BIOS has been 1398b3b2955SJan Beulich * skipped and devices will be in unknown state. 1408b3b2955SJan Beulich */ 1418b3b2955SJan Beulich unsigned int reset_devices; 1428b3b2955SJan Beulich EXPORT_SYMBOL(reset_devices); 1431da177e4SLinus Torvalds 1447e96287dSVivek Goyal static int __init set_reset_devices(char *str) 1457e96287dSVivek Goyal { 1467e96287dSVivek Goyal reset_devices = 1; 1477e96287dSVivek Goyal return 1; 1487e96287dSVivek Goyal } 1497e96287dSVivek Goyal 1507e96287dSVivek Goyal __setup("reset_devices", set_reset_devices); 1517e96287dSVivek Goyal 152d7627467SDavid Howells static const char * argv_init[MAX_INIT_ARGS+2] = { "init", NULL, }; 153d7627467SDavid Howells const char * envp_init[MAX_INIT_ENVS+2] = { "HOME=/", "TERM=linux", NULL, }; 1541da177e4SLinus Torvalds static const char *panic_later, *panic_param; 1551da177e4SLinus Torvalds 156914dcaa8SRusty Russell extern const struct obs_kernel_param __setup_start[], __setup_end[]; 1571da177e4SLinus Torvalds 1581da177e4SLinus Torvalds static int __init obsolete_checksetup(char *line) 1591da177e4SLinus Torvalds { 160914dcaa8SRusty Russell const struct obs_kernel_param *p; 16133df0d19SRusty Russell int had_early_param = 0; 1621da177e4SLinus Torvalds 1631da177e4SLinus Torvalds p = __setup_start; 1641da177e4SLinus Torvalds do { 1651da177e4SLinus Torvalds int n = strlen(p->str); 166b1e4d20cSMichal Schmidt if (parameqn(line, p->str, n)) { 1671da177e4SLinus Torvalds if (p->early) { 16833df0d19SRusty Russell /* Already done in parse_early_param? 16933df0d19SRusty Russell * (Needs exact match on param part). 17033df0d19SRusty Russell * Keep iterating, as we can have early 17133df0d19SRusty Russell * params and __setups of same names 8( */ 1721da177e4SLinus Torvalds if (line[n] == '\0' || line[n] == '=') 17333df0d19SRusty Russell had_early_param = 1; 1741da177e4SLinus Torvalds } else if (!p->setup_func) { 1751da177e4SLinus Torvalds printk(KERN_WARNING "Parameter %s is obsolete," 1761da177e4SLinus Torvalds " ignored\n", p->str); 1771da177e4SLinus Torvalds return 1; 1781da177e4SLinus Torvalds } else if (p->setup_func(line + n)) 1791da177e4SLinus Torvalds return 1; 1801da177e4SLinus Torvalds } 1811da177e4SLinus Torvalds p++; 1821da177e4SLinus Torvalds } while (p < __setup_end); 18333df0d19SRusty Russell 18433df0d19SRusty Russell return had_early_param; 1851da177e4SLinus Torvalds } 1861da177e4SLinus Torvalds 1871da177e4SLinus Torvalds /* 1881da177e4SLinus Torvalds * This should be approx 2 Bo*oMips to start (note initial shift), and will 1891da177e4SLinus Torvalds * still work even if initially too large, it will just take slightly longer 1901da177e4SLinus Torvalds */ 1911da177e4SLinus Torvalds unsigned long loops_per_jiffy = (1<<12); 1921da177e4SLinus Torvalds 1931da177e4SLinus Torvalds EXPORT_SYMBOL(loops_per_jiffy); 1941da177e4SLinus Torvalds 1951da177e4SLinus Torvalds static int __init debug_kernel(char *str) 1961da177e4SLinus Torvalds { 1971da177e4SLinus Torvalds console_loglevel = 10; 198f6f21c81SYinghai Lu return 0; 1991da177e4SLinus Torvalds } 2001da177e4SLinus Torvalds 2011da177e4SLinus Torvalds static int __init quiet_kernel(char *str) 2021da177e4SLinus Torvalds { 2031da177e4SLinus Torvalds console_loglevel = 4; 204f6f21c81SYinghai Lu return 0; 2051da177e4SLinus Torvalds } 2061da177e4SLinus Torvalds 207f6f21c81SYinghai Lu early_param("debug", debug_kernel); 208f6f21c81SYinghai Lu early_param("quiet", quiet_kernel); 2091da177e4SLinus Torvalds 2101da177e4SLinus Torvalds static int __init loglevel(char *str) 2111da177e4SLinus Torvalds { 212808bf29bSAlexander Sverdlin int newlevel; 213808bf29bSAlexander Sverdlin 214808bf29bSAlexander Sverdlin /* 215808bf29bSAlexander Sverdlin * Only update loglevel value when a correct setting was passed, 216808bf29bSAlexander Sverdlin * to prevent blind crashes (when loglevel being set to 0) that 217808bf29bSAlexander Sverdlin * are quite hard to debug 218808bf29bSAlexander Sverdlin */ 219808bf29bSAlexander Sverdlin if (get_option(&str, &newlevel)) { 220808bf29bSAlexander Sverdlin console_loglevel = newlevel; 221d9d4fcfeSAlex Riesen return 0; 2221da177e4SLinus Torvalds } 2231da177e4SLinus Torvalds 224808bf29bSAlexander Sverdlin return -EINVAL; 225808bf29bSAlexander Sverdlin } 226808bf29bSAlexander Sverdlin 227f6f21c81SYinghai Lu early_param("loglevel", loglevel); 2281da177e4SLinus Torvalds 2291da177e4SLinus Torvalds /* 2301da177e4SLinus Torvalds * Unknown boot options get handed to init, unless they look like 231f066a4f6SRusty Russell * unused parameters (modprobe will find them in /proc/cmdline). 2321da177e4SLinus Torvalds */ 2331da177e4SLinus Torvalds static int __init unknown_bootoption(char *param, char *val) 2341da177e4SLinus Torvalds { 2351da177e4SLinus Torvalds /* Change NUL term back to "=", to make "param" the whole string. */ 2361da177e4SLinus Torvalds if (val) { 2371da177e4SLinus Torvalds /* param=val or param="val"? */ 2381da177e4SLinus Torvalds if (val == param+strlen(param)+1) 2391da177e4SLinus Torvalds val[-1] = '='; 2401da177e4SLinus Torvalds else if (val == param+strlen(param)+2) { 2411da177e4SLinus Torvalds val[-2] = '='; 2421da177e4SLinus Torvalds memmove(val-1, val, strlen(val)+1); 2431da177e4SLinus Torvalds val--; 2441da177e4SLinus Torvalds } else 2451da177e4SLinus Torvalds BUG(); 2461da177e4SLinus Torvalds } 2471da177e4SLinus Torvalds 2481da177e4SLinus Torvalds /* Handle obsolete-style parameters */ 2491da177e4SLinus Torvalds if (obsolete_checksetup(param)) 2501da177e4SLinus Torvalds return 0; 2511da177e4SLinus Torvalds 252f066a4f6SRusty Russell /* Unused module parameter. */ 253f066a4f6SRusty Russell if (strchr(param, '.') && (!val || strchr(param, '.') < val)) 2541da177e4SLinus Torvalds return 0; 2551da177e4SLinus Torvalds 2561da177e4SLinus Torvalds if (panic_later) 2571da177e4SLinus Torvalds return 0; 2581da177e4SLinus Torvalds 2591da177e4SLinus Torvalds if (val) { 2601da177e4SLinus Torvalds /* Environment option */ 2611da177e4SLinus Torvalds unsigned int i; 2621da177e4SLinus Torvalds for (i = 0; envp_init[i]; i++) { 2631da177e4SLinus Torvalds if (i == MAX_INIT_ENVS) { 2641da177e4SLinus Torvalds panic_later = "Too many boot env vars at `%s'"; 2651da177e4SLinus Torvalds panic_param = param; 2661da177e4SLinus Torvalds } 2671da177e4SLinus Torvalds if (!strncmp(param, envp_init[i], val - param)) 2681da177e4SLinus Torvalds break; 2691da177e4SLinus Torvalds } 2701da177e4SLinus Torvalds envp_init[i] = param; 2711da177e4SLinus Torvalds } else { 2721da177e4SLinus Torvalds /* Command line option */ 2731da177e4SLinus Torvalds unsigned int i; 2741da177e4SLinus Torvalds for (i = 0; argv_init[i]; i++) { 2751da177e4SLinus Torvalds if (i == MAX_INIT_ARGS) { 2761da177e4SLinus Torvalds panic_later = "Too many boot init vars at `%s'"; 2771da177e4SLinus Torvalds panic_param = param; 2781da177e4SLinus Torvalds } 2791da177e4SLinus Torvalds } 2801da177e4SLinus Torvalds argv_init[i] = param; 2811da177e4SLinus Torvalds } 2821da177e4SLinus Torvalds return 0; 2831da177e4SLinus Torvalds } 2841da177e4SLinus Torvalds 2851da177e4SLinus Torvalds static int __init init_setup(char *str) 2861da177e4SLinus Torvalds { 2871da177e4SLinus Torvalds unsigned int i; 2881da177e4SLinus Torvalds 2891da177e4SLinus Torvalds execute_command = str; 2901da177e4SLinus Torvalds /* 2911da177e4SLinus Torvalds * In case LILO is going to boot us with default command line, 2921da177e4SLinus Torvalds * it prepends "auto" before the whole cmdline which makes 2931da177e4SLinus Torvalds * the shell think it should execute a script with such name. 2941da177e4SLinus Torvalds * So we ignore all arguments entered _before_ init=... [MJ] 2951da177e4SLinus Torvalds */ 2961da177e4SLinus Torvalds for (i = 1; i < MAX_INIT_ARGS; i++) 2971da177e4SLinus Torvalds argv_init[i] = NULL; 2981da177e4SLinus Torvalds return 1; 2991da177e4SLinus Torvalds } 3001da177e4SLinus Torvalds __setup("init=", init_setup); 3011da177e4SLinus Torvalds 302ffdfc409SOlof Johansson static int __init rdinit_setup(char *str) 303ffdfc409SOlof Johansson { 304ffdfc409SOlof Johansson unsigned int i; 305ffdfc409SOlof Johansson 306ffdfc409SOlof Johansson ramdisk_execute_command = str; 307ffdfc409SOlof Johansson /* See "auto" comment in init_setup */ 308ffdfc409SOlof Johansson for (i = 1; i < MAX_INIT_ARGS; i++) 309ffdfc409SOlof Johansson argv_init[i] = NULL; 310ffdfc409SOlof Johansson return 1; 311ffdfc409SOlof Johansson } 312ffdfc409SOlof Johansson __setup("rdinit=", rdinit_setup); 313ffdfc409SOlof Johansson 3141da177e4SLinus Torvalds #ifndef CONFIG_SMP 31534db18a0SAmerigo Wang static const unsigned int setup_max_cpus = NR_CPUS; 3161da177e4SLinus Torvalds #ifdef CONFIG_X86_LOCAL_APIC 3171da177e4SLinus Torvalds static void __init smp_init(void) 3181da177e4SLinus Torvalds { 3191da177e4SLinus Torvalds APIC_init_uniprocessor(); 3201da177e4SLinus Torvalds } 3211da177e4SLinus Torvalds #else 3221da177e4SLinus Torvalds #define smp_init() do { } while (0) 3231da177e4SLinus Torvalds #endif 3241da177e4SLinus Torvalds 325e0982e90SMike Travis static inline void setup_nr_cpu_ids(void) { } 3261da177e4SLinus Torvalds static inline void smp_prepare_cpus(unsigned int maxcpus) { } 3271da177e4SLinus Torvalds #endif 3281da177e4SLinus Torvalds 3291da177e4SLinus Torvalds /* 33030d7e0d4SAlon Bar-Lev * We need to store the untouched command line for future reference. 33130d7e0d4SAlon Bar-Lev * We also need to store the touched command line since the parameter 33230d7e0d4SAlon Bar-Lev * parsing is performed in place, and we should allow a component to 33330d7e0d4SAlon Bar-Lev * store reference of name/value for future reference. 33430d7e0d4SAlon Bar-Lev */ 33530d7e0d4SAlon Bar-Lev static void __init setup_command_line(char *command_line) 33630d7e0d4SAlon Bar-Lev { 33730d7e0d4SAlon Bar-Lev saved_command_line = alloc_bootmem(strlen (boot_command_line)+1); 33830d7e0d4SAlon Bar-Lev static_command_line = alloc_bootmem(strlen (command_line)+1); 33930d7e0d4SAlon Bar-Lev strcpy (saved_command_line, boot_command_line); 34030d7e0d4SAlon Bar-Lev strcpy (static_command_line, command_line); 34130d7e0d4SAlon Bar-Lev } 34230d7e0d4SAlon Bar-Lev 34330d7e0d4SAlon Bar-Lev /* 3441da177e4SLinus Torvalds * We need to finalize in a non-__init function or else race conditions 3451da177e4SLinus Torvalds * between the root thread and the init thread may cause start_kernel to 3461da177e4SLinus Torvalds * be reaped by free_initmem before the root thread has proceeded to 3471da177e4SLinus Torvalds * cpu_idle. 3481da177e4SLinus Torvalds * 3491da177e4SLinus Torvalds * gcc-3.4 accidentally inlines this function, so use noinline. 3501da177e4SLinus Torvalds */ 3511da177e4SLinus Torvalds 352b433c3d4SPeter Zijlstra static __initdata DECLARE_COMPLETION(kthreadd_done); 353b433c3d4SPeter Zijlstra 354f99ebf0aSRakib Mullick static noinline void __init_refok rest_init(void) 3551da177e4SLinus Torvalds { 35673c27992SEric W. Biederman int pid; 35773c27992SEric W. Biederman 3587db905e6SPaul E. McKenney rcu_scheduler_starting(); 359b433c3d4SPeter Zijlstra /* 36097158569SPeter Zijlstra * We need to spawn init first so that it obtains pid 1, however 361b433c3d4SPeter Zijlstra * the init task will end up wanting to create kthreads, which, if 362b433c3d4SPeter Zijlstra * we schedule it before we create kthreadd, will OOPS. 363b433c3d4SPeter Zijlstra */ 364aae5f662SSam Ravnborg kernel_thread(kernel_init, NULL, CLONE_FS | CLONE_SIGHAND); 3651da177e4SLinus Torvalds numa_default_policy(); 36673c27992SEric W. Biederman pid = kernel_thread(kthreadd, NULL, CLONE_FS | CLONE_FILES); 367d11c563dSPaul E. McKenney rcu_read_lock(); 3685cd20455SPavel Emelyanov kthreadd_task = find_task_by_pid_ns(pid, &init_pid_ns); 369d11c563dSPaul E. McKenney rcu_read_unlock(); 370b433c3d4SPeter Zijlstra complete(&kthreadd_done); 371f340c0d1SIngo Molnar 372f340c0d1SIngo Molnar /* 373f340c0d1SIngo Molnar * The boot idle thread must execute schedule() 3741df21055SIngo Molnar * at least once to get things moving: 375f340c0d1SIngo Molnar */ 3761df21055SIngo Molnar init_idle_bootup_task(current); 377bd2f5536SThomas Gleixner schedule_preempt_disabled(); 3785bfb5d69SNick Piggin /* Call into cpu_idle with preempt disabled */ 3791da177e4SLinus Torvalds cpu_idle(); 3801da177e4SLinus Torvalds } 3811da177e4SLinus Torvalds 3821da177e4SLinus Torvalds /* Check for early params. */ 3831da177e4SLinus Torvalds static int __init do_early_param(char *param, char *val) 3841da177e4SLinus Torvalds { 385914dcaa8SRusty Russell const struct obs_kernel_param *p; 3861da177e4SLinus Torvalds 3871da177e4SLinus Torvalds for (p = __setup_start; p < __setup_end; p++) { 388b1e4d20cSMichal Schmidt if ((p->early && parameq(param, p->str)) || 38918a8bd94SYinghai Lu (strcmp(param, "console") == 0 && 39018a8bd94SYinghai Lu strcmp(p->str, "earlycon") == 0) 39118a8bd94SYinghai Lu ) { 3921da177e4SLinus Torvalds if (p->setup_func(val) != 0) 3931da177e4SLinus Torvalds printk(KERN_WARNING 3941da177e4SLinus Torvalds "Malformed early option '%s'\n", param); 3951da177e4SLinus Torvalds } 3961da177e4SLinus Torvalds } 3971da177e4SLinus Torvalds /* We accept everything at this stage. */ 3981da177e4SLinus Torvalds return 0; 3991da177e4SLinus Torvalds } 4001da177e4SLinus Torvalds 40113977091SMagnus Damm void __init parse_early_options(char *cmdline) 40213977091SMagnus Damm { 403*026cee00SPawel Moll parse_args("early options", cmdline, NULL, 0, 0, 0, do_early_param); 40413977091SMagnus Damm } 40513977091SMagnus Damm 4061da177e4SLinus Torvalds /* Arch code calls this early on, or if not, just before other parsing. */ 4071da177e4SLinus Torvalds void __init parse_early_param(void) 4081da177e4SLinus Torvalds { 4091da177e4SLinus Torvalds static __initdata int done = 0; 4101da177e4SLinus Torvalds static __initdata char tmp_cmdline[COMMAND_LINE_SIZE]; 4111da177e4SLinus Torvalds 4121da177e4SLinus Torvalds if (done) 4131da177e4SLinus Torvalds return; 4141da177e4SLinus Torvalds 4151da177e4SLinus Torvalds /* All fall through to do_early_param. */ 41630d7e0d4SAlon Bar-Lev strlcpy(tmp_cmdline, boot_command_line, COMMAND_LINE_SIZE); 41713977091SMagnus Damm parse_early_options(tmp_cmdline); 4181da177e4SLinus Torvalds done = 1; 4191da177e4SLinus Torvalds } 4201da177e4SLinus Torvalds 4211da177e4SLinus Torvalds /* 4221da177e4SLinus Torvalds * Activate the first processor. 4231da177e4SLinus Torvalds */ 4241da177e4SLinus Torvalds 42544fd2299SStas Sergeev static void __init boot_cpu_init(void) 42644fd2299SStas Sergeev { 42744fd2299SStas Sergeev int cpu = smp_processor_id(); 42844fd2299SStas Sergeev /* Mark the boot cpu "present", "online" etc for SMP and UP case */ 429915441b6SRusty Russell set_cpu_online(cpu, true); 430933b0618SPeter Zijlstra set_cpu_active(cpu, true); 431915441b6SRusty Russell set_cpu_present(cpu, true); 432915441b6SRusty Russell set_cpu_possible(cpu, true); 43344fd2299SStas Sergeev } 43444fd2299SStas Sergeev 435839ad62eSBenjamin Herrenschmidt void __init __weak smp_setup_processor_id(void) 436033ab7f8SAndrew Morton { 437033ab7f8SAndrew Morton } 438033ab7f8SAndrew Morton 4398c9843e5SBenjamin Herrenschmidt void __init __weak thread_info_cache_init(void) 4408c9843e5SBenjamin Herrenschmidt { 4418c9843e5SBenjamin Herrenschmidt } 4428c9843e5SBenjamin Herrenschmidt 443444f478fSPekka Enberg /* 444444f478fSPekka Enberg * Set up kernel memory allocators 445444f478fSPekka Enberg */ 446444f478fSPekka Enberg static void __init mm_init(void) 447444f478fSPekka Enberg { 448ca371c0dSKAMEZAWA Hiroyuki /* 4497fa87ce7SJim Cromie * page_cgroup requires contiguous pages, 4507fa87ce7SJim Cromie * bigger than MAX_ORDER unless SPARSEMEM. 451ca371c0dSKAMEZAWA Hiroyuki */ 452ca371c0dSKAMEZAWA Hiroyuki page_cgroup_init_flatmem(); 453444f478fSPekka Enberg mem_init(); 454444f478fSPekka Enberg kmem_cache_init(); 455099a19d9STejun Heo percpu_init_late(); 456c868d550SBenjamin Herrenschmidt pgtable_cache_init(); 457444f478fSPekka Enberg vmalloc_init(); 458444f478fSPekka Enberg } 459444f478fSPekka Enberg 4601da177e4SLinus Torvalds asmlinkage void __init start_kernel(void) 4611da177e4SLinus Torvalds { 4621da177e4SLinus Torvalds char * command_line; 463914dcaa8SRusty Russell extern const struct kernel_param __start___param[], __stop___param[]; 464033ab7f8SAndrew Morton 4651da177e4SLinus Torvalds /* 466fbb9ce95SIngo Molnar * Need to run as early as possible, to initialize the 467fbb9ce95SIngo Molnar * lockdep hash: 468fbb9ce95SIngo Molnar */ 469fbb9ce95SIngo Molnar lockdep_init(); 47073839c5bSMing Lei smp_setup_processor_id(); 4713ac7fe5aSThomas Gleixner debug_objects_early_init(); 47242059429SIngo Molnar 47342059429SIngo Molnar /* 47442059429SIngo Molnar * Set up the the initial canary ASAP: 47542059429SIngo Molnar */ 47642059429SIngo Molnar boot_init_stack_canary(); 47742059429SIngo Molnar 478ddbcc7e8SPaul Menage cgroup_init_early(); 479fbb9ce95SIngo Molnar 480fbb9ce95SIngo Molnar local_irq_disable(); 4812ce802f6STejun Heo early_boot_irqs_disabled = true; 482fbb9ce95SIngo Molnar 483fbb9ce95SIngo Molnar /* 4841da177e4SLinus Torvalds * Interrupts are still disabled. Do necessary setups, then 4851da177e4SLinus Torvalds * enable them 4861da177e4SLinus Torvalds */ 487906568c9SThomas Gleixner tick_init(); 48844fd2299SStas Sergeev boot_cpu_init(); 4891da177e4SLinus Torvalds page_address_init(); 490657cafa6SAlex Riesen printk(KERN_NOTICE "%s", linux_banner); 4911da177e4SLinus Torvalds setup_arch(&command_line); 492cf475ad2SBalbir Singh mm_init_owner(&init_mm, &init_task); 4936345d24dSLinus Torvalds mm_init_cpumask(&init_mm); 49430d7e0d4SAlon Bar-Lev setup_command_line(command_line); 495e0982e90SMike Travis setup_nr_cpu_ids(); 496d6647bdfSTejun Heo setup_per_cpu_areas(); 49744fd2299SStas Sergeev smp_prepare_boot_cpu(); /* arch-specific boot-cpu hooks */ 4981da177e4SLinus Torvalds 4991f522509SHaicheng Li build_all_zonelists(NULL); 50083b519e8SPekka Enberg page_alloc_init(); 50183b519e8SPekka Enberg 50283b519e8SPekka Enberg printk(KERN_NOTICE "Kernel command line: %s\n", boot_command_line); 50383b519e8SPekka Enberg parse_early_param(); 50483b519e8SPekka Enberg parse_args("Booting kernel", static_command_line, __start___param, 50583b519e8SPekka Enberg __stop___param - __start___param, 506*026cee00SPawel Moll 0, 0, &unknown_bootoption); 50797ce2c88SJeremy Fitzhardinge 50897ce2c88SJeremy Fitzhardinge jump_label_init(); 50997ce2c88SJeremy Fitzhardinge 51083b519e8SPekka Enberg /* 51183b519e8SPekka Enberg * These use large bootmem allocations and must precede 51283b519e8SPekka Enberg * kmem_cache_init() 51383b519e8SPekka Enberg */ 514162a7e75SMike Travis setup_log_buf(0); 51583b519e8SPekka Enberg pidhash_init(); 51683b519e8SPekka Enberg vfs_caches_init_early(); 51783b519e8SPekka Enberg sort_main_extable(); 51883b519e8SPekka Enberg trap_init(); 519444f478fSPekka Enberg mm_init(); 520de03c72cSKOSAKI Motohiro 5211da177e4SLinus Torvalds /* 5221da177e4SLinus Torvalds * Set up the scheduler prior starting any interrupts (such as the 5231da177e4SLinus Torvalds * timer interrupt). Full topology setup happens at smp_init() 5241da177e4SLinus Torvalds * time - but meanwhile we still have a functioning scheduler. 5251da177e4SLinus Torvalds */ 5261da177e4SLinus Torvalds sched_init(); 5271da177e4SLinus Torvalds /* 5281da177e4SLinus Torvalds * Disable preemption - early bootup scheduling is extremely 5291da177e4SLinus Torvalds * fragile until we cpu_idle() for the first time. 5301da177e4SLinus Torvalds */ 5311da177e4SLinus Torvalds preempt_disable(); 532c4a68306SArd van Breemen if (!irqs_disabled()) { 533c4a68306SArd van Breemen printk(KERN_WARNING "start_kernel(): bug: interrupts were " 534c4a68306SArd van Breemen "enabled *very* early, fixing it\n"); 535c4a68306SArd van Breemen local_irq_disable(); 536c4a68306SArd van Breemen } 5379f58a205SPeter Zijlstra idr_init_cache(); 53824a24bb6SPeter Zijlstra perf_event_init(); 5391da177e4SLinus Torvalds rcu_init(); 540773e3eb7SYinghai Lu radix_tree_init(); 5410b8f1efaSYinghai Lu /* init some links before init_ISA_irqs() */ 5420b8f1efaSYinghai Lu early_irq_init(); 5431da177e4SLinus Torvalds init_IRQ(); 5443c7b4e6bSCatalin Marinas prio_tree_init(); 5451da177e4SLinus Torvalds init_timers(); 546c0a31329SThomas Gleixner hrtimers_init(); 5471da177e4SLinus Torvalds softirq_init(); 548ad596171Sjohn stultz timekeeping_init(); 54988fecaa2Sjohn stultz time_init(); 55093e02814SHeiko Carstens profile_init(); 551d8ad7d11STakao Indoh call_function_init(); 55293e02814SHeiko Carstens if (!irqs_disabled()) 55324d431d0SRon Lee printk(KERN_CRIT "start_kernel(): bug: interrupts were " 55424d431d0SRon Lee "enabled early\n"); 5552ce802f6STejun Heo early_boot_irqs_disabled = false; 55693e02814SHeiko Carstens local_irq_enable(); 557dcce284aSBenjamin Herrenschmidt 558dcce284aSBenjamin Herrenschmidt /* Interrupts are enabled now so all GFP allocations are safe. */ 559452aa699SRafael J. Wysocki gfp_allowed_mask = __GFP_BITS_MASK; 560dcce284aSBenjamin Herrenschmidt 5617e85ee0cSPekka Enberg kmem_cache_init_late(); 5621da177e4SLinus Torvalds 5631da177e4SLinus Torvalds /* 5641da177e4SLinus Torvalds * HACK ALERT! This is early. We're enabling the console before 5651da177e4SLinus Torvalds * we've done PCI setups etc, and console_init() must be aware of 5661da177e4SLinus Torvalds * this. But we do want output early, in case something goes wrong. 5671da177e4SLinus Torvalds */ 5681da177e4SLinus Torvalds console_init(); 5691da177e4SLinus Torvalds if (panic_later) 5701da177e4SLinus Torvalds panic(panic_later, panic_param); 571fbb9ce95SIngo Molnar 572fbb9ce95SIngo Molnar lockdep_info(); 573fbb9ce95SIngo Molnar 5749a11b49aSIngo Molnar /* 5759a11b49aSIngo Molnar * Need to run this when irqs are enabled, because it wants 5769a11b49aSIngo Molnar * to self-test [hard/soft]-irqs on/off lock inversion bugs 5779a11b49aSIngo Molnar * too: 5789a11b49aSIngo Molnar */ 5799a11b49aSIngo Molnar locking_selftest(); 5809a11b49aSIngo Molnar 5811da177e4SLinus Torvalds #ifdef CONFIG_BLK_DEV_INITRD 5821da177e4SLinus Torvalds if (initrd_start && !initrd_below_start_ok && 583bd673c7cSGeert Uytterhoeven page_to_pfn(virt_to_page((void *)initrd_start)) < min_low_pfn) { 5841da177e4SLinus Torvalds printk(KERN_CRIT "initrd overwritten (0x%08lx < 0x%08lx) - " 585fb6624ebSGeert Uytterhoeven "disabling it.\n", 586bd673c7cSGeert Uytterhoeven page_to_pfn(virt_to_page((void *)initrd_start)), 587bd673c7cSGeert Uytterhoeven min_low_pfn); 5881da177e4SLinus Torvalds initrd_start = 0; 5891da177e4SLinus Torvalds } 5901da177e4SLinus Torvalds #endif 59194b6da5aSKAMEZAWA Hiroyuki page_cgroup_init(); 5923ac7fe5aSThomas Gleixner debug_objects_mem_init(); 5939b090f2dSCatalin Marinas kmemleak_init(); 594e7c8d5c9SChristoph Lameter setup_per_cpu_pageset(); 5951da177e4SLinus Torvalds numa_policy_init(); 5961da177e4SLinus Torvalds if (late_time_init) 5971da177e4SLinus Torvalds late_time_init(); 598fa84e9eeSThomas Gleixner sched_clock_init(); 5991da177e4SLinus Torvalds calibrate_delay(); 6001da177e4SLinus Torvalds pidmap_init(); 6011da177e4SLinus Torvalds anon_vma_init(); 6021da177e4SLinus Torvalds #ifdef CONFIG_X86 6031da177e4SLinus Torvalds if (efi_enabled) 6041da177e4SLinus Torvalds efi_enter_virtual_mode(); 6051da177e4SLinus Torvalds #endif 6068c9843e5SBenjamin Herrenschmidt thread_info_cache_init(); 607d84f4f99SDavid Howells cred_init(); 6084481374cSJan Beulich fork_init(totalram_pages); 6091da177e4SLinus Torvalds proc_caches_init(); 6101da177e4SLinus Torvalds buffer_init(); 6111da177e4SLinus Torvalds key_init(); 6121da177e4SLinus Torvalds security_init(); 6130b4b3827SJason Wessel dbg_late_init(); 6144481374cSJan Beulich vfs_caches_init(totalram_pages); 6151da177e4SLinus Torvalds signals_init(); 6161da177e4SLinus Torvalds /* rootfs populating might need page-writeback */ 6171da177e4SLinus Torvalds page_writeback_init(); 6181da177e4SLinus Torvalds #ifdef CONFIG_PROC_FS 6191da177e4SLinus Torvalds proc_root_init(); 6201da177e4SLinus Torvalds #endif 621ddbcc7e8SPaul Menage cgroup_init(); 6221da177e4SLinus Torvalds cpuset_init(); 623c757249aSShailabh Nagar taskstats_init_early(); 624ca74e92bSShailabh Nagar delayacct_init(); 6251da177e4SLinus Torvalds 6261da177e4SLinus Torvalds check_bugs(); 6271da177e4SLinus Torvalds 6281da177e4SLinus Torvalds acpi_early_init(); /* before LAPIC and SMP init */ 6296ae6996aSFeng Tang sfi_init_late(); 6301da177e4SLinus Torvalds 63168bf21aaSSteven Rostedt ftrace_init(); 63268bf21aaSSteven Rostedt 6331da177e4SLinus Torvalds /* Do the rest non-__init'ed, we're now alive */ 6341da177e4SLinus Torvalds rest_init(); 6351da177e4SLinus Torvalds } 6361da177e4SLinus Torvalds 637b99b87f7SPeter Oberparleiter /* Call all constructor functions linked into the kernel. */ 638b99b87f7SPeter Oberparleiter static void __init do_ctors(void) 639b99b87f7SPeter Oberparleiter { 640b99b87f7SPeter Oberparleiter #ifdef CONFIG_CONSTRUCTORS 641196a15b4SH Hartley Sweeten ctor_fn_t *fn = (ctor_fn_t *) __ctors_start; 642b99b87f7SPeter Oberparleiter 643196a15b4SH Hartley Sweeten for (; fn < (ctor_fn_t *) __ctors_end; fn++) 644196a15b4SH Hartley Sweeten (*fn)(); 645b99b87f7SPeter Oberparleiter #endif 646b99b87f7SPeter Oberparleiter } 647b99b87f7SPeter Oberparleiter 6482329abfaSRusty Russell bool initcall_debug; 649d0ea3d7dSRusty Russell core_param(initcall_debug, initcall_debug, bool, 0644); 6501da177e4SLinus Torvalds 6514a683bf9SIngo Molnar static char msgbuf[64]; 6524a683bf9SIngo Molnar 653e4461271SKevin Winchester static int __init_or_module do_one_initcall_debug(initcall_t fn) 6541da177e4SLinus Torvalds { 65574239072SFrederic Weisbecker ktime_t calltime, delta, rettime; 65630dbb20eSAmérico Wang unsigned long long duration; 65730dbb20eSAmérico Wang int ret; 6581da177e4SLinus Torvalds 65922c5c03bSKevin Winchester printk(KERN_DEBUG "calling %pF @ %i\n", fn, task_pid_nr(current)); 66074239072SFrederic Weisbecker calltime = ktime_get(); 66130dbb20eSAmérico Wang ret = fn(); 66274239072SFrederic Weisbecker rettime = ktime_get(); 66374239072SFrederic Weisbecker delta = ktime_sub(rettime, calltime); 66430dbb20eSAmérico Wang duration = (unsigned long long) ktime_to_ns(delta) >> 10; 66522c5c03bSKevin Winchester printk(KERN_DEBUG "initcall %pF returned %d after %lld usecs\n", fn, 66630dbb20eSAmérico Wang ret, duration); 66722c5c03bSKevin Winchester 66822c5c03bSKevin Winchester return ret; 6698f0c45cdSIngo Molnar } 6708f0c45cdSIngo Molnar 671e4461271SKevin Winchester int __init_or_module do_one_initcall(initcall_t fn) 67222c5c03bSKevin Winchester { 67322c5c03bSKevin Winchester int count = preempt_count(); 67422c5c03bSKevin Winchester int ret; 67522c5c03bSKevin Winchester 67622c5c03bSKevin Winchester if (initcall_debug) 67722c5c03bSKevin Winchester ret = do_one_initcall_debug(fn); 67822c5c03bSKevin Winchester else 67922c5c03bSKevin Winchester ret = fn(); 68022c5c03bSKevin Winchester 681e662e1cfSCyrill Gorcunov msgbuf[0] = 0; 682e662e1cfSCyrill Gorcunov 68330dbb20eSAmérico Wang if (ret && ret != -ENODEV && initcall_debug) 68430dbb20eSAmérico Wang sprintf(msgbuf, "error code %d ", ret); 685e662e1cfSCyrill Gorcunov 6861da177e4SLinus Torvalds if (preempt_count() != count) { 687a76bfd0dSCyrill Gorcunov strlcat(msgbuf, "preemption imbalance ", sizeof(msgbuf)); 6881da177e4SLinus Torvalds preempt_count() = count; 6891da177e4SLinus Torvalds } 6901da177e4SLinus Torvalds if (irqs_disabled()) { 691a76bfd0dSCyrill Gorcunov strlcat(msgbuf, "disabled interrupts ", sizeof(msgbuf)); 6921da177e4SLinus Torvalds local_irq_enable(); 6931da177e4SLinus Torvalds } 694e662e1cfSCyrill Gorcunov if (msgbuf[0]) { 69596d746c6SLinus Torvalds printk("initcall %pF returned with %s\n", fn, msgbuf); 6961da177e4SLinus Torvalds } 69759f9415fSArjan van de Ven 69830dbb20eSAmérico Wang return ret; 6991da177e4SLinus Torvalds } 7001da177e4SLinus Torvalds 701e0df154fSLinus Torvalds 702*026cee00SPawel Moll extern initcall_t __initcall_start[]; 703*026cee00SPawel Moll extern initcall_t __initcall0_start[]; 704*026cee00SPawel Moll extern initcall_t __initcall1_start[]; 705*026cee00SPawel Moll extern initcall_t __initcall2_start[]; 706*026cee00SPawel Moll extern initcall_t __initcall3_start[]; 707*026cee00SPawel Moll extern initcall_t __initcall4_start[]; 708*026cee00SPawel Moll extern initcall_t __initcall5_start[]; 709*026cee00SPawel Moll extern initcall_t __initcall6_start[]; 710*026cee00SPawel Moll extern initcall_t __initcall7_start[]; 711*026cee00SPawel Moll extern initcall_t __initcall_end[]; 712*026cee00SPawel Moll 713*026cee00SPawel Moll static initcall_t *initcall_levels[] __initdata = { 714*026cee00SPawel Moll __initcall0_start, 715*026cee00SPawel Moll __initcall1_start, 716*026cee00SPawel Moll __initcall2_start, 717*026cee00SPawel Moll __initcall3_start, 718*026cee00SPawel Moll __initcall4_start, 719*026cee00SPawel Moll __initcall5_start, 720*026cee00SPawel Moll __initcall6_start, 721*026cee00SPawel Moll __initcall7_start, 722*026cee00SPawel Moll __initcall_end, 723*026cee00SPawel Moll }; 724*026cee00SPawel Moll 725*026cee00SPawel Moll static char *initcall_level_names[] __initdata = { 726*026cee00SPawel Moll "early parameters", 727*026cee00SPawel Moll "core parameters", 728*026cee00SPawel Moll "postcore parameters", 729*026cee00SPawel Moll "arch parameters", 730*026cee00SPawel Moll "subsys parameters", 731*026cee00SPawel Moll "fs parameters", 732*026cee00SPawel Moll "device parameters", 733*026cee00SPawel Moll "late parameters", 734*026cee00SPawel Moll }; 735*026cee00SPawel Moll 736*026cee00SPawel Moll static int __init ignore_unknown_bootoption(char *param, char *val) 737*026cee00SPawel Moll { 738*026cee00SPawel Moll return 0; 739*026cee00SPawel Moll } 740*026cee00SPawel Moll 741*026cee00SPawel Moll static void __init do_initcall_level(int level) 742*026cee00SPawel Moll { 743*026cee00SPawel Moll extern const struct kernel_param __start___param[], __stop___param[]; 744*026cee00SPawel Moll initcall_t *fn; 745*026cee00SPawel Moll 746*026cee00SPawel Moll strcpy(static_command_line, saved_command_line); 747*026cee00SPawel Moll parse_args(initcall_level_names[level], 748*026cee00SPawel Moll static_command_line, __start___param, 749*026cee00SPawel Moll __stop___param - __start___param, 750*026cee00SPawel Moll level, level, 751*026cee00SPawel Moll ignore_unknown_bootoption); 752*026cee00SPawel Moll 753*026cee00SPawel Moll for (fn = initcall_levels[level]; fn < initcall_levels[level+1]; fn++) 754*026cee00SPawel Moll do_one_initcall(*fn); 755*026cee00SPawel Moll } 756e0df154fSLinus Torvalds 757e0df154fSLinus Torvalds static void __init do_initcalls(void) 758e0df154fSLinus Torvalds { 759*026cee00SPawel Moll int level; 760e0df154fSLinus Torvalds 761*026cee00SPawel Moll for (level = 0; level < ARRAY_SIZE(initcall_levels) - 1; level++) 762*026cee00SPawel Moll do_initcall_level(level); 7631da177e4SLinus Torvalds } 7641da177e4SLinus Torvalds 7651da177e4SLinus Torvalds /* 7661da177e4SLinus Torvalds * Ok, the machine is now initialized. None of the devices 7671da177e4SLinus Torvalds * have been touched yet, but the CPU subsystem is up and 7681da177e4SLinus Torvalds * running, and memory and process management works. 7691da177e4SLinus Torvalds * 7701da177e4SLinus Torvalds * Now we can finally start doing some real work.. 7711da177e4SLinus Torvalds */ 7721da177e4SLinus Torvalds static void __init do_basic_setup(void) 7731da177e4SLinus Torvalds { 774759ee091SLai Jiangshan cpuset_init_smp(); 7751da177e4SLinus Torvalds usermodehelper_init(); 77641ffe5d5SHugh Dickins shmem_init(); 7771da177e4SLinus Torvalds driver_init(); 778b04c3afbSEric W. Biederman init_irq_proc(); 779b99b87f7SPeter Oberparleiter do_ctors(); 780d5767c53SLinus Torvalds usermodehelper_enable(); 781b0f84374Swangyanqing do_initcalls(); 7821da177e4SLinus Torvalds } 7831da177e4SLinus Torvalds 7847babe8dbSEduard - Gabriel Munteanu static void __init do_pre_smp_initcalls(void) 785c2147a50SEduard - Gabriel Munteanu { 786196a15b4SH Hartley Sweeten initcall_t *fn; 787c2147a50SEduard - Gabriel Munteanu 788*026cee00SPawel Moll for (fn = __initcall_start; fn < __initcall0_start; fn++) 789196a15b4SH Hartley Sweeten do_one_initcall(*fn); 790c2147a50SEduard - Gabriel Munteanu } 791c2147a50SEduard - Gabriel Munteanu 792d7627467SDavid Howells static void run_init_process(const char *init_filename) 7931da177e4SLinus Torvalds { 7941da177e4SLinus Torvalds argv_init[0] = init_filename; 79567608567SArnd Bergmann kernel_execve(init_filename, argv_init, envp_init); 7961da177e4SLinus Torvalds } 7971da177e4SLinus Torvalds 798ee5bfa64SVivek Goyal /* This is a non __init function. Force it to be noinline otherwise gcc 799ee5bfa64SVivek Goyal * makes it inline to init() and it becomes part of init.text section 800ee5bfa64SVivek Goyal */ 801f99ebf0aSRakib Mullick static noinline int init_post(void) 802ee5bfa64SVivek Goyal { 80322a9d645SArjan van de Ven /* need to finish all async __init code before freeing the memory */ 80422a9d645SArjan van de Ven async_synchronize_full(); 805ee5bfa64SVivek Goyal free_initmem(); 806ee5bfa64SVivek Goyal mark_rodata_ro(); 807ee5bfa64SVivek Goyal system_state = SYSTEM_RUNNING; 808ee5bfa64SVivek Goyal numa_default_policy(); 809ee5bfa64SVivek Goyal 810ee5bfa64SVivek Goyal 811fae5fa44SOleg Nesterov current->signal->flags |= SIGNAL_UNKILLABLE; 812fae5fa44SOleg Nesterov 813ee5bfa64SVivek Goyal if (ramdisk_execute_command) { 814ee5bfa64SVivek Goyal run_init_process(ramdisk_execute_command); 815ee5bfa64SVivek Goyal printk(KERN_WARNING "Failed to execute %s\n", 816ee5bfa64SVivek Goyal ramdisk_execute_command); 817ee5bfa64SVivek Goyal } 818ee5bfa64SVivek Goyal 819ee5bfa64SVivek Goyal /* 820ee5bfa64SVivek Goyal * We try each of these until one succeeds. 821ee5bfa64SVivek Goyal * 822ee5bfa64SVivek Goyal * The Bourne shell can be used instead of init if we are 823ee5bfa64SVivek Goyal * trying to recover a really broken machine. 824ee5bfa64SVivek Goyal */ 825ee5bfa64SVivek Goyal if (execute_command) { 826ee5bfa64SVivek Goyal run_init_process(execute_command); 827ee5bfa64SVivek Goyal printk(KERN_WARNING "Failed to execute %s. Attempting " 828ee5bfa64SVivek Goyal "defaults...\n", execute_command); 829ee5bfa64SVivek Goyal } 830ee5bfa64SVivek Goyal run_init_process("/sbin/init"); 831ee5bfa64SVivek Goyal run_init_process("/etc/init"); 832ee5bfa64SVivek Goyal run_init_process("/bin/init"); 833ee5bfa64SVivek Goyal run_init_process("/bin/sh"); 834ee5bfa64SVivek Goyal 8359a85b8d6SAndreas Mohr panic("No init found. Try passing init= option to kernel. " 8369a85b8d6SAndreas Mohr "See Linux Documentation/init.txt for guidance."); 837ee5bfa64SVivek Goyal } 838ee5bfa64SVivek Goyal 839aae5f662SSam Ravnborg static int __init kernel_init(void * unused) 8401da177e4SLinus Torvalds { 841b433c3d4SPeter Zijlstra /* 842b433c3d4SPeter Zijlstra * Wait until kthreadd is all set-up. 843b433c3d4SPeter Zijlstra */ 844b433c3d4SPeter Zijlstra wait_for_completion(&kthreadd_done); 84558568d2aSMiao Xie /* 84658568d2aSMiao Xie * init can allocate pages on any node 84758568d2aSMiao Xie */ 8485ab116c9SMiao Xie set_mems_allowed(node_states[N_HIGH_MEMORY]); 8491da177e4SLinus Torvalds /* 8501da177e4SLinus Torvalds * init can run on any cpu. 8511da177e4SLinus Torvalds */ 8521a2142afSRusty Russell set_cpus_allowed_ptr(current, cpu_all_mask); 8531da177e4SLinus Torvalds 8549ec52099SCedric Le Goater cad_pid = task_pid(current); 8559ec52099SCedric Le Goater 856ca74a6f8SAndi Kleen smp_prepare_cpus(setup_max_cpus); 8571da177e4SLinus Torvalds 8581da177e4SLinus Torvalds do_pre_smp_initcalls(); 859004417a6SPeter Zijlstra lockup_detector_init(); 8601da177e4SLinus Torvalds 8611da177e4SLinus Torvalds smp_init(); 8621da177e4SLinus Torvalds sched_init_smp(); 8631da177e4SLinus Torvalds 8641da177e4SLinus Torvalds do_basic_setup(); 8651da177e4SLinus Torvalds 8662bd3a997SEric W. Biederman /* Open the /dev/console on the rootfs, this should never fail */ 8672bd3a997SEric W. Biederman if (sys_open((const char __user *) "/dev/console", O_RDWR, 0) < 0) 8682bd3a997SEric W. Biederman printk(KERN_WARNING "Warning: unable to open an initial console.\n"); 8692bd3a997SEric W. Biederman 8702bd3a997SEric W. Biederman (void) sys_dup(0); 8712bd3a997SEric W. Biederman (void) sys_dup(0); 8721da177e4SLinus Torvalds /* 8731da177e4SLinus Torvalds * check if there is an early userspace init. If yes, let it do all 8741da177e4SLinus Torvalds * the work 8751da177e4SLinus Torvalds */ 876ffdfc409SOlof Johansson 877ffdfc409SOlof Johansson if (!ramdisk_execute_command) 878ffdfc409SOlof Johansson ramdisk_execute_command = "/init"; 879ffdfc409SOlof Johansson 880ffdfc409SOlof Johansson if (sys_access((const char __user *) ramdisk_execute_command, 0) != 0) { 881ffdfc409SOlof Johansson ramdisk_execute_command = NULL; 8821da177e4SLinus Torvalds prepare_namespace(); 883ffdfc409SOlof Johansson } 8841da177e4SLinus Torvalds 8851da177e4SLinus Torvalds /* 8861da177e4SLinus Torvalds * Ok, we have completed the initial bootup, and 8871da177e4SLinus Torvalds * we're essentially up and running. Get rid of the 8881da177e4SLinus Torvalds * initmem segments and start the user-mode stuff.. 8891da177e4SLinus Torvalds */ 89071566a0dSFrederic Weisbecker 891ee5bfa64SVivek Goyal init_post(); 892ee5bfa64SVivek Goyal return 0; 8931da177e4SLinus Torvalds } 894