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> 714a9d4b02SAl Viro #include <linux/file.h> 72a74fb73cSAl Viro #include <linux/ptrace.h> 73bb813f4cSTejun Heo #include <linux/blkdev.h> 74bb813f4cSTejun Heo #include <linux/elevator.h> 751da177e4SLinus Torvalds 761da177e4SLinus Torvalds #include <asm/io.h> 771da177e4SLinus Torvalds #include <asm/bugs.h> 781da177e4SLinus Torvalds #include <asm/setup.h> 79a940199fSAndi Kleen #include <asm/sections.h> 8037b73c82SArjan van de Ven #include <asm/cacheflush.h> 811da177e4SLinus Torvalds 821da177e4SLinus Torvalds #ifdef CONFIG_X86_LOCAL_APIC 831da177e4SLinus Torvalds #include <asm/smp.h> 841da177e4SLinus Torvalds #endif 851da177e4SLinus Torvalds 86aae5f662SSam Ravnborg static int kernel_init(void *); 871da177e4SLinus Torvalds 881da177e4SLinus Torvalds extern void init_IRQ(void); 891da177e4SLinus Torvalds extern void fork_init(unsigned long); 901da177e4SLinus Torvalds extern void mca_init(void); 911da177e4SLinus Torvalds extern void sbus_init(void); 921da177e4SLinus Torvalds extern void radix_tree_init(void); 9337b73c82SArjan van de Ven #ifndef CONFIG_DEBUG_RODATA 9437b73c82SArjan van de Ven static inline void mark_rodata_ro(void) { } 9537b73c82SArjan van de Ven #endif 961da177e4SLinus Torvalds 971da177e4SLinus Torvalds #ifdef CONFIG_TC 981da177e4SLinus Torvalds extern void tc_init(void); 991da177e4SLinus Torvalds #endif 1001da177e4SLinus Torvalds 1012ce802f6STejun Heo /* 1022ce802f6STejun Heo * Debug helper: via this flag we know that we are in 'early bootup code' 1032ce802f6STejun Heo * where only the boot processor is running with IRQ disabled. This means 1042ce802f6STejun Heo * two things - IRQ must not be enabled before the flag is cleared and some 1052ce802f6STejun Heo * operations which are not allowed with IRQ disabled are allowed while the 1062ce802f6STejun Heo * flag is set. 1072ce802f6STejun Heo */ 1082ce802f6STejun Heo bool early_boot_irqs_disabled __read_mostly; 1092ce802f6STejun Heo 110a6826048SPaul E. McKenney enum system_states system_state __read_mostly; 1111da177e4SLinus Torvalds EXPORT_SYMBOL(system_state); 1121da177e4SLinus Torvalds 1131da177e4SLinus Torvalds /* 1141da177e4SLinus Torvalds * Boot command-line arguments 1151da177e4SLinus Torvalds */ 1161da177e4SLinus Torvalds #define MAX_INIT_ARGS CONFIG_INIT_ENV_ARG_LIMIT 1171da177e4SLinus Torvalds #define MAX_INIT_ENVS CONFIG_INIT_ENV_ARG_LIMIT 1181da177e4SLinus Torvalds 1191da177e4SLinus Torvalds extern void time_init(void); 1201da177e4SLinus Torvalds /* Default late time init is NULL. archs can override this later. */ 121d2e3192bSJan Beulich void (*__initdata late_time_init)(void); 1221da177e4SLinus Torvalds extern void softirq_init(void); 1231da177e4SLinus Torvalds 12430d7e0d4SAlon Bar-Lev /* Untouched command line saved by arch-specific code. */ 12530d7e0d4SAlon Bar-Lev char __initdata boot_command_line[COMMAND_LINE_SIZE]; 12630d7e0d4SAlon Bar-Lev /* Untouched saved command line (eg. for /proc) */ 12730d7e0d4SAlon Bar-Lev char *saved_command_line; 12830d7e0d4SAlon Bar-Lev /* Command line for parameter parsing */ 12930d7e0d4SAlon Bar-Lev static char *static_command_line; 1301da177e4SLinus Torvalds 1311da177e4SLinus Torvalds static char *execute_command; 132ffdfc409SOlof Johansson static char *ramdisk_execute_command; 1331da177e4SLinus Torvalds 1348b3b2955SJan Beulich /* 1358b3b2955SJan Beulich * If set, this is an indication to the drivers that reset the underlying 1368b3b2955SJan Beulich * device before going ahead with the initialization otherwise driver might 1378b3b2955SJan Beulich * rely on the BIOS and skip the reset operation. 1388b3b2955SJan Beulich * 1398b3b2955SJan Beulich * This is useful if kernel is booting in an unreliable environment. 1408b3b2955SJan Beulich * For ex. kdump situaiton where previous kernel has crashed, BIOS has been 1418b3b2955SJan Beulich * skipped and devices will be in unknown state. 1428b3b2955SJan Beulich */ 1438b3b2955SJan Beulich unsigned int reset_devices; 1448b3b2955SJan Beulich EXPORT_SYMBOL(reset_devices); 1451da177e4SLinus Torvalds 1467e96287dSVivek Goyal static int __init set_reset_devices(char *str) 1477e96287dSVivek Goyal { 1487e96287dSVivek Goyal reset_devices = 1; 1497e96287dSVivek Goyal return 1; 1507e96287dSVivek Goyal } 1517e96287dSVivek Goyal 1527e96287dSVivek Goyal __setup("reset_devices", set_reset_devices); 1537e96287dSVivek Goyal 154d7627467SDavid Howells static const char * argv_init[MAX_INIT_ARGS+2] = { "init", NULL, }; 155d7627467SDavid Howells const char * envp_init[MAX_INIT_ENVS+2] = { "HOME=/", "TERM=linux", NULL, }; 1561da177e4SLinus Torvalds static const char *panic_later, *panic_param; 1571da177e4SLinus Torvalds 158914dcaa8SRusty Russell extern const struct obs_kernel_param __setup_start[], __setup_end[]; 1591da177e4SLinus Torvalds 1601da177e4SLinus Torvalds static int __init obsolete_checksetup(char *line) 1611da177e4SLinus Torvalds { 162914dcaa8SRusty Russell const struct obs_kernel_param *p; 16333df0d19SRusty Russell int had_early_param = 0; 1641da177e4SLinus Torvalds 1651da177e4SLinus Torvalds p = __setup_start; 1661da177e4SLinus Torvalds do { 1671da177e4SLinus Torvalds int n = strlen(p->str); 168b1e4d20cSMichal Schmidt if (parameqn(line, p->str, n)) { 1691da177e4SLinus Torvalds if (p->early) { 17033df0d19SRusty Russell /* Already done in parse_early_param? 17133df0d19SRusty Russell * (Needs exact match on param part). 17233df0d19SRusty Russell * Keep iterating, as we can have early 17333df0d19SRusty Russell * params and __setups of same names 8( */ 1741da177e4SLinus Torvalds if (line[n] == '\0' || line[n] == '=') 17533df0d19SRusty Russell had_early_param = 1; 1761da177e4SLinus Torvalds } else if (!p->setup_func) { 1771da177e4SLinus Torvalds printk(KERN_WARNING "Parameter %s is obsolete," 1781da177e4SLinus Torvalds " ignored\n", p->str); 1791da177e4SLinus Torvalds return 1; 1801da177e4SLinus Torvalds } else if (p->setup_func(line + n)) 1811da177e4SLinus Torvalds return 1; 1821da177e4SLinus Torvalds } 1831da177e4SLinus Torvalds p++; 1841da177e4SLinus Torvalds } while (p < __setup_end); 18533df0d19SRusty Russell 18633df0d19SRusty Russell return had_early_param; 1871da177e4SLinus Torvalds } 1881da177e4SLinus Torvalds 1891da177e4SLinus Torvalds /* 1901da177e4SLinus Torvalds * This should be approx 2 Bo*oMips to start (note initial shift), and will 1911da177e4SLinus Torvalds * still work even if initially too large, it will just take slightly longer 1921da177e4SLinus Torvalds */ 1931da177e4SLinus Torvalds unsigned long loops_per_jiffy = (1<<12); 1941da177e4SLinus Torvalds 1951da177e4SLinus Torvalds EXPORT_SYMBOL(loops_per_jiffy); 1961da177e4SLinus Torvalds 1971da177e4SLinus Torvalds static int __init debug_kernel(char *str) 1981da177e4SLinus Torvalds { 1991da177e4SLinus Torvalds console_loglevel = 10; 200f6f21c81SYinghai Lu return 0; 2011da177e4SLinus Torvalds } 2021da177e4SLinus Torvalds 2031da177e4SLinus Torvalds static int __init quiet_kernel(char *str) 2041da177e4SLinus Torvalds { 2051da177e4SLinus Torvalds console_loglevel = 4; 206f6f21c81SYinghai Lu return 0; 2071da177e4SLinus Torvalds } 2081da177e4SLinus Torvalds 209f6f21c81SYinghai Lu early_param("debug", debug_kernel); 210f6f21c81SYinghai Lu early_param("quiet", quiet_kernel); 2111da177e4SLinus Torvalds 2121da177e4SLinus Torvalds static int __init loglevel(char *str) 2131da177e4SLinus Torvalds { 214808bf29bSAlexander Sverdlin int newlevel; 215808bf29bSAlexander Sverdlin 216808bf29bSAlexander Sverdlin /* 217808bf29bSAlexander Sverdlin * Only update loglevel value when a correct setting was passed, 218808bf29bSAlexander Sverdlin * to prevent blind crashes (when loglevel being set to 0) that 219808bf29bSAlexander Sverdlin * are quite hard to debug 220808bf29bSAlexander Sverdlin */ 221808bf29bSAlexander Sverdlin if (get_option(&str, &newlevel)) { 222808bf29bSAlexander Sverdlin console_loglevel = newlevel; 223d9d4fcfeSAlex Riesen return 0; 2241da177e4SLinus Torvalds } 2251da177e4SLinus Torvalds 226808bf29bSAlexander Sverdlin return -EINVAL; 227808bf29bSAlexander Sverdlin } 228808bf29bSAlexander Sverdlin 229f6f21c81SYinghai Lu early_param("loglevel", loglevel); 2301da177e4SLinus Torvalds 2311da177e4SLinus Torvalds /* Change NUL term back to "=", to make "param" the whole string. */ 232eb157427SGreg Kroah-Hartman static int __init repair_env_string(char *param, char *val, const char *unused) 233a99cd112SChris Metcalf { 2341da177e4SLinus Torvalds if (val) { 2351da177e4SLinus Torvalds /* param=val or param="val"? */ 2361da177e4SLinus Torvalds if (val == param+strlen(param)+1) 2371da177e4SLinus Torvalds val[-1] = '='; 2381da177e4SLinus Torvalds else if (val == param+strlen(param)+2) { 2391da177e4SLinus Torvalds val[-2] = '='; 2401da177e4SLinus Torvalds memmove(val-1, val, strlen(val)+1); 2411da177e4SLinus Torvalds val--; 2421da177e4SLinus Torvalds } else 2431da177e4SLinus Torvalds BUG(); 2441da177e4SLinus Torvalds } 245a99cd112SChris Metcalf return 0; 246a99cd112SChris Metcalf } 247a99cd112SChris Metcalf 248a99cd112SChris Metcalf /* 249a99cd112SChris Metcalf * Unknown boot options get handed to init, unless they look like 250a99cd112SChris Metcalf * unused parameters (modprobe will find them in /proc/cmdline). 251a99cd112SChris Metcalf */ 252eb157427SGreg Kroah-Hartman static int __init unknown_bootoption(char *param, char *val, const char *unused) 253a99cd112SChris Metcalf { 254eb157427SGreg Kroah-Hartman repair_env_string(param, val, unused); 2551da177e4SLinus Torvalds 2561da177e4SLinus Torvalds /* Handle obsolete-style parameters */ 2571da177e4SLinus Torvalds if (obsolete_checksetup(param)) 2581da177e4SLinus Torvalds return 0; 2591da177e4SLinus Torvalds 260f066a4f6SRusty Russell /* Unused module parameter. */ 261f066a4f6SRusty Russell if (strchr(param, '.') && (!val || strchr(param, '.') < val)) 2621da177e4SLinus Torvalds return 0; 2631da177e4SLinus Torvalds 2641da177e4SLinus Torvalds if (panic_later) 2651da177e4SLinus Torvalds return 0; 2661da177e4SLinus Torvalds 2671da177e4SLinus Torvalds if (val) { 2681da177e4SLinus Torvalds /* Environment option */ 2691da177e4SLinus Torvalds unsigned int i; 2701da177e4SLinus Torvalds for (i = 0; envp_init[i]; i++) { 2711da177e4SLinus Torvalds if (i == MAX_INIT_ENVS) { 2721da177e4SLinus Torvalds panic_later = "Too many boot env vars at `%s'"; 2731da177e4SLinus Torvalds panic_param = param; 2741da177e4SLinus Torvalds } 2751da177e4SLinus Torvalds if (!strncmp(param, envp_init[i], val - param)) 2761da177e4SLinus Torvalds break; 2771da177e4SLinus Torvalds } 2781da177e4SLinus Torvalds envp_init[i] = param; 2791da177e4SLinus Torvalds } else { 2801da177e4SLinus Torvalds /* Command line option */ 2811da177e4SLinus Torvalds unsigned int i; 2821da177e4SLinus Torvalds for (i = 0; argv_init[i]; i++) { 2831da177e4SLinus Torvalds if (i == MAX_INIT_ARGS) { 2841da177e4SLinus Torvalds panic_later = "Too many boot init vars at `%s'"; 2851da177e4SLinus Torvalds panic_param = param; 2861da177e4SLinus Torvalds } 2871da177e4SLinus Torvalds } 2881da177e4SLinus Torvalds argv_init[i] = param; 2891da177e4SLinus Torvalds } 2901da177e4SLinus Torvalds return 0; 2911da177e4SLinus Torvalds } 2921da177e4SLinus Torvalds 2931da177e4SLinus Torvalds static int __init init_setup(char *str) 2941da177e4SLinus Torvalds { 2951da177e4SLinus Torvalds unsigned int i; 2961da177e4SLinus Torvalds 2971da177e4SLinus Torvalds execute_command = str; 2981da177e4SLinus Torvalds /* 2991da177e4SLinus Torvalds * In case LILO is going to boot us with default command line, 3001da177e4SLinus Torvalds * it prepends "auto" before the whole cmdline which makes 3011da177e4SLinus Torvalds * the shell think it should execute a script with such name. 3021da177e4SLinus Torvalds * So we ignore all arguments entered _before_ init=... [MJ] 3031da177e4SLinus Torvalds */ 3041da177e4SLinus Torvalds for (i = 1; i < MAX_INIT_ARGS; i++) 3051da177e4SLinus Torvalds argv_init[i] = NULL; 3061da177e4SLinus Torvalds return 1; 3071da177e4SLinus Torvalds } 3081da177e4SLinus Torvalds __setup("init=", init_setup); 3091da177e4SLinus Torvalds 310ffdfc409SOlof Johansson static int __init rdinit_setup(char *str) 311ffdfc409SOlof Johansson { 312ffdfc409SOlof Johansson unsigned int i; 313ffdfc409SOlof Johansson 314ffdfc409SOlof Johansson ramdisk_execute_command = str; 315ffdfc409SOlof Johansson /* See "auto" comment in init_setup */ 316ffdfc409SOlof Johansson for (i = 1; i < MAX_INIT_ARGS; i++) 317ffdfc409SOlof Johansson argv_init[i] = NULL; 318ffdfc409SOlof Johansson return 1; 319ffdfc409SOlof Johansson } 320ffdfc409SOlof Johansson __setup("rdinit=", rdinit_setup); 321ffdfc409SOlof Johansson 3221da177e4SLinus Torvalds #ifndef CONFIG_SMP 32334db18a0SAmerigo Wang static const unsigned int setup_max_cpus = NR_CPUS; 3241da177e4SLinus Torvalds #ifdef CONFIG_X86_LOCAL_APIC 3251da177e4SLinus Torvalds static void __init smp_init(void) 3261da177e4SLinus Torvalds { 3271da177e4SLinus Torvalds APIC_init_uniprocessor(); 3281da177e4SLinus Torvalds } 3291da177e4SLinus Torvalds #else 3301da177e4SLinus Torvalds #define smp_init() do { } while (0) 3311da177e4SLinus Torvalds #endif 3321da177e4SLinus Torvalds 333e0982e90SMike Travis static inline void setup_nr_cpu_ids(void) { } 3341da177e4SLinus Torvalds static inline void smp_prepare_cpus(unsigned int maxcpus) { } 3351da177e4SLinus Torvalds #endif 3361da177e4SLinus Torvalds 3371da177e4SLinus Torvalds /* 33830d7e0d4SAlon Bar-Lev * We need to store the untouched command line for future reference. 33930d7e0d4SAlon Bar-Lev * We also need to store the touched command line since the parameter 34030d7e0d4SAlon Bar-Lev * parsing is performed in place, and we should allow a component to 34130d7e0d4SAlon Bar-Lev * store reference of name/value for future reference. 34230d7e0d4SAlon Bar-Lev */ 34330d7e0d4SAlon Bar-Lev static void __init setup_command_line(char *command_line) 34430d7e0d4SAlon Bar-Lev { 34530d7e0d4SAlon Bar-Lev saved_command_line = alloc_bootmem(strlen (boot_command_line)+1); 34630d7e0d4SAlon Bar-Lev static_command_line = alloc_bootmem(strlen (command_line)+1); 34730d7e0d4SAlon Bar-Lev strcpy (saved_command_line, boot_command_line); 34830d7e0d4SAlon Bar-Lev strcpy (static_command_line, command_line); 34930d7e0d4SAlon Bar-Lev } 35030d7e0d4SAlon Bar-Lev 35130d7e0d4SAlon Bar-Lev /* 3521da177e4SLinus Torvalds * We need to finalize in a non-__init function or else race conditions 3531da177e4SLinus Torvalds * between the root thread and the init thread may cause start_kernel to 3541da177e4SLinus Torvalds * be reaped by free_initmem before the root thread has proceeded to 3551da177e4SLinus Torvalds * cpu_idle. 3561da177e4SLinus Torvalds * 3571da177e4SLinus Torvalds * gcc-3.4 accidentally inlines this function, so use noinline. 3581da177e4SLinus Torvalds */ 3591da177e4SLinus Torvalds 360b433c3d4SPeter Zijlstra static __initdata DECLARE_COMPLETION(kthreadd_done); 361b433c3d4SPeter Zijlstra 362f99ebf0aSRakib Mullick static noinline void __init_refok rest_init(void) 3631da177e4SLinus Torvalds { 36473c27992SEric W. Biederman int pid; 36573c27992SEric W. Biederman 3667db905e6SPaul E. McKenney rcu_scheduler_starting(); 367b433c3d4SPeter Zijlstra /* 36897158569SPeter Zijlstra * We need to spawn init first so that it obtains pid 1, however 369b433c3d4SPeter Zijlstra * the init task will end up wanting to create kthreads, which, if 370b433c3d4SPeter Zijlstra * we schedule it before we create kthreadd, will OOPS. 371b433c3d4SPeter Zijlstra */ 372aae5f662SSam Ravnborg kernel_thread(kernel_init, NULL, CLONE_FS | CLONE_SIGHAND); 3731da177e4SLinus Torvalds numa_default_policy(); 37473c27992SEric W. Biederman pid = kernel_thread(kthreadd, NULL, CLONE_FS | CLONE_FILES); 375d11c563dSPaul E. McKenney rcu_read_lock(); 3765cd20455SPavel Emelyanov kthreadd_task = find_task_by_pid_ns(pid, &init_pid_ns); 377d11c563dSPaul E. McKenney rcu_read_unlock(); 378b433c3d4SPeter Zijlstra complete(&kthreadd_done); 379f340c0d1SIngo Molnar 380f340c0d1SIngo Molnar /* 381f340c0d1SIngo Molnar * The boot idle thread must execute schedule() 3821df21055SIngo Molnar * at least once to get things moving: 383f340c0d1SIngo Molnar */ 3841df21055SIngo Molnar init_idle_bootup_task(current); 385bd2f5536SThomas Gleixner schedule_preempt_disabled(); 3865bfb5d69SNick Piggin /* Call into cpu_idle with preempt disabled */ 3871da177e4SLinus Torvalds cpu_idle(); 3881da177e4SLinus Torvalds } 3891da177e4SLinus Torvalds 3901da177e4SLinus Torvalds /* Check for early params. */ 3919fb48c74SJim Cromie static int __init do_early_param(char *param, char *val, const char *unused) 3921da177e4SLinus Torvalds { 393914dcaa8SRusty Russell const struct obs_kernel_param *p; 3941da177e4SLinus Torvalds 3951da177e4SLinus Torvalds for (p = __setup_start; p < __setup_end; p++) { 396b1e4d20cSMichal Schmidt if ((p->early && parameq(param, p->str)) || 39718a8bd94SYinghai Lu (strcmp(param, "console") == 0 && 39818a8bd94SYinghai Lu strcmp(p->str, "earlycon") == 0) 39918a8bd94SYinghai Lu ) { 4001da177e4SLinus Torvalds if (p->setup_func(val) != 0) 4011da177e4SLinus Torvalds printk(KERN_WARNING 4021da177e4SLinus Torvalds "Malformed early option '%s'\n", param); 4031da177e4SLinus Torvalds } 4041da177e4SLinus Torvalds } 4051da177e4SLinus Torvalds /* We accept everything at this stage. */ 4061da177e4SLinus Torvalds return 0; 4071da177e4SLinus Torvalds } 4081da177e4SLinus Torvalds 40913977091SMagnus Damm void __init parse_early_options(char *cmdline) 41013977091SMagnus Damm { 411026cee00SPawel Moll parse_args("early options", cmdline, NULL, 0, 0, 0, do_early_param); 41213977091SMagnus Damm } 41313977091SMagnus Damm 4141da177e4SLinus Torvalds /* Arch code calls this early on, or if not, just before other parsing. */ 4151da177e4SLinus Torvalds void __init parse_early_param(void) 4161da177e4SLinus Torvalds { 4171da177e4SLinus Torvalds static __initdata int done = 0; 4181da177e4SLinus Torvalds static __initdata char tmp_cmdline[COMMAND_LINE_SIZE]; 4191da177e4SLinus Torvalds 4201da177e4SLinus Torvalds if (done) 4211da177e4SLinus Torvalds return; 4221da177e4SLinus Torvalds 4231da177e4SLinus Torvalds /* All fall through to do_early_param. */ 42430d7e0d4SAlon Bar-Lev strlcpy(tmp_cmdline, boot_command_line, COMMAND_LINE_SIZE); 42513977091SMagnus Damm parse_early_options(tmp_cmdline); 4261da177e4SLinus Torvalds done = 1; 4271da177e4SLinus Torvalds } 4281da177e4SLinus Torvalds 4291da177e4SLinus Torvalds /* 4301da177e4SLinus Torvalds * Activate the first processor. 4311da177e4SLinus Torvalds */ 4321da177e4SLinus Torvalds 43344fd2299SStas Sergeev static void __init boot_cpu_init(void) 43444fd2299SStas Sergeev { 43544fd2299SStas Sergeev int cpu = smp_processor_id(); 43644fd2299SStas Sergeev /* Mark the boot cpu "present", "online" etc for SMP and UP case */ 437915441b6SRusty Russell set_cpu_online(cpu, true); 438933b0618SPeter Zijlstra set_cpu_active(cpu, true); 439915441b6SRusty Russell set_cpu_present(cpu, true); 440915441b6SRusty Russell set_cpu_possible(cpu, true); 44144fd2299SStas Sergeev } 44244fd2299SStas Sergeev 443839ad62eSBenjamin Herrenschmidt void __init __weak smp_setup_processor_id(void) 444033ab7f8SAndrew Morton { 445033ab7f8SAndrew Morton } 446033ab7f8SAndrew Morton 447eded09ccSDavid Howells # if THREAD_SIZE >= PAGE_SIZE 4488c9843e5SBenjamin Herrenschmidt void __init __weak thread_info_cache_init(void) 4498c9843e5SBenjamin Herrenschmidt { 4508c9843e5SBenjamin Herrenschmidt } 451eded09ccSDavid Howells #endif 4528c9843e5SBenjamin Herrenschmidt 453444f478fSPekka Enberg /* 454444f478fSPekka Enberg * Set up kernel memory allocators 455444f478fSPekka Enberg */ 456444f478fSPekka Enberg static void __init mm_init(void) 457444f478fSPekka Enberg { 458ca371c0dSKAMEZAWA Hiroyuki /* 4597fa87ce7SJim Cromie * page_cgroup requires contiguous pages, 4607fa87ce7SJim Cromie * bigger than MAX_ORDER unless SPARSEMEM. 461ca371c0dSKAMEZAWA Hiroyuki */ 462ca371c0dSKAMEZAWA Hiroyuki page_cgroup_init_flatmem(); 463444f478fSPekka Enberg mem_init(); 464444f478fSPekka Enberg kmem_cache_init(); 465099a19d9STejun Heo percpu_init_late(); 466c868d550SBenjamin Herrenschmidt pgtable_cache_init(); 467444f478fSPekka Enberg vmalloc_init(); 468444f478fSPekka Enberg } 469444f478fSPekka Enberg 4701da177e4SLinus Torvalds asmlinkage void __init start_kernel(void) 4711da177e4SLinus Torvalds { 4721da177e4SLinus Torvalds char * command_line; 473914dcaa8SRusty Russell extern const struct kernel_param __start___param[], __stop___param[]; 474033ab7f8SAndrew Morton 4751da177e4SLinus Torvalds /* 476fbb9ce95SIngo Molnar * Need to run as early as possible, to initialize the 477fbb9ce95SIngo Molnar * lockdep hash: 478fbb9ce95SIngo Molnar */ 479fbb9ce95SIngo Molnar lockdep_init(); 48073839c5bSMing Lei smp_setup_processor_id(); 4813ac7fe5aSThomas Gleixner debug_objects_early_init(); 48242059429SIngo Molnar 48342059429SIngo Molnar /* 48442059429SIngo Molnar * Set up the the initial canary ASAP: 48542059429SIngo Molnar */ 48642059429SIngo Molnar boot_init_stack_canary(); 48742059429SIngo Molnar 488ddbcc7e8SPaul Menage cgroup_init_early(); 489fbb9ce95SIngo Molnar 490fbb9ce95SIngo Molnar local_irq_disable(); 4912ce802f6STejun Heo early_boot_irqs_disabled = true; 492fbb9ce95SIngo Molnar 493fbb9ce95SIngo Molnar /* 4941da177e4SLinus Torvalds * Interrupts are still disabled. Do necessary setups, then 4951da177e4SLinus Torvalds * enable them 4961da177e4SLinus Torvalds */ 49744fd2299SStas Sergeev boot_cpu_init(); 4981da177e4SLinus Torvalds page_address_init(); 499657cafa6SAlex Riesen printk(KERN_NOTICE "%s", linux_banner); 5001da177e4SLinus Torvalds setup_arch(&command_line); 501cf475ad2SBalbir Singh mm_init_owner(&init_mm, &init_task); 5026345d24dSLinus Torvalds mm_init_cpumask(&init_mm); 50330d7e0d4SAlon Bar-Lev setup_command_line(command_line); 504e0982e90SMike Travis setup_nr_cpu_ids(); 505d6647bdfSTejun Heo setup_per_cpu_areas(); 50644fd2299SStas Sergeev smp_prepare_boot_cpu(); /* arch-specific boot-cpu hooks */ 5071da177e4SLinus Torvalds 5089adb62a5SJiang Liu build_all_zonelists(NULL, NULL); 50983b519e8SPekka Enberg page_alloc_init(); 51083b519e8SPekka Enberg 51183b519e8SPekka Enberg printk(KERN_NOTICE "Kernel command line: %s\n", boot_command_line); 51283b519e8SPekka Enberg parse_early_param(); 51383b519e8SPekka Enberg parse_args("Booting kernel", static_command_line, __start___param, 51483b519e8SPekka Enberg __stop___param - __start___param, 515ae82fdb1SRusty Russell -1, -1, &unknown_bootoption); 51697ce2c88SJeremy Fitzhardinge 51797ce2c88SJeremy Fitzhardinge jump_label_init(); 51897ce2c88SJeremy Fitzhardinge 51983b519e8SPekka Enberg /* 52083b519e8SPekka Enberg * These use large bootmem allocations and must precede 52183b519e8SPekka Enberg * kmem_cache_init() 52283b519e8SPekka Enberg */ 523162a7e75SMike Travis setup_log_buf(0); 52483b519e8SPekka Enberg pidhash_init(); 52583b519e8SPekka Enberg vfs_caches_init_early(); 52683b519e8SPekka Enberg sort_main_extable(); 52783b519e8SPekka Enberg trap_init(); 528444f478fSPekka Enberg mm_init(); 529de03c72cSKOSAKI Motohiro 5301da177e4SLinus Torvalds /* 5311da177e4SLinus Torvalds * Set up the scheduler prior starting any interrupts (such as the 5321da177e4SLinus Torvalds * timer interrupt). Full topology setup happens at smp_init() 5331da177e4SLinus Torvalds * time - but meanwhile we still have a functioning scheduler. 5341da177e4SLinus Torvalds */ 5351da177e4SLinus Torvalds sched_init(); 5361da177e4SLinus Torvalds /* 5371da177e4SLinus Torvalds * Disable preemption - early bootup scheduling is extremely 5381da177e4SLinus Torvalds * fragile until we cpu_idle() for the first time. 5391da177e4SLinus Torvalds */ 5401da177e4SLinus Torvalds preempt_disable(); 541c4a68306SArd van Breemen if (!irqs_disabled()) { 542c4a68306SArd van Breemen printk(KERN_WARNING "start_kernel(): bug: interrupts were " 543c4a68306SArd van Breemen "enabled *very* early, fixing it\n"); 544c4a68306SArd van Breemen local_irq_disable(); 545c4a68306SArd van Breemen } 5469f58a205SPeter Zijlstra idr_init_cache(); 54724a24bb6SPeter Zijlstra perf_event_init(); 5481da177e4SLinus Torvalds rcu_init(); 549773e3eb7SYinghai Lu radix_tree_init(); 5500b8f1efaSYinghai Lu /* init some links before init_ISA_irqs() */ 5510b8f1efaSYinghai Lu early_irq_init(); 5521da177e4SLinus Torvalds init_IRQ(); 553*ad2b1353SThomas Gleixner tick_init(); 5541da177e4SLinus Torvalds init_timers(); 555c0a31329SThomas Gleixner hrtimers_init(); 5561da177e4SLinus Torvalds softirq_init(); 557ad596171Sjohn stultz timekeeping_init(); 55888fecaa2Sjohn stultz time_init(); 55993e02814SHeiko Carstens profile_init(); 560d8ad7d11STakao Indoh call_function_init(); 56193e02814SHeiko Carstens if (!irqs_disabled()) 56224d431d0SRon Lee printk(KERN_CRIT "start_kernel(): bug: interrupts were " 56324d431d0SRon Lee "enabled early\n"); 5642ce802f6STejun Heo early_boot_irqs_disabled = false; 56593e02814SHeiko Carstens local_irq_enable(); 566dcce284aSBenjamin Herrenschmidt 5677e85ee0cSPekka Enberg kmem_cache_init_late(); 5681da177e4SLinus Torvalds 5691da177e4SLinus Torvalds /* 5701da177e4SLinus Torvalds * HACK ALERT! This is early. We're enabling the console before 5711da177e4SLinus Torvalds * we've done PCI setups etc, and console_init() must be aware of 5721da177e4SLinus Torvalds * this. But we do want output early, in case something goes wrong. 5731da177e4SLinus Torvalds */ 5741da177e4SLinus Torvalds console_init(); 5751da177e4SLinus Torvalds if (panic_later) 5761da177e4SLinus Torvalds panic(panic_later, panic_param); 577fbb9ce95SIngo Molnar 578fbb9ce95SIngo Molnar lockdep_info(); 579fbb9ce95SIngo Molnar 5809a11b49aSIngo Molnar /* 5819a11b49aSIngo Molnar * Need to run this when irqs are enabled, because it wants 5829a11b49aSIngo Molnar * to self-test [hard/soft]-irqs on/off lock inversion bugs 5839a11b49aSIngo Molnar * too: 5849a11b49aSIngo Molnar */ 5859a11b49aSIngo Molnar locking_selftest(); 5869a11b49aSIngo Molnar 5871da177e4SLinus Torvalds #ifdef CONFIG_BLK_DEV_INITRD 5881da177e4SLinus Torvalds if (initrd_start && !initrd_below_start_ok && 589bd673c7cSGeert Uytterhoeven page_to_pfn(virt_to_page((void *)initrd_start)) < min_low_pfn) { 5901da177e4SLinus Torvalds printk(KERN_CRIT "initrd overwritten (0x%08lx < 0x%08lx) - " 591fb6624ebSGeert Uytterhoeven "disabling it.\n", 592bd673c7cSGeert Uytterhoeven page_to_pfn(virt_to_page((void *)initrd_start)), 593bd673c7cSGeert Uytterhoeven min_low_pfn); 5941da177e4SLinus Torvalds initrd_start = 0; 5951da177e4SLinus Torvalds } 5961da177e4SLinus Torvalds #endif 59794b6da5aSKAMEZAWA Hiroyuki page_cgroup_init(); 5983ac7fe5aSThomas Gleixner debug_objects_mem_init(); 5999b090f2dSCatalin Marinas kmemleak_init(); 600e7c8d5c9SChristoph Lameter setup_per_cpu_pageset(); 6011da177e4SLinus Torvalds numa_policy_init(); 6021da177e4SLinus Torvalds if (late_time_init) 6031da177e4SLinus Torvalds late_time_init(); 604fa84e9eeSThomas Gleixner sched_clock_init(); 6051da177e4SLinus Torvalds calibrate_delay(); 6061da177e4SLinus Torvalds pidmap_init(); 6071da177e4SLinus Torvalds anon_vma_init(); 60811520e5eSLinus Torvalds #ifdef CONFIG_X86 60983e68189SMatt Fleming if (efi_enabled(EFI_RUNTIME_SERVICES)) 61011520e5eSLinus Torvalds efi_enter_virtual_mode(); 61111520e5eSLinus Torvalds #endif 6128c9843e5SBenjamin Herrenschmidt thread_info_cache_init(); 613d84f4f99SDavid Howells cred_init(); 6144481374cSJan Beulich fork_init(totalram_pages); 6151da177e4SLinus Torvalds proc_caches_init(); 6161da177e4SLinus Torvalds buffer_init(); 6171da177e4SLinus Torvalds key_init(); 6181da177e4SLinus Torvalds security_init(); 6190b4b3827SJason Wessel dbg_late_init(); 6204481374cSJan Beulich vfs_caches_init(totalram_pages); 6211da177e4SLinus Torvalds signals_init(); 6221da177e4SLinus Torvalds /* rootfs populating might need page-writeback */ 6231da177e4SLinus Torvalds page_writeback_init(); 6241da177e4SLinus Torvalds #ifdef CONFIG_PROC_FS 6251da177e4SLinus Torvalds proc_root_init(); 6261da177e4SLinus Torvalds #endif 627ddbcc7e8SPaul Menage cgroup_init(); 6281da177e4SLinus Torvalds cpuset_init(); 629c757249aSShailabh Nagar taskstats_init_early(); 630ca74e92bSShailabh Nagar delayacct_init(); 6311da177e4SLinus Torvalds 6321da177e4SLinus Torvalds check_bugs(); 6331da177e4SLinus Torvalds 6341da177e4SLinus Torvalds acpi_early_init(); /* before LAPIC and SMP init */ 6356ae6996aSFeng Tang sfi_init_late(); 6361da177e4SLinus Torvalds 63783e68189SMatt Fleming if (efi_enabled(EFI_RUNTIME_SERVICES)) { 6382223af38SJosh Triplett efi_late_init(); 63978510792SJosh Triplett efi_free_boot_services(); 6402223af38SJosh Triplett } 64178510792SJosh Triplett 64268bf21aaSSteven Rostedt ftrace_init(); 64368bf21aaSSteven Rostedt 6441da177e4SLinus Torvalds /* Do the rest non-__init'ed, we're now alive */ 6451da177e4SLinus Torvalds rest_init(); 6461da177e4SLinus Torvalds } 6471da177e4SLinus Torvalds 648b99b87f7SPeter Oberparleiter /* Call all constructor functions linked into the kernel. */ 649b99b87f7SPeter Oberparleiter static void __init do_ctors(void) 650b99b87f7SPeter Oberparleiter { 651b99b87f7SPeter Oberparleiter #ifdef CONFIG_CONSTRUCTORS 652196a15b4SH Hartley Sweeten ctor_fn_t *fn = (ctor_fn_t *) __ctors_start; 653b99b87f7SPeter Oberparleiter 654196a15b4SH Hartley Sweeten for (; fn < (ctor_fn_t *) __ctors_end; fn++) 655196a15b4SH Hartley Sweeten (*fn)(); 656b99b87f7SPeter Oberparleiter #endif 657b99b87f7SPeter Oberparleiter } 658b99b87f7SPeter Oberparleiter 6592329abfaSRusty Russell bool initcall_debug; 660d0ea3d7dSRusty Russell core_param(initcall_debug, initcall_debug, bool, 0644); 6611da177e4SLinus Torvalds 6624a683bf9SIngo Molnar static char msgbuf[64]; 6634a683bf9SIngo Molnar 664e4461271SKevin Winchester static int __init_or_module do_one_initcall_debug(initcall_t fn) 6651da177e4SLinus Torvalds { 66674239072SFrederic Weisbecker ktime_t calltime, delta, rettime; 66730dbb20eSAmérico Wang unsigned long long duration; 66830dbb20eSAmérico Wang int ret; 6691da177e4SLinus Torvalds 67022c5c03bSKevin Winchester printk(KERN_DEBUG "calling %pF @ %i\n", fn, task_pid_nr(current)); 67174239072SFrederic Weisbecker calltime = ktime_get(); 67230dbb20eSAmérico Wang ret = fn(); 67374239072SFrederic Weisbecker rettime = ktime_get(); 67474239072SFrederic Weisbecker delta = ktime_sub(rettime, calltime); 67530dbb20eSAmérico Wang duration = (unsigned long long) ktime_to_ns(delta) >> 10; 67622c5c03bSKevin Winchester printk(KERN_DEBUG "initcall %pF returned %d after %lld usecs\n", fn, 67730dbb20eSAmérico Wang ret, duration); 67822c5c03bSKevin Winchester 67922c5c03bSKevin Winchester return ret; 6808f0c45cdSIngo Molnar } 6818f0c45cdSIngo Molnar 682e4461271SKevin Winchester int __init_or_module do_one_initcall(initcall_t fn) 68322c5c03bSKevin Winchester { 68422c5c03bSKevin Winchester int count = preempt_count(); 68522c5c03bSKevin Winchester int ret; 68622c5c03bSKevin Winchester 68722c5c03bSKevin Winchester if (initcall_debug) 68822c5c03bSKevin Winchester ret = do_one_initcall_debug(fn); 68922c5c03bSKevin Winchester else 69022c5c03bSKevin Winchester ret = fn(); 69122c5c03bSKevin Winchester 692e662e1cfSCyrill Gorcunov msgbuf[0] = 0; 693e662e1cfSCyrill Gorcunov 69430dbb20eSAmérico Wang if (ret && ret != -ENODEV && initcall_debug) 69530dbb20eSAmérico Wang sprintf(msgbuf, "error code %d ", ret); 696e662e1cfSCyrill Gorcunov 6971da177e4SLinus Torvalds if (preempt_count() != count) { 698a76bfd0dSCyrill Gorcunov strlcat(msgbuf, "preemption imbalance ", sizeof(msgbuf)); 6991da177e4SLinus Torvalds preempt_count() = count; 7001da177e4SLinus Torvalds } 7011da177e4SLinus Torvalds if (irqs_disabled()) { 702a76bfd0dSCyrill Gorcunov strlcat(msgbuf, "disabled interrupts ", sizeof(msgbuf)); 7031da177e4SLinus Torvalds local_irq_enable(); 7041da177e4SLinus Torvalds } 705e662e1cfSCyrill Gorcunov if (msgbuf[0]) { 70696d746c6SLinus Torvalds printk("initcall %pF returned with %s\n", fn, msgbuf); 7071da177e4SLinus Torvalds } 70859f9415fSArjan van de Ven 70930dbb20eSAmérico Wang return ret; 7101da177e4SLinus Torvalds } 7111da177e4SLinus Torvalds 712e0df154fSLinus Torvalds 713026cee00SPawel Moll extern initcall_t __initcall_start[]; 714026cee00SPawel Moll extern initcall_t __initcall0_start[]; 715026cee00SPawel Moll extern initcall_t __initcall1_start[]; 716026cee00SPawel Moll extern initcall_t __initcall2_start[]; 717026cee00SPawel Moll extern initcall_t __initcall3_start[]; 718026cee00SPawel Moll extern initcall_t __initcall4_start[]; 719026cee00SPawel Moll extern initcall_t __initcall5_start[]; 720026cee00SPawel Moll extern initcall_t __initcall6_start[]; 721026cee00SPawel Moll extern initcall_t __initcall7_start[]; 722026cee00SPawel Moll extern initcall_t __initcall_end[]; 723026cee00SPawel Moll 724026cee00SPawel Moll static initcall_t *initcall_levels[] __initdata = { 725026cee00SPawel Moll __initcall0_start, 726026cee00SPawel Moll __initcall1_start, 727026cee00SPawel Moll __initcall2_start, 728026cee00SPawel Moll __initcall3_start, 729026cee00SPawel Moll __initcall4_start, 730026cee00SPawel Moll __initcall5_start, 731026cee00SPawel Moll __initcall6_start, 732026cee00SPawel Moll __initcall7_start, 733026cee00SPawel Moll __initcall_end, 734026cee00SPawel Moll }; 735026cee00SPawel Moll 73696263d28SJim Cromie /* Keep these in sync with initcalls in include/linux/init.h */ 737026cee00SPawel Moll static char *initcall_level_names[] __initdata = { 7389fb48c74SJim Cromie "early", 7399fb48c74SJim Cromie "core", 7409fb48c74SJim Cromie "postcore", 7419fb48c74SJim Cromie "arch", 7429fb48c74SJim Cromie "subsys", 7439fb48c74SJim Cromie "fs", 7449fb48c74SJim Cromie "device", 7459fb48c74SJim Cromie "late", 746026cee00SPawel Moll }; 747026cee00SPawel Moll 748026cee00SPawel Moll static void __init do_initcall_level(int level) 749026cee00SPawel Moll { 750026cee00SPawel Moll extern const struct kernel_param __start___param[], __stop___param[]; 751026cee00SPawel Moll initcall_t *fn; 752026cee00SPawel Moll 753026cee00SPawel Moll strcpy(static_command_line, saved_command_line); 754026cee00SPawel Moll parse_args(initcall_level_names[level], 755026cee00SPawel Moll static_command_line, __start___param, 756026cee00SPawel Moll __stop___param - __start___param, 757026cee00SPawel Moll level, level, 758eb157427SGreg Kroah-Hartman &repair_env_string); 759026cee00SPawel Moll 760026cee00SPawel Moll for (fn = initcall_levels[level]; fn < initcall_levels[level+1]; fn++) 761026cee00SPawel Moll do_one_initcall(*fn); 762026cee00SPawel Moll } 763e0df154fSLinus Torvalds 764e0df154fSLinus Torvalds static void __init do_initcalls(void) 765e0df154fSLinus Torvalds { 766026cee00SPawel Moll int level; 767e0df154fSLinus Torvalds 76819efb72fSBorislav Petkov for (level = 0; level < ARRAY_SIZE(initcall_levels) - 1; level++) 769026cee00SPawel Moll do_initcall_level(level); 7701da177e4SLinus Torvalds } 7711da177e4SLinus Torvalds 7721da177e4SLinus Torvalds /* 7731da177e4SLinus Torvalds * Ok, the machine is now initialized. None of the devices 7741da177e4SLinus Torvalds * have been touched yet, but the CPU subsystem is up and 7751da177e4SLinus Torvalds * running, and memory and process management works. 7761da177e4SLinus Torvalds * 7771da177e4SLinus Torvalds * Now we can finally start doing some real work.. 7781da177e4SLinus Torvalds */ 7791da177e4SLinus Torvalds static void __init do_basic_setup(void) 7801da177e4SLinus Torvalds { 781759ee091SLai Jiangshan cpuset_init_smp(); 7821da177e4SLinus Torvalds usermodehelper_init(); 78341ffe5d5SHugh Dickins shmem_init(); 7841da177e4SLinus Torvalds driver_init(); 785b04c3afbSEric W. Biederman init_irq_proc(); 786b99b87f7SPeter Oberparleiter do_ctors(); 787d5767c53SLinus Torvalds usermodehelper_enable(); 788b0f84374Swangyanqing do_initcalls(); 7891da177e4SLinus Torvalds } 7901da177e4SLinus Torvalds 7917babe8dbSEduard - Gabriel Munteanu static void __init do_pre_smp_initcalls(void) 792c2147a50SEduard - Gabriel Munteanu { 793196a15b4SH Hartley Sweeten initcall_t *fn; 794c2147a50SEduard - Gabriel Munteanu 795026cee00SPawel Moll for (fn = __initcall_start; fn < __initcall0_start; fn++) 796196a15b4SH Hartley Sweeten do_one_initcall(*fn); 797c2147a50SEduard - Gabriel Munteanu } 798c2147a50SEduard - Gabriel Munteanu 799bb813f4cSTejun Heo /* 800bb813f4cSTejun Heo * This function requests modules which should be loaded by default and is 801bb813f4cSTejun Heo * called twice right after initrd is mounted and right before init is 802bb813f4cSTejun Heo * exec'd. If such modules are on either initrd or rootfs, they will be 803bb813f4cSTejun Heo * loaded before control is passed to userland. 804bb813f4cSTejun Heo */ 805bb813f4cSTejun Heo void __init load_default_modules(void) 806bb813f4cSTejun Heo { 807bb813f4cSTejun Heo load_default_elevator_module(); 808bb813f4cSTejun Heo } 809bb813f4cSTejun Heo 810a74fb73cSAl Viro static int run_init_process(const char *init_filename) 8111da177e4SLinus Torvalds { 8121da177e4SLinus Torvalds argv_init[0] = init_filename; 813ae903caaSAl Viro return do_execve(init_filename, 814ae903caaSAl Viro (const char __user *const __user *)argv_init, 815ae903caaSAl Viro (const char __user *const __user *)envp_init); 8161da177e4SLinus Torvalds } 8171da177e4SLinus Torvalds 818f80b0c90SVineet Gupta static noinline void __init kernel_init_freeable(void); 819d6b21238SAl Viro 820d6b21238SAl Viro static int __ref kernel_init(void *unused) 821ee5bfa64SVivek Goyal { 822d6b21238SAl Viro kernel_init_freeable(); 82322a9d645SArjan van de Ven /* need to finish all async __init code before freeing the memory */ 82422a9d645SArjan van de Ven async_synchronize_full(); 825ee5bfa64SVivek Goyal free_initmem(); 826ee5bfa64SVivek Goyal mark_rodata_ro(); 827ee5bfa64SVivek Goyal system_state = SYSTEM_RUNNING; 828ee5bfa64SVivek Goyal numa_default_policy(); 829ee5bfa64SVivek Goyal 8304a9d4b02SAl Viro flush_delayed_fput(); 831fae5fa44SOleg Nesterov 832ee5bfa64SVivek Goyal if (ramdisk_execute_command) { 833a74fb73cSAl Viro if (!run_init_process(ramdisk_execute_command)) 834a74fb73cSAl Viro return 0; 835ee5bfa64SVivek Goyal printk(KERN_WARNING "Failed to execute %s\n", 836ee5bfa64SVivek Goyal ramdisk_execute_command); 837ee5bfa64SVivek Goyal } 838ee5bfa64SVivek Goyal 839ee5bfa64SVivek Goyal /* 840ee5bfa64SVivek Goyal * We try each of these until one succeeds. 841ee5bfa64SVivek Goyal * 842ee5bfa64SVivek Goyal * The Bourne shell can be used instead of init if we are 843ee5bfa64SVivek Goyal * trying to recover a really broken machine. 844ee5bfa64SVivek Goyal */ 845ee5bfa64SVivek Goyal if (execute_command) { 846a74fb73cSAl Viro if (!run_init_process(execute_command)) 847a74fb73cSAl Viro return 0; 848ee5bfa64SVivek Goyal printk(KERN_WARNING "Failed to execute %s. Attempting " 849ee5bfa64SVivek Goyal "defaults...\n", execute_command); 850ee5bfa64SVivek Goyal } 851a74fb73cSAl Viro if (!run_init_process("/sbin/init") || 852a74fb73cSAl Viro !run_init_process("/etc/init") || 853a74fb73cSAl Viro !run_init_process("/bin/init") || 854a74fb73cSAl Viro !run_init_process("/bin/sh")) 855a74fb73cSAl Viro return 0; 856ee5bfa64SVivek Goyal 8579a85b8d6SAndreas Mohr panic("No init found. Try passing init= option to kernel. " 8589a85b8d6SAndreas Mohr "See Linux Documentation/init.txt for guidance."); 859ee5bfa64SVivek Goyal } 860ee5bfa64SVivek Goyal 861f80b0c90SVineet Gupta static noinline void __init kernel_init_freeable(void) 8621da177e4SLinus Torvalds { 863b433c3d4SPeter Zijlstra /* 864b433c3d4SPeter Zijlstra * Wait until kthreadd is all set-up. 865b433c3d4SPeter Zijlstra */ 866b433c3d4SPeter Zijlstra wait_for_completion(&kthreadd_done); 86731a67102SLinus Torvalds 86831a67102SLinus Torvalds /* Now the scheduler is fully set up and can do blocking allocations */ 86931a67102SLinus Torvalds gfp_allowed_mask = __GFP_BITS_MASK; 87031a67102SLinus Torvalds 87158568d2aSMiao Xie /* 87258568d2aSMiao Xie * init can allocate pages on any node 87358568d2aSMiao Xie */ 8743c466d46SLai Jiangshan set_mems_allowed(node_states[N_MEMORY]); 8751da177e4SLinus Torvalds /* 8761da177e4SLinus Torvalds * init can run on any cpu. 8771da177e4SLinus Torvalds */ 8781a2142afSRusty Russell set_cpus_allowed_ptr(current, cpu_all_mask); 8791da177e4SLinus Torvalds 8809ec52099SCedric Le Goater cad_pid = task_pid(current); 8819ec52099SCedric Le Goater 882ca74a6f8SAndi Kleen smp_prepare_cpus(setup_max_cpus); 8831da177e4SLinus Torvalds 8841da177e4SLinus Torvalds do_pre_smp_initcalls(); 885004417a6SPeter Zijlstra lockup_detector_init(); 8861da177e4SLinus Torvalds 8871da177e4SLinus Torvalds smp_init(); 8881da177e4SLinus Torvalds sched_init_smp(); 8891da177e4SLinus Torvalds 8901da177e4SLinus Torvalds do_basic_setup(); 8911da177e4SLinus Torvalds 8922bd3a997SEric W. Biederman /* Open the /dev/console on the rootfs, this should never fail */ 8932bd3a997SEric W. Biederman if (sys_open((const char __user *) "/dev/console", O_RDWR, 0) < 0) 8942bd3a997SEric W. Biederman printk(KERN_WARNING "Warning: unable to open an initial console.\n"); 8952bd3a997SEric W. Biederman 8962bd3a997SEric W. Biederman (void) sys_dup(0); 8972bd3a997SEric W. Biederman (void) sys_dup(0); 8981da177e4SLinus Torvalds /* 8991da177e4SLinus Torvalds * check if there is an early userspace init. If yes, let it do all 9001da177e4SLinus Torvalds * the work 9011da177e4SLinus Torvalds */ 902ffdfc409SOlof Johansson 903ffdfc409SOlof Johansson if (!ramdisk_execute_command) 904ffdfc409SOlof Johansson ramdisk_execute_command = "/init"; 905ffdfc409SOlof Johansson 906ffdfc409SOlof Johansson if (sys_access((const char __user *) ramdisk_execute_command, 0) != 0) { 907ffdfc409SOlof Johansson ramdisk_execute_command = NULL; 9081da177e4SLinus Torvalds prepare_namespace(); 909ffdfc409SOlof Johansson } 9101da177e4SLinus Torvalds 9111da177e4SLinus Torvalds /* 9121da177e4SLinus Torvalds * Ok, we have completed the initial bootup, and 9131da177e4SLinus Torvalds * we're essentially up and running. Get rid of the 9141da177e4SLinus Torvalds * initmem segments and start the user-mode stuff.. 9151da177e4SLinus Torvalds */ 916bb813f4cSTejun Heo 917bb813f4cSTejun Heo /* rootfs is available now, try loading default modules */ 918bb813f4cSTejun Heo load_default_modules(); 9191da177e4SLinus Torvalds } 920