亚洲av成人无遮挡网站在线观看,少妇性bbb搡bbb爽爽爽,亚洲av日韩精品久久久久久,兔费看少妇性l交大片免费,无码少妇一区二区三区

  免費(fèi)注冊(cè) 查看新帖 |

Chinaunix

  平臺(tái) 論壇 博客 文庫(kù)
最近訪問板塊 發(fā)新帖
查看: 33680 | 回復(fù): 0
打印 上一主題 下一主題

[實(shí)踐] 最近看vfs部分,寫了個(gè)筆記, 大家看看.(更新了mount 文件系 統(tǒng)部分) [復(fù)制鏈接]

論壇徽章:
169
申猴
日期:2013-10-09 10:10:16天秤座
日期:2013-10-10 15:28:08天蝎座
日期:2014-07-17 14:02:54丑牛
日期:2014-07-17 14:03:04處女座
日期:2014-07-17 14:03:12雙子座
日期:2014-07-17 14:03:21天秤座
日期:2014-07-17 14:03:29酉雞
日期:2014-07-17 14:03:39金牛座
日期:2014-07-21 10:37:54水瓶座
日期:2014-07-22 16:56:09巳蛇
日期:2014-07-23 11:48:03天蝎座
日期:2014-07-31 10:16:36
跳轉(zhuǎn)到指定樓層
1 [收藏(0)] [報(bào)告]
發(fā)表于 2019-07-26 16:45 |只看該作者 |倒序?yàn)g覽


最近看vfs部分,寫了個(gè)筆記, 大家看看.(更新了mount 文件系 統(tǒng)部分)  原作者:xpl

mount 文件系統(tǒng)

主要數(shù)據(jù)結(jié)構(gòu):

struct namespace {
        atomic_t                count;                /* 引用技術(shù) */
        struct vfsmount *        root;        /* 根目錄的 vfsmount */
        struct list_head        list;        /* 所有已經(jīng)mount的 文件系統(tǒng)的 list */
        struct rw_semaphore        sem;        /* 讀寫信號(hào)量 */
};



struct vfsmount
{
        struct list_head mnt_hash;                /* 用于散列表的list */
        struct vfsmount *mnt_parent;        /* 指向父文件系統(tǒng)的 vfsmount */
        struct dentry *mnt_mountpoint;        /* mountpoint 的 dentry */
        struct dentry *mnt_root;                /* mount的文件系統(tǒng)的根目錄的 dentry */
        struct super_block *mnt_sb;                /* 指向該文件系統(tǒng)的 superblock */
        struct list_head mnt_mounts;        /* 所有mount到該文件系統(tǒng)的 vfsmount 的 list */
        struct list_head mnt_child;                /* 子文件系統(tǒng)的 list  */
        atomic_t mnt_count;                                /* 引用計(jì)數(shù) */
        int mnt_flags;                                        /* mount 的 flag */
        int mnt_expiry_mark;                        /* 到期標(biāo)志 */
        char *mnt_devname;                                /* mount設(shè)備的名稱 比如: /dev/dsk/hda1 */
        struct list_head mnt_list;                /* 在namespace上的vfsmount的 list */
        struct list_head mnt_fslink;        
        struct namespace *mnt_namespace; /* 指向mount文件系統(tǒng)的進(jìn)程的 namespace */
};

struct nameidata {
        struct dentry        *dentry;                /* 目錄的dentry */
        struct vfsmount *mnt;                        /* 目錄所在的文件系統(tǒng)的 vfsmount */
        struct qstr        last;                                /* 在LOOKUP_PARENT 標(biāo)志被設(shè)置時(shí)使用,存儲(chǔ)路徑中最后一個(gè)分量 */
        unsigned int        flags;                        /* 查找的 flags */
        int                last_type;                                /* 在LOOKUP_PARENT 標(biāo)志被設(shè)置時(shí)使用,存儲(chǔ)路徑名的最后一個(gè)分量的類型,值見下面的enum */
        unsigned        depth;                                /* 符號(hào)鏈接的嵌套深度 */
        char *saved_names[MAX_NESTED_LINKS + 1];        /* 存儲(chǔ)嵌套的符號(hào)鏈接所關(guān)聯(lián)的路徑名數(shù)組, 通過depth作為下標(biāo)來索引 */

        /* Intent data */
        union {
                struct open_intent open;
        } intent;
};

/*
* Type of the last component on LOOKUP_PARENT
*/
enum {LAST_NORM, LAST_ROOT, LAST_DOT, LAST_DOTDOT, LAST_BIND};        /* nameidata 的 last_type 的值 */

