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> 171da177e4SLinus Torvalds #include <linux/string.h> 181da177e4SLinus Torvalds #include <linux/ctype.h> 191da177e4SLinus Torvalds #include <linux/delay.h> 201da177e4SLinus Torvalds #include <linux/utsname.h> 211da177e4SLinus Torvalds #include <linux/ioport.h> 221da177e4SLinus Torvalds #include <linux/init.h> 231da177e4SLinus Torvalds #include <linux/smp_lock.h> 241da177e4SLinus Torvalds #include <linux/initrd.h> 251da177e4SLinus Torvalds #include <linux/hdreg.h> 261da177e4SLinus Torvalds #include <linux/bootmem.h> 271da177e4SLinus Torvalds #include <linux/tty.h> 281da177e4SLinus Torvalds #include <linux/gfp.h> 291da177e4SLinus Torvalds #include <linux/percpu.h> 301da177e4SLinus Torvalds #include <linux/kmod.h> 311da177e4SLinus Torvalds #include <linux/kernel_stat.h> 32d7cd5611SRusty Russell #include <linux/start_kernel.h> 331da177e4SLinus Torvalds #include <linux/security.h> 341da177e4SLinus Torvalds #include <linux/workqueue.h> 351da177e4SLinus Torvalds #include <linux/profile.h> 361da177e4SLinus Torvalds #include <linux/rcupdate.h> 371da177e4SLinus Torvalds #include <linux/moduleparam.h> 381da177e4SLinus Torvalds #include <linux/kallsyms.h> 391da177e4SLinus Torvalds #include <linux/writeback.h> 401da177e4SLinus Torvalds #include <linux/cpu.h> 411da177e4SLinus Torvalds #include <linux/cpuset.h> 42ddbcc7e8SPaul Menage #include <linux/cgroup.h> 431da177e4SLinus Torvalds #include <linux/efi.h> 44906568c9SThomas Gleixner #include <linux/tick.h> 456168a702SAndrew Morton #include <linux/interrupt.h> 46c757249aSShailabh Nagar #include <linux/taskstats_kern.h> 47ca74e92bSShailabh Nagar #include <linux/delayacct.h> 481da177e4SLinus Torvalds #include <linux/unistd.h> 491da177e4SLinus Torvalds #include <linux/rmap.h> 501da177e4SLinus Torvalds #include <linux/mempolicy.h> 511da177e4SLinus Torvalds #include <linux/key.h> 524552d5dcSJan Beulich #include <linux/unwind.h> 53b6cd0b77SAdrian Bunk #include <linux/buffer_head.h> 549a11b49aSIngo Molnar #include <linux/debug_locks.h> 55fbb9ce95SIngo Molnar #include <linux/lockdep.h> 5684d73786SSukadev Bhattiprolu #include <linux/pid_namespace.h> 571f21782eSAdrian Bunk #include <linux/device.h> 5873c27992SEric W. Biederman #include <linux/kthread.h> 59e6fe6649SAdrian Bunk #include <linux/sched.h> 60a1c9eea9SAdrian Bunk #include <linux/signal.h> 611da177e4SLinus Torvalds 621da177e4SLinus Torvalds #include <asm/io.h> 631da177e4SLinus Torvalds #include <asm/bugs.h> 641da177e4SLinus Torvalds #include <asm/setup.h> 65a940199fSAndi Kleen #include <asm/sections.h> 6637b73c82SArjan van de Ven #include <asm/cacheflush.h> 671da177e4SLinus Torvalds 681da177e4SLinus Torvalds #ifdef CONFIG_X86_LOCAL_APIC 691da177e4SLinus Torvalds #include <asm/smp.h> 701da177e4SLinus Torvalds #endif 711da177e4SLinus Torvalds 721da177e4SLinus Torvalds /* 7369c99ac1SCoywolf Qi Hunt * This is one of the first .c files built. Error out early if we have compiler 7469c99ac1SCoywolf Qi Hunt * trouble. 751da177e4SLinus Torvalds */ 761da177e4SLinus Torvalds 77f1a60dbfSAndrew Morton #if __GNUC__ == 4 && __GNUC_MINOR__ == 1 && __GNUC_PATCHLEVEL__ == 0 78f1a60dbfSAndrew Morton #warning gcc-4.1.0 is known to miscompile the kernel. A different compiler version is recommended. 79f1a60dbfSAndrew Morton #endif 80f1a60dbfSAndrew Morton 81aae5f662SSam Ravnborg static int kernel_init(void *); 821da177e4SLinus Torvalds 831da177e4SLinus Torvalds extern void init_IRQ(void); 841da177e4SLinus Torvalds extern void fork_init(unsigned long); 851da177e4SLinus Torvalds extern void mca_init(void); 861da177e4SLinus Torvalds extern void sbus_init(void); 871da177e4SLinus Torvalds extern void pidhash_init(void); 881da177e4SLinus Torvalds extern void pidmap_init(void); 891da177e4SLinus Torvalds extern void prio_tree_init(void); 901da177e4SLinus Torvalds extern void radix_tree_init(void); 911da177e4SLinus Torvalds extern void free_initmem(void); 921da177e4SLinus Torvalds #ifdef CONFIG_ACPI 931da177e4SLinus Torvalds extern void acpi_early_init(void); 941da177e4SLinus Torvalds #else 951da177e4SLinus Torvalds static inline void acpi_early_init(void) { } 961da177e4SLinus Torvalds #endif 9737b73c82SArjan van de Ven #ifndef CONFIG_DEBUG_RODATA 9837b73c82SArjan van de Ven static inline void mark_rodata_ro(void) { } 9937b73c82SArjan van de Ven #endif 1001da177e4SLinus Torvalds 1011da177e4SLinus Torvalds #ifdef CONFIG_TC 1021da177e4SLinus Torvalds extern void tc_init(void); 1031da177e4SLinus Torvalds #endif 1041da177e4SLinus Torvalds 10571fc47a9SMarkus Gaugusch #ifdef CONFIG_ACPI_CUSTOM_DSDT_INITRD 10671fc47a9SMarkus Gaugusch extern int populate_rootfs(void); 10771fc47a9SMarkus Gaugusch #else 10871fc47a9SMarkus Gaugusch static inline void populate_rootfs(void) {} 10971fc47a9SMarkus Gaugusch #endif 11071fc47a9SMarkus Gaugusch 1111da177e4SLinus Torvalds enum system_states system_state; 1121da177e4SLinus Torvalds EXPORT_SYMBOL(system_state); 1131da177e4SLinus Torvalds 1141da177e4SLinus Torvalds /* 1151da177e4SLinus Torvalds * Boot command-line arguments 1161da177e4SLinus Torvalds */ 1171da177e4SLinus Torvalds #define MAX_INIT_ARGS CONFIG_INIT_ENV_ARG_LIMIT 1181da177e4SLinus Torvalds #define MAX_INIT_ENVS CONFIG_INIT_ENV_ARG_LIMIT 1191da177e4SLinus Torvalds 1201da177e4SLinus Torvalds extern void time_init(void); 1211da177e4SLinus Torvalds /* Default late time init is NULL. archs can override this later. */ 1221da177e4SLinus Torvalds void (*late_time_init)(void); 1231da177e4SLinus Torvalds extern void softirq_init(void); 1241da177e4SLinus Torvalds 12530d7e0d4SAlon Bar-Lev /* Untouched command line saved by arch-specific code. */ 12630d7e0d4SAlon Bar-Lev char __initdata boot_command_line[COMMAND_LINE_SIZE]; 12730d7e0d4SAlon Bar-Lev /* Untouched saved command line (eg. for /proc) */ 12830d7e0d4SAlon Bar-Lev char *saved_command_line; 12930d7e0d4SAlon Bar-Lev /* Command line for parameter parsing */ 13030d7e0d4SAlon Bar-Lev static char *static_command_line; 1311da177e4SLinus Torvalds 1321da177e4SLinus Torvalds static char *execute_command; 133ffdfc409SOlof Johansson static char *ramdisk_execute_command; 1341da177e4SLinus Torvalds 1358b3b2955SJan Beulich #ifdef CONFIG_SMP 1361da177e4SLinus Torvalds /* Setup configured maximum number of CPUs to activate */ 137ca74a6f8SAndi Kleen unsigned int __initdata setup_max_cpus = NR_CPUS; 1387e96287dSVivek Goyal 1397e96287dSVivek Goyal /* 1401da177e4SLinus Torvalds * Setup routine for controlling SMP activation 1411da177e4SLinus Torvalds * 1421da177e4SLinus Torvalds * Command-line option of "nosmp" or "maxcpus=0" will disable SMP 1431da177e4SLinus Torvalds * activation entirely (the MPS table probe still happens, though). 1441da177e4SLinus Torvalds * 1451da177e4SLinus Torvalds * Command-line option of "maxcpus=<NUM>", where <NUM> is an integer 1461da177e4SLinus Torvalds * greater than 0, limits the maximum number of CPUs activated in 1471da177e4SLinus Torvalds * SMP mode to <NUM>. 1481da177e4SLinus Torvalds */ 14961ec7567SLen Brown #ifndef CONFIG_X86_IO_APIC 15061ec7567SLen Brown static inline void disable_ioapic_setup(void) {}; 15161ec7567SLen Brown #endif 15261ec7567SLen Brown 1531da177e4SLinus Torvalds static int __init nosmp(char *str) 1541da177e4SLinus Torvalds { 155ca74a6f8SAndi Kleen setup_max_cpus = 0; 15661ec7567SLen Brown disable_ioapic_setup(); 1578b3b2955SJan Beulich return 0; 1581da177e4SLinus Torvalds } 1591da177e4SLinus Torvalds 1608b3b2955SJan Beulich early_param("nosmp", nosmp); 1611da177e4SLinus Torvalds 1621da177e4SLinus Torvalds static int __init maxcpus(char *str) 1631da177e4SLinus Torvalds { 164ca74a6f8SAndi Kleen get_option(&str, &setup_max_cpus); 165ca74a6f8SAndi Kleen if (setup_max_cpus == 0) 16661ec7567SLen Brown disable_ioapic_setup(); 16761ec7567SLen Brown 16861ec7567SLen Brown return 0; 1691da177e4SLinus Torvalds } 1701da177e4SLinus Torvalds 17181340977SHugh Dickins early_param("maxcpus", maxcpus); 1728b3b2955SJan Beulich #else 173ca74a6f8SAndi Kleen #define setup_max_cpus NR_CPUS 1748b3b2955SJan Beulich #endif 1758b3b2955SJan Beulich 1768b3b2955SJan Beulich /* 1778b3b2955SJan Beulich * If set, this is an indication to the drivers that reset the underlying 1788b3b2955SJan Beulich * device before going ahead with the initialization otherwise driver might 1798b3b2955SJan Beulich * rely on the BIOS and skip the reset operation. 1808b3b2955SJan Beulich * 1818b3b2955SJan Beulich * This is useful if kernel is booting in an unreliable environment. 1828b3b2955SJan Beulich * For ex. kdump situaiton where previous kernel has crashed, BIOS has been 1838b3b2955SJan Beulich * skipped and devices will be in unknown state. 1848b3b2955SJan Beulich */ 1858b3b2955SJan Beulich unsigned int reset_devices; 1868b3b2955SJan Beulich EXPORT_SYMBOL(reset_devices); 1871da177e4SLinus Torvalds 1887e96287dSVivek Goyal static int __init set_reset_devices(char *str) 1897e96287dSVivek Goyal { 1907e96287dSVivek Goyal reset_devices = 1; 1917e96287dSVivek Goyal return 1; 1927e96287dSVivek Goyal } 1937e96287dSVivek Goyal 1947e96287dSVivek Goyal __setup("reset_devices", set_reset_devices); 1957e96287dSVivek Goyal 1961da177e4SLinus Torvalds static char * argv_init[MAX_INIT_ARGS+2] = { "init", NULL, }; 1971da177e4SLinus Torvalds char * envp_init[MAX_INIT_ENVS+2] = { "HOME=/", "TERM=linux", NULL, }; 1981da177e4SLinus Torvalds static const char *panic_later, *panic_param; 1991da177e4SLinus Torvalds 2001da177e4SLinus Torvalds extern struct obs_kernel_param __setup_start[], __setup_end[]; 2011da177e4SLinus Torvalds 2021da177e4SLinus Torvalds static int __init obsolete_checksetup(char *line) 2031da177e4SLinus Torvalds { 2041da177e4SLinus Torvalds struct obs_kernel_param *p; 20533df0d19SRusty Russell int had_early_param = 0; 2061da177e4SLinus Torvalds 2071da177e4SLinus Torvalds p = __setup_start; 2081da177e4SLinus Torvalds do { 2091da177e4SLinus Torvalds int n = strlen(p->str); 2101da177e4SLinus Torvalds if (!strncmp(line, p->str, n)) { 2111da177e4SLinus Torvalds if (p->early) { 21233df0d19SRusty Russell /* Already done in parse_early_param? 21333df0d19SRusty Russell * (Needs exact match on param part). 21433df0d19SRusty Russell * Keep iterating, as we can have early 21533df0d19SRusty Russell * params and __setups of same names 8( */ 2161da177e4SLinus Torvalds if (line[n] == '\0' || line[n] == '=') 21733df0d19SRusty Russell had_early_param = 1; 2181da177e4SLinus Torvalds } else if (!p->setup_func) { 2191da177e4SLinus Torvalds printk(KERN_WARNING "Parameter %s is obsolete," 2201da177e4SLinus Torvalds " ignored\n", p->str); 2211da177e4SLinus Torvalds return 1; 2221da177e4SLinus Torvalds } else if (p->setup_func(line + n)) 2231da177e4SLinus Torvalds return 1; 2241da177e4SLinus Torvalds } 2251da177e4SLinus Torvalds p++; 2261da177e4SLinus Torvalds } while (p < __setup_end); 22733df0d19SRusty Russell 22833df0d19SRusty Russell return had_early_param; 2291da177e4SLinus Torvalds } 2301da177e4SLinus Torvalds 2311da177e4SLinus Torvalds /* 2321da177e4SLinus Torvalds * This should be approx 2 Bo*oMips to start (note initial shift), and will 2331da177e4SLinus Torvalds * still work even if initially too large, it will just take slightly longer 2341da177e4SLinus Torvalds */ 2351da177e4SLinus Torvalds unsigned long loops_per_jiffy = (1<<12); 2361da177e4SLinus Torvalds 2371da177e4SLinus Torvalds EXPORT_SYMBOL(loops_per_jiffy); 2381da177e4SLinus Torvalds 2391da177e4SLinus Torvalds static int __init debug_kernel(char *str) 2401da177e4SLinus Torvalds { 2411da177e4SLinus Torvalds console_loglevel = 10; 242*f6f21c81SYinghai Lu return 0; 2431da177e4SLinus Torvalds } 2441da177e4SLinus Torvalds 2451da177e4SLinus Torvalds static int __init quiet_kernel(char *str) 2461da177e4SLinus Torvalds { 2471da177e4SLinus Torvalds console_loglevel = 4; 248*f6f21c81SYinghai Lu return 0; 2491da177e4SLinus Torvalds } 2501da177e4SLinus Torvalds 251*f6f21c81SYinghai Lu early_param("debug", debug_kernel); 252*f6f21c81SYinghai Lu early_param("quiet", quiet_kernel); 2531da177e4SLinus Torvalds 2541da177e4SLinus Torvalds static int __init loglevel(char *str) 2551da177e4SLinus Torvalds { 2561da177e4SLinus Torvalds get_option(&str, &console_loglevel); 2571da177e4SLinus Torvalds return 1; 2581da177e4SLinus Torvalds } 2591da177e4SLinus Torvalds 260*f6f21c81SYinghai Lu early_param("loglevel", loglevel); 2611da177e4SLinus Torvalds 2621da177e4SLinus Torvalds /* 2631da177e4SLinus Torvalds * Unknown boot options get handed to init, unless they look like 2641da177e4SLinus Torvalds * failed parameters 2651da177e4SLinus Torvalds */ 2661da177e4SLinus Torvalds static int __init unknown_bootoption(char *param, char *val) 2671da177e4SLinus Torvalds { 2681da177e4SLinus Torvalds /* Change NUL term back to "=", to make "param" the whole string. */ 2691da177e4SLinus Torvalds if (val) { 2701da177e4SLinus Torvalds /* param=val or param="val"? */ 2711da177e4SLinus Torvalds if (val == param+strlen(param)+1) 2721da177e4SLinus Torvalds val[-1] = '='; 2731da177e4SLinus Torvalds else if (val == param+strlen(param)+2) { 2741da177e4SLinus Torvalds val[-2] = '='; 2751da177e4SLinus Torvalds memmove(val-1, val, strlen(val)+1); 2761da177e4SLinus Torvalds val--; 2771da177e4SLinus Torvalds } else 2781da177e4SLinus Torvalds BUG(); 2791da177e4SLinus Torvalds } 2801da177e4SLinus Torvalds 2811da177e4SLinus Torvalds /* Handle obsolete-style parameters */ 2821da177e4SLinus Torvalds if (obsolete_checksetup(param)) 2831da177e4SLinus Torvalds return 0; 2841da177e4SLinus Torvalds 2851da177e4SLinus Torvalds /* 286211fee8aSSimon Arlott * Preemptive maintenance for "why didn't my misspelled command 2871da177e4SLinus Torvalds * line work?" 2881da177e4SLinus Torvalds */ 2891da177e4SLinus Torvalds if (strchr(param, '.') && (!val || strchr(param, '.') < val)) { 2901da177e4SLinus Torvalds printk(KERN_ERR "Unknown boot option `%s': ignoring\n", param); 2911da177e4SLinus Torvalds return 0; 2921da177e4SLinus Torvalds } 2931da177e4SLinus Torvalds 2941da177e4SLinus Torvalds if (panic_later) 2951da177e4SLinus Torvalds return 0; 2961da177e4SLinus Torvalds 2971da177e4SLinus Torvalds if (val) { 2981da177e4SLinus Torvalds /* Environment option */ 2991da177e4SLinus Torvalds unsigned int i; 3001da177e4SLinus Torvalds for (i = 0; envp_init[i]; i++) { 3011da177e4SLinus Torvalds if (i == MAX_INIT_ENVS) { 3021da177e4SLinus Torvalds panic_later = "Too many boot env vars at `%s'"; 3031da177e4SLinus Torvalds panic_param = param; 3041da177e4SLinus Torvalds } 3051da177e4SLinus Torvalds if (!strncmp(param, envp_init[i], val - param)) 3061da177e4SLinus Torvalds break; 3071da177e4SLinus Torvalds } 3081da177e4SLinus Torvalds envp_init[i] = param; 3091da177e4SLinus Torvalds } else { 3101da177e4SLinus Torvalds /* Command line option */ 3111da177e4SLinus Torvalds unsigned int i; 3121da177e4SLinus Torvalds for (i = 0; argv_init[i]; i++) { 3131da177e4SLinus Torvalds if (i == MAX_INIT_ARGS) { 3141da177e4SLinus Torvalds panic_later = "Too many boot init vars at `%s'"; 3151da177e4SLinus Torvalds panic_param = param; 3161da177e4SLinus Torvalds } 3171da177e4SLinus Torvalds } 3181da177e4SLinus Torvalds argv_init[i] = param; 3191da177e4SLinus Torvalds } 3201da177e4SLinus Torvalds return 0; 3211da177e4SLinus Torvalds } 3221da177e4SLinus Torvalds 32312d6f21eSIngo Molnar #ifdef CONFIG_DEBUG_PAGEALLOC 32412d6f21eSIngo Molnar int __read_mostly debug_pagealloc_enabled = 0; 32512d6f21eSIngo Molnar #endif 32612d6f21eSIngo Molnar 3271da177e4SLinus Torvalds static int __init init_setup(char *str) 3281da177e4SLinus Torvalds { 3291da177e4SLinus Torvalds unsigned int i; 3301da177e4SLinus Torvalds 3311da177e4SLinus Torvalds execute_command = str; 3321da177e4SLinus Torvalds /* 3331da177e4SLinus Torvalds * In case LILO is going to boot us with default command line, 3341da177e4SLinus Torvalds * it prepends "auto" before the whole cmdline which makes 3351da177e4SLinus Torvalds * the shell think it should execute a script with such name. 3361da177e4SLinus Torvalds * So we ignore all arguments entered _before_ init=... [MJ] 3371da177e4SLinus Torvalds */ 3381da177e4SLinus Torvalds for (i = 1; i < MAX_INIT_ARGS; i++) 3391da177e4SLinus Torvalds argv_init[i] = NULL; 3401da177e4SLinus Torvalds return 1; 3411da177e4SLinus Torvalds } 3421da177e4SLinus Torvalds __setup("init=", init_setup); 3431da177e4SLinus Torvalds 344ffdfc409SOlof Johansson static int __init rdinit_setup(char *str) 345ffdfc409SOlof Johansson { 346ffdfc409SOlof Johansson unsigned int i; 347ffdfc409SOlof Johansson 348ffdfc409SOlof Johansson ramdisk_execute_command = str; 349ffdfc409SOlof Johansson /* See "auto" comment in init_setup */ 350ffdfc409SOlof Johansson for (i = 1; i < MAX_INIT_ARGS; i++) 351ffdfc409SOlof Johansson argv_init[i] = NULL; 352ffdfc409SOlof Johansson return 1; 353ffdfc409SOlof Johansson } 354ffdfc409SOlof Johansson __setup("rdinit=", rdinit_setup); 355ffdfc409SOlof Johansson 3561da177e4SLinus Torvalds #ifndef CONFIG_SMP 3571da177e4SLinus Torvalds 3581da177e4SLinus Torvalds #ifdef CONFIG_X86_LOCAL_APIC 3591da177e4SLinus Torvalds static void __init smp_init(void) 3601da177e4SLinus Torvalds { 3611da177e4SLinus Torvalds APIC_init_uniprocessor(); 3621da177e4SLinus Torvalds } 3631da177e4SLinus Torvalds #else 3641da177e4SLinus Torvalds #define smp_init() do { } while (0) 3651da177e4SLinus Torvalds #endif 3661da177e4SLinus Torvalds 3671da177e4SLinus Torvalds static inline void setup_per_cpu_areas(void) { } 3681da177e4SLinus Torvalds static inline void smp_prepare_cpus(unsigned int maxcpus) { } 3691da177e4SLinus Torvalds 3701da177e4SLinus Torvalds #else 3711da177e4SLinus Torvalds 372dd5af90aSMike Travis #ifndef CONFIG_HAVE_SETUP_PER_CPU_AREA 373b73b459fSEric Dumazet unsigned long __per_cpu_offset[NR_CPUS] __read_mostly; 3741da177e4SLinus Torvalds 3751da177e4SLinus Torvalds EXPORT_SYMBOL(__per_cpu_offset); 3761da177e4SLinus Torvalds 3771da177e4SLinus Torvalds static void __init setup_per_cpu_areas(void) 3781da177e4SLinus Torvalds { 3791da177e4SLinus Torvalds unsigned long size, i; 3801da177e4SLinus Torvalds char *ptr; 38163872f87SEric Dumazet unsigned long nr_possible_cpus = num_possible_cpus(); 3821da177e4SLinus Torvalds 3831da177e4SLinus Torvalds /* Copy section for each CPU (we discard the original) */ 384b6e3590fSJeremy Fitzhardinge size = ALIGN(PERCPU_ENOUGH_ROOM, PAGE_SIZE); 385b6e3590fSJeremy Fitzhardinge ptr = alloc_bootmem_pages(size * nr_possible_cpus); 3861da177e4SLinus Torvalds 3870a945022SKAMEZAWA Hiroyuki for_each_possible_cpu(i) { 3881da177e4SLinus Torvalds __per_cpu_offset[i] = ptr - __per_cpu_start; 3891da177e4SLinus Torvalds memcpy(ptr, __per_cpu_start, __per_cpu_end - __per_cpu_start); 39063872f87SEric Dumazet ptr += size; 3911da177e4SLinus Torvalds } 3921da177e4SLinus Torvalds } 393dd5af90aSMike Travis #endif /* CONFIG_HAVE_SETUP_PER_CPU_AREA */ 3941da177e4SLinus Torvalds 3951da177e4SLinus Torvalds /* Called by boot processor to activate the rest. */ 3961da177e4SLinus Torvalds static void __init smp_init(void) 3971da177e4SLinus Torvalds { 39853b8a315SChristoph Lameter unsigned int cpu; 3991da177e4SLinus Torvalds 4001da177e4SLinus Torvalds /* FIXME: This should be done in userspace --RR */ 40153b8a315SChristoph Lameter for_each_present_cpu(cpu) { 402ca74a6f8SAndi Kleen if (num_online_cpus() >= setup_max_cpus) 4031da177e4SLinus Torvalds break; 40453b8a315SChristoph Lameter if (!cpu_online(cpu)) 40553b8a315SChristoph Lameter cpu_up(cpu); 4061da177e4SLinus Torvalds } 4071da177e4SLinus Torvalds 4081da177e4SLinus Torvalds /* Any cleanup work */ 4091da177e4SLinus Torvalds printk(KERN_INFO "Brought up %ld CPUs\n", (long)num_online_cpus()); 410ca74a6f8SAndi Kleen smp_cpus_done(setup_max_cpus); 4111da177e4SLinus Torvalds } 4121da177e4SLinus Torvalds 4131da177e4SLinus Torvalds #endif 4141da177e4SLinus Torvalds 4151da177e4SLinus Torvalds /* 41630d7e0d4SAlon Bar-Lev * We need to store the untouched command line for future reference. 41730d7e0d4SAlon Bar-Lev * We also need to store the touched command line since the parameter 41830d7e0d4SAlon Bar-Lev * parsing is performed in place, and we should allow a component to 41930d7e0d4SAlon Bar-Lev * store reference of name/value for future reference. 42030d7e0d4SAlon Bar-Lev */ 42130d7e0d4SAlon Bar-Lev static void __init setup_command_line(char *command_line) 42230d7e0d4SAlon Bar-Lev { 42330d7e0d4SAlon Bar-Lev saved_command_line = alloc_bootmem(strlen (boot_command_line)+1); 42430d7e0d4SAlon Bar-Lev static_command_line = alloc_bootmem(strlen (command_line)+1); 42530d7e0d4SAlon Bar-Lev strcpy (saved_command_line, boot_command_line); 42630d7e0d4SAlon Bar-Lev strcpy (static_command_line, command_line); 42730d7e0d4SAlon Bar-Lev } 42830d7e0d4SAlon Bar-Lev 42930d7e0d4SAlon Bar-Lev /* 4301da177e4SLinus Torvalds * We need to finalize in a non-__init function or else race conditions 4311da177e4SLinus Torvalds * between the root thread and the init thread may cause start_kernel to 4321da177e4SLinus Torvalds * be reaped by free_initmem before the root thread has proceeded to 4331da177e4SLinus Torvalds * cpu_idle. 4341da177e4SLinus Torvalds * 4351da177e4SLinus Torvalds * gcc-3.4 accidentally inlines this function, so use noinline. 4361da177e4SLinus Torvalds */ 4371da177e4SLinus Torvalds 43892080309SSam Ravnborg static void noinline __init_refok rest_init(void) 4391da177e4SLinus Torvalds __releases(kernel_lock) 4401da177e4SLinus Torvalds { 44173c27992SEric W. Biederman int pid; 44273c27992SEric W. Biederman 443aae5f662SSam Ravnborg kernel_thread(kernel_init, NULL, CLONE_FS | CLONE_SIGHAND); 4441da177e4SLinus Torvalds numa_default_policy(); 44573c27992SEric W. Biederman pid = kernel_thread(kthreadd, NULL, CLONE_FS | CLONE_FILES); 44673c27992SEric W. Biederman kthreadd_task = find_task_by_pid(pid); 4471da177e4SLinus Torvalds unlock_kernel(); 448f340c0d1SIngo Molnar 449f340c0d1SIngo Molnar /* 450f340c0d1SIngo Molnar * The boot idle thread must execute schedule() 4511df21055SIngo Molnar * at least once to get things moving: 452f340c0d1SIngo Molnar */ 4531df21055SIngo Molnar init_idle_bootup_task(current); 4545bfb5d69SNick Piggin preempt_enable_no_resched(); 455f340c0d1SIngo Molnar schedule(); 4565bfb5d69SNick Piggin preempt_disable(); 457f340c0d1SIngo Molnar 4585bfb5d69SNick Piggin /* Call into cpu_idle with preempt disabled */ 4591da177e4SLinus Torvalds cpu_idle(); 4601da177e4SLinus Torvalds } 4611da177e4SLinus Torvalds 4621da177e4SLinus Torvalds /* Check for early params. */ 4631da177e4SLinus Torvalds static int __init do_early_param(char *param, char *val) 4641da177e4SLinus Torvalds { 4651da177e4SLinus Torvalds struct obs_kernel_param *p; 4661da177e4SLinus Torvalds 4671da177e4SLinus Torvalds for (p = __setup_start; p < __setup_end; p++) { 46818a8bd94SYinghai Lu if ((p->early && strcmp(param, p->str) == 0) || 46918a8bd94SYinghai Lu (strcmp(param, "console") == 0 && 47018a8bd94SYinghai Lu strcmp(p->str, "earlycon") == 0) 47118a8bd94SYinghai Lu ) { 4721da177e4SLinus Torvalds if (p->setup_func(val) != 0) 4731da177e4SLinus Torvalds printk(KERN_WARNING 4741da177e4SLinus Torvalds "Malformed early option '%s'\n", param); 4751da177e4SLinus Torvalds } 4761da177e4SLinus Torvalds } 4771da177e4SLinus Torvalds /* We accept everything at this stage. */ 4781da177e4SLinus Torvalds return 0; 4791da177e4SLinus Torvalds } 4801da177e4SLinus Torvalds 4811da177e4SLinus Torvalds /* Arch code calls this early on, or if not, just before other parsing. */ 4821da177e4SLinus Torvalds void __init parse_early_param(void) 4831da177e4SLinus Torvalds { 4841da177e4SLinus Torvalds static __initdata int done = 0; 4851da177e4SLinus Torvalds static __initdata char tmp_cmdline[COMMAND_LINE_SIZE]; 4861da177e4SLinus Torvalds 4871da177e4SLinus Torvalds if (done) 4881da177e4SLinus Torvalds return; 4891da177e4SLinus Torvalds 4901da177e4SLinus Torvalds /* All fall through to do_early_param. */ 49130d7e0d4SAlon Bar-Lev strlcpy(tmp_cmdline, boot_command_line, COMMAND_LINE_SIZE); 4921da177e4SLinus Torvalds parse_args("early options", tmp_cmdline, NULL, 0, do_early_param); 4931da177e4SLinus Torvalds done = 1; 4941da177e4SLinus Torvalds } 4951da177e4SLinus Torvalds 4961da177e4SLinus Torvalds /* 4971da177e4SLinus Torvalds * Activate the first processor. 4981da177e4SLinus Torvalds */ 4991da177e4SLinus Torvalds 50044fd2299SStas Sergeev static void __init boot_cpu_init(void) 50144fd2299SStas Sergeev { 50244fd2299SStas Sergeev int cpu = smp_processor_id(); 50344fd2299SStas Sergeev /* Mark the boot cpu "present", "online" etc for SMP and UP case */ 50444fd2299SStas Sergeev cpu_set(cpu, cpu_online_map); 50544fd2299SStas Sergeev cpu_set(cpu, cpu_present_map); 50644fd2299SStas Sergeev cpu_set(cpu, cpu_possible_map); 50744fd2299SStas Sergeev } 50844fd2299SStas Sergeev 509033ab7f8SAndrew Morton void __init __attribute__((weak)) smp_setup_processor_id(void) 510033ab7f8SAndrew Morton { 511033ab7f8SAndrew Morton } 512033ab7f8SAndrew Morton 5131da177e4SLinus Torvalds asmlinkage void __init start_kernel(void) 5141da177e4SLinus Torvalds { 5151da177e4SLinus Torvalds char * command_line; 5161da177e4SLinus Torvalds extern struct kernel_param __start___param[], __stop___param[]; 517033ab7f8SAndrew Morton 518033ab7f8SAndrew Morton smp_setup_processor_id(); 519033ab7f8SAndrew Morton 5201da177e4SLinus Torvalds /* 521fbb9ce95SIngo Molnar * Need to run as early as possible, to initialize the 522fbb9ce95SIngo Molnar * lockdep hash: 523fbb9ce95SIngo Molnar */ 524c9538ed4SAndi Kleen unwind_init(); 525fbb9ce95SIngo Molnar lockdep_init(); 526ddbcc7e8SPaul Menage cgroup_init_early(); 527fbb9ce95SIngo Molnar 528fbb9ce95SIngo Molnar local_irq_disable(); 529fbb9ce95SIngo Molnar early_boot_irqs_off(); 530243c7621SIngo Molnar early_init_irq_lock_class(); 531fbb9ce95SIngo Molnar 532fbb9ce95SIngo Molnar /* 5331da177e4SLinus Torvalds * Interrupts are still disabled. Do necessary setups, then 5341da177e4SLinus Torvalds * enable them 5351da177e4SLinus Torvalds */ 5361da177e4SLinus Torvalds lock_kernel(); 537906568c9SThomas Gleixner tick_init(); 53844fd2299SStas Sergeev boot_cpu_init(); 5391da177e4SLinus Torvalds page_address_init(); 5401da177e4SLinus Torvalds printk(KERN_NOTICE); 5418993780aSLinus Torvalds printk(linux_banner); 5421da177e4SLinus Torvalds setup_arch(&command_line); 54330d7e0d4SAlon Bar-Lev setup_command_line(command_line); 544690a973fSJan Beulich unwind_setup(); 5451da177e4SLinus Torvalds setup_per_cpu_areas(); 54644fd2299SStas Sergeev smp_prepare_boot_cpu(); /* arch-specific boot-cpu hooks */ 5471da177e4SLinus Torvalds 5481da177e4SLinus Torvalds /* 5491da177e4SLinus Torvalds * Set up the scheduler prior starting any interrupts (such as the 5501da177e4SLinus Torvalds * timer interrupt). Full topology setup happens at smp_init() 5511da177e4SLinus Torvalds * time - but meanwhile we still have a functioning scheduler. 5521da177e4SLinus Torvalds */ 5531da177e4SLinus Torvalds sched_init(); 5541da177e4SLinus Torvalds /* 5551da177e4SLinus Torvalds * Disable preemption - early bootup scheduling is extremely 5561da177e4SLinus Torvalds * fragile until we cpu_idle() for the first time. 5571da177e4SLinus Torvalds */ 5581da177e4SLinus Torvalds preempt_disable(); 5591da177e4SLinus Torvalds build_all_zonelists(); 5601da177e4SLinus Torvalds page_alloc_init(); 56112d6f21eSIngo Molnar enable_debug_pagealloc(); 56230d7e0d4SAlon Bar-Lev printk(KERN_NOTICE "Kernel command line: %s\n", boot_command_line); 5631da177e4SLinus Torvalds parse_early_param(); 56430d7e0d4SAlon Bar-Lev parse_args("Booting kernel", static_command_line, __start___param, 5651da177e4SLinus Torvalds __stop___param - __start___param, 5661da177e4SLinus Torvalds &unknown_bootoption); 567c4a68306SArd van Breemen if (!irqs_disabled()) { 568c4a68306SArd van Breemen printk(KERN_WARNING "start_kernel(): bug: interrupts were " 569c4a68306SArd van Breemen "enabled *very* early, fixing it\n"); 570c4a68306SArd van Breemen local_irq_disable(); 571c4a68306SArd van Breemen } 5721da177e4SLinus Torvalds sort_main_extable(); 5731da177e4SLinus Torvalds trap_init(); 5741da177e4SLinus Torvalds rcu_init(); 5751da177e4SLinus Torvalds init_IRQ(); 5761da177e4SLinus Torvalds pidhash_init(); 5771da177e4SLinus Torvalds init_timers(); 578c0a31329SThomas Gleixner hrtimers_init(); 5791da177e4SLinus Torvalds softirq_init(); 580ad596171Sjohn stultz timekeeping_init(); 58188fecaa2Sjohn stultz time_init(); 58293e02814SHeiko Carstens profile_init(); 58393e02814SHeiko Carstens if (!irqs_disabled()) 58493e02814SHeiko Carstens printk("start_kernel(): bug: interrupts were enabled early\n"); 585fbb9ce95SIngo Molnar early_boot_irqs_on(); 58693e02814SHeiko Carstens local_irq_enable(); 5871da177e4SLinus Torvalds 5881da177e4SLinus Torvalds /* 5891da177e4SLinus Torvalds * HACK ALERT! This is early. We're enabling the console before 5901da177e4SLinus Torvalds * we've done PCI setups etc, and console_init() must be aware of 5911da177e4SLinus Torvalds * this. But we do want output early, in case something goes wrong. 5921da177e4SLinus Torvalds */ 5931da177e4SLinus Torvalds console_init(); 5941da177e4SLinus Torvalds if (panic_later) 5951da177e4SLinus Torvalds panic(panic_later, panic_param); 596fbb9ce95SIngo Molnar 597fbb9ce95SIngo Molnar lockdep_info(); 598fbb9ce95SIngo Molnar 5999a11b49aSIngo Molnar /* 6009a11b49aSIngo Molnar * Need to run this when irqs are enabled, because it wants 6019a11b49aSIngo Molnar * to self-test [hard/soft]-irqs on/off lock inversion bugs 6029a11b49aSIngo Molnar * too: 6039a11b49aSIngo Molnar */ 6049a11b49aSIngo Molnar locking_selftest(); 6059a11b49aSIngo Molnar 6061da177e4SLinus Torvalds #ifdef CONFIG_BLK_DEV_INITRD 6071da177e4SLinus Torvalds if (initrd_start && !initrd_below_start_ok && 6081da177e4SLinus Torvalds initrd_start < min_low_pfn << PAGE_SHIFT) { 6091da177e4SLinus Torvalds printk(KERN_CRIT "initrd overwritten (0x%08lx < 0x%08lx) - " 6101da177e4SLinus Torvalds "disabling it.\n",initrd_start,min_low_pfn << PAGE_SHIFT); 6111da177e4SLinus Torvalds initrd_start = 0; 6121da177e4SLinus Torvalds } 6131da177e4SLinus Torvalds #endif 6141da177e4SLinus Torvalds vfs_caches_init_early(); 615c417f024SPaul Jackson cpuset_init_early(); 6161da177e4SLinus Torvalds mem_init(); 617d221938cSGautham R Shenoy cpu_hotplug_init(); 6181da177e4SLinus Torvalds kmem_cache_init(); 619e7c8d5c9SChristoph Lameter setup_per_cpu_pageset(); 6201da177e4SLinus Torvalds numa_policy_init(); 6211da177e4SLinus Torvalds if (late_time_init) 6221da177e4SLinus Torvalds late_time_init(); 6231da177e4SLinus Torvalds calibrate_delay(); 6241da177e4SLinus Torvalds pidmap_init(); 6251da177e4SLinus Torvalds pgtable_cache_init(); 6261da177e4SLinus Torvalds prio_tree_init(); 6271da177e4SLinus Torvalds anon_vma_init(); 6281da177e4SLinus Torvalds #ifdef CONFIG_X86 6291da177e4SLinus Torvalds if (efi_enabled) 6301da177e4SLinus Torvalds efi_enter_virtual_mode(); 6311da177e4SLinus Torvalds #endif 6321da177e4SLinus Torvalds fork_init(num_physpages); 6331da177e4SLinus Torvalds proc_caches_init(); 6341da177e4SLinus Torvalds buffer_init(); 6351da177e4SLinus Torvalds unnamed_dev_init(); 6361da177e4SLinus Torvalds key_init(); 6371da177e4SLinus Torvalds security_init(); 6381da177e4SLinus Torvalds vfs_caches_init(num_physpages); 6391da177e4SLinus Torvalds radix_tree_init(); 6401da177e4SLinus Torvalds signals_init(); 6411da177e4SLinus Torvalds /* rootfs populating might need page-writeback */ 6421da177e4SLinus Torvalds page_writeback_init(); 6431da177e4SLinus Torvalds #ifdef CONFIG_PROC_FS 6441da177e4SLinus Torvalds proc_root_init(); 6451da177e4SLinus Torvalds #endif 646ddbcc7e8SPaul Menage cgroup_init(); 6471da177e4SLinus Torvalds cpuset_init(); 648c757249aSShailabh Nagar taskstats_init_early(); 649ca74e92bSShailabh Nagar delayacct_init(); 6501da177e4SLinus Torvalds 6511da177e4SLinus Torvalds check_bugs(); 6521da177e4SLinus Torvalds 65371fc47a9SMarkus Gaugusch populate_rootfs(); /* For DSDT override from initramfs */ 6541da177e4SLinus Torvalds acpi_early_init(); /* before LAPIC and SMP init */ 6551da177e4SLinus Torvalds 6561da177e4SLinus Torvalds /* Do the rest non-__init'ed, we're now alive */ 6571da177e4SLinus Torvalds rest_init(); 6581da177e4SLinus Torvalds } 6591da177e4SLinus Torvalds 6601da177e4SLinus Torvalds static int __initdata initcall_debug; 6611da177e4SLinus Torvalds 6621da177e4SLinus Torvalds static int __init initcall_debug_setup(char *str) 6631da177e4SLinus Torvalds { 6641da177e4SLinus Torvalds initcall_debug = 1; 6651da177e4SLinus Torvalds return 1; 6661da177e4SLinus Torvalds } 6671da177e4SLinus Torvalds __setup("initcall_debug", initcall_debug_setup); 6681da177e4SLinus Torvalds 6691da177e4SLinus Torvalds extern initcall_t __initcall_start[], __initcall_end[]; 6701da177e4SLinus Torvalds 6711da177e4SLinus Torvalds static void __init do_initcalls(void) 6721da177e4SLinus Torvalds { 6731da177e4SLinus Torvalds initcall_t *call; 6741da177e4SLinus Torvalds int count = preempt_count(); 6751da177e4SLinus Torvalds 6761da177e4SLinus Torvalds for (call = __initcall_start; call < __initcall_end; call++) { 6778f0c45cdSIngo Molnar ktime_t t0, t1, delta; 678c1cda48aSAndrew Morton char *msg = NULL; 679c1cda48aSAndrew Morton char msgbuf[40]; 680c1cda48aSAndrew Morton int result; 6811da177e4SLinus Torvalds 6821da177e4SLinus Torvalds if (initcall_debug) { 683c1cda48aSAndrew Morton printk("Calling initcall 0x%p", *call); 684c1cda48aSAndrew Morton print_fn_descriptor_symbol(": %s()", 685c1cda48aSAndrew Morton (unsigned long) *call); 6861da177e4SLinus Torvalds printk("\n"); 6878f0c45cdSIngo Molnar t0 = ktime_get(); 6881da177e4SLinus Torvalds } 6891da177e4SLinus Torvalds 690c1cda48aSAndrew Morton result = (*call)(); 6911da177e4SLinus Torvalds 6928f0c45cdSIngo Molnar if (initcall_debug) { 6938f0c45cdSIngo Molnar t1 = ktime_get(); 6948f0c45cdSIngo Molnar delta = ktime_sub(t1, t0); 6958f0c45cdSIngo Molnar 6968f0c45cdSIngo Molnar printk("initcall 0x%p", *call); 6978f0c45cdSIngo Molnar print_fn_descriptor_symbol(": %s()", 6988f0c45cdSIngo Molnar (unsigned long) *call); 6998f0c45cdSIngo Molnar printk(" returned %d.\n", result); 7008f0c45cdSIngo Molnar 7018f0c45cdSIngo Molnar printk("initcall 0x%p ran for %Ld msecs: ", 7028f0c45cdSIngo Molnar *call, (unsigned long long)delta.tv64 >> 20); 7038f0c45cdSIngo Molnar print_fn_descriptor_symbol("%s()\n", 7048f0c45cdSIngo Molnar (unsigned long) *call); 7058f0c45cdSIngo Molnar } 7068f0c45cdSIngo Molnar 707f3a19cb4SAndrew Morton if (result && result != -ENODEV && initcall_debug) { 708c1cda48aSAndrew Morton sprintf(msgbuf, "error code %d", result); 709c1cda48aSAndrew Morton msg = msgbuf; 710c1cda48aSAndrew Morton } 7111da177e4SLinus Torvalds if (preempt_count() != count) { 7121da177e4SLinus Torvalds msg = "preemption imbalance"; 7131da177e4SLinus Torvalds preempt_count() = count; 7141da177e4SLinus Torvalds } 7151da177e4SLinus Torvalds if (irqs_disabled()) { 7161da177e4SLinus Torvalds msg = "disabled interrupts"; 7171da177e4SLinus Torvalds local_irq_enable(); 7181da177e4SLinus Torvalds } 7191da177e4SLinus Torvalds if (msg) { 720c1cda48aSAndrew Morton printk(KERN_WARNING "initcall at 0x%p", *call); 721c1cda48aSAndrew Morton print_fn_descriptor_symbol(": %s()", 722c1cda48aSAndrew Morton (unsigned long) *call); 723c1cda48aSAndrew Morton printk(": returned with %s\n", msg); 7241da177e4SLinus Torvalds } 7251da177e4SLinus Torvalds } 7261da177e4SLinus Torvalds 7271da177e4SLinus Torvalds /* Make sure there is no pending stuff from the initcall sequence */ 7281da177e4SLinus Torvalds flush_scheduled_work(); 7291da177e4SLinus Torvalds } 7301da177e4SLinus Torvalds 7311da177e4SLinus Torvalds /* 7321da177e4SLinus Torvalds * Ok, the machine is now initialized. None of the devices 7331da177e4SLinus Torvalds * have been touched yet, but the CPU subsystem is up and 7341da177e4SLinus Torvalds * running, and memory and process management works. 7351da177e4SLinus Torvalds * 7361da177e4SLinus Torvalds * Now we can finally start doing some real work.. 7371da177e4SLinus Torvalds */ 7381da177e4SLinus Torvalds static void __init do_basic_setup(void) 7391da177e4SLinus Torvalds { 7401da177e4SLinus Torvalds /* drivers will send hotplug events */ 7411da177e4SLinus Torvalds init_workqueues(); 7421da177e4SLinus Torvalds usermodehelper_init(); 7431da177e4SLinus Torvalds driver_init(); 744b04c3afbSEric W. Biederman init_irq_proc(); 7451da177e4SLinus Torvalds do_initcalls(); 7461da177e4SLinus Torvalds } 7471da177e4SLinus Torvalds 74897842216SDave Jones static int __initdata nosoftlockup; 74997842216SDave Jones 75097842216SDave Jones static int __init nosoftlockup_setup(char *str) 75197842216SDave Jones { 75297842216SDave Jones nosoftlockup = 1; 75397842216SDave Jones return 1; 75497842216SDave Jones } 75597842216SDave Jones __setup("nosoftlockup", nosoftlockup_setup); 75697842216SDave Jones 75788d20328SVivek Goyal static void __init do_pre_smp_initcalls(void) 7581da177e4SLinus Torvalds { 7591da177e4SLinus Torvalds extern int spawn_ksoftirqd(void); 7601da177e4SLinus Torvalds 7611da177e4SLinus Torvalds migration_init(); 7621da177e4SLinus Torvalds spawn_ksoftirqd(); 76397842216SDave Jones if (!nosoftlockup) 7648446f1d3SIngo Molnar spawn_softlockup_task(); 7651da177e4SLinus Torvalds } 7661da177e4SLinus Torvalds 7671da177e4SLinus Torvalds static void run_init_process(char *init_filename) 7681da177e4SLinus Torvalds { 7691da177e4SLinus Torvalds argv_init[0] = init_filename; 77067608567SArnd Bergmann kernel_execve(init_filename, argv_init, envp_init); 7711da177e4SLinus Torvalds } 7721da177e4SLinus Torvalds 773ee5bfa64SVivek Goyal /* This is a non __init function. Force it to be noinline otherwise gcc 774ee5bfa64SVivek Goyal * makes it inline to init() and it becomes part of init.text section 775ee5bfa64SVivek Goyal */ 776ee5bfa64SVivek Goyal static int noinline init_post(void) 777ee5bfa64SVivek Goyal { 778ee5bfa64SVivek Goyal free_initmem(); 779ee5bfa64SVivek Goyal unlock_kernel(); 780ee5bfa64SVivek Goyal mark_rodata_ro(); 781ee5bfa64SVivek Goyal system_state = SYSTEM_RUNNING; 782ee5bfa64SVivek Goyal numa_default_policy(); 783ee5bfa64SVivek Goyal 784ee5bfa64SVivek Goyal if (sys_open((const char __user *) "/dev/console", O_RDWR, 0) < 0) 785ee5bfa64SVivek Goyal printk(KERN_WARNING "Warning: unable to open an initial console.\n"); 786ee5bfa64SVivek Goyal 787ee5bfa64SVivek Goyal (void) sys_dup(0); 788ee5bfa64SVivek Goyal (void) sys_dup(0); 789ee5bfa64SVivek Goyal 790ee5bfa64SVivek Goyal if (ramdisk_execute_command) { 791ee5bfa64SVivek Goyal run_init_process(ramdisk_execute_command); 792ee5bfa64SVivek Goyal printk(KERN_WARNING "Failed to execute %s\n", 793ee5bfa64SVivek Goyal ramdisk_execute_command); 794ee5bfa64SVivek Goyal } 795ee5bfa64SVivek Goyal 796ee5bfa64SVivek Goyal /* 797ee5bfa64SVivek Goyal * We try each of these until one succeeds. 798ee5bfa64SVivek Goyal * 799ee5bfa64SVivek Goyal * The Bourne shell can be used instead of init if we are 800ee5bfa64SVivek Goyal * trying to recover a really broken machine. 801ee5bfa64SVivek Goyal */ 802ee5bfa64SVivek Goyal if (execute_command) { 803ee5bfa64SVivek Goyal run_init_process(execute_command); 804ee5bfa64SVivek Goyal printk(KERN_WARNING "Failed to execute %s. Attempting " 805ee5bfa64SVivek Goyal "defaults...\n", execute_command); 806ee5bfa64SVivek Goyal } 807ee5bfa64SVivek Goyal run_init_process("/sbin/init"); 808ee5bfa64SVivek Goyal run_init_process("/etc/init"); 809ee5bfa64SVivek Goyal run_init_process("/bin/init"); 810ee5bfa64SVivek Goyal run_init_process("/bin/sh"); 811ee5bfa64SVivek Goyal 812ee5bfa64SVivek Goyal panic("No init found. Try passing init= option to kernel."); 813ee5bfa64SVivek Goyal } 814ee5bfa64SVivek Goyal 815aae5f662SSam Ravnborg static int __init kernel_init(void * unused) 8161da177e4SLinus Torvalds { 8171da177e4SLinus Torvalds lock_kernel(); 8181da177e4SLinus Torvalds /* 8191da177e4SLinus Torvalds * init can run on any cpu. 8201da177e4SLinus Torvalds */ 8211da177e4SLinus Torvalds set_cpus_allowed(current, CPU_MASK_ALL); 8221da177e4SLinus Torvalds /* 8231da177e4SLinus Torvalds * Tell the world that we're going to be the grim 8241da177e4SLinus Torvalds * reaper of innocent orphaned children. 8251da177e4SLinus Torvalds * 8261da177e4SLinus Torvalds * We don't want people to have to make incorrect 8271da177e4SLinus Torvalds * assumptions about where in the task array this 8281da177e4SLinus Torvalds * can be found. 8291da177e4SLinus Torvalds */ 83084d73786SSukadev Bhattiprolu init_pid_ns.child_reaper = current; 8311da177e4SLinus Torvalds 8329ec52099SCedric Le Goater cad_pid = task_pid(current); 8339ec52099SCedric Le Goater 834ca74a6f8SAndi Kleen smp_prepare_cpus(setup_max_cpus); 8351da177e4SLinus Torvalds 8361da177e4SLinus Torvalds do_pre_smp_initcalls(); 8371da177e4SLinus Torvalds 8381da177e4SLinus Torvalds smp_init(); 8391da177e4SLinus Torvalds sched_init_smp(); 8401da177e4SLinus Torvalds 8411da177e4SLinus Torvalds cpuset_init_smp(); 8421da177e4SLinus Torvalds 8431da177e4SLinus Torvalds do_basic_setup(); 8441da177e4SLinus Torvalds 8451da177e4SLinus Torvalds /* 8461da177e4SLinus Torvalds * check if there is an early userspace init. If yes, let it do all 8471da177e4SLinus Torvalds * the work 8481da177e4SLinus Torvalds */ 849ffdfc409SOlof Johansson 850ffdfc409SOlof Johansson if (!ramdisk_execute_command) 851ffdfc409SOlof Johansson ramdisk_execute_command = "/init"; 852ffdfc409SOlof Johansson 853ffdfc409SOlof Johansson if (sys_access((const char __user *) ramdisk_execute_command, 0) != 0) { 854ffdfc409SOlof Johansson ramdisk_execute_command = NULL; 8551da177e4SLinus Torvalds prepare_namespace(); 856ffdfc409SOlof Johansson } 8571da177e4SLinus Torvalds 8581da177e4SLinus Torvalds /* 8591da177e4SLinus Torvalds * Ok, we have completed the initial bootup, and 8601da177e4SLinus Torvalds * we're essentially up and running. Get rid of the 8611da177e4SLinus Torvalds * initmem segments and start the user-mode stuff.. 8621da177e4SLinus Torvalds */ 863ee5bfa64SVivek Goyal init_post(); 864ee5bfa64SVivek Goyal return 0; 8651da177e4SLinus Torvalds } 866