Linux-0.11 文件系统super.c详解

news/2025/1/15 23:38:45/

Linux-0.11 文件系统super.c详解

模块简介

该模块主要包含了对超级块的一些读写操作。

函数详解

lock_super

static void lock_super(struct super_block * sb)

该函数的作用是锁定bh块。

    cli();//关中断while (sb->s_lock)//如果已经被锁定sleep_on(&(sb->s_wait));//将当前任务置为不可中断的等待状态,并添加到该超级快等待队列。sb->s_lock = 1;//锁定该超级快sti();//关中断

free_super

static void free_super(struct super_block * sb)

对指定超级块进行解锁。

	cli();//关中断sb->s_lock = 0;//对超级块进行解锁wake_up(&(sb->s_wait));//唤醒等待该超级块的进程sti();//开中断

wait_on_super

static void wait_on_super(struct super_block * sb)

该函数的作用是等待超级块解锁。

	cli();//关中断while (sb->s_lock)//如果已经被锁定sleep_on(&(sb->s_wait));//将当前任务置为不可中断的等待状态,并添加到该超级快等待队列。sti();//开中断

get_super

struct super_block * get_super(int dev)

获取指定设备的超级块。

	struct super_block * s;if (!dev)return NULL;s = 0+super_block;while (s < NR_SUPER+super_block)if (s->s_dev == dev) {wait_on_super(s);if (s->s_dev == dev)return s;s = 0+super_block;} elses++;return NULL;

put_super

void put_super(int dev)

该函数用于放回指定设备的超级块。

	struct super_block * sb;/* struct m_inode * inode;*/int i;

首先判断该设备是否是根文件系统设备,如果是,则直接返回。

	if (dev == ROOT_DEV) {printk("root diskette changed: prepare for armageddon\n\r");return;}

接着开始读取该设备的超级块。如果该超级块的挂载位置i节点s_imount还没有处理,则打印告警并返回。

	if (!(sb = get_super(dev)))return;if (sb->s_imount) {printk("Mounted disk changed - tssk, tssk\n\r");return;}

找到该设备的超级块这时候,首先需要锁定该超级块。接着释放i节点位图和逻辑块位图。结束之后,对该超级块进行解锁,并返回。

	lock_super(sb);sb->s_dev = 0;for(i=0;i<I_MAP_SLOTS;i++)brelse(sb->s_imap[i]);for(i=0;i<Z_MAP_SLOTS;i++)brelse(sb->s_zmap[i]);free_super(sb);return;

read_super

static struct super_block * read_super(int dev)

该函数的作用是用于读取指定设备的超级块。

程序的开始定义了一些变量,对进行一些校验。

	struct super_block * s;struct buffer_head * bh;int i,block;if (!dev)return NULL;check_disk_change(dev);

接着从超级块数组中获取一个超级块。如果已经存在,则直接返回该超级块的指针。如果不存在,则找出一个空闲的位置(s_dev=0)。找到之后边进行初始化。

	if ((s = get_super(dev)))return s;for (s = 0+super_block ;; s++) {if (s >= NR_SUPER+super_block)return NULL;if (!s->s_dev)break;}s->s_dev = dev;s->s_isup = NULL;s->s_imount = NULL;s->s_time = 0;s->s_rd_only = 0;s->s_dirt = 0;

接下来就是读取该设备的第一个磁盘块到内存中。(第0个磁盘块是引导,第一个磁盘块是超级块。)

	if (!(bh = bread(dev,1))) {s->s_dev=0;free_super(s);return NULL;}*((struct d_super_block *) s) =*((struct d_super_block *) bh->b_data);brelse(bh);

接下俩检查其魔数,如果不是0x137F,则不能处理。

	if (s->s_magic != SUPER_MAGIC) {s->s_dev = 0;free_super(s);return NULL;}

接下来开始读取i节点位图和逻辑块位图数据。i节点位图在设备的第2号块开始,共占用s_imap_blocks块。逻辑块位图在i节点位图之后,共占用s_zmap_blocks块。

	for (i=0;i<I_MAP_SLOTS;i++)s->s_imap[i] = NULL;for (i=0;i<Z_MAP_SLOTS;i++)s->s_zmap[i] = NULL;block=2;for (i=0 ; i < s->s_imap_blocks ; i++)if ((s->s_imap[i]=bread(dev,block)))block++;elsebreak;for (i=0 ; i < s->s_zmap_blocks ; i++)if ((s->s_zmap[i]=bread(dev,block)))block++;elsebreak;

接下来如果文件系统信息有问题,则意味着初始化失败,则进行失败的处理。

	if (block != 2+s->s_imap_blocks+s->s_zmap_blocks) {for(i=0;i<I_MAP_SLOTS;i++)brelse(s->s_imap[i]);for(i=0;i<Z_MAP_SLOTS;i++)brelse(s->s_zmap[i]);s->s_dev=0;free_super(s);return NULL;}