/*
* The bitmask for a lookup event:
*  - follow links at the end
*  - require a directory
*  - ending slashes ok even for nonexistent files
*  - internal "there are more path compnents" flag
*  - locked when lookup done with dcache_lock held
*  - dentry cache is untrusted; force a real lookup
*/
#define LOOKUP_FOLLOW                 1
#define LOOKUP_DIRECTORY         2
#define LOOKUP_CONTINUE                 4
#define LOOKUP_PARENT                16
#define LOOKUP_NOALT                32
#define LOOKUP_REVAL                64
/*
* Intent data
*/
#define LOOKUP_OPEN                (0x0100)
#define LOOKUP_CREATE                (0x0200)
#define LOOKUP_ACCESS                (0x0400)



/*
* "quick string" -- eases parameter passing, but more importantly
* saves "metadata" about the string (ie length and the hash).
*
* hash comes first so it snuggles against d_parent in the
* dentry.
*/
struct qstr {
        unsigned int hash;                        /* hash值 */
        unsigned int len;                        /* 路徑中分量名稱的長(zhǎng)度 */
        const unsigned char *name;        /* 路徑中分量名稱的字符串地址 */
};


--------------------------------------------------------------------------

需要補(bǔ)習(xí)的內(nèi)容:

對(duì)于一個(gè)文件(在Linux下所有都是文件,包括目錄等等) ,如何判斷該文件 是不是目錄,或者是不是符號(hào)鏈接, 是通過inode :

如果是目錄,則一定有 inode->i_op->lookup 方法, 即 inode->i_op->lookup 一定不是NULL

如果是符號(hào)鏈接, 則一定有 inode->i_op->follow_link 方法,即 inode->i_op->follow_link 一定不是NULL




--------------------------------------------------------------------------

對(duì)于每一個(gè) mount 的文件系統(tǒng),都由一個(gè) vfsmount 實(shí)例來表示。

對(duì)于每一個(gè)進(jìn)程,都有自己的 namespace , 這可以理解為這個(gè)進(jìn)程的地盤。
在這里,所有的文件系統(tǒng)都要掛上來統(tǒng)一管理, 如下所示:
  圖(1)


同時(shí),對(duì)于所有的vfsmount,都存在于 一個(gè)hash table中,他們通過一個(gè) hash 數(shù)組組織在一起:


對(duì)于mount的文件系統(tǒng),會(huì)有在一個(gè)文件系統(tǒng)上 mount 另外一個(gè)文件系統(tǒng)的情況,這種情況,可以稱原文件系統(tǒng)為 父vfsmount, 對(duì)于mount上的文件系統(tǒng),稱之位子文件系統(tǒng)。
他們的關(guān)系如下:

  圖(3)

                                                               

下面我以一個(gè)例子來說明:
例如我們要mount一個(gè)設(shè)備 /dev/hdb1 到 /home/xpl 目錄下
我們假設(shè) /home/xpl 就是當(dāng)前進(jìn)程的根文件系統(tǒng)中的目錄(即 home 和 xpl 都沒有mount任何文件系統(tǒng)),

我們mount的時(shí)候,傳入的參數(shù)有三個(gè): 要mount的設(shè)備( /dev/hdb1 ) , 設(shè)備的文件系統(tǒng) ( ext2 之類的), mount到什么目錄 ( /home/xpl )

首先,我們要根據(jù)要mount的目錄的路徑名( 我們傳入的只是路徑名稱的字符串),來找到mount的目錄 disk 的dentry (即 mountpoint )
這個(gè)查找過程如下所示:
1. 首先確定查找路徑的起始目錄,要么是根目錄,要么是當(dāng)前目錄。
        如果是根目錄,則根目錄的 dentry 和 vfsmount 的信息在: current->fs->root 和 current->fs->rootmnt
        如果是當(dāng)前目錄,則當(dāng)前目錄的 dentry 和 vfsmount 的信息在:current->fs->pwd 和 current->fs->pwdmnt
2. 然后,從路徑的起始目錄開始逐級(jí)的往下找。
        對(duì)于我們的例子,我們首先要查找根目錄下的 home 目錄( 就是要找到 home 目錄的 dentry 和 vfsmount 對(duì)象)
        1) 首先在 hashtable 中查找,就是在上面的圖(2)中的鏈表中找。
        2) 如果這個(gè)目錄沒有在 hashtable 中,則需要到磁盤上(我們假設(shè)根文件系統(tǒng)是在一個(gè)磁盤上,如果不是,就是去根文件系統(tǒng)對(duì)應(yīng)的存儲(chǔ)介質(zhì)中查找)去查找
                        這通過調(diào)用 根目錄的 inode 的 lookup 方法來查找。
                        通過根目錄的 dentry->d_inode 得到根目錄的inode,然后調(diào)用 inode->i_ops->lookup 方法,將要查找目錄的名稱作為參數(shù)傳遞進(jìn)去。

