设为首页 - 加入收藏 ASP站长网(Aspzz.Cn)- 科技、建站、经验、云计算、5G、大数据,站长网!
热搜: 手机 数据 公司
当前位置: 首页 > 服务器 > 安全 > 正文

深度解析Linux根文件系统的挂载过程(3)

发布时间:2020-12-30 15:54 所属栏目:53 来源:网络整理
导读:从代码中可以看中,会依次执行指定的init文件,如果失败,就会执行/sbin/init,/etc/init,/bin/init,/bin/sh 注意的是,run_init_process在调用相应程序运行的时候,用的是kernel_execve.也就是说调用进程会替换当前进程.

从代码中可以看中,会依次执行指定的init文件,如果失败,就会执行/sbin/init,/etc/init,/bin/init,/bin/sh
注意的是,run_init_process在调用相应程序运行的时候,用的是kernel_execve.也就是说调用进程会替换当前进程.只要上述任意一个文件调用成功,就不会返回到这个函数.如果上面几个文件都无法执行.打印出没有找到init文件的错误.
对于image-hdr或者是虚拟文件系统中没有包含 /init的情况,会由prepare_namespace()处理.代码如下:

[code lang=”c”]void __init prepare_namespace(void)
{
int is_floppy;

if (root_delay) {
printk(KERN_INFO "Waiting %dsec before mounting root device…\n",
root_delay);
ssleep(root_delay);
}

/* wait for the known devices to complete their probing */
while (driver_probe_done() != 0)
msleep(100);

//mtd的处理
md_run_setup();

if (saved_root_name[0]) {
root_device_name = saved_root_name;
if (!strncmp(root_device_name,"mtd",3)) {
mount_block_root(root_device_name,root_mountflags);
goto out;
}
ROOT_DEV = name_to_dev_t(root_device_name);
if (strncmp(root_device_name,"/dev/",5) == 0)
root_device_name += 5;
}

if (initrd_load())
goto out;

/* wait for any asynchronous scanning to complete */
if ((ROOT_DEV == 0) && root_wait) {
printk(KERN_INFO "Waiting for root device %s…\n",
saved_root_name);
while (driver_probe_done() != 0 ||
(ROOT_DEV = name_to_dev_t(saved_root_name)) == 0)
msleep(100);
}

is_floppy = MAJOR(ROOT_DEV) == FLOPPY_MAJOR;

if (is_floppy && rd_doload && rd_load_disk(0))
ROOT_DEV = Root_RAM0;

mount_root();
out:
sys_mount(".","/",NULL,MS_MOVE,NULL);
sys_chroot(".");
}[/code]

这里有几个比较有意思的处理,首先用户可以用root=来指定根文件系统.它的值保存在saved_root_name中.如果用户指定了以mtd开始的字串做为它的根文件系统.就会直接去挂载.这个文件是mtdblock的设备文件.
否则将设备结点文件转换为ROOT_DEV即设备节点号
然后,转向initrd_load()执行initrd预处理后,再将具体的根文件系统挂载.
注意到,在这个函数末尾.会调用sys_mount()来移动当前文件系统挂载点到”/”目录下.然后将根目录切换到当前目录.这样,根文件系统的挂载点就成为了我们在用户空间所看到的”/”了.
对于其它根文件系统的情况,会先经过initrd的处理.即

[code lang=”c”]int __init initrd_load(void)
{
if (mount_initrd) {
create_dev("/dev/ram",Root_RAM0);
/*
* Load the initrd data into /dev/ram0. Execute it as initrd
* unless /dev/ram0 is supposed to be our actual root device,
* in that case the ram disk is just set up here,and gets
* mounted in the normal path.
*/
if (rd_load_image("/initrd.image") && ROOT_DEV != Root_RAM0) {
sys_unlink("/initrd.image");
handle_initrd();
return 1;
}
}
sys_unlink("/initrd.image");
return 0;
}[/code]

建立一个ROOT_RAM)的设备节点,并将/initrd/.image释放到这个节点中,/initrd.image的内容,就是我们之前分析的image-initrd.
如果根文件设备号不是ROOT_RAM0( 用户指定的根文件系统不是/dev/ram0就会转入到handle_initrd()
如果当前根文件系统是/dev/ram0.将其直接挂载就好了.

handle_initrd()代码如下:

[code lang=”c”]static void __init handle_initrd(void)
{
int error;
int pid;

(编辑:ASP站长网)

网友评论
推荐文章
    热点阅读