程序运行到这里,说明一切正常。因为0号i节点和0号数据块是不能使用的,因此将i节点位图和逻辑块位图的0号位置设置为1。

	s->s_imap[0]->b_data[0] |= 1;s->s_zmap[0]->b_data[0] |= 1;free_super(s);return s;

sys_umount

int sys_umount(char * dev_name)

该函数用于卸载一个文件系统。

程序的开始定义了一些变量,接着通过设备名称获取其i节点。

	struct m_inode * inode;struct super_block * sb;int dev;if (!(inode=namei(dev_name)))return -ENOENT;

对于设备节点而言,其设备号存放在i节点的i_zone[0]中。如果该i节点不是一个块设备文件,则不能进行卸载,于是返回错误。

	dev = inode->i_zone[0];if (!S_ISBLK(inode->i_mode)) {iput(inode);return -ENOTBLK;}iput(inode);

如果该设备号存放的是根文件系统,则不能进行卸载。

	if (dev==ROOT_DEV)return -EBUSY;

接下来根据设备号取出其超级块,如果该超级块中的s_imount为NULL,也就是该设备没有被挂载,则不能进行卸载。

	if (!(sb=get_super(dev)) || !(sb->s_imount))return -ENOENT;

如果挂载的目录节点的i_mount字段为0,则需要打印日志提示。

	if (!sb->s_imount->i_mount)printk("Mounted inode has i_mount=0\n");

接下来遍历inode表,查看该设备节点是否被占用,如果被占用则返回错误。

	for (inode=inode_table+0 ; inode<inode_table+NR_INODE ; inode++)if (inode->i_dev==dev && inode->i_count)return -EBUSY;

程序运行到这里,就开始正式的卸载。首先将挂载的目录节点的i_mount标记为0,随后放回该挂载目录节点。随后将超级块的s_imount字段标记为NULL,并将该文件系统的根i节点进行放回。

	sb->s_imount->i_mount=0;iput(sb->s_imount);sb->s_imount = NULL;iput(sb->s_isup);sb->s_isup = NULL;

最后释放该设备上的超级块以及位图中占用的高速缓冲块,并对该设备进行数据同步。

	put_super(dev);sync_dev(dev);

sys_mount

int sys_mount(char * dev_name, char * dir_name, int rw_flag)

该函数的作用是用于安装文件系统。该函数是sys_开头的,是一个系统调用。

函数的开始定义了一些变量,接着通过设备名称获取其i节点。

	struct m_inode * dev_i, * dir_i;struct super_block * sb;int dev;if (!(dev_i=namei(dev_name)))return -ENOENT;

对于设备类型的i节点,其i_zone[0]存放的是设备号dev。接着判断该设备是否是一个块设备文件,如果不是块设备文件,是不能挂在文件系统的,就会返回权限错误。如果权限校验没有问题,就把dev_i节点放回。

	dev = dev_i->i_zone[0];   //获取其设备号devif (!S_ISBLK(dev_i->i_mode)) { //判断该i节点是否是一个设备节点,如果不是则返回权限错误iput(dev_i);return -EPERM;}iput(dev_i);  //获取到dev_i

接下来获取dir_name对应的i节点dir_i。

	if (!(dir_i=namei(dir_name))) //获取dir_name的i节点return -ENOENT;

如果dir_i这个i节点的引用计数不为1,也就是说这个i节点还被其他进程使用,或者dir_i节点是根文件系统的1号i节点,则不能进行挂载。

	if (dir_i->i_count != 1 || dir_i->i_num == ROOT_INO) {iput(dir_i);return -EBUSY;}

