Blame view

init/main.c 21.2 KB
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1
2
3
4
5
6
7
8
9
10
  /*
   *  linux/init/main.c
   *
   *  Copyright (C) 1991, 1992  Linus Torvalds
   *
   *  GK 2/5/95  -  Changed to support mounting root fs via NFS
   *  Added initrd & change_root: Werner Almesberger & Hans Lermen, Feb '96
   *  Moan early if gcc is old, avoiding bogus kernels - Paul Gortmaker, May '96
   *  Simplified starting of init:  Michael A. Griffith <grif@acm.org> 
   */
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
11
12
13
  #include <linux/types.h>
  #include <linux/module.h>
  #include <linux/proc_fs.h>
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
14
15
  #include <linux/kernel.h>
  #include <linux/syscalls.h>
9b5609fd7   Ingo Molnar   stackprotector: i...
16
  #include <linux/stackprotector.h>
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
17
18
19
20
21
22
23
24
  #include <linux/string.h>
  #include <linux/ctype.h>
  #include <linux/delay.h>
  #include <linux/utsname.h>
  #include <linux/ioport.h>
  #include <linux/init.h>
  #include <linux/smp_lock.h>
  #include <linux/initrd.h>
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
25
26
27
28
29
  #include <linux/bootmem.h>
  #include <linux/tty.h>
  #include <linux/gfp.h>
  #include <linux/percpu.h>
  #include <linux/kmod.h>
db64fe022   Nick Piggin   mm: rewrite vmap ...
30
  #include <linux/vmalloc.h>
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
31
  #include <linux/kernel_stat.h>
d7cd56111   Rusty Russell   [PATCH] i386: cpu...
32
  #include <linux/start_kernel.h>
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
33
  #include <linux/security.h>
3d4422332   Jens Axboe   Add generic helpe...
34
  #include <linux/smp.h>
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
35
36
37
38
39
40
41
42
  #include <linux/workqueue.h>
  #include <linux/profile.h>
  #include <linux/rcupdate.h>
  #include <linux/moduleparam.h>
  #include <linux/kallsyms.h>
  #include <linux/writeback.h>
  #include <linux/cpu.h>
  #include <linux/cpuset.h>
ddbcc7e8e   Paul Menage   Task Control Grou...
43
  #include <linux/cgroup.h>
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
44
  #include <linux/efi.h>
906568c9c   Thomas Gleixner   [PATCH] tick-mana...
45
  #include <linux/tick.h>
6168a702a   Andrew Morton   [PATCH] Declare i...
46
  #include <linux/interrupt.h>
c757249af   Shailabh Nagar   [PATCH] per-task-...
47
  #include <linux/taskstats_kern.h>
ca74e92b4   Shailabh Nagar   [PATCH] per-task-...
48
  #include <linux/delayacct.h>
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
49
50
51
52
  #include <linux/unistd.h>
  #include <linux/rmap.h>
  #include <linux/mempolicy.h>
  #include <linux/key.h>
b6cd0b772   Adrian Bunk   [PATCH] fs/buffer...
53
  #include <linux/buffer_head.h>
94b6da5ab   KAMEZAWA Hiroyuki   memcg: fix page_c...
54
  #include <linux/page_cgroup.h>
9a11b49a8   Ingo Molnar   [PATCH] lockdep: ...
55
  #include <linux/debug_locks.h>
3ac7fe5a4   Thomas Gleixner   infrastructure to...
56
  #include <linux/debugobjects.h>
fbb9ce953   Ingo Molnar   [PATCH] lockdep: ...
57
  #include <linux/lockdep.h>
84d737866   Sukadev Bhattiprolu   [PATCH] add child...
58
  #include <linux/pid_namespace.h>
1f21782e6   Adrian Bunk   Driver core: prop...
59
  #include <linux/device.h>
73c279927   Eric W. Biederman   kthread: don't de...
60
  #include <linux/kthread.h>
e6fe6649b   Adrian Bunk   sched: proper pro...
61
  #include <linux/sched.h>
a1c9eea9e   Adrian Bunk   proper prototype ...
62
  #include <linux/signal.h>
199f0ca51   Akinobu Mita   idr: create idr_l...
63
  #include <linux/idr.h>
68bf21aa1   Steven Rostedt   ftrace: mcount ca...
64
  #include <linux/ftrace.h>
22a9d6456   Arjan van de Ven   async: Asynchrono...
65
  #include <linux/async.h>
3f5ec1369   Frederic Weisbecker   tracing/fastboot:...
66
  #include <trace/boot.h>
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
67
68
69
70
  
  #include <asm/io.h>
  #include <asm/bugs.h>
  #include <asm/setup.h>
a940199f2   Andi Kleen   [PATCH] x86_64: S...
71
  #include <asm/sections.h>
37b73c828   Arjan van de Ven   [PATCH] x86/x86_6...
72
  #include <asm/cacheflush.h>
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
73

1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
74
75
76
  #ifdef CONFIG_X86_LOCAL_APIC
  #include <asm/smp.h>
  #endif
aae5f662a   Sam Ravnborg   kbuild: whitelist...
77
  static int kernel_init(void *);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
78
79
  
  extern void init_IRQ(void);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
80
81
82
  extern void fork_init(unsigned long);
  extern void mca_init(void);
  extern void sbus_init(void);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
83
84
85
  extern void prio_tree_init(void);
  extern void radix_tree_init(void);
  extern void free_initmem(void);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
86
87
88
89
90
  #ifdef	CONFIG_ACPI
  extern void acpi_early_init(void);
  #else
  static inline void acpi_early_init(void) { }
  #endif
37b73c828   Arjan van de Ven   [PATCH] x86/x86_6...
91
92
93
  #ifndef CONFIG_DEBUG_RODATA
  static inline void mark_rodata_ro(void) { }
  #endif
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
94
95
96
97
  
  #ifdef CONFIG_TC
  extern void tc_init(void);
  #endif
