挂载VFS是在mnt_init(见第一部分)里,分两步:
1、init_rootfs();
2、init_mount_tree();
init_rootfs:
int__init init_rootfs(void)
{
returnregister_filesystem(&rootfs_fs_type);
}
简单明了,这个是文件系统初始化的第一步,将rootfs挂到大表下
init_mount_tree:
static void __initinit_mount_tree(void)
{
structvfsmount *mnt;
structmnt_namespace *ns;
mnt= do_kern_mount("rootfs", 0, "rootfs",NULL); //初始化rootfs这个文件系统,返回mnt为所需的格式
if(IS_ERR(mnt))
panic("Can'tcreate rootfs");
ns= kmalloc(sizeof(*ns), GFP_KERNEL);
if(!ns)
panic("Can'tallocate initial namespace");
atomic_set(&ns->count,1);
INIT_LIST_HEAD(&ns->list);
init_waitqueue_head(&ns->poll);
ns->event= 0;
list_add(&mnt->mnt_list,&ns->list);
ns->root= mnt; //指向mnt
mnt->mnt_ns= ns;
init_task.nsproxy->mnt_ns= ns;
get_mnt_ns(ns);
set_fs_pwd(current->fs,ns->root,ns->root->mnt_root); //设置当前pwd为ns->root
//fs->pwdmnt=mnt fs->pwd=dentry
set_fs_root(current->fs,ns->root, ns->root->mnt_root);//设置当前root为ns->root
//fs->rootmnt=mnt fs->root=dentry
//mnt_root:superblock
//root:mnt
//root挂至fs->rootmnt;mnt_root挂至fs->root
}
个人觉得最关键的代码是
set_fs_pwd(current->fs,ns->root, ns->root->mnt_root);
set_fs_root(current->fs, ns->root,ns->root->mnt_root);
这两行代码鉴定了VFS的基础。由于从前面一篇的超级块中可以得知,返回的mnt中,它的mnt_root就是sb->s_root,即在获得超级块过程中的第一个dentry。