Linux2430内核文件系统学习(多图).doc
《Linux2430内核文件系统学习(多图).doc》由会员分享,可在线阅读,更多相关《Linux2430内核文件系统学习(多图).doc(30页珍藏版)》请在淘文阁 - 分享文档赚钱的网站上搜索。
1、Linux 2.4.30 内核文件系统学习(多图) 1: 关键数据结构1. 概述根据以前学习内核源码的经验,在学习文件系统实现之前,我大概定了个目标:1、 建立一个清晰的全局概念。为将来需要研究代码细节打下坚实基础。2、 只研究虚拟文件系统 VFS 的实现,不研究具体文件系统。为什么选择 Linux 2.4.30?因为可以参考Linux 源码情景分析一书,减少学习难度。1.1. 基本概念1、 一块磁盘(块设备),首先要按照某种文件系统(如 NTFS)格式进行格式化,然后才能在其上进行创建目录、保存文件等操作。在 Linux 中,有“安装”文件系统和“卸载”文件系统的概念。一块经过格式化的“块设
2、备”(不管是刚刚格式化完的,没有创建任何名录和文件;还是已经创建了目录和文件),只有先被“安装”,才能融入 Linux 的文件系统中,用户才可以在它上面进行正常的文件操作。2、 Linux 把目录或普通文件,统一看成“目录节点”。通常一个“目录节点”具有两个重要属性:名称以及磁盘上实际对应的数据。本文中,“目录节点”有时简称为“节点”“符号链接”是一种特殊的目录节点,它只有一个名称,没有实际数据。这个名称指向一个实际的目录节点。3、 “接口结构”:在 内核代码中,经常可以看到一种结构,其成员全部是函数指针,例如:struct file_operations struct module *own
3、er; loff_t (*llseek) (struct file *, loff_t, int); ssize_t (*read) (struct file *, char *, size_t, loff_t *); ssize_t (*write) (struct file *, const char *, size_t, loff_t *); int (*readdir) (struct file *, void *, filldir_t); unsigned int (*poll) (struct file *, struct poll_table_struct *); int (*i
4、octl) (struct inode *, struct file *, unsigned int, unsigned long); int (*mmap) (struct file *, struct vm_area_struct *); int (*open) (struct inode *, struct file *); int (*flush) (struct file *); int (*release) (struct inode *, struct file *); int (*fsync) (struct file *, struct dentry *, int datas
5、ync); int (*fasync) (int, struct file *, int); int (*lock) (struct file *, int, struct file_lock *); ssize_t (*readv) (struct file *, const struct iovec *, unsigned long, loff_t *); ssize_t (*writev) (struct file *, const struct iovec *, unsigned long, loff_t *); ssize_t (*sendpage) (struct file *,
6、struct page *, int, size_t, loff_t *, int); unsigned long (*get_unmapped_area)(struct file *, unsigned long, unsigned long, unsigned long, unsigned long);这种结构的作用类似与 C+ 中的“接口类”,它是用 C 语言进行软件抽象设计时最重要的工具。通过它,将一组通用的操作抽象出来,核心的代码只针对这种“接口结构”进行操作,而这些函数的具体实现由不同的“子类”去完成。以这个 file_operations“接口”为例,它是“目录节点”提供的操作接
7、口。不同的文件系统需要提供这些函数的具体实现。本文中,“接口结构”有时简称“接口”。1.2. 虚拟文件系统Linux 通过虚拟文件系统 (VFS) 来支持不同的具体的文件系统,那么 VFS 到底是什么?从程序员的角度看,我认为 VFS 就是一套代码框架(framework),它将用户与具体的文件系统隔离开来,使得用户能够通过这套框架,以统一的接口在不同的具体的文件系统上进行操作。这套框架包括:1、 为用户提供统一的文件和目录的操作接口,如 open, read, write2、 抽象出文件系统共有的一些结构,包括“目录节点”inode、“超级块”super_block 等。3、 面向具体的文件
8、系统,定义一系列统一的操作“接口”, 如 file_operations, inode_operations, dentry_operation,具体的文件系统必须提供它们的实现。4、 提供一套机制,让具体的文件系统融入 VFS 框架中,包括文件系统的“注册”和“安装”5、 实现这套框架逻辑的核心代码我对文件系统的学习,实际上就是学习虚拟文件系统这套框架是如何实现的。 2. 核心数据结构数据结构是代码的灵魂,要分析一个复杂的系统,关键是掌握那些核心的数据结构,这包括:1、 弄清数据结构的核心功能。一个数据结构通常具有比较复杂的成员,此外,还有一些成员用于建立数据结构之间的关系。如果要一个个去理
9、解,就会陷入细节。2、 弄清数据结构之间的静态关系3、 弄清数据结构之间是如何建立起动态的关系的本文重点分析文件系统中的关键数据结构以及它们之间的关系。 2.1. inode 和 file_operations1、 inode 用以描述“目录节点” ,它描述了一个目录节点物理上的属性,例如大小,创建时间,修改时间、uid、gid 等2、 file_operations 是“目录节点”提供的操作“接口”。它包括 open, read, wirte, ioctl, llseek, mmap 等操作。3、 一个 inode 通过成员 i_fop 对应一个 file_operations4、 打开文件
10、的过程就是寻找 “目录节点”对应的 inode 的过程5、 文件被打开后,inode 和 file_operation 都已经在内存中建立,file_operations 的指针也已经指向了具体文件系统提供的函数,此后都文件的操作,都由这些函数来完成。例如打开了一个普通文件 /root/file,其所在文件系统格式是 ext2,那么,内存中结构如下: 2.2. 目录节点入口dentry本来,inode 中应该包括“目录节点”的名称,但由于符号链接的存在,导致一个物理文件可能有多个文件名,因此把和“目录节点”名称相关的部分从 inode 中分开,放在一个专门的 dentry 结构中。这样:1、
11、一个dentry 通过成员 d_inode 对应到一个 inode上,寻找 inode 的过程变成了寻找 dentry 的过程。因此,dentry 变得更加关键,inode 常常被 dentry 所遮掩。可以说, dentry 是文件系统中最核心的数据结构,它的身影无处不在。2、 由于符号链接的存在,导致多个 dentry 可能对应到同一个 inode 上例如,有一个符号链接 /tmp/abc 指向一个普通文件 /root/file,那么 dentry 与 inode 之间的关系大致如下:d_inodefile dentryfile inoded_inodeabc dentryopenstru
12、ct file_operationsreadwriteioctlllseekmmapi_fop2.3. super_block 和 super_operations一个存放在磁盘上的文件系统如 EXT2 等,在它的格式中通常包括一个“超级块”或者“控制块”的部分,用于从整体上描述文件系统,例如文件系统的大小、是否可读可写等等。虚拟文件系统中也通过“超级块”这种概念来描述文件系统整体的信息,对应的结构是 struct super_block。super_block 除了要记录文件大小、访问权限等信息外,更重要的是提供一个操作“接口”super_operations。structsuper_ope
13、rations structinode*(*alloc_inode)(structsuper_block*sb); void(*destroy_inode)(structinode*); void(*read_inode)(structinode*); void(*read_inode2)(structinode*,void*); void(*dirty_inode)(structinode*); void(*write_inode)(structinode*,int); void(*put_inode)(structinode*); void(*delete_inode)(structino
14、de*); void(*put_super)(structsuper_block*); void(*write_super)(structsuper_block*); int(*sync_fs)(structsuper_block*); void(*write_super_lockfs)(structsuper_block*); void(*unlockfs)(structsuper_block*); int(*statfs)(structsuper_block*,structstatfs*); int(*remount_fs)(structsuper_block*,int*,char*);
15、void(*clear_inode)(structinode*); void(*umount_begin)(structsuper_block*); structdentry*(*fh_to_dentry)(structsuper_block*sb,_u32*fh,intlen,intfhtype,intparent); int(*dentry_to_fh)(structdentry*,_u32*fh,int*lenp,intneed_parent); int(*show_options)(structseq_file*,structvfsmount*);我们通过分析“获取一个 inode ”
16、的过程来只理解这个“接口”中两个成员 alloc_inode 和 read_inode 的作用。 在文件系统的操作中,经常需要获得一个“目录节点”对应的 inode,这个 inode 有可能已经存在于内存中了,也可能还没有,需要创建一个新的 inode,并从磁盘上读取相应的信息来填充。对应的代码是 iget() (inlcude/linux/fs.h)过程如下:1、 通过 iget4_locked() 获取 inode。如果 inode 在内存中已经存在,则直接返回;否则创建一个新的 inode2、 如果是新创建的 inode,通过 super_block-s_op-read_inode()
17、来填充它。也就是说,如何填充一个新创建的 inode, 是由具体文件系统提供的函数实现的。 iget4_locked() 首先在全局的 inode hash table 中寻找,如果找不到,则调用 get_new_inode() ,进而调用 alloc_inode() 来创建一个新的 inode在 alloc_inode() 中可以看到,如果具体文件系统提供了创建 inode 的方法,则由具体文件系统来负责创建,否则采用系统默认的的创建方法。 staticstructinode*alloc_inode(structsuper_block*sb)staticstructaddress_space
- 配套讲稿:
如PPT文件的首页显示word图标,表示该PPT已包含配套word讲稿。双击word图标可打开word文档。
- 特殊限制:
部分文档作品中含有的国旗、国徽等图片,仅作为作品整体效果示例展示,禁止商用。设计者仅对作品中独创性部分享有著作权。
- 关 键 词:
- Linux2430 内核 文件系统 学习 多图
限制150内