3. 找到了第一個(gè)目錄,下面的過程就是簡(jiǎn)單的遞歸了,最后,找到 目錄 xpl 的 dentry 和 vfsmount

找到了要 mount 的目錄,下面就開始實(shí)際的mount過程
mount的過程就是把設(shè)備的文件系統(tǒng)加入到 vfs 框架中
1. 首先,要mount一個(gè)新的設(shè)備,需要?jiǎng)?chuàng)建一個(gè)新的 super block。
        這通過要mount的文件系統(tǒng)的 file_system_type, 調(diào)用其 get_sb 方法來創(chuàng)建一個(gè)新的 super block

2. 對(duì)于任何一個(gè) mount 的文件系統(tǒng),都要有一個(gè) vfsmount, 創(chuàng)建這個(gè)vfsmount, 并設(shè)置好其屬性(就是 vfsmount 中的各個(gè)成員)

3. 將創(chuàng)建好的 vfsmount 加入到系統(tǒng)中。

整個(gè)過程如下所示:

從這張圖可以看到,三個(gè)目錄:  "/", "home" 和 "xpl" 的dentry
我們新mount的設(shè)備為/dev/hdb1, 新創(chuàng)建了一個(gè)super_block 和 一個(gè) vfsmount (new)
新的super_block 在創(chuàng)建的時(shí)候已經(jīng)加入到整個(gè) vfs 的架構(gòu)中(可參看前面的 super block 一節(jié))
對(duì)于新的vfsmount:
        其mountpoint為 目錄 "xpl" 的dentry,
        其mnt_root 是設(shè)備hdb1上的根目錄的 dentry
        其父 vfsmount 就是原文件系統(tǒng)中的那個(gè) vfsmount
        
同時(shí),我們將新的這個(gè)vfsmount加入到了進(jìn)程的namespace中。

至此,我們已經(jīng)完成了整個(gè)mount的過程。

另外,對(duì)于Linux,可以將多個(gè)文件系統(tǒng) mount 到同一個(gè)目錄上,這樣的話, 新 mount 的文件系統(tǒng)會(huì)覆蓋原來mount的文件系統(tǒng)。
比如我們?cè)侔岩粋(gè) "/dve/hdb2" 的設(shè)備 mount 到 "/home/xpl"  目錄下,
這樣,如果我們?cè)L問 "/home/xpl" 的時(shí)候,就會(huì)訪問到 "/dev/hdb2"
當(dāng)新mount的文件系統(tǒng)被 unmount 之后,原來被覆蓋的文件系統(tǒng)就會(huì)再次顯露出來了。

這個(gè)過程的之所以是這樣的,是因?yàn)樵诼窂讲檎业臅r(shí)候,如果發(fā)現(xiàn)要查找的目錄上mount了 文件系統(tǒng)(其dentry的d_mounted 不為0),會(huì)切換文件系統(tǒng)。
比如我們上面的例子,當(dāng)路徑查找 "/home/xpl" 的時(shí)候,當(dāng)查找到根文件系統(tǒng)的目錄 "xpl" 的dentry時(shí),發(fā)現(xiàn)有一個(gè)文件系統(tǒng)已經(jīng)mount到這個(gè)目錄上,就會(huì)切換文件系統(tǒng),進(jìn)而找到 "/dev/hdb1" 的根目錄的 dentry
這樣,路徑查找"/home/xpl" 返回的時(shí)候,返回的是 "/dve/hdb1" 根目錄的 dentry

這樣,在我們 mount 第二個(gè)文件系統(tǒng) "/dev/hdb2" 的時(shí)候,其mountpoint 就是 "/dev/hdb1" 的根目錄的 dentry。
這個(gè)過程如下圖所示:


以此類推,如果在該目錄上再mount一個(gè)新的文件系統(tǒng),基本邏輯是相同的。
因此在路徑查找過程中,每查找到一個(gè)新的目錄,都要對(duì)該目錄的d_mounted 進(jìn)行判斷,看是否在該目錄上mount的了文件系統(tǒng),如果是,則要切換文件系統(tǒng)(切換查找過程中的 dentry 和 vfsmount)
這個(gè)過程是要遞歸的,即如果一個(gè)路徑上mount了多個(gè)文件系統(tǒng)(如上面的例子),要遞歸到最后一個(gè)mount的文件系統(tǒng)(其dentry 的 d_mounted 為0)為止。