此外,如果该i节点不是一个目录节点,则也不能进行挂载。

	if (!S_ISDIR(dir_i->i_mode)) { //判断dir_i的类型,如果不是目录节点则不能进行挂载iput(dir_i);return -EPERM;}

当运行到这里时,就意味着对挂载设备和挂载目录的校验就通过了。接下来,读取设备dev的超级块。

	if (!(sb=read_super(dev))) {//读取dev设备的超级块iput(dir_i);return -EBUSY;}

如果该超级块已经挂载到某个目录下,那么将返回错误。如果该目录已经挂载了其他的块设备,也返回错误。

	if (sb->s_imount) {//超级块的s_imount不为空,代表该超级块已经挂载到某个目录下,则返回EBUSYiput(dir_i);return -EBUSY;}if (dir_i->i_mount) {//目标的目录节点已经挂载了其他的设备,则返回EPERMiput(dir_i);return -EPERM;}

最后便开始进行真正的挂载步骤,其实就是将超级块sb的s_imount指向挂载的目录i节点dir_i。接着将目录i节点dir_i的i_mount字段标记为1,也将i_dirt标记为1。这些执行完毕之后,将返回0。

	sb->s_imount=dir_i;dir_i->i_mount=1;dir_i->i_dirt=1;return 0;

mount_root

void mount_root(void)

该函数的作用是安装根文件系统。

该函数的最初定义了一些变量,对d_inode的结构的长度进行校验。

	int i,free;struct super_block * p;struct m_inode * mi;if (32 != sizeof (struct d_inode))panic("bad i-node size");

如果根文件系统在软盘中,就提示插入跟文件系统盘。

	for(i=0;i<NR_FILE;i++)file_table[i].f_count=0;if (MAJOR(ROOT_DEV) == 2) {printk("Insert root floppy and press ENTER");wait_for_keypress();}

初始化超级块数组。

	for(p = &super_block[0] ; p < &super_block[NR_SUPER] ; p++) {p->s_dev = 0;p->s_lock = 0;p->s_wait = NULL;}

读取根文件系统的超级块。

	if (!(p=read_super(ROOT_DEV)))panic("Unable to mount root");

读取根目录的i节点。

	if (!(mi=iget(ROOT_DEV,ROOT_INO)))panic("Unable to read root i-node");

该函数在init进程中调用,下面设置init进程PCB的pwd和root。

	mi->i_count += 3 ;	/* NOTE! it is logically used 4 times, not 1 */p->s_isup = p->s_imount = mi;current->pwd = mi;current->root = mi;

统计空闲的i节点数量。

	free=0;i=p->s_nzones;while (-- i >= 0)if (!set_bit(i&8191,p->s_zmap[i>>13]->b_data))free++;printk("%d/%d free blocks\n\r",free,p->s_nzones);

统计空闲的逻辑块节点。

	free=0;i=p->s_ninodes+1;while (-- i >= 0)if (!set_bit(i&8191,p->s_imap[i>>13]->b_data))free++;printk("%d/%d free inodes\n\r",free,p->s_ninodes);

Q & A


http://www.ppmy.cn/news/97392.html

相关文章

日常 - HttpURLConnection 网络请求 TLS 1.2

文章目录 环境前言HTTPS 请求流程服务端支持JDK 验证资源 环境 JDK 8 Hutool 4.5.1 前言 应供应商 DD 的 TLS 版本升级通知&#xff0c;企业版接口升级后 TLS 1.0 及 1.1 版本请求将无法连接&#xff0c;仅支持 TLS 1.2 及以上版本的客户端发起请求。 当前项目使用 Hutool …

协程切换原理与实践 -- 从ucontext api到x86_64汇编

目录 1.协程切换原理理解 2.ucontext实现协程切换 2.1 实现流程 2.2 根据ucontext流程看协程实现 2.3 回答开头提出的问题 3.x86_64汇编实现协程切换 3.1libco x86_64汇编代码分析 3.2.保存程序返回代码地址流程 3.3.恢复程序地址以及上下文 4.实现简单协程框架 1.协程…

python+django协同过滤算法的美食O2O外卖点餐系统vue

当然使用的数据库是mysql。尽管没有面向对象的数据库的作用强大&#xff0c;但是在Python开发上还是比较的灵活和方便的。系统功能主要介绍以下几点&#xff1a; 本外卖点餐系统主要包括二大功能模块&#xff0c;即用户功能模块和管理员功能模块。 &#xff08;1&#xff09;管…

1. Tomcat整体架构及其设计精髓分析

tomcat 1.Tomcat的整体架构1.1 Tomcat介绍1.1.1 Servlet基础回顾 1.2 目录结构1.3 web应用部署的方式1.4 结合Server.xml理解Tomcat架构1.5 架构图 2. Tomcat核心组件详解2.1 Server 组件2.2 Service组件2.3 连接器Connector组件2.3.1 ProtocolHandler 组件2.3.1.1 EndPoint2.3…

Java中常见锁的分类及概念分析

基于线程对同一把锁的获取情况分类 可重入锁 同一个线程可以多次获取锁 每次获取锁&#xff0c;锁的计数器加1&#xff0c;每次释放锁锁的计数器减1 锁的计数器归零&#xff0c;锁完全释放 Java中提供的synchronized&#xff0c;ReentrantLock&#xff0c;ReentrantReadWriteL…

文档处理新探究成果——前沿技术CCIG文档图像智能分析论坛分享

目录 前言 一、文档分析与识别最新研究 二、视觉-语言预训练模型及迁移学习 三、篡改文本图像的生成与检测技术 四、智能文档处理技术在工业界的应用与挑战 总结 前言 图文智能处理前沿技术一直是我所关注的技术&#xff0c;尤其在现在集成多态大模型的基础之上&#xff0…

深入了解平均精度(mAP):通过精确率-召回率曲线评估目标检测性能

平均精度&#xff08;Average Precision&#xff0c;mAP&#xff09;是一种常用的用于评估目标检测模型性能的指标。在目标检测任务中&#xff0c;模型需要识别图像中的不同目标&#xff0c;并返回它们的边界框&#xff08;bounding box&#xff09;和类别。mAP用于综合考虑模型…

【原创】浅谈EtherCAT主站EOE(上)-EOE网络

版权声明&#xff1a;本文为本文为博主原创文章&#xff0c;未经同意&#xff0c;禁止转载。如有问题&#xff0c;欢迎指正。博客地址&#xff1a;https://www.cnblogs.com/wsg1100/ 文章目录 一、EoE二、EoE服务规范EtherCAT主站如何提供EoE服务&#xff1f; 三、EoE网络EOE网…