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 #define __KERNEL_SYSCALLS__ 131da177e4SLinus Torvalds 141da177e4SLinus Torvalds #include <linux/config.h> 151da177e4SLinus Torvalds #include <linux/types.h> 161da177e4SLinus Torvalds #include <linux/module.h> 171da177e4SLinus Torvalds #include <linux/proc_fs.h> 181da177e4SLinus Torvalds #include <linux/devfs_fs_kernel.h> 191da177e4SLinus Torvalds #include <linux/kernel.h> 201da177e4SLinus Torvalds #include <linux/syscalls.h> 211da177e4SLinus Torvalds #include <linux/string.h> 221da177e4SLinus Torvalds #include <linux/ctype.h> 231da177e4SLinus Torvalds #include <linux/delay.h> 241da177e4SLinus Torvalds #include <linux/utsname.h> 251da177e4SLinus Torvalds #include <linux/ioport.h> 261da177e4SLinus Torvalds #include <linux/init.h> 271da177e4SLinus Torvalds #include <linux/smp_lock.h> 281da177e4SLinus Torvalds #include <linux/initrd.h> 291da177e4SLinus Torvalds #include <linux/hdreg.h> 301da177e4SLinus Torvalds #include <linux/bootmem.h> 311da177e4SLinus Torvalds #include <linux/tty.h> 321da177e4SLinus Torvalds #include <linux/gfp.h> 331da177e4SLinus Torvalds #include <linux/percpu.h> 341da177e4SLinus Torvalds #include <linux/kmod.h> 351da177e4SLinus Torvalds #include <linux/kernel_stat.h> 361da177e4SLinus Torvalds #include <linux/security.h> 371da177e4SLinus Torvalds #include <linux/workqueue.h> 381da177e4SLinus Torvalds #include <linux/profile.h> 391da177e4SLinus Torvalds #include <linux/rcupdate.h> 401da177e4SLinus Torvalds #include <linux/moduleparam.h> 411da177e4SLinus Torvalds #include <linux/kallsyms.h> 421da177e4SLinus Torvalds #include <linux/writeback.h> 431da177e4SLinus Torvalds #include <linux/cpu.h> 441da177e4SLinus Torvalds #include <linux/cpuset.h> 451da177e4SLinus Torvalds #include <linux/efi.h> 461da177e4SLinus Torvalds #include <linux/unistd.h> 471da177e4SLinus Torvalds #include <linux/rmap.h> 481da177e4SLinus Torvalds #include <linux/mempolicy.h> 491da177e4SLinus Torvalds #include <linux/key.h> 501da177e4SLinus Torvalds 511da177e4SLinus Torvalds #include <asm/io.h> 521da177e4SLinus Torvalds #include <asm/bugs.h> 531da177e4SLinus Torvalds #include <asm/setup.h> 54a940199fSAndi Kleen #include <asm/sections.h> 5537b73c82SArjan van de Ven #include <asm/cacheflush.h> 561da177e4SLinus Torvalds 571da177e4SLinus Torvalds /* 581da177e4SLinus Torvalds * This is one of the first .c files built. Error out early 591da177e4SLinus Torvalds * if we have compiler trouble.. 601da177e4SLinus Torvalds */ 611da177e4SLinus Torvalds 621da177e4SLinus Torvalds #ifdef CONFIG_X86_LOCAL_APIC 631da177e4SLinus Torvalds #include <asm/smp.h> 641da177e4SLinus Torvalds #endif 651da177e4SLinus Torvalds 661da177e4SLinus Torvalds /* 671da177e4SLinus Torvalds * Versions of gcc older than that listed below may actually compile 681da177e4SLinus Torvalds * and link okay, but the end product can have subtle run time bugs. 691da177e4SLinus Torvalds * To avoid associated bogus bug reports, we flatly refuse to compile 701da177e4SLinus Torvalds * with a gcc that is known to be too old from the very beginning. 711da177e4SLinus Torvalds */ 72fd285bb5SAndrew Morton #if (__GNUC__ < 3) || (__GNUC__ == 3 && __GNUC_MINOR__ < 2) 731da177e4SLinus Torvalds #error Sorry, your GCC is too old. It builds incorrect kernels. 741da177e4SLinus Torvalds #endif 751da177e4SLinus Torvalds 761da177e4SLinus Torvalds static int init(void *); 771da177e4SLinus Torvalds 781da177e4SLinus Torvalds extern void init_IRQ(void); 791da177e4SLinus Torvalds extern void fork_init(unsigned long); 801da177e4SLinus Torvalds extern void mca_init(void); 811da177e4SLinus Torvalds extern void sbus_init(void); 821da177e4SLinus Torvalds extern void sysctl_init(void); 831da177e4SLinus Torvalds extern void signals_init(void); 841da177e4SLinus Torvalds extern void buffer_init(void); 851da177e4SLinus Torvalds extern void pidhash_init(void); 861da177e4SLinus Torvalds extern void pidmap_init(void); 871da177e4SLinus Torvalds extern void prio_tree_init(void); 881da177e4SLinus Torvalds extern void radix_tree_init(void); 891da177e4SLinus Torvalds extern void free_initmem(void); 901da177e4SLinus Torvalds extern void populate_rootfs(void); 911da177e4SLinus Torvalds extern void driver_init(void); 921da177e4SLinus Torvalds extern void prepare_namespace(void); 931da177e4SLinus Torvalds #ifdef CONFIG_ACPI 941da177e4SLinus Torvalds extern void acpi_early_init(void); 951da177e4SLinus Torvalds #else 961da177e4SLinus Torvalds static inline void acpi_early_init(void) { } 971da177e4SLinus Torvalds #endif 9837b73c82SArjan van de Ven #ifndef CONFIG_DEBUG_RODATA 9937b73c82SArjan van de Ven static inline void mark_rodata_ro(void) { } 10037b73c82SArjan van de Ven #endif 1011da177e4SLinus Torvalds 1021da177e4SLinus Torvalds #ifdef CONFIG_TC 1031da177e4SLinus Torvalds extern void tc_init(void); 1041da177e4SLinus Torvalds #endif 1051da177e4SLinus Torvalds 1061da177e4SLinus Torvalds enum system_states system_state; 1071da177e4SLinus Torvalds EXPORT_SYMBOL(system_state); 1081da177e4SLinus Torvalds 1091da177e4SLinus Torvalds /* 1101da177e4SLinus Torvalds * Boot command-line arguments 1111da177e4SLinus Torvalds */ 1121da177e4SLinus Torvalds #define MAX_INIT_ARGS CONFIG_INIT_ENV_ARG_LIMIT 1131da177e4SLinus Torvalds #define MAX_INIT_ENVS CONFIG_INIT_ENV_ARG_LIMIT 1141da177e4SLinus Torvalds 1151da177e4SLinus Torvalds extern void time_init(void); 1161da177e4SLinus Torvalds /* Default late time init is NULL. archs can override this later. */ 1171da177e4SLinus Torvalds void (*late_time_init)(void); 1181da177e4SLinus Torvalds extern void softirq_init(void); 1191da177e4SLinus Torvalds 1201da177e4SLinus Torvalds /* Untouched command line (eg. for /proc) saved by arch-specific code. */ 1211da177e4SLinus Torvalds char saved_command_line[COMMAND_LINE_SIZE]; 1221da177e4SLinus Torvalds 1231da177e4SLinus Torvalds static char *execute_command; 124ffdfc409SOlof Johansson static char *ramdisk_execute_command; 1251da177e4SLinus Torvalds 1261da177e4SLinus Torvalds /* Setup configured maximum number of CPUs to activate */ 1271da177e4SLinus Torvalds static unsigned int max_cpus = NR_CPUS; 1281da177e4SLinus Torvalds 1291da177e4SLinus Torvalds /* 1301da177e4SLinus Torvalds * Setup routine for controlling SMP activation 1311da177e4SLinus Torvalds * 1321da177e4SLinus Torvalds * Command-line option of "nosmp" or "maxcpus=0" will disable SMP 1331da177e4SLinus Torvalds * activation entirely (the MPS table probe still happens, though). 1341da177e4SLinus Torvalds * 1351da177e4SLinus Torvalds * Command-line option of "maxcpus=<NUM>", where <NUM> is an integer 1361da177e4SLinus Torvalds * greater than 0, limits the maximum number of CPUs activated in 1371da177e4SLinus Torvalds * SMP mode to <NUM>. 1381da177e4SLinus Torvalds */ 1391da177e4SLinus Torvalds static int __init nosmp(char *str) 1401da177e4SLinus Torvalds { 1411da177e4SLinus Torvalds max_cpus = 0; 1421da177e4SLinus Torvalds return 1; 1431da177e4SLinus Torvalds } 1441da177e4SLinus Torvalds 1451da177e4SLinus Torvalds __setup("nosmp", nosmp); 1461da177e4SLinus Torvalds 1471da177e4SLinus Torvalds static int __init maxcpus(char *str) 1481da177e4SLinus Torvalds { 1491da177e4SLinus Torvalds get_option(&str, &max_cpus); 1501da177e4SLinus Torvalds return 1; 1511da177e4SLinus Torvalds } 1521da177e4SLinus Torvalds 1531da177e4SLinus Torvalds __setup("maxcpus=", maxcpus); 1541da177e4SLinus Torvalds 1551da177e4SLinus Torvalds static char * argv_init[MAX_INIT_ARGS+2] = { "init", NULL, }; 1561da177e4SLinus Torvalds char * envp_init[MAX_INIT_ENVS+2] = { "HOME=/", "TERM=linux", NULL, }; 1571da177e4SLinus Torvalds static const char *panic_later, *panic_param; 1581da177e4SLinus Torvalds 1591da177e4SLinus Torvalds extern struct obs_kernel_param __setup_start[], __setup_end[]; 1601da177e4SLinus Torvalds 1611da177e4SLinus Torvalds static int __init obsolete_checksetup(char *line) 1621da177e4SLinus Torvalds { 1631da177e4SLinus Torvalds struct obs_kernel_param *p; 1641da177e4SLinus Torvalds 1651da177e4SLinus Torvalds p = __setup_start; 1661da177e4SLinus Torvalds do { 1671da177e4SLinus Torvalds int n = strlen(p->str); 1681da177e4SLinus Torvalds if (!strncmp(line, p->str, n)) { 1691da177e4SLinus Torvalds if (p->early) { 1701da177e4SLinus Torvalds /* Already done in parse_early_param? (Needs 1711da177e4SLinus Torvalds * exact match on param part) */ 1721da177e4SLinus Torvalds if (line[n] == '\0' || line[n] == '=') 1731da177e4SLinus Torvalds return 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); 1831da177e4SLinus Torvalds return 0; 1841da177e4SLinus Torvalds } 1851da177e4SLinus Torvalds 1861da177e4SLinus Torvalds /* 1871da177e4SLinus Torvalds * This should be approx 2 Bo*oMips to start (note initial shift), and will 1881da177e4SLinus Torvalds * still work even if initially too large, it will just take slightly longer 1891da177e4SLinus Torvalds */ 1901da177e4SLinus Torvalds unsigned long loops_per_jiffy = (1<<12); 1911da177e4SLinus Torvalds 1921da177e4SLinus Torvalds EXPORT_SYMBOL(loops_per_jiffy); 1931da177e4SLinus Torvalds 1941da177e4SLinus Torvalds static int __init debug_kernel(char *str) 1951da177e4SLinus Torvalds { 1961da177e4SLinus Torvalds if (*str) 1971da177e4SLinus Torvalds return 0; 1981da177e4SLinus Torvalds console_loglevel = 10; 1991da177e4SLinus Torvalds return 1; 2001da177e4SLinus Torvalds } 2011da177e4SLinus Torvalds 2021da177e4SLinus Torvalds static int __init quiet_kernel(char *str) 2031da177e4SLinus Torvalds { 2041da177e4SLinus Torvalds if (*str) 2051da177e4SLinus Torvalds return 0; 2061da177e4SLinus Torvalds console_loglevel = 4; 2071da177e4SLinus Torvalds return 1; 2081da177e4SLinus Torvalds } 2091da177e4SLinus Torvalds 2101da177e4SLinus Torvalds __setup("debug", debug_kernel); 2111da177e4SLinus Torvalds __setup("quiet", quiet_kernel); 2121da177e4SLinus Torvalds 2131da177e4SLinus Torvalds static int __init loglevel(char *str) 2141da177e4SLinus Torvalds { 2151da177e4SLinus Torvalds get_option(&str, &console_loglevel); 2161da177e4SLinus Torvalds return 1; 2171da177e4SLinus Torvalds } 2181da177e4SLinus Torvalds 2191da177e4SLinus Torvalds __setup("loglevel=", loglevel); 2201da177e4SLinus Torvalds 2211da177e4SLinus Torvalds /* 2221da177e4SLinus Torvalds * Unknown boot options get handed to init, unless they look like 2231da177e4SLinus Torvalds * failed parameters 2241da177e4SLinus Torvalds */ 2251da177e4SLinus Torvalds static int __init unknown_bootoption(char *param, char *val) 2261da177e4SLinus Torvalds { 2271da177e4SLinus Torvalds /* Change NUL term back to "=", to make "param" the whole string. */ 2281da177e4SLinus Torvalds if (val) { 2291da177e4SLinus Torvalds /* param=val or param="val"? */ 2301da177e4SLinus Torvalds if (val == param+strlen(param)+1) 2311da177e4SLinus Torvalds val[-1] = '='; 2321da177e4SLinus Torvalds else if (val == param+strlen(param)+2) { 2331da177e4SLinus Torvalds val[-2] = '='; 2341da177e4SLinus Torvalds memmove(val-1, val, strlen(val)+1); 2351da177e4SLinus Torvalds val--; 2361da177e4SLinus Torvalds } else 2371da177e4SLinus Torvalds BUG(); 2381da177e4SLinus Torvalds } 2391da177e4SLinus Torvalds 2401da177e4SLinus Torvalds /* Handle obsolete-style parameters */ 2411da177e4SLinus Torvalds if (obsolete_checksetup(param)) 2421da177e4SLinus Torvalds return 0; 2431da177e4SLinus Torvalds 2441da177e4SLinus Torvalds /* 2451da177e4SLinus Torvalds * Preemptive maintenance for "why didn't my mispelled command 2461da177e4SLinus Torvalds * line work?" 2471da177e4SLinus Torvalds */ 2481da177e4SLinus Torvalds if (strchr(param, '.') && (!val || strchr(param, '.') < val)) { 2491da177e4SLinus Torvalds printk(KERN_ERR "Unknown boot option `%s': ignoring\n", param); 2501da177e4SLinus Torvalds return 0; 2511da177e4SLinus Torvalds } 2521da177e4SLinus Torvalds 2531da177e4SLinus Torvalds if (panic_later) 2541da177e4SLinus Torvalds return 0; 2551da177e4SLinus Torvalds 2561da177e4SLinus Torvalds if (val) { 2571da177e4SLinus Torvalds /* Environment option */ 2581da177e4SLinus Torvalds unsigned int i; 2591da177e4SLinus Torvalds for (i = 0; envp_init[i]; i++) { 2601da177e4SLinus Torvalds if (i == MAX_INIT_ENVS) { 2611da177e4SLinus Torvalds panic_later = "Too many boot env vars at `%s'"; 2621da177e4SLinus Torvalds panic_param = param; 2631da177e4SLinus Torvalds } 2641da177e4SLinus Torvalds if (!strncmp(param, envp_init[i], val - param)) 2651da177e4SLinus Torvalds break; 2661da177e4SLinus Torvalds } 2671da177e4SLinus Torvalds envp_init[i] = param; 2681da177e4SLinus Torvalds } else { 2691da177e4SLinus Torvalds /* Command line option */ 2701da177e4SLinus Torvalds unsigned int i; 2711da177e4SLinus Torvalds for (i = 0; argv_init[i]; i++) { 2721da177e4SLinus Torvalds if (i == MAX_INIT_ARGS) { 2731da177e4SLinus Torvalds panic_later = "Too many boot init vars at `%s'"; 2741da177e4SLinus Torvalds panic_param = param; 2751da177e4SLinus Torvalds } 2761da177e4SLinus Torvalds } 2771da177e4SLinus Torvalds argv_init[i] = param; 2781da177e4SLinus Torvalds } 2791da177e4SLinus Torvalds return 0; 2801da177e4SLinus Torvalds } 2811da177e4SLinus Torvalds 2821da177e4SLinus Torvalds static int __init init_setup(char *str) 2831da177e4SLinus Torvalds { 2841da177e4SLinus Torvalds unsigned int i; 2851da177e4SLinus Torvalds 2861da177e4SLinus Torvalds execute_command = str; 2871da177e4SLinus Torvalds /* 2881da177e4SLinus Torvalds * In case LILO is going to boot us with default command line, 2891da177e4SLinus Torvalds * it prepends "auto" before the whole cmdline which makes 2901da177e4SLinus Torvalds * the shell think it should execute a script with such name. 2911da177e4SLinus Torvalds * So we ignore all arguments entered _before_ init=... [MJ] 2921da177e4SLinus Torvalds */ 2931da177e4SLinus Torvalds for (i = 1; i < MAX_INIT_ARGS; i++) 2941da177e4SLinus Torvalds argv_init[i] = NULL; 2951da177e4SLinus Torvalds return 1; 2961da177e4SLinus Torvalds } 2971da177e4SLinus Torvalds __setup("init=", init_setup); 2981da177e4SLinus Torvalds 299ffdfc409SOlof Johansson static int __init rdinit_setup(char *str) 300ffdfc409SOlof Johansson { 301ffdfc409SOlof Johansson unsigned int i; 302ffdfc409SOlof Johansson 303ffdfc409SOlof Johansson ramdisk_execute_command = str; 304ffdfc409SOlof Johansson /* See "auto" comment in init_setup */ 305ffdfc409SOlof Johansson for (i = 1; i < MAX_INIT_ARGS; i++) 306ffdfc409SOlof Johansson argv_init[i] = NULL; 307ffdfc409SOlof Johansson return 1; 308ffdfc409SOlof Johansson } 309ffdfc409SOlof Johansson __setup("rdinit=", rdinit_setup); 310ffdfc409SOlof Johansson 3111da177e4SLinus Torvalds extern void setup_arch(char **); 3121da177e4SLinus Torvalds 3131da177e4SLinus Torvalds #ifndef CONFIG_SMP 3141da177e4SLinus Torvalds 3151da177e4SLinus Torvalds #ifdef CONFIG_X86_LOCAL_APIC 3161da177e4SLinus Torvalds static void __init smp_init(void) 3171da177e4SLinus Torvalds { 3181da177e4SLinus Torvalds APIC_init_uniprocessor(); 3191da177e4SLinus Torvalds } 3201da177e4SLinus Torvalds #else 3211da177e4SLinus Torvalds #define smp_init() do { } while (0) 3221da177e4SLinus Torvalds #endif 3231da177e4SLinus Torvalds 3241da177e4SLinus Torvalds static inline void setup_per_cpu_areas(void) { } 3251da177e4SLinus Torvalds static inline void smp_prepare_cpus(unsigned int maxcpus) { } 3261da177e4SLinus Torvalds 3271da177e4SLinus Torvalds #else 3281da177e4SLinus Torvalds 3291da177e4SLinus Torvalds #ifdef __GENERIC_PER_CPU 3301da177e4SLinus Torvalds unsigned long __per_cpu_offset[NR_CPUS]; 3311da177e4SLinus Torvalds 3321da177e4SLinus Torvalds EXPORT_SYMBOL(__per_cpu_offset); 3331da177e4SLinus Torvalds 3341da177e4SLinus Torvalds static void __init setup_per_cpu_areas(void) 3351da177e4SLinus Torvalds { 3361da177e4SLinus Torvalds unsigned long size, i; 3371da177e4SLinus Torvalds char *ptr; 3381da177e4SLinus Torvalds 3391da177e4SLinus Torvalds /* Copy section for each CPU (we discard the original) */ 3401da177e4SLinus Torvalds size = ALIGN(__per_cpu_end - __per_cpu_start, SMP_CACHE_BYTES); 3411da177e4SLinus Torvalds #ifdef CONFIG_MODULES 3421da177e4SLinus Torvalds if (size < PERCPU_ENOUGH_ROOM) 3431da177e4SLinus Torvalds size = PERCPU_ENOUGH_ROOM; 3441da177e4SLinus Torvalds #endif 3451da177e4SLinus Torvalds 3461da177e4SLinus Torvalds ptr = alloc_bootmem(size * NR_CPUS); 3471da177e4SLinus Torvalds 3481da177e4SLinus Torvalds for (i = 0; i < NR_CPUS; i++, ptr += size) { 3491da177e4SLinus Torvalds __per_cpu_offset[i] = ptr - __per_cpu_start; 3501da177e4SLinus Torvalds memcpy(ptr, __per_cpu_start, __per_cpu_end - __per_cpu_start); 3511da177e4SLinus Torvalds } 3521da177e4SLinus Torvalds } 3531da177e4SLinus Torvalds #endif /* !__GENERIC_PER_CPU */ 3541da177e4SLinus Torvalds 3551da177e4SLinus Torvalds /* Called by boot processor to activate the rest. */ 3561da177e4SLinus Torvalds static void __init smp_init(void) 3571da177e4SLinus Torvalds { 3581da177e4SLinus Torvalds unsigned int i; 3591da177e4SLinus Torvalds 3601da177e4SLinus Torvalds /* FIXME: This should be done in userspace --RR */ 3611da177e4SLinus Torvalds for_each_present_cpu(i) { 3621da177e4SLinus Torvalds if (num_online_cpus() >= max_cpus) 3631da177e4SLinus Torvalds break; 3641da177e4SLinus Torvalds if (!cpu_online(i)) 3651da177e4SLinus Torvalds cpu_up(i); 3661da177e4SLinus Torvalds } 3671da177e4SLinus Torvalds 3681da177e4SLinus Torvalds /* Any cleanup work */ 3691da177e4SLinus Torvalds printk(KERN_INFO "Brought up %ld CPUs\n", (long)num_online_cpus()); 3701da177e4SLinus Torvalds smp_cpus_done(max_cpus); 3711da177e4SLinus Torvalds #if 0 3721da177e4SLinus Torvalds /* Get other processors into their bootup holding patterns. */ 3731da177e4SLinus Torvalds 3741da177e4SLinus Torvalds smp_commence(); 3751da177e4SLinus Torvalds #endif 3761da177e4SLinus Torvalds } 3771da177e4SLinus Torvalds 3781da177e4SLinus Torvalds #endif 3791da177e4SLinus Torvalds 3801da177e4SLinus Torvalds /* 3811da177e4SLinus Torvalds * We need to finalize in a non-__init function or else race conditions 3821da177e4SLinus Torvalds * between the root thread and the init thread may cause start_kernel to 3831da177e4SLinus Torvalds * be reaped by free_initmem before the root thread has proceeded to 3841da177e4SLinus Torvalds * cpu_idle. 3851da177e4SLinus Torvalds * 3861da177e4SLinus Torvalds * gcc-3.4 accidentally inlines this function, so use noinline. 3871da177e4SLinus Torvalds */ 3881da177e4SLinus Torvalds 3891da177e4SLinus Torvalds static void noinline rest_init(void) 3901da177e4SLinus Torvalds __releases(kernel_lock) 3911da177e4SLinus Torvalds { 3921da177e4SLinus Torvalds kernel_thread(init, NULL, CLONE_FS | CLONE_SIGHAND); 3931da177e4SLinus Torvalds numa_default_policy(); 3941da177e4SLinus Torvalds unlock_kernel(); 395f340c0d1SIngo Molnar 396f340c0d1SIngo Molnar /* 397f340c0d1SIngo Molnar * The boot idle thread must execute schedule() 398f340c0d1SIngo Molnar * at least one to get things moving: 399f340c0d1SIngo Molnar */ 4005bfb5d69SNick Piggin preempt_enable_no_resched(); 401f340c0d1SIngo Molnar schedule(); 4025bfb5d69SNick Piggin preempt_disable(); 403f340c0d1SIngo Molnar 4045bfb5d69SNick Piggin /* Call into cpu_idle with preempt disabled */ 4051da177e4SLinus Torvalds cpu_idle(); 4061da177e4SLinus Torvalds } 4071da177e4SLinus Torvalds 4081da177e4SLinus Torvalds /* Check for early params. */ 4091da177e4SLinus Torvalds static int __init do_early_param(char *param, char *val) 4101da177e4SLinus Torvalds { 4111da177e4SLinus Torvalds struct obs_kernel_param *p; 4121da177e4SLinus Torvalds 4131da177e4SLinus Torvalds for (p = __setup_start; p < __setup_end; p++) { 4141da177e4SLinus Torvalds if (p->early && strcmp(param, p->str) == 0) { 4151da177e4SLinus Torvalds if (p->setup_func(val) != 0) 4161da177e4SLinus Torvalds printk(KERN_WARNING 4171da177e4SLinus Torvalds "Malformed early option '%s'\n", param); 4181da177e4SLinus Torvalds } 4191da177e4SLinus Torvalds } 4201da177e4SLinus Torvalds /* We accept everything at this stage. */ 4211da177e4SLinus Torvalds return 0; 4221da177e4SLinus Torvalds } 4231da177e4SLinus Torvalds 4241da177e4SLinus Torvalds /* Arch code calls this early on, or if not, just before other parsing. */ 4251da177e4SLinus Torvalds void __init parse_early_param(void) 4261da177e4SLinus Torvalds { 4271da177e4SLinus Torvalds static __initdata int done = 0; 4281da177e4SLinus Torvalds static __initdata char tmp_cmdline[COMMAND_LINE_SIZE]; 4291da177e4SLinus Torvalds 4301da177e4SLinus Torvalds if (done) 4311da177e4SLinus Torvalds return; 4321da177e4SLinus Torvalds 4331da177e4SLinus Torvalds /* All fall through to do_early_param. */ 4341da177e4SLinus Torvalds strlcpy(tmp_cmdline, saved_command_line, COMMAND_LINE_SIZE); 4351da177e4SLinus Torvalds parse_args("early options", tmp_cmdline, NULL, 0, do_early_param); 4361da177e4SLinus Torvalds done = 1; 4371da177e4SLinus Torvalds } 4381da177e4SLinus Torvalds 4391da177e4SLinus Torvalds /* 4401da177e4SLinus Torvalds * Activate the first processor. 4411da177e4SLinus Torvalds */ 4421da177e4SLinus Torvalds 4431da177e4SLinus Torvalds asmlinkage void __init start_kernel(void) 4441da177e4SLinus Torvalds { 4451da177e4SLinus Torvalds char * command_line; 4461da177e4SLinus Torvalds extern struct kernel_param __start___param[], __stop___param[]; 4471da177e4SLinus Torvalds /* 4481da177e4SLinus Torvalds * Interrupts are still disabled. Do necessary setups, then 4491da177e4SLinus Torvalds * enable them 4501da177e4SLinus Torvalds */ 4511da177e4SLinus Torvalds lock_kernel(); 4521da177e4SLinus Torvalds page_address_init(); 4531da177e4SLinus Torvalds printk(KERN_NOTICE); 4541da177e4SLinus Torvalds printk(linux_banner); 4551da177e4SLinus Torvalds setup_arch(&command_line); 4561da177e4SLinus Torvalds setup_per_cpu_areas(); 4571da177e4SLinus Torvalds 4581da177e4SLinus Torvalds /* 4591da177e4SLinus Torvalds * Mark the boot cpu "online" so that it can call console drivers in 4601da177e4SLinus Torvalds * printk() and can access its per-cpu storage. 4611da177e4SLinus Torvalds */ 4621da177e4SLinus Torvalds smp_prepare_boot_cpu(); 4631da177e4SLinus Torvalds 4641da177e4SLinus Torvalds /* 4651da177e4SLinus Torvalds * Set up the scheduler prior starting any interrupts (such as the 4661da177e4SLinus Torvalds * timer interrupt). Full topology setup happens at smp_init() 4671da177e4SLinus Torvalds * time - but meanwhile we still have a functioning scheduler. 4681da177e4SLinus Torvalds */ 4691da177e4SLinus Torvalds sched_init(); 4701da177e4SLinus Torvalds /* 4711da177e4SLinus Torvalds * Disable preemption - early bootup scheduling is extremely 4721da177e4SLinus Torvalds * fragile until we cpu_idle() for the first time. 4731da177e4SLinus Torvalds */ 4741da177e4SLinus Torvalds preempt_disable(); 4751da177e4SLinus Torvalds build_all_zonelists(); 4761da177e4SLinus Torvalds page_alloc_init(); 4771da177e4SLinus Torvalds printk(KERN_NOTICE "Kernel command line: %s\n", saved_command_line); 4781da177e4SLinus Torvalds parse_early_param(); 4791da177e4SLinus Torvalds parse_args("Booting kernel", command_line, __start___param, 4801da177e4SLinus Torvalds __stop___param - __start___param, 4811da177e4SLinus Torvalds &unknown_bootoption); 4821da177e4SLinus Torvalds sort_main_extable(); 4831da177e4SLinus Torvalds trap_init(); 4841da177e4SLinus Torvalds rcu_init(); 4851da177e4SLinus Torvalds init_IRQ(); 4861da177e4SLinus Torvalds pidhash_init(); 4871da177e4SLinus Torvalds init_timers(); 488*c0a31329SThomas Gleixner hrtimers_init(); 4891da177e4SLinus Torvalds softirq_init(); 4901da177e4SLinus Torvalds time_init(); 4911da177e4SLinus Torvalds 4921da177e4SLinus Torvalds /* 4931da177e4SLinus Torvalds * HACK ALERT! This is early. We're enabling the console before 4941da177e4SLinus Torvalds * we've done PCI setups etc, and console_init() must be aware of 4951da177e4SLinus Torvalds * this. But we do want output early, in case something goes wrong. 4961da177e4SLinus Torvalds */ 4971da177e4SLinus Torvalds console_init(); 4981da177e4SLinus Torvalds if (panic_later) 4991da177e4SLinus Torvalds panic(panic_later, panic_param); 5001da177e4SLinus Torvalds profile_init(); 5011da177e4SLinus Torvalds local_irq_enable(); 5021da177e4SLinus Torvalds #ifdef CONFIG_BLK_DEV_INITRD 5031da177e4SLinus Torvalds if (initrd_start && !initrd_below_start_ok && 5041da177e4SLinus Torvalds initrd_start < min_low_pfn << PAGE_SHIFT) { 5051da177e4SLinus Torvalds printk(KERN_CRIT "initrd overwritten (0x%08lx < 0x%08lx) - " 5061da177e4SLinus Torvalds "disabling it.\n",initrd_start,min_low_pfn << PAGE_SHIFT); 5071da177e4SLinus Torvalds initrd_start = 0; 5081da177e4SLinus Torvalds } 5091da177e4SLinus Torvalds #endif 5101da177e4SLinus Torvalds vfs_caches_init_early(); 511c417f024SPaul Jackson cpuset_init_early(); 5121da177e4SLinus Torvalds mem_init(); 5131da177e4SLinus Torvalds kmem_cache_init(); 514e7c8d5c9SChristoph Lameter setup_per_cpu_pageset(); 5151da177e4SLinus Torvalds numa_policy_init(); 5161da177e4SLinus Torvalds if (late_time_init) 5171da177e4SLinus Torvalds late_time_init(); 5181da177e4SLinus Torvalds calibrate_delay(); 5191da177e4SLinus Torvalds pidmap_init(); 5201da177e4SLinus Torvalds pgtable_cache_init(); 5211da177e4SLinus Torvalds prio_tree_init(); 5221da177e4SLinus Torvalds anon_vma_init(); 5231da177e4SLinus Torvalds #ifdef CONFIG_X86 5241da177e4SLinus Torvalds if (efi_enabled) 5251da177e4SLinus Torvalds efi_enter_virtual_mode(); 5261da177e4SLinus Torvalds #endif 5271da177e4SLinus Torvalds fork_init(num_physpages); 5281da177e4SLinus Torvalds proc_caches_init(); 5291da177e4SLinus Torvalds buffer_init(); 5301da177e4SLinus Torvalds unnamed_dev_init(); 5311da177e4SLinus Torvalds key_init(); 5321da177e4SLinus Torvalds security_init(); 5331da177e4SLinus Torvalds vfs_caches_init(num_physpages); 5341da177e4SLinus Torvalds radix_tree_init(); 5351da177e4SLinus Torvalds signals_init(); 5361da177e4SLinus Torvalds /* rootfs populating might need page-writeback */ 5371da177e4SLinus Torvalds page_writeback_init(); 5381da177e4SLinus Torvalds #ifdef CONFIG_PROC_FS 5391da177e4SLinus Torvalds proc_root_init(); 5401da177e4SLinus Torvalds #endif 5411da177e4SLinus Torvalds cpuset_init(); 5421da177e4SLinus Torvalds 5431da177e4SLinus Torvalds check_bugs(); 5441da177e4SLinus Torvalds 5451da177e4SLinus Torvalds acpi_early_init(); /* before LAPIC and SMP init */ 5461da177e4SLinus Torvalds 5471da177e4SLinus Torvalds /* Do the rest non-__init'ed, we're now alive */ 5481da177e4SLinus Torvalds rest_init(); 5491da177e4SLinus Torvalds } 5501da177e4SLinus Torvalds 5511da177e4SLinus Torvalds static int __initdata initcall_debug; 5521da177e4SLinus Torvalds 5531da177e4SLinus Torvalds static int __init initcall_debug_setup(char *str) 5541da177e4SLinus Torvalds { 5551da177e4SLinus Torvalds initcall_debug = 1; 5561da177e4SLinus Torvalds return 1; 5571da177e4SLinus Torvalds } 5581da177e4SLinus Torvalds __setup("initcall_debug", initcall_debug_setup); 5591da177e4SLinus Torvalds 5601da177e4SLinus Torvalds struct task_struct *child_reaper = &init_task; 5611da177e4SLinus Torvalds 5621da177e4SLinus Torvalds extern initcall_t __initcall_start[], __initcall_end[]; 5631da177e4SLinus Torvalds 5641da177e4SLinus Torvalds static void __init do_initcalls(void) 5651da177e4SLinus Torvalds { 5661da177e4SLinus Torvalds initcall_t *call; 5671da177e4SLinus Torvalds int count = preempt_count(); 5681da177e4SLinus Torvalds 5691da177e4SLinus Torvalds for (call = __initcall_start; call < __initcall_end; call++) { 5701da177e4SLinus Torvalds char *msg; 5711da177e4SLinus Torvalds 5721da177e4SLinus Torvalds if (initcall_debug) { 5731da177e4SLinus Torvalds printk(KERN_DEBUG "Calling initcall 0x%p", *call); 5741da177e4SLinus Torvalds print_fn_descriptor_symbol(": %s()", (unsigned long) *call); 5751da177e4SLinus Torvalds printk("\n"); 5761da177e4SLinus Torvalds } 5771da177e4SLinus Torvalds 5781da177e4SLinus Torvalds (*call)(); 5791da177e4SLinus Torvalds 5801da177e4SLinus Torvalds msg = NULL; 5811da177e4SLinus Torvalds if (preempt_count() != count) { 5821da177e4SLinus Torvalds msg = "preemption imbalance"; 5831da177e4SLinus Torvalds preempt_count() = count; 5841da177e4SLinus Torvalds } 5851da177e4SLinus Torvalds if (irqs_disabled()) { 5861da177e4SLinus Torvalds msg = "disabled interrupts"; 5871da177e4SLinus Torvalds local_irq_enable(); 5881da177e4SLinus Torvalds } 5891da177e4SLinus Torvalds if (msg) { 5901da177e4SLinus Torvalds printk(KERN_WARNING "error in initcall at 0x%p: " 5911da177e4SLinus Torvalds "returned with %s\n", *call, msg); 5921da177e4SLinus Torvalds } 5931da177e4SLinus Torvalds } 5941da177e4SLinus Torvalds 5951da177e4SLinus Torvalds /* Make sure there is no pending stuff from the initcall sequence */ 5961da177e4SLinus Torvalds flush_scheduled_work(); 5971da177e4SLinus Torvalds } 5981da177e4SLinus Torvalds 5991da177e4SLinus Torvalds /* 6001da177e4SLinus Torvalds * Ok, the machine is now initialized. None of the devices 6011da177e4SLinus Torvalds * have been touched yet, but the CPU subsystem is up and 6021da177e4SLinus Torvalds * running, and memory and process management works. 6031da177e4SLinus Torvalds * 6041da177e4SLinus Torvalds * Now we can finally start doing some real work.. 6051da177e4SLinus Torvalds */ 6061da177e4SLinus Torvalds static void __init do_basic_setup(void) 6071da177e4SLinus Torvalds { 6081da177e4SLinus Torvalds /* drivers will send hotplug events */ 6091da177e4SLinus Torvalds init_workqueues(); 6101da177e4SLinus Torvalds usermodehelper_init(); 6111da177e4SLinus Torvalds driver_init(); 6121da177e4SLinus Torvalds 6131da177e4SLinus Torvalds #ifdef CONFIG_SYSCTL 6141da177e4SLinus Torvalds sysctl_init(); 6151da177e4SLinus Torvalds #endif 6161da177e4SLinus Torvalds 6171da177e4SLinus Torvalds do_initcalls(); 6181da177e4SLinus Torvalds } 6191da177e4SLinus Torvalds 6201da177e4SLinus Torvalds static void do_pre_smp_initcalls(void) 6211da177e4SLinus Torvalds { 6221da177e4SLinus Torvalds extern int spawn_ksoftirqd(void); 6231da177e4SLinus Torvalds #ifdef CONFIG_SMP 6241da177e4SLinus Torvalds extern int migration_init(void); 6251da177e4SLinus Torvalds 6261da177e4SLinus Torvalds migration_init(); 6271da177e4SLinus Torvalds #endif 6281da177e4SLinus Torvalds spawn_ksoftirqd(); 6298446f1d3SIngo Molnar spawn_softlockup_task(); 6301da177e4SLinus Torvalds } 6311da177e4SLinus Torvalds 6321da177e4SLinus Torvalds static void run_init_process(char *init_filename) 6331da177e4SLinus Torvalds { 6341da177e4SLinus Torvalds argv_init[0] = init_filename; 6351da177e4SLinus Torvalds execve(init_filename, argv_init, envp_init); 6361da177e4SLinus Torvalds } 6371da177e4SLinus Torvalds 6381da177e4SLinus Torvalds static inline void fixup_cpu_present_map(void) 6391da177e4SLinus Torvalds { 6401da177e4SLinus Torvalds #ifdef CONFIG_SMP 6411da177e4SLinus Torvalds int i; 6421da177e4SLinus Torvalds 6431da177e4SLinus Torvalds /* 6441da177e4SLinus Torvalds * If arch is not hotplug ready and did not populate 6451da177e4SLinus Torvalds * cpu_present_map, just make cpu_present_map same as cpu_possible_map 6461da177e4SLinus Torvalds * for other cpu bringup code to function as normal. e.g smp_init() etc. 6471da177e4SLinus Torvalds */ 6481da177e4SLinus Torvalds if (cpus_empty(cpu_present_map)) { 6491da177e4SLinus Torvalds for_each_cpu(i) { 6501da177e4SLinus Torvalds cpu_set(i, cpu_present_map); 6511da177e4SLinus Torvalds } 6521da177e4SLinus Torvalds } 6531da177e4SLinus Torvalds #endif 6541da177e4SLinus Torvalds } 6551da177e4SLinus Torvalds 6561da177e4SLinus Torvalds static int init(void * unused) 6571da177e4SLinus Torvalds { 6581da177e4SLinus Torvalds lock_kernel(); 6591da177e4SLinus Torvalds /* 6601da177e4SLinus Torvalds * init can run on any cpu. 6611da177e4SLinus Torvalds */ 6621da177e4SLinus Torvalds set_cpus_allowed(current, CPU_MASK_ALL); 6631da177e4SLinus Torvalds /* 6641da177e4SLinus Torvalds * Tell the world that we're going to be the grim 6651da177e4SLinus Torvalds * reaper of innocent orphaned children. 6661da177e4SLinus Torvalds * 6671da177e4SLinus Torvalds * We don't want people to have to make incorrect 6681da177e4SLinus Torvalds * assumptions about where in the task array this 6691da177e4SLinus Torvalds * can be found. 6701da177e4SLinus Torvalds */ 6711da177e4SLinus Torvalds child_reaper = current; 6721da177e4SLinus Torvalds 6731da177e4SLinus Torvalds /* Sets up cpus_possible() */ 6741da177e4SLinus Torvalds smp_prepare_cpus(max_cpus); 6751da177e4SLinus Torvalds 6761da177e4SLinus Torvalds do_pre_smp_initcalls(); 6771da177e4SLinus Torvalds 6781da177e4SLinus Torvalds fixup_cpu_present_map(); 6791da177e4SLinus Torvalds smp_init(); 6801da177e4SLinus Torvalds sched_init_smp(); 6811da177e4SLinus Torvalds 6821da177e4SLinus Torvalds cpuset_init_smp(); 6831da177e4SLinus Torvalds 6841da177e4SLinus Torvalds /* 6851da177e4SLinus Torvalds * Do this before initcalls, because some drivers want to access 6861da177e4SLinus Torvalds * firmware files. 6871da177e4SLinus Torvalds */ 6881da177e4SLinus Torvalds populate_rootfs(); 6891da177e4SLinus Torvalds 6901da177e4SLinus Torvalds do_basic_setup(); 6911da177e4SLinus Torvalds 6921da177e4SLinus Torvalds /* 6931da177e4SLinus Torvalds * check if there is an early userspace init. If yes, let it do all 6941da177e4SLinus Torvalds * the work 6951da177e4SLinus Torvalds */ 696ffdfc409SOlof Johansson 697ffdfc409SOlof Johansson if (!ramdisk_execute_command) 698ffdfc409SOlof Johansson ramdisk_execute_command = "/init"; 699ffdfc409SOlof Johansson 700ffdfc409SOlof Johansson if (sys_access((const char __user *) ramdisk_execute_command, 0) != 0) { 701ffdfc409SOlof Johansson ramdisk_execute_command = NULL; 7021da177e4SLinus Torvalds prepare_namespace(); 703ffdfc409SOlof Johansson } 7041da177e4SLinus Torvalds 7051da177e4SLinus Torvalds /* 7061da177e4SLinus Torvalds * Ok, we have completed the initial bootup, and 7071da177e4SLinus Torvalds * we're essentially up and running. Get rid of the 7081da177e4SLinus Torvalds * initmem segments and start the user-mode stuff.. 7091da177e4SLinus Torvalds */ 7101da177e4SLinus Torvalds free_initmem(); 7111da177e4SLinus Torvalds unlock_kernel(); 71237b73c82SArjan van de Ven mark_rodata_ro(); 7131da177e4SLinus Torvalds system_state = SYSTEM_RUNNING; 7141da177e4SLinus Torvalds numa_default_policy(); 7151da177e4SLinus Torvalds 7161da177e4SLinus Torvalds if (sys_open((const char __user *) "/dev/console", O_RDWR, 0) < 0) 7171da177e4SLinus Torvalds printk(KERN_WARNING "Warning: unable to open an initial console.\n"); 7181da177e4SLinus Torvalds 7191da177e4SLinus Torvalds (void) sys_dup(0); 7201da177e4SLinus Torvalds (void) sys_dup(0); 7211da177e4SLinus Torvalds 722ffdfc409SOlof Johansson if (ramdisk_execute_command) { 723ffdfc409SOlof Johansson run_init_process(ramdisk_execute_command); 724ffdfc409SOlof Johansson printk(KERN_WARNING "Failed to execute %s\n", 725ffdfc409SOlof Johansson ramdisk_execute_command); 726ffdfc409SOlof Johansson } 727ffdfc409SOlof Johansson 7281da177e4SLinus Torvalds /* 7291da177e4SLinus Torvalds * We try each of these until one succeeds. 7301da177e4SLinus Torvalds * 7311da177e4SLinus Torvalds * The Bourne shell can be used instead of init if we are 7321da177e4SLinus Torvalds * trying to recover a really broken machine. 7331da177e4SLinus Torvalds */ 734c1d7ef70SAvery, Brian if (execute_command) { 7351da177e4SLinus Torvalds run_init_process(execute_command); 736c1d7ef70SAvery, Brian printk(KERN_WARNING "Failed to execute %s. Attempting " 737c1d7ef70SAvery, Brian "defaults...\n", execute_command); 738c1d7ef70SAvery, Brian } 7391da177e4SLinus Torvalds run_init_process("/sbin/init"); 7401da177e4SLinus Torvalds run_init_process("/etc/init"); 7411da177e4SLinus Torvalds run_init_process("/bin/init"); 7421da177e4SLinus Torvalds run_init_process("/bin/sh"); 7431da177e4SLinus Torvalds 7441da177e4SLinus Torvalds panic("No init found. Try passing init= option to kernel."); 7451da177e4SLinus Torvalds } 746