图解Linux虚拟文件系统(VFS)之关系篇

目录

1.什么是虚拟文件系统?

2.Linux系统文件树

3.文件系统注册

4.文件系统挂载

4.1 索引挂载点

4.2 创建新文件系统挂载实例

4.3 新旧挂载实例对接

总结:


大家好,今天和大家探讨一下Linux虚拟文件系统,虚拟文件系统是我一直想要去聊的一个知识点,如果你想从事Linux开发相关的工作,一定要了解虚拟文件系统。

1.什么是虚拟文件系统?

Linux虚拟文件系统(Virtual File System,VFS)是Linux操作系统中的一个重要组成部分,它提供了一个统一的接口,使得用户和应用程序可以通过相同的方式访问不同类型的文件系统。

VFS的设计目标是将不同类型的文件系统抽象为一个统一的接口,使得用户和应用程序无需关心底层文件系统的具体实现细节。通过VFS用户可以使用相同的系统调用(如open、read、write等)来访问不同类型的文件系统,包括本地文件系统(如ext4、XFS等)、网络文件系统(如NFS、CIFS等)以及虚拟文件系统(如procfs、sysfs等)。

VFS由以下几个主要组件组成:

  • 虚拟文件系统接口:VFS定义了一组通用的文件系统操作接口。

  • 超级块(super_block):每个文件系统都有一个超级块,它包含了文件系统的元数据信息,如文件系统类型、块大小、inode表等,超级块提供了对文件系统的整体描述和管理。

  • 目录项(dentry):dentry是目录项的缩写,用于表示文件系统中的目录和文件,dentry包含了目录和文件对应的inode指针,通过它可以快速定位到目录下的文件或子目录。

  • 文件节点(inode):inode是文件系统中的一个数据结构,用于存储文件或目录的元数据信息,如文件大小、权限、所有者等,每个文件或目录都对应一个唯一的inode。

  • 文件对象(file):file是表示打开文件的数据结构,它包含了对应的inode指针、当前读写位置等信息,通过file可以进行文件的读写操作。

2.Linux系统文件树

对于一个普通的Linux用户或者运维人员,Linux系统文件树通常的样子如下图,以根文件系统根目录为起点,通过根目录遍历整个文件树。

而在系统开发人员眼中,Linux系统文件树则变成这样一个结构,每个文件和目录都对应一个dentry结构体。

dentry到底是什么?

dentry结构体的主要作用是提供文件系统层次结构的表示,它们通过形成一个树状结构来组织目录和文件,每个dentry都有一个唯一的路径名,可以通过遍历dentry树来找到特定文件或目录。

struct dentry结构体定义:

struct dentry {struct dentry *d_parent;struct qstr d_name;struct inode *d_inode;const struct dentry_operations *d_op;struct super_block *d_sb; struct list_head d_child;struct list_head d_subdirs;....
};

struct dentry结构体通过d_parent,d_child,d_subdirs等成员将文件系统组成一颗文件树,要了解Linux文件系统,我们得学会运用dentry。

小节:dentry是VFS重要的组成部分,要理解VFS先从dentry开始。

3.文件系统注册

通过前面的学习,我们了解到dentry结构的重要性,接下来围绕dentry结构体来解析文件VFS各组件之间的关系,我们先来看一下整体架构图:

Linux文件系统对应一个file_system_type结构体对象,file_system_type结构体定义如下:

struct file_system_type {const char *name;int fs_flags;int (*init_fs_context)(struct fs_context *);const struct fs_parameter_spec *parameters;struct dentry *(*mount) (struct file_system_type *, int,const char *, void *);void (*kill_sb) (struct super_block *);......
};

ramfs文件系统定义如下,name表示文件系统类型,当ramfs文件系统需要实例化,需要通过name查找全局文件系统链表头找到对应的已注册文件系统,再通过已注册文件系统创建超级块(super block)。