a68260483   Paul E. McKenney   rcu: Teach RCU th...
98
  enum system_states system_state __read_mostly;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
99
100
101
102
103
104
105
106
107
108
  EXPORT_SYMBOL(system_state);
  
  /*
   * Boot command-line arguments
   */
  #define MAX_INIT_ARGS CONFIG_INIT_ENV_ARG_LIMIT
  #define MAX_INIT_ENVS CONFIG_INIT_ENV_ARG_LIMIT
  
  extern void time_init(void);
  /* Default late time init is NULL. archs can override this later. */
d2e3192b6   Jan Beulich   init/main.c: mark...
109
  void (*__initdata late_time_init)(void);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
110
  extern void softirq_init(void);
30d7e0d46   Alon Bar-Lev   [PATCH] Dynamic k...
111
112
113
114
115
116
  /* Untouched command line saved by arch-specific code. */
  char __initdata boot_command_line[COMMAND_LINE_SIZE];
  /* Untouched saved command line (eg. for /proc) */
  char *saved_command_line;
  /* Command line for parameter parsing */
  static char *static_command_line;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
117
118
  
  static char *execute_command;
ffdfc4097   Olof Johansson   [PATCH] Add rdini...
119
  static char *ramdisk_execute_command;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
120

8b3b29550   Jan Beulich   adjust nosmp hand...
121
  #ifdef CONFIG_SMP
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
122
  /* Setup configured maximum number of CPUs to activate */
ca74a6f84   Andi Kleen   x86: optimize loc...
123
  unsigned int __initdata setup_max_cpus = NR_CPUS;
7e96287dd   Vivek Goyal   [PATCH] kdump: in...
124
125
  
  /*
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
126
127
128
129
130
131
132
133
134
   * Setup routine for controlling SMP activation
   *
   * Command-line option of "nosmp" or "maxcpus=0" will disable SMP
   * activation entirely (the MPS table probe still happens, though).
   *
   * Command-line option of "maxcpus=<NUM>", where <NUM> is an integer
   * greater than 0, limits the maximum number of CPUs activated in
   * SMP mode to <NUM>.
   */
65a4e574d   Ingo Molnar   smp, generic: int...
135
136
  
  void __weak arch_disable_smp_support(void) { }
61ec7567d   Len Brown   ACPI: boot correc...
137

1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
138
139
  static int __init nosmp(char *str)
  {
ca74a6f84   Andi Kleen   x86: optimize loc...
140
  	setup_max_cpus = 0;
65a4e574d   Ingo Molnar   smp, generic: int...
141
  	arch_disable_smp_support();
8b3b29550   Jan Beulich   adjust nosmp hand...
142
  	return 0;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
143
  }
8b3b29550   Jan Beulich   adjust nosmp hand...
144
  early_param("nosmp", nosmp);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
145
146
147
  
  static int __init maxcpus(char *str)
  {
ca74a6f84   Andi Kleen   x86: optimize loc...
148
149
  	get_option(&str, &setup_max_cpus);
  	if (setup_max_cpus == 0)
65a4e574d   Ingo Molnar   smp, generic: int...
150
  		arch_disable_smp_support();
61ec7567d   Len Brown   ACPI: boot correc...
151
152
  
  	return 0;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
153
  }
813409771   Hugh Dickins   fix maxcpus=N par...
154
  early_param("maxcpus", maxcpus);
8b3b29550   Jan Beulich   adjust nosmp hand...
155
  #else
65a4e574d   Ingo Molnar   smp, generic: int...
156
  const unsigned int setup_max_cpus = NR_CPUS;
8b3b29550   Jan Beulich   adjust nosmp hand...
157
158
159
160
161
162
163
164
165
166
167
168
169
  #endif
  
  /*
   * If set, this is an indication to the drivers that reset the underlying
   * device before going ahead with the initialization otherwise driver might
   * rely on the BIOS and skip the reset operation.
   *
   * This is useful if kernel is booting in an unreliable environment.
   * For ex. kdump situaiton where previous kernel has crashed, BIOS has been
   * skipped and devices will be in unknown state.
   */
  unsigned int reset_devices;
  EXPORT_SYMBOL(reset_devices);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
170

7e96287dd   Vivek Goyal   [PATCH] kdump: in...
171
172
173
174
175
176
177
  static int __init set_reset_devices(char *str)
  {
  	reset_devices = 1;
  	return 1;
  }
  
  __setup("reset_devices", set_reset_devices);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
