linux进程的启动分析:
1.分析start_kernel函数
路径 linux-2.6.22.6/init/main.c
start_kernel(void)调用
2.分析rest_init函数
路径 linux-2.6.22.6/init/main.c
static void noinline __init_refok rest_init(void)
__releases(kernel_lock)
{
int pid;
函数里面创建内核线程
kernel_thread(kernel_init, NULL, CLONE_FS | CLONE_SIGHAND);
kernel_init:创建的线程对应的函数名
....
....
....
会调度内核线程的
schedule();
preempt_disable();
/* Call into cpu_idle with preempt disabled */
cpu_idle();
}
3.分析内核线程:kernel_init
路径 linux-2.6.22.6/init/main.c
static int __init kernel_init(void * unused)
{
...
...
init_post();
return 0;
}
4.分析init_post函数
路径: linux-2.6.22.6/init/main.c
static int noinline init_post(void)
{
...
...
打开标准输入,标准输出,标准错误设备
linux中最先打开的3个文件分别为标准输入,标准输出 ,标准错误,它们对应的
设备描述符是0,1,2.标准输入就是程序中使用的scanf(),fscanf(stdin,...)获取数据
。标准输出,标准错误都是输出设备 前者对应是是printf,fprintf(stdout,...),后者对应fprintf(stderr,..).
if (sys_open((const char __user *) "/dev/console", O_RDWR, 0) < 0)
printk(KERN_WARNING "Warning: unable to open an initial console.\n");
将文件描述符0复制给文件描述符1,2,所以标准输入,标准输出,标准错误,都对应
同一个文件(设备)。
(void) sys_dup(0);
(void) sys_dup(0);
ramdisk_execute_command为空,不执行。
if (ramdisk_execute_command) { 判断变量ramdisk_execute_command是否指定的运行程序。
run_init_process(ramdisk_execute_command);
printk(KERN_WARNING "Failed to execute %s\n",
ramdisk_execute_command);
}
启动可执行程序函数:
static void run_init_process(char *init_filename)
{
argv_init[0] = init_filename;
kernel_execve(init_filename, argv_init, envp_init)
}
/*
* 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.
*/
变量execute_command为空,不执行。
if (execute_command) { 判断变量execute_command是否指定的运行程序。
run_init_process(execute_command);
printk(KERN_WARNING "Failed to execute %s. Attempting "
"defaults...\n", execute_command);
}
run_init_process("/sbin/init");//如果/sbin/init存在,系统的控制权交给/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.");
}