static struct file_system_type ramfs_fs_type = {.name = "ramfs",.init_fs_context = ramfs_init_fs_context,.parameters = ramfs_fs_parameters,.kill_sb = ramfs_kill_sb,.fs_flags = FS_USERNS_MOUNT,
};

定义好文件系统后,通过register_filesystem函数将文件系统注册至Linux系统,注册成功的文件系统会插入全局文件系统链表,已注册的文件系统能够用来创建超级块(super block)。

通过cat /proc/filesystems查看系统所有已注册文件系统

图片

4.文件系统挂载

文件系统挂载就是新文件系统生成一个挂载实例(struct mount),让新挂载实例和父文件系统的挂载实例建立父子关系。

一个新的挂载实例包括几个重要部分:

  • 超级块(super_block)

超级块用来指示新的文件系统对应的设备。

  • 父挂载实例(mount)

父挂载实例表示挂载点所处的文件系统挂载实例。

  • 挂载点(mountpoint)

挂载点是新文件系统和父文件系统之间连接的纽带。

  • 文件系统根目录(dentry)

每个文件系统都有一个根目录,当索引一个文件路径进入到一个新的文件系统后,会从新的文件系统根目录开始索引。

4.1 索引挂载点

索引挂载点的目的是为了获取挂载点的struct path记录信息,挂载点索引的过程就是struct path记录信息不断被替换的过程。

以挂载点/mnt/test/dir为例来讲解:

  1. 索引/目录,获取/目录的path记录信息。

  2. 索引mnt目录,获取mnt目录的path记录信息,并覆盖/目录的path记录信息。

  3. 索引test目录,获取test目录的path记录信息,并覆盖mnt目录的path记录信息。

  4. 索引dir目录,获取dir目录的path记录信息,并覆盖test目录的path记录信息。

  5. 最终获取到挂载点dir的struct path记录信息。

struct path结构体定义如下:

struct path {struct vfsmount *mnt;struct dentry *dentry;
};

mnt:记录挂载点所在文件系统的挂载实例。

dentry:挂载点目录dentry。

4.2 创建新文件系统挂载实例

  • 创建超级块

要创建超级块首先要知道文件系统类型,mount命令通过-t参数指定文件系统类型,通过mount命令传入的文件系统类型,可以遍历全局文件系统链表找到已注册的文件系统,通过已注册的文件系统创建超级块。

  • 创建新文件系统挂载实例

创建超级块后,通过超级块的信息创建新文件系统挂载实例。

  • 创建挂载点

通过挂载点dentry创建一个挂载点。

4.3 新旧挂载实例对接

通过前面的过程,我们已经具备文件系统挂载三要素:

  • 新文件系统挂载实例。

  • 父文件系统挂载实例。

  • 挂载点。

通过挂载三要素,我们就能完成新旧挂载实例对接,完成对接后,新文件系统挂载实例的mnt_parent指向父挂载实例,整个挂载过程就已经完成。

新文件系统挂载成功后,Linux系统文件树将新文件系统嫁接进来,如下图:

此时我们想要操作新文件系统中的文件,只需要根据路径名层层索引获取文件path信息,path信息记录dentry信息,dentry绑定了inode对象。

最终获取到inode文件节点就能操作文件了。


总结:

如果你想系统学习Linux软件开发,我这里有两套完整的Linux编程课程(图解Linux环境编程和图解Linux网络编程)

课程总共近150课时,时长近60小时,课程都是以图解的方式进行讲解,希望能帮助到你。

点击文末卡片,获取相关资料

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/448396.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

人脸识别的多样化和稀疏关注对姿势变化和遮挡具有鲁棒性

DSA-Face: Diverse and Sparse Attentions for Face Recognition Robust to Pose Variation and Occlusion 一、创新点 1.提出了成对自我对比注意力来强制模型提取不同的局部特征; 2.设计注意力稀疏性损失是为了鼓励注意力图中的稀疏反应,阻止对分散注意…

LeetCode:141和142,环形链表之追及相遇和快慢指针的运用