178
179
180
181
182
183
184
185
186
  static char * argv_init[MAX_INIT_ARGS+2] = { "init", NULL, };
  char * envp_init[MAX_INIT_ENVS+2] = { "HOME=/", "TERM=linux", NULL, };
  static const char *panic_later, *panic_param;
  
  extern struct obs_kernel_param __setup_start[], __setup_end[];
  
  static int __init obsolete_checksetup(char *line)
  {
  	struct obs_kernel_param *p;
33df0d19e   Rusty Russell   [PATCH] Allow ear...
187
  	int had_early_param = 0;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
188
189
190
191
192
193
  
  	p = __setup_start;
  	do {
  		int n = strlen(p->str);
  		if (!strncmp(line, p->str, n)) {
  			if (p->early) {
33df0d19e   Rusty Russell   [PATCH] Allow ear...
194
195
196
197
  				/* Already done in parse_early_param?
  				 * (Needs exact match on param part).
  				 * Keep iterating, as we can have early
  				 * params and __setups of same names 8( */
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
198
  				if (line[n] == '\0' || line[n] == '=')
33df0d19e   Rusty Russell   [PATCH] Allow ear...
199
  					had_early_param = 1;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
200
201
202
203
204
205
206
207
208
209
  			} else if (!p->setup_func) {
  				printk(KERN_WARNING "Parameter %s is obsolete,"
  				       " ignored
  ", p->str);
  				return 1;
  			} else if (p->setup_func(line + n))
  				return 1;
  		}
  		p++;
  	} while (p < __setup_end);
33df0d19e   Rusty Russell   [PATCH] Allow ear...
210
211
  
  	return had_early_param;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
212
213
214
215
216
217
218
219
220
221
222
223
  }
  
  /*
   * This should be approx 2 Bo*oMips to start (note initial shift), and will
   * still work even if initially too large, it will just take slightly longer
   */
  unsigned long loops_per_jiffy = (1<<12);
  
  EXPORT_SYMBOL(loops_per_jiffy);
  
  static int __init debug_kernel(char *str)
  {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
224
  	console_loglevel = 10;
f6f21c814   Yinghai Lu   Convert loglevel-...
225
  	return 0;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
226
227
228
229
  }
  
  static int __init quiet_kernel(char *str)
  {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
230
  	console_loglevel = 4;
f6f21c814   Yinghai Lu   Convert loglevel-...
231
  	return 0;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
232
  }
f6f21c814   Yinghai Lu   Convert loglevel-...
233
234
  early_param("debug", debug_kernel);
  early_param("quiet", quiet_kernel);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
235
236
237
238
  
  static int __init loglevel(char *str)
  {
  	get_option(&str, &console_loglevel);
d9d4fcfe5   Alex Riesen   Fix "Malformed ea...
239
  	return 0;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
240
  }
f6f21c814   Yinghai Lu   Convert loglevel-...
241
  early_param("loglevel", loglevel);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
  
  /*
   * Unknown boot options get handed to init, unless they look like
   * failed parameters
   */
  static int __init unknown_bootoption(char *param, char *val)
  {
  	/* Change NUL term back to "=", to make "param" the whole string. */
  	if (val) {
  		/* param=val or param="val"? */
  		if (val == param+strlen(param)+1)
  			val[-1] = '=';
  		else if (val == param+strlen(param)+2) {
  			val[-2] = '=';
  			memmove(val-1, val, strlen(val)+1);
  			val--;
  		} else
  			BUG();
  	}
  
  	/* Handle obsolete-style parameters */
  	if (obsolete_checksetup(param))
  		return 0;
  
  	/*
211fee8a8   Simon Arlott   spelling fixes: i...
267
  	 * Preemptive maintenance for "why didn't my misspelled command
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
  	 * line work?"
  	 */
  	if (strchr(param, '.') && (!val || strchr(param, '.') < val)) {
  		printk(KERN_ERR "Unknown boot option `%s': ignoring
  ", param);
  		return 0;
  	}
  
  	if (panic_later)
  		return 0;
  
  	if (val) {
  		/* Environment option */
  		unsigned int i;
  		for (i = 0; envp_init[i]; i++) {
  			if (i == MAX_INIT_ENVS) {
  				panic_later = "Too many boot env vars at `%s'";
  				panic_param = param;
  			}
  			if (!strncmp(param, envp_init[i], val - param))
  				break;
  		}
  		envp_init[i] = param;
  	} else {
  		/* Command line option */
  		unsigned int i;
  		for (i = 0; argv_init[i]; i++) {
  			if (i == MAX_INIT_ARGS) {
  				panic_later = "Too many boot init vars at `%s'";
  				panic_param = param;
  			}
  		}
  		argv_init[i] = param;
  	}
  	return 0;
  }
12d6f21ea   Ingo Molnar   x86: do not PSE o...
304
305
306
  #ifdef CONFIG_DEBUG_PAGEALLOC
  int __read_mostly debug_pagealloc_enabled = 0;
  #endif
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
  static int __init init_setup(char *str)
  {
  	unsigned int i;
  
  	execute_command = str;
  	/*
  	 * In case LILO is going to boot us with default command line,
  	 * it prepends "auto" before the whole cmdline which makes
  	 * the shell think it should execute a script with such name.
  	 * So we ignore all arguments entered _before_ init=... [MJ]
  	 */
  	for (i = 1; i < MAX_INIT_ARGS; i++)
  		argv_init[i] = NULL;
  	return 1;
  }
  __setup("init=", init_setup);
ffdfc4097   Olof Johansson   [PATCH] Add rdini...
323
324
325
326
327
328
329
330
331
332
333
  static int __init rdinit_setup(char *str)
  {
  	unsigned int i;
  
  	ramdisk_execute_command = str;
  	/* See "auto" comment in init_setup */
  	for (i = 1; i < MAX_INIT_ARGS; i++)
  		argv_init[i] = NULL;
  	return 1;
  }
  __setup("rdinit=", rdinit_setup);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
334
335
336
337
338
339
340
341
342
343
344
345
  #ifndef CONFIG_SMP
  
  #ifdef CONFIG_X86_LOCAL_APIC
  static void __init smp_init(void)
  {
  	APIC_init_uniprocessor();
  }
  #else
  #define smp_init()	do { } while (0)
  #endif
  
  static inline void setup_per_cpu_areas(void) { }
e0982e90c   Mike Travis   init: move setup ...
346
  static inline void setup_nr_cpu_ids(void) { }
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
347
348
349
  static inline void smp_prepare_cpus(unsigned int maxcpus) { }
  
  #else
321a8e9dc   Mike Travis   cpumask: add CPU_...
350
351
352
353
  #if NR_CPUS > BITS_PER_LONG
  cpumask_t cpu_mask_all __read_mostly = CPU_MASK_ALL;
  EXPORT_SYMBOL(cpu_mask_all);
  #endif
e0982e90c   Mike Travis   init: move setup ...
354
355
356
357
358
359
360
  /* Setup number of possible processor ids */
  int nr_cpu_ids __read_mostly = NR_CPUS;
  EXPORT_SYMBOL(nr_cpu_ids);
  
  /* An arch may set nr_cpu_ids earlier if needed, so this would be redundant */
  static void __init setup_nr_cpu_ids(void)
  {
e0c0ba736   Rusty Russell   cpumask: Use find...
361
  	nr_cpu_ids = find_last_bit(cpumask_bits(cpu_possible_mask),NR_CPUS) + 1;
e0982e90c   Mike Travis   init: move setup ...
362
  }
dd5af90a7   Mike Travis   x86/non-x86: perc...
363
  #ifndef CONFIG_HAVE_SETUP_PER_CPU_AREA
b73b459f7   Eric Dumazet   [PATCH] __GENERIC...
364
  unsigned long __per_cpu_offset[NR_CPUS] __read_mostly;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
365
366
367
368
369
370
371
  
  EXPORT_SYMBOL(__per_cpu_offset);
  
  static void __init setup_per_cpu_areas(void)
  {
  	unsigned long size, i;
  	char *ptr;
63872f87a   Eric Dumazet   [PATCH] Only allo...
372
  	unsigned long nr_possible_cpus = num_possible_cpus();
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
373
374
  
  	/* Copy section for each CPU (we discard the original) */
b6e3590f8   Jeremy Fitzhardinge   [PATCH] x86: Allo...
375
376
  	size = ALIGN(PERCPU_ENOUGH_ROOM, PAGE_SIZE);
  	ptr = alloc_bootmem_pages(size * nr_possible_cpus);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
377

0a9450227   KAMEZAWA Hiroyuki   [PATCH] for_each_...
378
  	for_each_possible_cpu(i) {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
379
380
  		__per_cpu_offset[i] = ptr - __per_cpu_start;
  		memcpy(ptr, __per_cpu_start, __per_cpu_end - __per_cpu_start);
63872f87a   Eric Dumazet   [PATCH] Only allo...
381
  		ptr += size;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
382
383
  	}
  }
dd5af90a7   Mike Travis   x86/non-x86: perc...
384
  #endif /* CONFIG_HAVE_SETUP_PER_CPU_AREA */
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
385
386
387
388
  
  /* Called by boot processor to activate the rest. */
  static void __init smp_init(void)
  {
53b8a315b   Christoph Lameter   [PATCH] Convert h...
389
  	unsigned int cpu;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
390

e761b7725   Max Krasnyansky   cpu hotplug, sche...
391
392
393
394
  	/*
  	 * Set up the current CPU as possible to migrate to.
  	 * The other ones will be done by cpu_up/cpu_down()
  	 */
2b17fa506   Rusty Russell   cpumask: use set_...
395
  	set_cpu_active(smp_processor_id(), true);
e761b7725   Max Krasnyansky   cpu hotplug, sche...
396

1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
397
  	/* FIXME: This should be done in userspace --RR */
53b8a315b   Christoph Lameter   [PATCH] Convert h...
398
  	for_each_present_cpu(cpu) {
ca74a6f84   Andi Kleen   x86: optimize loc...
399
  		if (num_online_cpus() >= setup_max_cpus)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
400
  			break;
53b8a315b   Christoph Lameter   [PATCH] Convert h...
401
402
  		if (!cpu_online(cpu))
  			cpu_up(cpu);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
403
404
405
406
407
  	}
  
  	/* Any cleanup work */
  	printk(KERN_INFO "Brought up %ld CPUs
  ", (long)num_online_cpus());
ca74a6f84   Andi Kleen   x86: optimize loc...
408
  	smp_cpus_done(setup_max_cpus);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
409
410
411
412
413
  }
  
  #endif
  
  /*
30d7e0d46   Alon Bar-Lev   [PATCH] Dynamic k...
414
415
416
417
418
419
420
421
422
423
424
425
426
427
   * We need to store the untouched command line for future reference.
   * We also need to store the touched command line since the parameter
   * parsing is performed in place, and we should allow a component to
   * store reference of name/value for future reference.
   */
  static void __init setup_command_line(char *command_line)
  {
  	saved_command_line = alloc_bootmem(strlen (boot_command_line)+1);
  	static_command_line = alloc_bootmem(strlen (command_line)+1);
  	strcpy (saved_command_line, boot_command_line);
  	strcpy (static_command_line, command_line);
  }
  
  /*
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
428
429
430
431
432
433
434
   * We need to finalize in a non-__init function or else race conditions
   * between the root thread and the init thread may cause start_kernel to
   * be reaped by free_initmem before the root thread has proceeded to
   * cpu_idle.
   *
   * gcc-3.4 accidentally inlines this function, so use noinline.
   */
f99ebf0a8   Rakib Mullick   init: properly pl...
435
  static noinline void __init_refok rest_init(void)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
436
437
  	__releases(kernel_lock)
  {
73c279927   Eric W. Biederman   kthread: don't de...
438
  	int pid;
aae5f662a   Sam Ravnborg   kbuild: whitelist...
439
  	kernel_thread(kernel_init, NULL, CLONE_FS | CLONE_SIGHAND);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
440
  	numa_default_policy();
73c279927   Eric W. Biederman   kthread: don't de...
441
  	pid = kernel_thread(kthreadd, NULL, CLONE_FS | CLONE_FILES);
5cd204550   Pavel Emelyanov   Deprecate find_ta...
442
  	kthreadd_task = find_task_by_pid_ns(pid, &init_pid_ns);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
443
  	unlock_kernel();
f340c0d1a   Ingo Molnar   [PATCH] Tweak idl...
444
445
446
  
  	/*
  	 * The boot idle thread must execute schedule()
1df21055e   Ingo Molnar   sched: add init_i...
447
  	 * at least once to get things moving:
f340c0d1a   Ingo Molnar   [PATCH] Tweak idl...
448
  	 */
1df21055e   Ingo Molnar   sched: add init_i...
449
  	init_idle_bootup_task(current);
a68260483   Paul E. McKenney   rcu: Teach RCU th...
450
  	rcu_scheduler_starting();
5bfb5d690   Nick Piggin   [PATCH] sched: di...
451
  	preempt_enable_no_resched();
f340c0d1a   Ingo Molnar   [PATCH] Tweak idl...
452
  	schedule();
5bfb5d690   Nick Piggin   [PATCH] sched: di...
453
  	preempt_disable();
f340c0d1a   Ingo Molnar   [PATCH] Tweak idl...
454

5bfb5d690   Nick Piggin   [PATCH] sched: di...
455
  	/* Call into cpu_idle with preempt disabled */
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
456
  	cpu_idle();
1df21055e   Ingo Molnar   sched: add init_i...
457
  }
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
458
459
460
461
462
463
464
  
  /* Check for early params. */
  static int __init do_early_param(char *param, char *val)
  {
  	struct obs_kernel_param *p;
  
  	for (p = __setup_start; p < __setup_end; p++) {
18a8bd949   Yinghai Lu   serial: convert e...
465
466
467
468
  		if ((p->early && strcmp(param, p->str) == 0) ||
  		    (strcmp(param, "console") == 0 &&
  		     strcmp(p->str, "earlycon") == 0)
  		) {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
  			if (p->setup_func(val) != 0)
  				printk(KERN_WARNING
  				       "Malformed early option '%s'
  ", param);
  		}
  	}
  	/* We accept everything at this stage. */
  	return 0;
  }
  
  /* Arch code calls this early on, or if not, just before other parsing. */
  void __init parse_early_param(void)
  {
  	static __initdata int done = 0;
  	static __initdata char tmp_cmdline[COMMAND_LINE_SIZE];
  
  	if (done)
  		return;
  
  	/* All fall through to do_early_param. */
30d7e0d46   Alon Bar-Lev   [PATCH] Dynamic k...
489
  	strlcpy(tmp_cmdline, boot_command_line, COMMAND_LINE_SIZE);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
490
491
492
493
494
495
496
  	parse_args("early options", tmp_cmdline, NULL, 0, do_early_param);
  	done = 1;
  }
  
  /*
   *	Activate the first processor.
   */
44fd22992   Stas Sergeev   [PATCH] Register ...
497
498
499
500
  static void __init boot_cpu_init(void)
  {
  	int cpu = smp_processor_id();
  	/* Mark the boot cpu "present", "online" etc for SMP and UP case */
915441b60   Rusty Russell   cpumask: Use acce...
501
502
503
  	set_cpu_online(cpu, true);
  	set_cpu_present(cpu, true);
  	set_cpu_possible(cpu, true);
44fd22992   Stas Sergeev   [PATCH] Register ...
504
  }
839ad62e7   Benjamin Herrenschmidt   [POWERPC] Use __w...
505
  void __init __weak smp_setup_processor_id(void)
033ab7f8e   Andrew Morton   [PATCH] add smp_s...
506
507
  {
  }
8c9843e57   Benjamin Herrenschmidt   [POWERPC] Add thr...
508
509
510
  void __init __weak thread_info_cache_init(void)
  {
  }
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
511
512
513
514
  asmlinkage void __init start_kernel(void)
  {
  	char * command_line;
  	extern struct kernel_param __start___param[], __stop___param[];
033ab7f8e   Andrew Morton   [PATCH] add smp_s...
515
516
  
  	smp_setup_processor_id();
fbb9ce953   Ingo Molnar   [PATCH] lockdep: ...
517
518
519
520
521
  	/*
  	 * Need to run as early as possible, to initialize the
  	 * lockdep hash:
  	 */
  	lockdep_init();
3ac7fe5a4   Thomas Gleixner   infrastructure to...
522
  	debug_objects_early_init();
420594296   Ingo Molnar   x86: fix the stac...
523
524
525
526
527
  
  	/*
  	 * Set up the the initial canary ASAP:
  	 */
  	boot_init_stack_canary();
ddbcc7e8e   Paul Menage   Task Control Grou...
528
  	cgroup_init_early();
fbb9ce953   Ingo Molnar   [PATCH] lockdep: ...
529
530
531
  
  	local_irq_disable();
  	early_boot_irqs_off();
243c7621a   Ingo Molnar   [PATCH] lockdep: ...
532
  	early_init_irq_lock_class();
fbb9ce953   Ingo Molnar   [PATCH] lockdep: ...
533

1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
534
535
536
537
538
  /*
   * Interrupts are still disabled. Do necessary setups, then
   * enable them
   */
  	lock_kernel();
906568c9c   Thomas Gleixner   [PATCH] tick-mana...
539
  	tick_init();
44fd22992   Stas Sergeev   [PATCH] Register ...
540
  	boot_cpu_init();
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
541
542
  	page_address_init();
  	printk(KERN_NOTICE);
8993780a6   Linus Torvalds   Make SLES9 "get_k...
543
  	printk(linux_banner);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
544
  	setup_arch(&command_line);
cf475ad28   Balbir Singh   cgroups: add an o...
545
  	mm_init_owner(&init_mm, &init_task);
30d7e0d46   Alon Bar-Lev   [PATCH] Dynamic k...
546
  	setup_command_line(command_line);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
547
  	setup_per_cpu_areas();
e0982e90c   Mike Travis   init: move setup ...
548
  	setup_nr_cpu_ids();
44fd22992   Stas Sergeev   [PATCH] Register ...
549
  	smp_prepare_boot_cpu();	/* arch-specific boot-cpu hooks */
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
550
551
552
553
554
555
556
557
558
559
560
561
562
563
  
  	/*
  	 * Set up the scheduler prior starting any interrupts (such as the
  	 * timer interrupt). Full topology setup happens at smp_init()
  	 * time - but meanwhile we still have a functioning scheduler.
  	 */
  	sched_init();
  	/*
  	 * Disable preemption - early bootup scheduling is extremely
  	 * fragile until we cpu_idle() for the first time.
  	 */
  	preempt_disable();
  	build_all_zonelists();
  	page_alloc_init();
30d7e0d46   Alon Bar-Lev   [PATCH] Dynamic k...
564
565
  	printk(KERN_NOTICE "Kernel command line: %s
  ", boot_command_line);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
566
  	parse_early_param();
30d7e0d46   Alon Bar-Lev   [PATCH] Dynamic k...
567
  	parse_args("Booting kernel", static_command_line, __start___param,
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
568
569
  		   __stop___param - __start___param,
  		   &unknown_bootoption);
c4a68306b   Ard van Breemen   [PATCH] start_ker...
570
571
572
573
574
575
  	if (!irqs_disabled()) {
  		printk(KERN_WARNING "start_kernel(): bug: interrupts were "
  				"enabled *very* early, fixing it
  ");
  		local_irq_disable();
  	}
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
576
577
578
  	sort_main_extable();
  	trap_init();
  	rcu_init();
0b8f1efad   Yinghai Lu   sparse irq_desc[]...
579
580
  	/* init some links before init_ISA_irqs() */
  	early_irq_init();
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
581
582
583
  	init_IRQ();
  	pidhash_init();
  	init_timers();
c0a313296   Thomas Gleixner   [PATCH] hrtimer: ...
584
  	hrtimers_init();
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
585
  	softirq_init();
ad596171e   John Stultz   [PATCH] Time: Use...
586
  	timekeeping_init();
88fecaa27   John Stultz   [PATCH] time init...
587
  	time_init();
3e51f33fc   Peter Zijlstra   sched: add option...
588
  	sched_clock_init();
93e028148   Heiko Carstens   [PATCH] lockdep: ...
589
590
  	profile_init();
  	if (!irqs_disabled())
24d431d06   Ron Lee   trivial: add miss...
591
592
593
  		printk(KERN_CRIT "start_kernel(): bug: interrupts were "
  				 "enabled early
  ");
fbb9ce953   Ingo Molnar   [PATCH] lockdep: ...
594
  	early_boot_irqs_on();
93e028148   Heiko Carstens   [PATCH] lockdep: ...
595
  	local_irq_enable();
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
596
597
598
599
600
601
602
603
604
  
  	/*
  	 * HACK ALERT! This is early. We're enabling the console before
  	 * we've done PCI setups etc, and console_init() must be aware of
  	 * this. But we do want output early, in case something goes wrong.
  	 */
  	console_init();
  	if (panic_later)
  		panic(panic_later, panic_param);
fbb9ce953   Ingo Molnar   [PATCH] lockdep: ...
605
606
  
  	lockdep_info();
9a11b49a8   Ingo Molnar   [PATCH] lockdep: ...
607
608
609
610
611
612
  	/*
  	 * Need to run this when irqs are enabled, because it wants
  	 * to self-test [hard/soft]-irqs on/off lock inversion bugs
  	 * too:
  	 */
  	locking_selftest();
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
613
614
  #ifdef CONFIG_BLK_DEV_INITRD
  	if (initrd_start && !initrd_below_start_ok &&
bd673c7c3   Geert Uytterhoeven   initrd: cast `ini...
615
  	    page_to_pfn(virt_to_page((void *)initrd_start)) < min_low_pfn) {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
616
  		printk(KERN_CRIT "initrd overwritten (0x%08lx < 0x%08lx) - "
fb6624ebd   Geert Uytterhoeven   initrd: Fix virtu...
617
618
  		    "disabling it.
  ",
bd673c7c3   Geert Uytterhoeven   initrd: cast `ini...
619
620
  		    page_to_pfn(virt_to_page((void *)initrd_start)),
  		    min_low_pfn);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
621
622
623
  		initrd_start = 0;
  	}
  #endif
db64fe022   Nick Piggin   mm: rewrite vmap ...
624
  	vmalloc_init();
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
625
  	vfs_caches_init_early();
c417f0242   Paul Jackson   [PATCH] cpuset: r...
626
  	cpuset_init_early();
94b6da5ab   KAMEZAWA Hiroyuki   memcg: fix page_c...
627
  	page_cgroup_init();
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
628
  	mem_init();
a03c2a48e   Thomas Gleixner   x86: DEBUG_PAGEAL...
629
  	enable_debug_pagealloc();
d221938c0   Gautham R Shenoy   cpu-hotplug: refc...
630
  	cpu_hotplug_init();
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
631
  	kmem_cache_init();
3ac7fe5a4   Thomas Gleixner   infrastructure to...
632
  	debug_objects_mem_init();
199f0ca51   Akinobu Mita   idr: create idr_l...
633
  	idr_init_cache();
e7c8d5c99   Christoph Lameter   [PATCH] node loca...
634
  	setup_per_cpu_pageset();
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
635
636
637
638
639
640
641
642
643
644
645
646
  	numa_policy_init();
  	if (late_time_init)
  		late_time_init();
  	calibrate_delay();
  	pidmap_init();
  	pgtable_cache_init();
  	prio_tree_init();
  	anon_vma_init();
  #ifdef CONFIG_X86
  	if (efi_enabled)
  		efi_enter_virtual_mode();
  #endif
8c9843e57   Benjamin Herrenschmidt   [POWERPC] Add thr...
647
  	thread_info_cache_init();
d84f4f992   David Howells   CRED: Inaugurate ...
648
  	cred_init();
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
649
650
651
  	fork_init(num_physpages);
  	proc_caches_init();
  	buffer_init();
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
652
653
654
655
656
657
658
659
660
661
  	key_init();
  	security_init();
  	vfs_caches_init(num_physpages);
  	radix_tree_init();
  	signals_init();
  	/* rootfs populating might need page-writeback */
  	page_writeback_init();
  #ifdef CONFIG_PROC_FS
  	proc_root_init();
  #endif
ddbcc7e8e   Paul Menage   Task Control Grou...
662
  	cgroup_init();
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
663
  	cpuset_init();
c757249af   Shailabh Nagar   [PATCH] per-task-...
664
  	taskstats_init_early();
ca74e92b4   Shailabh Nagar   [PATCH] per-task-...
665
  	delayacct_init();
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
666
667
668
669
  
  	check_bugs();
  
  	acpi_early_init(); /* before LAPIC and SMP init */
68bf21aa1   Steven Rostedt   ftrace: mcount ca...
670
  	ftrace_init();
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
671
672
673
  	/* Do the rest non-__init'ed, we're now alive */
  	rest_init();
  }
22a9d6456   Arjan van de Ven   async: Asynchrono...
674
  int initcall_debug;
d0ea3d7d2   Rusty Russell   Make initcall_deb...
675
  core_param(initcall_debug, initcall_debug, bool, 0644);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
676

59f9415ff   Arjan van de Ven   modules: extend i...
677
  int do_one_initcall(initcall_t fn)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
678
  {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
679
  	int count = preempt_count();
742390728   Frederic Weisbecker   tracing/fastboot:...
680
  	ktime_t calltime, delta, rettime;
a76bfd0da   Cyrill Gorcunov   initcalls: Fix m6...
681
  	char msgbuf[64];
742390728   Frederic Weisbecker   tracing/fastboot:...
682
683
  	struct boot_trace_call call;
  	struct boot_trace_ret ret;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
684

e0df154f4   Linus Torvalds   Split up 'do_init...
685
  	if (initcall_debug) {
742390728   Frederic Weisbecker   tracing/fastboot:...
686
687
688
689
690
  		call.caller = task_pid_nr(current);
  		printk("calling  %pF @ %i
  ", fn, call.caller);
  		calltime = ktime_get();
  		trace_boot_call(&call, fn);
71566a0d1   Frederic Weisbecker   tracing/fastboot:...
691
  		enable_boot_trace();
e0df154f4   Linus Torvalds   Split up 'do_init...
692
  	}
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
693

742390728   Frederic Weisbecker   tracing/fastboot:...
694
  	ret.result = fn();
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
695

e0df154f4   Linus Torvalds   Split up 'do_init...
696
  	if (initcall_debug) {
71566a0d1   Frederic Weisbecker   tracing/fastboot:...
697
  		disable_boot_trace();
742390728   Frederic Weisbecker   tracing/fastboot:...
698
699
  		rettime = ktime_get();
  		delta = ktime_sub(rettime, calltime);
1d926f275   Will Newton   init/main.c: use ...
700
  		ret.duration = (unsigned long long) ktime_to_ns(delta) >> 10;
742390728   Frederic Weisbecker   tracing/fastboot:...
701
  		trace_boot_ret(&ret, fn);
ca538f6bb   Tim Bird   tracing/fastboot:...
702
703
  		printk("initcall %pF returned %d after %Ld usecs
  ", fn,
742390728   Frederic Weisbecker   tracing/fastboot:...
704
  			ret.result, ret.duration);
e0df154f4   Linus Torvalds   Split up 'do_init...
705
  	}
8f0c45cdf   Ingo Molnar   enhance initcall_...
706

e0df154f4   Linus Torvalds   Split up 'do_init...
707
  	msgbuf[0] = 0;
e662e1cfd   Cyrill Gorcunov   init: don't lose ...
708

742390728   Frederic Weisbecker   tracing/fastboot:...
709
710
  	if (ret.result && ret.result != -ENODEV && initcall_debug)
  		sprintf(msgbuf, "error code %d ", ret.result);
e662e1cfd   Cyrill Gorcunov   init: don't lose ...
711

e0df154f4   Linus Torvalds   Split up 'do_init...
712
  	if (preempt_count() != count) {
a76bfd0da   Cyrill Gorcunov   initcalls: Fix m6...
713
  		strlcat(msgbuf, "preemption imbalance ", sizeof(msgbuf));
e0df154f4   Linus Torvalds   Split up 'do_init...
714
  		preempt_count() = count;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
715
  	}
e0df154f4   Linus Torvalds   Split up 'do_init...
716
  	if (irqs_disabled()) {
a76bfd0da   Cyrill Gorcunov   initcalls: Fix m6...
717
  		strlcat(msgbuf, "disabled interrupts ", sizeof(msgbuf));
e0df154f4   Linus Torvalds   Split up 'do_init...
718
719
720
  		local_irq_enable();
  	}
  	if (msgbuf[0]) {
96d746c68   Linus Torvalds   Fix init/main.c t...
721
722
  		printk("initcall %pF returned with %s
  ", fn, msgbuf);
e0df154f4   Linus Torvalds   Split up 'do_init...
723
  	}
59f9415ff   Arjan van de Ven   modules: extend i...
724

742390728   Frederic Weisbecker   tracing/fastboot:...
725
  	return ret.result;
e0df154f4   Linus Torvalds   Split up 'do_init...
726
  }
c2147a509   Eduard - Gabriel Munteanu   Better interface ...
727
  extern initcall_t __initcall_start[], __initcall_end[], __early_initcall_end[];
e0df154f4   Linus Torvalds   Split up 'do_init...
728
729
730
731
  
  static void __init do_initcalls(void)
  {
  	initcall_t *call;
c2147a509   Eduard - Gabriel Munteanu   Better interface ...
732
  	for (call = __early_initcall_end; call < __initcall_end; call++)
e0df154f4   Linus Torvalds   Split up 'do_init...
733
  		do_one_initcall(*call);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
734
735
736
737
738
739
740
741
742
743
744
745
746
747
  
  	/* Make sure there is no pending stuff from the initcall sequence */
  	flush_scheduled_work();
  }
  
  /*
   * Ok, the machine is now initialized. None of the devices
   * have been touched yet, but the CPU subsystem is up and
   * running, and memory and process management works.
   *
   * Now we can finally start doing some real work..
   */
  static void __init do_basic_setup(void)
  {
4446a36ff   Paul E. McKenney   rcu: add call_rcu...
748
  	rcu_init_sched(); /* needed by module_init stage. */
4403b406d   Linus Torvalds   Revert "Call init...
749
  	init_workqueues();
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
750
751
  	usermodehelper_init();
  	driver_init();
b04c3afb2   Eric W. Biederman   [PATCH] sysctl: m...
752
  	init_irq_proc();
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
753
754
  	do_initcalls();
  }
7babe8db9   Eduard - Gabriel Munteanu   Full conversion t...
755
  static void __init do_pre_smp_initcalls(void)
c2147a509   Eduard - Gabriel Munteanu   Better interface ...
756
757
758
759
760
761
  {
  	initcall_t *call;
  
  	for (call = __initcall_start; call < __early_initcall_end; call++)
  		do_one_initcall(*call);
  }
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
762
763
764
  static void run_init_process(char *init_filename)
  {
  	argv_init[0] = init_filename;
676085679   Arnd Bergmann   [PATCH] introduce...
765
  	kernel_execve(init_filename, argv_init, envp_init);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
766
  }
ee5bfa642   Vivek Goyal   [PATCH] generic: ...
767
768
769
  /* This is a non __init function. Force it to be noinline otherwise gcc
   * makes it inline to init() and it becomes part of init.text section
   */
f99ebf0a8   Rakib Mullick   init: properly pl...
770
  static noinline int init_post(void)
acdd052a2   Hannes Eder   init/main.c: fix ...
771
  	__releases(kernel_lock)
ee5bfa642   Vivek Goyal   [PATCH] generic: ...
772
  {
22a9d6456   Arjan van de Ven   async: Asynchrono...
773
774
  	/* need to finish all async __init code before freeing the memory */
  	async_synchronize_full();
ee5bfa642   Vivek Goyal   [PATCH] generic: ...
775
776
777
778
779
780
781
782
783
784
785
786
  	free_initmem();
  	unlock_kernel();
  	mark_rodata_ro();
  	system_state = SYSTEM_RUNNING;
  	numa_default_policy();
  
  	if (sys_open((const char __user *) "/dev/console", O_RDWR, 0) < 0)
  		printk(KERN_WARNING "Warning: unable to open an initial console.
  ");
  
  	(void) sys_dup(0);
  	(void) sys_dup(0);
fae5fa44f   Oleg Nesterov   signals: fix /sbi...
787
  	current->signal->flags |= SIGNAL_UNKILLABLE;
ee5bfa642   Vivek Goyal   [PATCH] generic: ...
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
  	if (ramdisk_execute_command) {
  		run_init_process(ramdisk_execute_command);
  		printk(KERN_WARNING "Failed to execute %s
  ",
  				ramdisk_execute_command);
  	}
  
  	/*
  	 * We try each of these until one succeeds.
  	 *
  	 * The Bourne shell can be used instead of init if we are
  	 * trying to recover a really broken machine.
  	 */
  	if (execute_command) {
  		run_init_process(execute_command);
  		printk(KERN_WARNING "Failed to execute %s.  Attempting "
  					"defaults...
  ", execute_command);
  	}
  	run_init_process("/sbin/init");
  	run_init_process("/etc/init");
  	run_init_process("/bin/init");
  	run_init_process("/bin/sh");
  
  	panic("No init found.  Try passing init= option to kernel.");
  }
aae5f662a   Sam Ravnborg   kbuild: whitelist...
814
  static int __init kernel_init(void * unused)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
815
816
817
818
819
  {
  	lock_kernel();
  	/*
  	 * init can run on any cpu.
  	 */
1a2142afa   Rusty Russell   cpumask: remove d...
820
  	set_cpus_allowed_ptr(current, cpu_all_mask);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
821
822
823
824
825
826
827
828
  	/*
  	 * Tell the world that we're going to be the grim
  	 * reaper of innocent orphaned children.
  	 *
  	 * We don't want people to have to make incorrect
  	 * assumptions about where in the task array this
  	 * can be found.
  	 */
84d737866   Sukadev Bhattiprolu   [PATCH] add child...
829
  	init_pid_ns.child_reaper = current;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
830

9ec52099e   Cedric Le Goater   [PATCH] replace c...
831
  	cad_pid = task_pid(current);
ca74a6f84   Andi Kleen   x86: optimize loc...
832
  	smp_prepare_cpus(setup_max_cpus);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
833
834
  
  	do_pre_smp_initcalls();
3bf77af6e   Frédéric Weisbecker   tracing/ftrace: l...
835
  	start_boot_trace();
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
836

1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
837
838
839
840
  	smp_init();
  	sched_init_smp();
  
  	cpuset_init_smp();
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
841
842
843
844
845
846
  	do_basic_setup();
  
  	/*
  	 * check if there is an early userspace init.  If yes, let it do all
  	 * the work
  	 */
ffdfc4097   Olof Johansson   [PATCH] Add rdini...
847
848
849
850
851
852
  
  	if (!ramdisk_execute_command)
  		ramdisk_execute_command = "/init";
  
  	if (sys_access((const char __user *) ramdisk_execute_command, 0) != 0) {
  		ramdisk_execute_command = NULL;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
853
  		prepare_namespace();
ffdfc4097   Olof Johansson   [PATCH] Add rdini...
854
  	}
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
855
856
857
858
859
860
  
  	/*
  	 * Ok, we have completed the initial bootup, and
  	 * we're essentially up and running. Get rid of the
  	 * initmem segments and start the user-mode stuff..
  	 */
71566a0d1   Frederic Weisbecker   tracing/fastboot:...
861

ee5bfa642   Vivek Goyal   [PATCH] generic: ...
862
863
  	init_post();
  	return 0;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
864
  }