Linux 中容許衆多不一樣的文件系統共存,如 ext2, ext3, vfat 等。經過使用同一套文件 I/O 系統 調用便可對 Linux 中的任意文件進行操做而無需考慮其所在的具體文件系統格式;更進一步,對文件的 操做能夠跨文件系統而執行。如圖 1 所示,咱們可使用 cp 命令從 vfat 文件系統格式的硬盤拷貝數據到 ext3 文件系統格式的硬盤;而這樣的操做涉及到兩個不一樣的文件系統。html
「一切皆是文件」是 Unix/Linux 的基本哲學之一。不只普通的文件,目錄、字符設備、塊設備、 套接字等在 Unix/Linux 中都是以文件被對待;它們雖然類型不一樣,可是對其提供的倒是同一套操做界面。node
而虛擬文件系統正是實現上述兩點 Linux 特性的關鍵所在。虛擬文件系統(Virtual File System, 簡稱 VFS), 是 Linux 內核中的一個軟件層,用於給用戶空間的程序提供文件系統接口;同時,它也提供了內核中的一個 抽象功能,容許不一樣的文件系統共存。系統中全部的文件系統不但依賴 VFS 共存,並且也依靠 VFS 協同工做。linux
爲了可以支持各類實際文件系統,VFS 定義了全部文件系統都支持的基本的、概念上的接口和數據 結構;同時實際文件系統也提供 VFS 所指望的抽象接口和數據結構,將自身的諸如文件、目錄等概念在形式 上與VFS的定義保持一致。換句話說,一個實際的文件系統想要被 Linux 支持,就必須提供一個符合VFS標準 的接口,才能與 VFS 協同工做。實際文件系統在統一的接口和數據結構下隱藏了具體的實現細節,因此在VFS 層和內核的其餘部分看來,全部文件系統都是相同的。圖3顯示了VFS在內核中與實際的文件系統的協同關係。編程
咱們已經知道,正是因爲在內核中引入了VFS,跨文件系統的文件操做才能實現,「一切皆是文件」 的口號才能承諾。而爲何引入了VFS,就能實現這兩個特性呢?在接下來,咱們將以這樣的一個思路來切入 文章的正題:咱們將先簡要介紹下用以描述VFS模型的一些數據結構,總結出這些數據結構相互間的關係;而後 選擇兩個具備表明性的文件I/O操做sys_open()和sys_read()來詳細說明內核是如何藉助VFS和具體的文件系統打 交道以實現跨文件系統的文件操做和承諾「一切皆是文件」的口號。數組
從本質上講,文件系統是特殊的數據分層存儲結構,它包含文件、目錄和相關的控制信息。爲了描述 這個結構,Linux引入了一些基本概念:緩存
文件 一組在邏輯上具備完整意義的信息項的系列。在Linux中,除了普通文件,其餘諸如目錄、設備、套接字等 也以文件被對待。總之,「一切皆文件」。數據結構
目錄 目錄比如一個文件夾,用來容納相關文件。由於目錄能夠包含子目錄,因此目錄是能夠層層嵌套,造成 文件路徑。在Linux中,目錄也是以一種特殊文件被對待的,因此用於文件的操做一樣也能夠用在目錄上。app
目錄項 在一個文件路徑中,路徑中的每一部分都被稱爲目錄項;如路徑/home/source/helloworld.c中,目錄 /, home, source和文件 helloworld.c都是一個目錄項。函數
索引節點 用於存儲文件的元數據的一個數據結構。文件的元數據,也就是文件的相關信息,和文件自己是兩個不一樣 的概念。它包含的是諸如文件的大小、擁有者、建立時間、磁盤位置等和文件相關的信息。atom
超級塊 用於存儲文件系統的控制信息的數據結構。描述文件系統的狀態、文件系統類型、大小、區塊數、索引節 點數等,存放於磁盤的特定扇區中。
如上的幾個概念在磁盤中的位置關係如圖4所示。
關於文件系統的三個易混淆的概念:
建立 以某種方式格式化磁盤的過程就是在其之上創建一個文件系統的過程。建立文現系統時,會在磁盤的特定位置寫入 關於該文件系統的控制信息。
註冊 向內核報到,聲明本身能被內核支持。通常在編譯內核的時侯註冊;也能夠加載模塊的方式手動註冊。註冊過程實 際上是將表示各實際文件系統的數據結構struct file_system_type 實例化。
安裝 也就是咱們熟悉的mount操做,將文件系統加入到Linux的根文件系統的目錄樹結構上;這樣文件系統才能被訪問。
VFS依靠四個主要的數據結構和一些輔助的數據結構來描述其結構信息,這些數據結構表現得就像是對象; 每一個主要對象中都包含由操做函數表構成的操做對象,這些操做對象描述了內核針對這幾個主要的對象能夠進行的操做。
存儲一個已安裝的文件系統的控制信息,表明一個已安裝的文件系統;每次一個實際的文件系統被安裝時, 內核會從磁盤的特定位置讀取一些控制信息來填充內存中的超級塊對象。一個安裝實例和一個超級塊對象一一對應。 超級塊經過其結構中的一個域s_type記錄它所屬的文件系統類型。
根據第三部分追蹤源代碼的須要,如下是對該超級塊結構的部分相關成員域的描述,(以下同):
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
|
struct super_block { //超級塊數據結構
struct list_head s_list; /*指向超級塊鏈表的指針*/
……
struct file_system_type *s_type; /*文件系統類型*/
struct super_operations *s_op; /*超級塊方法*/
……
struct list_head s_instances; /*該類型文件系統*/
……
};
struct super_operations { //超級塊方法
……
//該函數在給定的超級塊下建立並初始化一個新的索引節點對象
struct inode *(*alloc_inode)(struct super_block *sb);
……
//該函數從磁盤上讀取索引節點,並動態填充內存中對應的索引節點對象的剩餘部分
void (*read_inode) (struct inode *);
……
};
|
索引節點對象存儲了文件的相關信息,表明了存儲設備上的一個實際的物理文件。當一個 文件首次被訪問時,內核會在內存中組裝相應的索引節點對象,以便向內核提供對一個文件進行操 做時所必需的所有信息;這些信息一部分存儲在磁盤特定位置,另一部分是在加載時動態填充的。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
|
struct inode {//索引節點結構
……
struct inode_operations *i_op; /*索引節點操做表*/
struct file_operations *i_fop; /*該索引節點對應文件的文件操做集*/
struct super_block *i_sb; /*相關的超級塊*/
……
};
struct inode_operations { //索引節點方法
……
//該函數爲dentry對象所對應的文件建立一個新的索引節點,主要是由open()系統調用來調用
int (*create) (struct inode *,struct dentry *,int, struct nameidata *);
//在特定目錄中尋找dentry對象所對應的索引節點
struct dentry * (*lookup) (struct inode *,struct dentry *, struct nameidata *);
……
};
|
引入目錄項的概念主要是出於方便查找文件的目的。一個路徑的各個組成部分,無論是目錄仍是 普通的文件,都是一個目錄項對象。如,在路徑/home/source/test.c中,目錄 /, home, source和文件 test.c都對應一個目錄項對象。不一樣於前面的兩個對象,目錄項對象沒有對應的磁盤數據結構,VFS在遍 歷路徑名的過程當中現場將它們逐個地解析成目錄項對象。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
|
struct dentry {//目錄項結構
……
struct inode *d_inode; /*相關的索引節點*/
struct dentry *d_parent; /*父目錄的目錄項對象*/
struct qstr d_name; /*目錄項的名字*/
……
struct list_head d_subdirs; /*子目錄*/
……
struct dentry_operations *d_op; /*目錄項操做表*/
struct super_block *d_sb; /*文件超級塊*/
……
};
struct dentry_operations {
//判斷目錄項是否有效;
int (*d_revalidate)(struct dentry *, struct nameidata *);
//爲目錄項生成散列值;
int (*d_hash) (struct dentry *, struct qstr *);
……
};
|
文件對象是已打開的文件在內存中的表示,主要用於創建進程和磁盤上的文件的對應關係。它由sys_open() 現場建立,由sys_close()銷燬。文件對象和物理文件的關係有點像進程和程序的關係同樣。當咱們站在用戶空間來看 待VFS,咱們像是隻需與文件對象打交道,而無須關心超級塊,索引節點或目錄項。由於多個進程能夠同時打開和操做 同一個文件,因此同一個文件也可能存在多個對應的文件對象。文件對象僅僅在進程觀點上表明已經打開的文件,它 反過來指向目錄項對象(反過來指向索引節點)。一個文件對應的文件對象可能不是唯一的,可是其對應的索引節點和 目錄項對象無疑是唯一的。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
|
struct file {
……
struct list_head f_list; /*文件對象鏈表*/
struct dentry *f_dentry; /*相關目錄項對象*/
struct vfsmount *f_vfsmnt; /*相關的安裝文件系統*/
struct file_operations *f_op; /*文件操做表*/
……
};
struct file_operations {
……
//文件讀操做
ssize_t (*read) (struct file *, char __user *, size_t, loff_t *);
……
//文件寫操做
ssize_t (*write) (struct file *, const char __user *, size_t, loff_t *);
……
int (*readdir) (struct file *, void *, filldir_t);
……
//文件打開操做
int (*open) (struct inode *, struct file *);
……
};
|
根據文件系統所在的物理介質和數據在物理介質上的組織方式來區分不一樣的文件系統類型的。 file_system_type結構用於描述具體的文件系統的類型信息。被Linux支持的文件系統,都有且僅有一 個file_system_type結構而無論它有零個或多個實例被安裝到系統中。
而與此對應的是每當一個文件系統被實際安裝,就有一個vfsmount結構體被建立,這個結構體對應一個安裝點。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
|
struct file_system_type {
const char *name; /*文件系統的名字*/
struct subsystem subsys; /*sysfs子系統對象*/
int fs_flags; /*文件系統類型標誌*/
/*在文件系統被安裝時,從磁盤中讀取超級塊,在內存中組裝超級塊對象*/
struct super_block *(*get_sb) (struct file_system_type*,
int, const char*, void *);
void (*kill_sb) (struct super_block *); /*終止訪問超級塊*/
struct module *owner; /*文件系統模塊*/
struct file_system_type * next; /*鏈表中的下一個文件系統類型*/
struct list_head fs_supers; /*具備同一種文件系統類型的超級塊對象鏈表*/
};
struct vfsmount
{
struct list_head mnt_hash; /*散列表*/
struct vfsmount *mnt_parent; /*父文件系統*/
struct dentry *mnt_mountpoint; /*安裝點的目錄項對象*/
struct dentry *mnt_root; /*該文件系統的根目錄項對象*/
struct super_block *mnt_sb; /*該文件系統的超級塊*/
struct list_head mnt_mounts; /*子文件系統鏈表*/
struct list_head mnt_child; /*子文件系統鏈表*/
atomic_t mnt_count; /*使用計數*/
int mnt_flags; /*安裝標誌*/
char *mnt_devname; /*設備文件名*/
struct list_head mnt_list; /*描述符鏈表*/
struct list_head mnt_fslink; /*具體文件系統的到期列表*/
struct namespace *mnt_namespace; /*相關的名字空間*/
};
|
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
|
struct files_struct {//打開的文件集
atomic_t count; /*結構的使用計數*/
……
int max_fds; /*文件對象數的上限*/
int max_fdset; /*文件描述符的上限*/
int next_fd; /*下一個文件描述符*/
struct file ** fd; /*所有文件對象數組*/
……
};
struct fs_struct {//創建進程與文件系統的關係
atomic_t count; /*結構的使用計數*/
rwlock_t lock; /*保護該結構體的鎖*/
int umask; /*默認的文件訪問權限*/
struct dentry * root; /*根目錄的目錄項對象*/
struct dentry * pwd; /*當前工做目錄的目錄項對象*/
struct dentry * altroot; /*可供選擇的根目錄的目錄項對象*/
struct vfsmount * rootmnt; /*根目錄的安裝點對象*/
struct vfsmount * pwdmnt; /*pwd的安裝點對象*/
struct vfsmount * altrootmnt;/*可供選擇的根目錄的安裝點對象*/
};
|
1
2
3
4
5
6
7
8
9
10
11
12
13
|
struct nameidata {
struct dentry *dentry; /*目錄項對象的地址*/
struct vfsmount *mnt; /*安裝點的數據*/
struct qstr last; /*路徑中的最後一個component*/
unsigned int flags; /*查找標識*/
int last_type; /*路徑中的最後一個component的類型*/
unsigned depth; /*當前symbolic link的嵌套深度,不能大於6*/
char *saved_names[MAX_NESTED_LINKS + 1];/
/*和嵌套symbolic link 相關的pathname*/
union {
struct open_intent open; /*說明文件該如何訪問*/
} intent; /*專用數據*/
};
|
如上的數據結構並非孤立存在的。正是經過它們的有機聯繫,VFS才能正常工做。以下的幾張圖是對它們之間的聯繫的描述。
如圖5所示,被Linux支持的文件系統,都有且僅有一個file_system_type結構而無論它有零個或多個實例被安裝到系統 中。每安裝一個文件系統,就對應有一個超級塊和安裝點。超級塊經過它的一個域s_type指向其對應的具體的文件系統類型。具體的 文件系統經過file_system_type中的一個域fs_supers連接具備同一種文件類型的超級塊。同一種文件系統類型的超級塊經過域s_instances鏈 接。
從圖6可知:進程經過task_struct中的一個域files_struct files來了解它當前所打開的文件對象;而咱們一般所說的文件 描述符實際上是進程打開的文件對象數組的索引值。文件對象經過域f_dentry找到它對應的dentry對象,再由dentry對象的域d_inode找 到它對應的索引結點,這樣就創建了文件對象與實際的物理文件的關聯。最後,還有一點很重要的是, 文件對象所對應的文件操做函數 列表是經過索引結點的域i_fop獲得的。圖6對第三部分源碼的理解起到很大的做用。
到目前爲止,文章主要都是從理論上來說述VFS的運行機制;接下來咱們將深刻源代碼層中,經過闡述兩個具備表明性的系統 調用sys_open()和sys_read()來更好地理解VFS向具體文件系統提供的接口機制。因爲本文更關注的是文件操做的整個流程體制,因此我 們在追蹤源代碼時,對一些細節性的處理不予關心。又因爲篇幅所限,只列出相關代碼。本文中的源代碼來自於linux-2.6.17內核版本。
在深刻sys_open()和sys_read()以前,咱們先概覽下調用sys_read()的上下文。圖7描述了從用戶空間的read()調用到數據從 磁盤讀出的整個流程。當在用戶應用程序調用文件I/O read()操做時,系統調用sys_read()被激發,sys_read()找到文件所在的具體文件 系統,把控制權傳給該文件系統,最後由具體文件系統與物理介質交互,從介質中讀出數據。
sys_open()系統調用打開或建立一個文件,成功返回該文件的文件描述符。圖8是sys_open()實現代碼中主要的函數調用關係圖。
因爲sys_open()的代碼量大,函數調用關係複雜,如下主要是對該函數作總體的解析;而對其中的一些關鍵點,則列出其關鍵代碼。
a. 從sys_open()的函數調用關係圖能夠看到,sys_open()在作了一些簡單的參數檢驗後,就把接力棒傳給do_sys_open():
1)、首先,get_unused_fd()獲得一個可用的文件描述符;經過該函數,可知文件描述符實質是進程打開文件列表中對應某個文件對象的索引值;
2)、接着,do_filp_open()打開文件,返回一個file對象,表明由該進程打開的一個文件;進程經過這樣的一個數據結構對物理文件進行讀寫操做。
3)、最後,fd_install()創建文件描述符與file對象的聯繫,之後進程對文件的讀寫都是經過操縱該文件描述符而進行。
b. do_filp_open()用於打開文件,返回一個file對象;而打開以前須要先找到該文件:
1)、open_namei()用於根據文件路徑名查找文件,藉助一個持有路徑信息的數據結構nameidata而進行;
2)、查找結束後將填充有路徑信息的nameidata返回給接下來的函數nameidata_to_filp()從而獲得最終的file對象;當達到目的後,nameidata這個數據結構將會立刻被釋放。
c.open_namei()用於查找一個文件:
1)、path_lookup_open()實現文件的查找功能;要打開的文件若不存在,還須要有一個新建的過程,則調用path_lookup_create(),後者和前者封裝的是同一個實際的路徑查找函數,只是參數不同,使它們在處理細節上有所誤差;
2)、當是以新建文件的方式打開文件時,即設置了O_CREAT標識時須要建立一個新的索引節點,表明建立一個文件。在vfs_create()裏的一句核心語句dir->i_op->create(dir, dentry, mode, nd)可知它調用了具體的文件系統所提供的建立索引節點的方法。注意:這邊的索引節點的概念,還只是位於內存之中,它和磁盤上的物理的索引節點的關係就像位於內存中和位於磁盤中的文件同樣。此時新建的索引節點還不能徹底標誌一個物理文件的成功建立,只有當把索引節點回寫到磁盤上纔是一個物理文件的真正建立。想一想咱們以新建的方式打開一個文件,對其讀寫但最終沒有保存而關閉,則位於內存中的索引節點會經歷重新建到消失的過程,而磁盤卻始終不知道有人曾經想過建立一個文件,這是由於索引節點沒有回寫的緣故。
3)、path_to_nameidata()填充nameidata數據結構;
4)、may_open()檢查是否能夠打開該文件;一些文件如連接文件和只有寫權限的目錄是不能被打開的,先檢查nd->dentry->inode所指的文件是不是這一類文件,是的話則錯誤返回。還有一些文件是不能以TRUNC的方式打開的,若nd->dentry->inode所指的文件屬於這一類,則顯式地關閉TRUNC標誌位。接着若是有以TRUNC方式打開文件的,則更新nd->dentry->inode的信息
無論是path_lookup_open()仍是path_lookup_create()最終都是調用__path_lookup_intent_open()來實現查找文件的功能。 查找時,在遍歷路徑的過程當中,會逐層地將各個路徑組成部分解析成目錄項對象,若是此目錄項對象在目錄項緩存中,則直接從緩存中得到;若是該目錄項在緩存中不存在,則進行一次實際的讀盤操做,從磁盤中讀取該目錄項所對應的索引節點。獲得索引節點後,則創建索引節點與該目錄項的聯繫。如此循環,直到最終找到目標文件對應的目錄項,也就找到了索引節點,而由索引節點找到對應的超級塊對象就可知道該文件所在的文件系統的類型。 從磁盤中讀取該目錄項所對應的索引節點;這將引起VFS和實際的文件系統的一次交互。從前面的VFS理論介紹可知,讀索引節點方法是由超級塊來提供的。而當安裝一個實際的文件系統時,在內存中建立的超級塊的信息是由一個實際文件系統的相關信息來填充的,這裏的相關信息就包括了實際文件系統所定義的超級塊的操做函數列表,固然也就包括了讀索引節點的具體執行方式。 當繼續追蹤一個實際文件系統ext3的ext3_read_inode()時,可發現這個函數很重要的一個工做就是爲不一樣的文件類型設置不一樣的索引節點操做函數表和文件操做函數表。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
|
void ext3_read_inode(struct inode * inode)
{
……
//是普通文件
if (S_ISREG(inode->i_mode)) {
inode->i_op = &ext3_file_inode_operations;
inode->i_fop = &ext3_file_operations;
ext3_set_aops(inode);
} else if (S_ISDIR(inode->i_mode)) {
//是目錄文件
inode->i_op = &ext3_dir_inode_operations;
inode->i_fop = &ext3_dir_operations;
} else if (S_ISLNK(inode->i_mode)) {
// 是鏈接文件
……
} else {
// 若是以上三種狀況都排除了,則是設備驅動
//這裏的設備還包括套結字、FIFO等僞設備
……
}
|
這是VFS與實際的文件系統聯繫的一個關鍵點。從3.1.1小節分析中可知,調用實際文件系統讀取索引節點的方法讀取索引節點時,實際文件系統會根據文件的不一樣類型賦予索引節點不一樣的文件操做函數集,如普通文件有普通文件對應的一套操做函數,設備文件有設備文件對應的一套操做函數。這樣當把對應的索引節點的文件操做函數集賦予文件對象,之後對該文件進行操做時,好比讀操做,VFS雖然對各類不一樣文件都是執行同一個read()操做界面,可是真正讀時,內核卻知道怎麼區分對待不一樣的文件類型。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
|
static struct file *__dentry_open(struct dentry *dentry, struct vfsmount *mnt,
int flags, struct file *f,
int (*open)(struct inode *, struct file *))
{
struct inode *inode;
……
//整個函數的工做在於填充一個file對象
……
f->f_mapping = inode->i_mapping;
f->f_dentry = dentry;
f->f_vfsmnt = mnt;
f->f_pos = 0;
//將對應的索引節點的文件操做函數集賦予文件對象的操做列表
f->f_op = fops_get(inode->i_fop);
……
//若文件本身定義了open操做,則執行這個特定的open操做。
if (!open && f->f_op)
open = f->f_op->open;
if (open) {
error = open(inode, f);
if (error)
goto cleanup_all;
……
return f;
}
|
sys_read()系統調用用於從已打開的文件讀取數據。如read成功,則返回讀到的字節數。如已到達文件的尾端,則返回0。圖9是sys_read()實現代碼中的函數調用關係圖。
對文件進行讀操做時,須要先打開它。從3.1小結可知,打開一個文件時,會在內存組裝一個文件對象,但願對該文件執行的操做方法已在文件對象設置好。因此對文件進行讀操做時,VFS在作了一些簡單的轉換後(由文件描述符獲得其對應的文件對象;其核心思想是返回current->files->fd[fd]所指向的文件對象),就能夠經過語句file->f_op->read(file, buf, count, pos)輕鬆調用實際文件系統的相應方法對文件進行讀操做了。
到此,咱們也就可以解釋在Linux中爲何可以跨文件系統地操做文件了。舉個例子,將vfat格式的磁盤上的一個文件a.txt拷貝到ext3格式的磁盤上,命名爲b.txt。這包含兩個過程,對a.txt進行讀操做,對b.txt進行寫操做。讀寫操做前,須要先打開文件。由前面的分析可知,打開文件時,VFS會知道該文件對應的文件系統格式,之後操做該文件時,VFS會調用其對應的實際文件系統的操做方法。因此,VFS調用vfat的讀文件方法將a.txt的數據讀入內存;在將a.txt在內存中的數據映射到b.txt對應的內存空間後,VFS調用ext3的寫文件方法將b.txt寫入磁盤;從而實現了最終的跨文件系統的複製操做。
不管是普通的文件,仍是特殊的目錄、設備等,VFS都將它們同等看待成文件,經過同一套文件操做界面來對它們進行操做。操做文件時需先打開;打開文件時,VFS會知道該文件對應的文件系統格式;當VFS把控制權傳給實際的文件系統時,實際的文件系統再作出具體區分,對不一樣的文件類型執行不一樣的操做。這也就是「一切皆是文件」的根本所在。
VFS即虛擬文件系統是Linux文件系統中的一個抽象軟件層;由於它的支持,衆多不一樣的實際文件系統才能在Linux中共存,跨文件系統操做才能實現。VFS藉助它四個主要的數據結構即超級塊、索引節點、目錄項和文件對象以及一些輔助的數據結構,向Linux中無論是普通的文件仍是目錄、設備、套接字等都提供一樣的操做界面,如打開、讀寫、關閉等。只有當把控制權傳給實際的文件系統時,實際的文件系統纔會作出區分,對不一樣的文件類型執行不一樣的操做。因而可知,正是有了VFS的存在,跨文件系統操做才能執行,Unix/Linux中的「一切皆是文件」的口號纔可以得以實現。
[1].Claudia Salzberg Rodriguez, Gordon Fischer, Steven Smolski. The Linux Kernel Primer.機械工業出版社.2006.7
[2].Robert Love.Linux內核設計與實現(第二版).機械工業出版社.2007.1
[3].Stevens W.Richard.Unix環境高級編程(第二版).人民郵電出版社.2006
[4].楊芙清,陳向羣.操做系統教程.北京大學出版社.2005.7
[5].Linux-2.6.17.13內核源代碼
https://www.ibm.com/developerworks/cn/linux/l-cn-vfs/