这两个题是相关联的,主要做法为哈希和快慢指针,当然像博主我,不看解析只会O(n^2)的暴力遍历,太惨了,不过,快慢指针还是很好理解的,是一个追及的问题, 目录 …

基于STM32的DMA在外设数据交换中的应用案例

如何使用STM32的DMA在外设数据交换中实现高效的数据传输呢?下面,我将提供一个应用案例,涉及使用STM32的DMA在UART外设和内存之间进行数据传输的示例。 ✅作者简介:热爱科研的嵌入式开发者,修心和技术同步精进 ❤欢迎关…

【代码随想录】LC 1. 两数之和

文章目录 前言一、题目1、原题链接2、题目描述 二、解题报告1、思路分析2、时间复杂度3、代码详解 前言 本专栏文章为《代码随想录》书籍的刷题题解以及读书笔记,如有侵权,立即删除。 一、题目 1、原题链接 1. 两数之和 2、题目描述 二、解题报告 1、思…

(6)Elastix图像配准:可视化配准工具

文章目录 前言一、配准工具1.1、基于Elastix的可视化配准工具1.1.1、elastix-napari:基于napari的Elastix图像配准1.1.2、SlicerElastix:基于3D slice的Elastix图像配准1.1.3、BIRDS:基于ImageJ的Elastix双通道图像配准 1.2、基于ITK开发的配…

input框前面名字长短不一时,让上下input框对齐方法

没设置之前 设置之后&#xff1a; 代码如下&#xff1a; <style>div{width: 500px;}label {display: block; /* 设置 label 元素为块级元素 */text-align: right; /* 设置文本右对齐 */margin-bottom: 10px; /* 设置标签之间的间距 */} </style> </head><…

springboot 文件差异化对比以及可视化展示

maven依赖 <!-- 文件内容对比--><dependency><groupId>io.github.java-diff-utils</groupId><artifactId>java-diff-utils</artifactId><version>4.11</version></dependency>创建Diff 工具类 package com.system.ut…

C++ copy()函数详细介绍

copy() 是一个标准库函数&#xff0c;位于 头文件中。它用于将一个容器中的元素复制到另一个容器中&#xff0c;或者将一个范围内的元素复制到另一个范围中。 函数参数介绍 copy( first, last, d_first );first 和 last&#xff1a;表示输入范围的迭代器。 first 指向要复制的…

python脚本将照片按时间线整理

说明&#xff1a;有一次自己瞎折腾&#xff0c;然后把服务器相册搞崩了&#xff0c;后来做了备份同步给找了回来&#xff0c;但是相册的时间线全乱了&#xff0c;看起来非常难受。所以就想通过文件夹的形式把照片重新分类&#xff0c;分类后的结构如下(红色字体为文件夹)&#…

openmax

通过EmptyThisBuffer传递未解码的buffer给component&#xff0c;component收到该命令后会去读取input port buffer中的数据&#xff0c;将其组装为帧之后进行解码&#xff0c;buffer处理完成后会通过EmptyBufferDone通知上层输入使用完成&#xff0c;上层收到命令可以继续送输入…

Duplicate entry ‘2020045-2-1‘ for key ‘index_uid‘ 解决方案

项目场景&#xff1a; 今天小编在工作中编写接口对数据库增加相同的非主键数据的时候&#xff0c;突然出现了这样的一个错误&#xff1a; 下面我来给大家解答这个错误的出现原因以及解决办法。 问题描述 Duplicate entry 2020045-2-1 for key index_uid 这个错误大概意思就是…

举个栗子!Tableau 技巧(263):按需突出显示文本表的 N 个行

我们分享过 &#x1f330; &#xff1a;突出显示文本表的行或列&#xff0c;可以突出显示文本表中的某一行或某一列。有数据粉提出新的问题&#xff1a;如果想突出显示多行数据&#xff0c;该如何实现呢&#xff1f; 在 Tableau 中是可以实现的&#xff08;如上图&#xff09;&…