對(duì)于路徑查找,我們補(bǔ)充說明一下兩種特殊的情況:

一、路徑中的 "." 和 ".."
        對(duì)于 "." ,很明顯,就是當(dāng)前目錄,不需要額外的處理,簡(jiǎn)單跳過即可
         例如: /home/./xpl 當(dāng)查找到 "." 的時(shí)候,還是 home 目錄,因而 "." 的 dentry 和vfsmount 還是 目錄 home的 dentry 和 vfsmount
        對(duì)于 "..", 這個(gè)需要跳到上一級(jí)目錄,在這里,要注意:
        1. 如果發(fā)現(xiàn)已經(jīng)沒法向上了,就不再向上,而保持當(dāng)前的路徑。比如:"/../" ,已經(jīng)是根目錄了,返回的結(jié)果仍然是根目錄。
        2. 如果發(fā)現(xiàn)當(dāng)前的 dentry 是當(dāng)前文件系統(tǒng)的根目錄,并且該文件系統(tǒng)是mount到其他文件系統(tǒng)上的,這個(gè)時(shí)候就要反溯文件系統(tǒng),要切換到原來mount的文件系統(tǒng)后,再向上一級(jí)目錄。這個(gè)反溯過程也是要遞歸的。
                比如我們上面的例子中:"/home/xpl" 先mount了一個(gè) hdb1 的文件系統(tǒng),然后又mount 了一個(gè) hdb2 的文件系統(tǒng)。
                在執(zhí)行".." 查找的時(shí)候:
                        1)我們發(fā)現(xiàn)當(dāng)前的目錄是hdb2 的根目錄(vfsmount->mnt_root),并且mount到了hdb1上(vfsmount != vfsmount->parent),這個(gè)時(shí)候我們就要先切換到 hdb1 的文件系統(tǒng)中, 此時(shí)vfsmount 換為hdb1 的vfsmount, dentry換成 hdb2的mountpoint。
                        2) 接下來要遞歸上面的文件系統(tǒng)切換,我們發(fā)現(xiàn) 當(dāng)前的dentry 是hdb1文件系統(tǒng)的根目錄,并且mount到了別的文件系統(tǒng)上,這個(gè)時(shí)候就要繼續(xù)切換。
                        3) 當(dāng)所有的遞歸完成以后,我們得到了根文件系統(tǒng)的 xpl目錄的dentry,然后再執(zhí)行向上的操作(".."),最后得到 "/home"

        
二、路徑中的符號(hào)鏈接
        如果路徑中有符號(hào)鏈接,我們要跟隨符號(hào)鏈接的路徑。其實(shí)這個(gè)過程只是一個(gè)簡(jiǎn)單的遞歸。
        在路徑查找中,我們得到一個(gè)dentry后,要判斷這個(gè)dentry是否是一個(gè)符號(hào)鏈接。通過判斷 dentry->d_inode->i_op->follow_link 是否是NULL,即是否有follow_link方法來判斷是不是符號(hào)鏈接。
        如果是符號(hào)鏈接的話,我們需要通過這個(gè)inode的follow_link 來獲得鏈接的路徑。
        例如有這么一個(gè)符號(hào)鏈接: /home/xpl/link -> /mnt/disk
        我們會(huì)獲得到其鏈接的路徑  "/mnt/disk", 接下來就是繼續(xù)解析鏈接的路徑名,即查找 "/mnt/disk", 這個(gè)查找過程和普通的查找是一樣的,遞歸下去,最終找到需要的目標(biāo)。

您需要登錄后才可以回帖 登錄 | 注冊(cè)

本版積分規(guī)則 發(fā)表回復(fù)

  

北京盛拓優(yōu)訊信息技術(shù)有限公司. 版權(quán)所有 京ICP備16024965號(hào)-6 北京市公安局海淀分局網(wǎng)監(jiān)中心備案編號(hào):11010802020122 niuxiaotong@pcpop.com 17352615567
未成年舉報(bào)專區(qū)
中國(guó)互聯(lián)網(wǎng)協(xié)會(huì)會(huì)員  聯(lián)系我們:huangweiwei@itpub.net
感謝所有關(guān)心和支持過ChinaUnix的朋友們 轉(zhuǎn)載本站內(nèi)容請(qǐng)注明原作者名及出處

清除 Cookies - ChinaUnix - Archiver - WAP - TOP