- 论坛徽章:
- 16
|
在linux 2.6.11的代码中sys_umount
sys_umount
|---->__user_walk
| |-------->path_lookup
|
|---->do_umount
|--------->umount_tree
首先调用path_lookup获得挂载点相关的nameidata,然后调用umount_tree卸载文件系统(及其所有子系统)。
但是对umount_tree有个疑问,从代码中的for循环来看,这里使用的mnt应该是从path_lookup得到的,也就是nd.mnt
这就是父文件系统的mnt,怎么把这个删掉呢???- 340void umount_tree(struct vfsmount *mnt)
- 341{
- 342 struct vfsmount *p;
- 343 LIST_HEAD(kill);
- 344
- 345 for (p = mnt; p; p = next_mnt(p, mnt)) {
- 346 list_del(&p->mnt_list);
- 347 list_add(&p->mnt_list, &kill);
- 348 }
- 349
- 350 while (!list_empty(&kill)) {
- 351 mnt = list_entry(kill.next, struct vfsmount, mnt_list);
- 352 list_del_init(&mnt->mnt_list);
- 353 list_del_init(&mnt->mnt_fslink);
- 354 if (mnt->mnt_parent == mnt) {
- 355 spin_unlock(&vfsmount_lock);
- 356 } else {
- 357 struct nameidata old_nd;
- 358 detach_mnt(mnt, &old_nd);
- 359 spin_unlock(&vfsmount_lock);
- 360 path_release(&old_nd);
- 361 }
- 362 mntput(mnt);
- 363 spin_lock(&vfsmount_lock);
- 364 }
- 365}
复制代码 从detach_mnt的第116~121行来看,这个mnt又貌似是指子文件系统的vfsmount,而不是父文件系统的vfsmount,
但是第122行,减少dentry->d_mounted,应该是父文件系统挂载点的dentry的d_mounted减少,而不是子文件系统的
这段代码为什么这么矛盾呢,这里的mnt到底是父文件系统的,还是子文件系统的呢?- 114static void detach_mnt(struct vfsmount *mnt, struct nameidata *old_nd)
- 115{
- 116 old_nd->dentry = mnt->mnt_mountpoint;
- 117 old_nd->mnt = mnt->mnt_parent;
- 118 mnt->mnt_parent = mnt;
- 119 mnt->mnt_mountpoint = mnt->mnt_root;
- 120 list_del_init(&mnt->mnt_child);
- 121 list_del_init(&mnt->mnt_hash);
- 122 old_nd->dentry->d_mounted--;
- 123}
复制代码 |
|