挂载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。