<?xml version="1.0" encoding="utf-8" standalone="yes"?>
<rss version="2.0" xmlns:atom="http://www.w3.org/2005/Atom" xmlns:content="http://purl.org/rss/1.0/modules/content/">
  <channel>
    <title>VFS on 大飞的博客</title>
    <link>https://www.dafei.me/tags/vfs/</link>
    <description>Recent content in VFS on 大飞的博客</description>
    <generator>Hugo</generator>
    <language>zh-cn</language>
    <lastBuildDate>Thu, 14 May 2026 07:00:00 +0000</lastBuildDate>
    <atom:link href="https://www.dafei.me/tags/vfs/index.xml" rel="self" type="application/rss+xml" />
    <item>
      <title>从零写OS（二十七）：字符设备 —— /dev/null 和 /dev/zero</title>
      <link>https://www.dafei.me/posts/os-27-chardev/</link>
      <pubDate>Thu, 14 May 2026 07:00:00 +0000</pubDate>
      <guid>https://www.dafei.me/posts/os-27-chardev/</guid>
      <description>&lt;p&gt;Linux 里有个特殊目录 &lt;code&gt;/dev/&lt;/code&gt;，里面住着各种设备文件：&lt;/p&gt;
&lt;pre tabindex=&#34;0&#34;&gt;&lt;code&gt;/dev/null     ← 写什么扔什么，读永远 EOF
/dev/zero     ← 读出来全是 \0
/dev/random   ← 读出来是随机字节
/dev/tty      ← 当前终端
&lt;/code&gt;&lt;/pre&gt;&lt;p&gt;对用户程序来说，它们和普通文件没区别：&lt;code&gt;open&lt;/code&gt;、&lt;code&gt;read&lt;/code&gt;、&lt;code&gt;write&lt;/code&gt;、&lt;code&gt;close&lt;/code&gt;，完全一样的接口。&lt;/p&gt;
&lt;p&gt;这一章实现字符设备（char device）框架，让 VFS 能路由 &lt;code&gt;/dev/&lt;/code&gt; 路径，并内置 &lt;code&gt;null&lt;/code&gt; 和 &lt;code&gt;zero&lt;/code&gt; 两个最基础的设备。&lt;/p&gt;
&lt;hr&gt;
&lt;h2 id=&#34;什么是字符设备&#34;&gt;什么是字符设备&lt;/h2&gt;
&lt;p&gt;字符设备（character device）的特点：&lt;/p&gt;
&lt;ul&gt;
&lt;li&gt;&lt;strong&gt;按字节读写&lt;/strong&gt;，没有块/扇区的概念&lt;/li&gt;
&lt;li&gt;&lt;strong&gt;没有随机寻址&lt;/strong&gt;（不像磁盘文件可以 seek）&lt;/li&gt;
&lt;li&gt;&lt;strong&gt;读写是即时的&lt;/strong&gt;：写进去就消失（null）或立刻可读（zero/tty）&lt;/li&gt;
&lt;/ul&gt;
&lt;p&gt;与之对应的是块设备（block device），如硬盘，以固定大小的块为单位操作。&lt;/p&gt;
&lt;hr&gt;
&lt;h2 id=&#34;设备注册表&#34;&gt;设备注册表&lt;/h2&gt;
&lt;p&gt;核心数据结构 &lt;code&gt;cdev_t&lt;/code&gt;：&lt;/p&gt;
&lt;div class=&#34;highlight&#34;&gt;&lt;pre tabindex=&#34;0&#34; style=&#34;color:#f8f8f2;background-color:#272822;-moz-tab-size:4;-o-tab-size:4;tab-size:4;-webkit-text-size-adjust:none;&#34;&gt;&lt;code class=&#34;language-c&#34; data-lang=&#34;c&#34;&gt;&lt;span style=&#34;display:flex;&#34;&gt;&lt;span&gt;&lt;span style=&#34;color:#66d9ef&#34;&gt;typedef&lt;/span&gt; &lt;span style=&#34;color:#66d9ef&#34;&gt;struct&lt;/span&gt; {
&lt;/span&gt;&lt;/span&gt;&lt;span style=&#34;display:flex;&#34;&gt;&lt;span&gt;    &lt;span style=&#34;color:#66d9ef&#34;&gt;char&lt;/span&gt;    name[&lt;span style=&#34;color:#ae81ff&#34;&gt;16&lt;/span&gt;];              &lt;span style=&#34;color:#75715e&#34;&gt;// 设备名，如 &amp;#34;null&amp;#34;、&amp;#34;zero&amp;#34;
&lt;/span&gt;&lt;/span&gt;&lt;/span&gt;&lt;span style=&#34;display:flex;&#34;&gt;&lt;span&gt;    &lt;span style=&#34;color:#66d9ef&#34;&gt;int&lt;/span&gt;     (&lt;span style=&#34;color:#f92672&#34;&gt;*&lt;/span&gt;open) (&lt;span style=&#34;color:#66d9ef&#34;&gt;void&lt;/span&gt;);
&lt;/span&gt;&lt;/span&gt;&lt;span style=&#34;display:flex;&#34;&gt;&lt;span&gt;    &lt;span style=&#34;color:#66d9ef&#34;&gt;int&lt;/span&gt;     (&lt;span style=&#34;color:#f92672&#34;&gt;*&lt;/span&gt;read) (&lt;span style=&#34;color:#66d9ef&#34;&gt;void&lt;/span&gt; &lt;span style=&#34;color:#f92672&#34;&gt;*&lt;/span&gt;buf, &lt;span style=&#34;color:#66d9ef&#34;&gt;uint32_t&lt;/span&gt; len);
&lt;/span&gt;&lt;/span&gt;&lt;span style=&#34;display:flex;&#34;&gt;&lt;span&gt;    &lt;span style=&#34;color:#66d9ef&#34;&gt;int&lt;/span&gt;     (&lt;span style=&#34;color:#f92672&#34;&gt;*&lt;/span&gt;write)(&lt;span style=&#34;color:#66d9ef&#34;&gt;const&lt;/span&gt; &lt;span style=&#34;color:#66d9ef&#34;&gt;void&lt;/span&gt; &lt;span style=&#34;color:#f92672&#34;&gt;*&lt;/span&gt;buf, &lt;span style=&#34;color:#66d9ef&#34;&gt;uint32_t&lt;/span&gt; len);
&lt;/span&gt;&lt;/span&gt;&lt;span style=&#34;display:flex;&#34;&gt;&lt;span&gt;    &lt;span style=&#34;color:#66d9ef&#34;&gt;void&lt;/span&gt;    (&lt;span style=&#34;color:#f92672&#34;&gt;*&lt;/span&gt;close)(&lt;span style=&#34;color:#66d9ef&#34;&gt;int&lt;/span&gt; fd);
&lt;/span&gt;&lt;/span&gt;&lt;span style=&#34;display:flex;&#34;&gt;&lt;span&gt;    &lt;span style=&#34;color:#66d9ef&#34;&gt;int&lt;/span&gt;     used;
&lt;/span&gt;&lt;/span&gt;&lt;span style=&#34;display:flex;&#34;&gt;&lt;span&gt;} &lt;span style=&#34;color:#66d9ef&#34;&gt;cdev_t&lt;/span&gt;;
&lt;/span&gt;&lt;/span&gt;&lt;span style=&#34;display:flex;&#34;&gt;&lt;span&gt;
&lt;/span&gt;&lt;/span&gt;&lt;span style=&#34;display:flex;&#34;&gt;&lt;span&gt;&lt;span style=&#34;color:#66d9ef&#34;&gt;static&lt;/span&gt; &lt;span style=&#34;color:#66d9ef&#34;&gt;cdev_t&lt;/span&gt; devs[CDEV_MAX];     &lt;span style=&#34;color:#75715e&#34;&gt;// 全局设备表
&lt;/span&gt;&lt;/span&gt;&lt;/span&gt;&lt;/code&gt;&lt;/pre&gt;&lt;/div&gt;&lt;p&gt;通过函数指针实现多态——不同设备注册不同的 read/write 实现。&lt;/p&gt;
&lt;p&gt;&lt;code&gt;chardev_register&lt;/code&gt; 向 &lt;code&gt;devs[]&lt;/code&gt; 添加一个设备，&lt;code&gt;chardev_open&lt;/code&gt; 按名字查找并调用 &lt;code&gt;open()&lt;/code&gt;。&lt;/p&gt;</description>
    </item>
    <item>
      <title>从零写OS（二十三）：procfs 进程文件系统</title>
      <link>https://www.dafei.me/posts/os-23-procfs/</link>
      <pubDate>Thu, 14 May 2026 03:00:00 +0000</pubDate>
      <guid>https://www.dafei.me/posts/os-23-procfs/</guid>
      <description>&lt;p&gt;上一章有了 pipe 和 fd，现在进程可以互相传数据了。但还有一个问题：用户程序没有办法查询&amp;quot;系统里现在有哪些进程&amp;quot;——这类信息只存在内核里，外面拿不到。&lt;/p&gt;
&lt;p&gt;加专用 syscall 当然可以，但 Linux 选择了一个更优雅的方案：&lt;strong&gt;procfs&lt;/strong&gt;。&lt;/p&gt;
&lt;hr&gt;
&lt;h2 id=&#34;一切皆文件&#34;&gt;一切皆文件&lt;/h2&gt;
&lt;p&gt;procfs 的思路是：把内核状态&lt;strong&gt;伪装成文件&lt;/strong&gt;，挂在 &lt;code&gt;/proc/&lt;/code&gt; 目录下。&lt;/p&gt;
&lt;pre tabindex=&#34;0&#34;&gt;&lt;code&gt;/proc/0/status   → 进程 0 的信息
/proc/1/status   → 进程 1 的信息
&lt;/code&gt;&lt;/pre&gt;&lt;p&gt;用户程序用完全一样的 &lt;code&gt;open/read/close&lt;/code&gt; 就能读到，不需要学新 API：&lt;/p&gt;
&lt;div class=&#34;highlight&#34;&gt;&lt;pre tabindex=&#34;0&#34; style=&#34;color:#f8f8f2;background-color:#272822;-moz-tab-size:4;-o-tab-size:4;tab-size:4;-webkit-text-size-adjust:none;&#34;&gt;&lt;code class=&#34;language-c&#34; data-lang=&#34;c&#34;&gt;&lt;span style=&#34;display:flex;&#34;&gt;&lt;span&gt;&lt;span style=&#34;color:#66d9ef&#34;&gt;int&lt;/span&gt; fd &lt;span style=&#34;color:#f92672&#34;&gt;=&lt;/span&gt; &lt;span style=&#34;color:#a6e22e&#34;&gt;open&lt;/span&gt;(&lt;span style=&#34;color:#e6db74&#34;&gt;&amp;#34;/proc/1/status&amp;#34;&lt;/span&gt;);
&lt;/span&gt;&lt;/span&gt;&lt;span style=&#34;display:flex;&#34;&gt;&lt;span&gt;&lt;span style=&#34;color:#a6e22e&#34;&gt;read&lt;/span&gt;(fd, buf, &lt;span style=&#34;color:#ae81ff&#34;&gt;128&lt;/span&gt;);
&lt;/span&gt;&lt;/span&gt;&lt;span style=&#34;display:flex;&#34;&gt;&lt;span&gt;&lt;span style=&#34;color:#75715e&#34;&gt;// buf 里是 &amp;#34;pid: 1\nstate: running\nparent: 0\n&amp;#34;
&lt;/span&gt;&lt;/span&gt;&lt;/span&gt;&lt;/code&gt;&lt;/pre&gt;&lt;/div&gt;&lt;p&gt;这就是 Linux &amp;ldquo;一切皆文件&amp;rdquo; 哲学的体现——统一接口，背后实现可以完全不同。&lt;/p&gt;
&lt;hr&gt;
&lt;h2 id=&#34;虚拟文件和真实文件的区别&#34;&gt;虚拟文件和真实文件的区别&lt;/h2&gt;
&lt;p&gt;普通文件的 &lt;code&gt;read&lt;/code&gt;：从磁盘读数据。&lt;/p&gt;
&lt;p&gt;procfs 文件的 &lt;code&gt;read&lt;/code&gt;：&lt;strong&gt;内核现场生成内容&lt;/strong&gt;，没有磁盘，没有 inode，数据就是 &lt;code&gt;procs[]&lt;/code&gt; 数组里的字段格式化出来的字符串。&lt;/p&gt;
&lt;pre tabindex=&#34;0&#34;&gt;&lt;code&gt;open(&amp;#34;/proc/1/status&amp;#34;)
  → vfs_open 识别 &amp;#34;/proc/&amp;#34; 前缀
  → procfs_open 解析 pid，格式化状态字符串存入内核 buf
  → 返回 VFile（type=VFILE_PROC）

read(fd, buf, len)
  → vfs_read 识别 VFILE_PROC
  → 从内核 buf 按 offset 拷贝给用户
&lt;/code&gt;&lt;/pre&gt;&lt;hr&gt;
&lt;h2 id=&#34;实现&#34;&gt;实现&lt;/h2&gt;
&lt;h3 id=&#34;procfs-内核结构&#34;&gt;procfs 内核结构&lt;/h3&gt;
&lt;div class=&#34;highlight&#34;&gt;&lt;pre tabindex=&#34;0&#34; style=&#34;color:#f8f8f2;background-color:#272822;-moz-tab-size:4;-o-tab-size:4;tab-size:4;-webkit-text-size-adjust:none;&#34;&gt;&lt;code class=&#34;language-c&#34; data-lang=&#34;c&#34;&gt;&lt;span style=&#34;display:flex;&#34;&gt;&lt;span&gt;&lt;span style=&#34;color:#66d9ef&#34;&gt;typedef&lt;/span&gt; &lt;span style=&#34;color:#66d9ef&#34;&gt;struct&lt;/span&gt; {
&lt;/span&gt;&lt;/span&gt;&lt;span style=&#34;display:flex;&#34;&gt;&lt;span&gt;    &lt;span style=&#34;color:#66d9ef&#34;&gt;int&lt;/span&gt;      used;
&lt;/span&gt;&lt;/span&gt;&lt;span style=&#34;display:flex;&#34;&gt;&lt;span&gt;    &lt;span style=&#34;color:#66d9ef&#34;&gt;uint32_t&lt;/span&gt; pid;
&lt;/span&gt;&lt;/span&gt;&lt;span style=&#34;display:flex;&#34;&gt;&lt;span&gt;    &lt;span style=&#34;color:#66d9ef&#34;&gt;char&lt;/span&gt;     buf[&lt;span style=&#34;color:#ae81ff&#34;&gt;256&lt;/span&gt;];   &lt;span style=&#34;color:#75715e&#34;&gt;// open 时格式化好的字符串
&lt;/span&gt;&lt;/span&gt;&lt;/span&gt;&lt;span style=&#34;display:flex;&#34;&gt;&lt;span&gt;    &lt;span style=&#34;color:#66d9ef&#34;&gt;uint32_t&lt;/span&gt; buf_len;
&lt;/span&gt;&lt;/span&gt;&lt;span style=&#34;display:flex;&#34;&gt;&lt;span&gt;    &lt;span style=&#34;color:#66d9ef&#34;&gt;uint32_t&lt;/span&gt; offset;     &lt;span style=&#34;color:#75715e&#34;&gt;// 支持分段 read
&lt;/span&gt;&lt;/span&gt;&lt;/span&gt;&lt;span style=&#34;display:flex;&#34;&gt;&lt;span&gt;} &lt;span style=&#34;color:#66d9ef&#34;&gt;proc_fd_t&lt;/span&gt;;
&lt;/span&gt;&lt;/span&gt;&lt;/code&gt;&lt;/pre&gt;&lt;/div&gt;&lt;p&gt;&lt;code&gt;open&lt;/code&gt; 时一次性生成完整字符串，后续 &lt;code&gt;read&lt;/code&gt; 只是按 offset 切片拷贝——和普通文件行为完全一致，可以多次 read。&lt;/p&gt;</description>
    </item>
    <item>
      <title>从零写OS（二十一）：文件描述符 fd</title>
      <link>https://www.dafei.me/posts/os-21-fd/</link>
      <pubDate>Thu, 14 May 2026 01:00:00 +0000</pubDate>
      <guid>https://www.dafei.me/posts/os-21-fd/</guid>
      <description>&lt;p&gt;前几章的用户程序如果想读文件，得直接传 inode 号给内核——&lt;code&gt;read(ino, offset, buf, len)&lt;/code&gt;。这意味着用户程序要自己记当前读到哪了，而且根本不知道文件名，只有一个数字。&lt;/p&gt;
&lt;p&gt;这一章引入 &lt;strong&gt;文件描述符（fd）&lt;/strong&gt;，让用户程序用熟悉的方式操作文件：&lt;/p&gt;
&lt;div class=&#34;highlight&#34;&gt;&lt;pre tabindex=&#34;0&#34; style=&#34;color:#f8f8f2;background-color:#272822;-moz-tab-size:4;-o-tab-size:4;tab-size:4;-webkit-text-size-adjust:none;&#34;&gt;&lt;code class=&#34;language-c&#34; data-lang=&#34;c&#34;&gt;&lt;span style=&#34;display:flex;&#34;&gt;&lt;span&gt;&lt;span style=&#34;color:#66d9ef&#34;&gt;int&lt;/span&gt; fd &lt;span style=&#34;color:#f92672&#34;&gt;=&lt;/span&gt; &lt;span style=&#34;color:#a6e22e&#34;&gt;open&lt;/span&gt;(&lt;span style=&#34;color:#e6db74&#34;&gt;&amp;#34;hello.txt&amp;#34;&lt;/span&gt;);
&lt;/span&gt;&lt;/span&gt;&lt;span style=&#34;display:flex;&#34;&gt;&lt;span&gt;&lt;span style=&#34;color:#66d9ef&#34;&gt;int&lt;/span&gt; n  &lt;span style=&#34;color:#f92672&#34;&gt;=&lt;/span&gt; &lt;span style=&#34;color:#a6e22e&#34;&gt;read&lt;/span&gt;(fd, buf, &lt;span style=&#34;color:#ae81ff&#34;&gt;64&lt;/span&gt;);
&lt;/span&gt;&lt;/span&gt;&lt;span style=&#34;display:flex;&#34;&gt;&lt;span&gt;&lt;span style=&#34;color:#a6e22e&#34;&gt;close&lt;/span&gt;(fd);
&lt;/span&gt;&lt;/span&gt;&lt;/code&gt;&lt;/pre&gt;&lt;/div&gt;&lt;hr&gt;
&lt;h2 id=&#34;fd-是什么&#34;&gt;fd 是什么&lt;/h2&gt;
&lt;p&gt;fd 是一个&lt;strong&gt;进程内的整数&lt;/strong&gt;，通常从 0 开始分配（0=stdin，1=stdout，2=stderr，之后是普通文件）。&lt;/p&gt;
&lt;p&gt;它背后有三层：&lt;/p&gt;
&lt;pre tabindex=&#34;0&#34;&gt;&lt;code&gt;进程 fd_table[]          内核 open_file             磁盘
  fd=3  ──────────→   { ino=42, offset=0 }  ──→  hello.txt
  fd=4  ──────────→   { ino=43, offset=0 }  ──→  readme.txt
&lt;/code&gt;&lt;/pre&gt;&lt;p&gt;&lt;strong&gt;fd 本身只是下标&lt;/strong&gt;，真正存 offset 的是内核里的&amp;quot;打开文件&amp;quot;结构。这样的好处：&lt;/p&gt;
&lt;ul&gt;
&lt;li&gt;offset 由内核自动推进，用户不用管&lt;/li&gt;
&lt;li&gt;fork 后父子可以共享同一个打开文件（共享 offset）&lt;/li&gt;
&lt;li&gt;文件、管道、设备用同一套接口，背后实现不同&lt;/li&gt;
&lt;/ul&gt;
&lt;hr&gt;
&lt;h2 id=&#34;实现每个进程一张-fd-表&#34;&gt;实现：每个进程一张 fd 表&lt;/h2&gt;
&lt;p&gt;在 &lt;code&gt;process_t&lt;/code&gt; 里加一个数组：&lt;/p&gt;
&lt;div class=&#34;highlight&#34;&gt;&lt;pre tabindex=&#34;0&#34; style=&#34;color:#f8f8f2;background-color:#272822;-moz-tab-size:4;-o-tab-size:4;tab-size:4;-webkit-text-size-adjust:none;&#34;&gt;&lt;code class=&#34;language-c&#34; data-lang=&#34;c&#34;&gt;&lt;span style=&#34;display:flex;&#34;&gt;&lt;span&gt;&lt;span style=&#34;color:#66d9ef&#34;&gt;typedef&lt;/span&gt; &lt;span style=&#34;color:#66d9ef&#34;&gt;struct&lt;/span&gt; {
&lt;/span&gt;&lt;/span&gt;&lt;span style=&#34;display:flex;&#34;&gt;&lt;span&gt;    &lt;span style=&#34;color:#75715e&#34;&gt;// ...
&lt;/span&gt;&lt;/span&gt;&lt;/span&gt;&lt;span style=&#34;display:flex;&#34;&gt;&lt;span&gt;    &lt;span style=&#34;color:#66d9ef&#34;&gt;int32_t&lt;/span&gt; fd_table[PROC_MAX_FD];  &lt;span style=&#34;color:#75715e&#34;&gt;// fd → vfs_fd，-1 表示空槽
&lt;/span&gt;&lt;/span&gt;&lt;/span&gt;&lt;span style=&#34;display:flex;&#34;&gt;&lt;span&gt;} &lt;span style=&#34;color:#66d9ef&#34;&gt;process_t&lt;/span&gt;;
&lt;/span&gt;&lt;/span&gt;&lt;/code&gt;&lt;/pre&gt;&lt;/div&gt;&lt;p&gt;&lt;code&gt;fd_table[fd]&lt;/code&gt; 存的是 VFS 层的内部 fd（VFile 数组下标）。&lt;/p&gt;</description>
    </item>
    <item>
      <title>从零写OS（十五）：挂载 ext2，读真实文件系统</title>
      <link>https://www.dafei.me/posts/os-15-ext2/</link>
      <pubDate>Wed, 06 May 2026 15:00:00 +0000</pubDate>
      <guid>https://www.dafei.me/posts/os-15-ext2/</guid>
      <description>&lt;p&gt;前几章的文件系统是存在内存里的——重启数据就没了，文件名也是硬编码的。这一章做真实的：挂载一个 ext2 磁盘镜像，让 Shell 能读取里面的文件。&lt;/p&gt;
&lt;p&gt;上一章已经有了 ATA 驱动，能按扇区号读磁盘。现在的问题是：磁盘上的数据是怎么组织的？&lt;/p&gt;
&lt;hr&gt;
&lt;h2 id=&#34;ext2-的结构&#34;&gt;ext2 的结构&lt;/h2&gt;
&lt;p&gt;ext2 是 Linux 最经典的文件系统，ext3/ext4 都是在它基础上演化来的。理解 ext2，基本上就理解了现代文件系统的核心思路。&lt;/p&gt;
&lt;p&gt;磁盘从偏移 1024 字节开始是 &lt;strong&gt;Superblock&lt;/strong&gt;，存整个文件系统的基本参数：block 大小是多少、有多少 inode、magic number 是 &lt;code&gt;0xEF53&lt;/code&gt;（用来确认这确实是 ext2）。&lt;/p&gt;
&lt;p&gt;接下来是 &lt;strong&gt;Group Descriptor&lt;/strong&gt;，告诉你 inode table 在哪个 block。&lt;/p&gt;
&lt;p&gt;然后才是真正的数据区：inode table 和数据块。&lt;/p&gt;
&lt;p&gt;&lt;strong&gt;Inode&lt;/strong&gt; 是文件的&amp;quot;身份证&amp;quot;。每个文件有一个唯一的 inode 号，inode 里存着文件大小、权限，以及最重要的——&lt;code&gt;i_block[0..11]&lt;/code&gt;，12 个直接指向数据块的指针。想读文件内容，就顺着这些指针去读对应的数据块。&lt;/p&gt;
&lt;p&gt;&lt;strong&gt;目录&lt;/strong&gt;也是文件，它的数据块里存的是一条条 &lt;code&gt;ext2_dir_entry&lt;/code&gt;：每条记录包含 inode 号、文件名长度、文件名。&lt;code&gt;ls&lt;/code&gt; 就是读根目录的数据块，遍历这些记录。&lt;/p&gt;
&lt;hr&gt;
&lt;h2 id=&#34;读文件的完整路径&#34;&gt;读文件的完整路径&lt;/h2&gt;
&lt;pre tabindex=&#34;0&#34;&gt;&lt;code&gt;Superblock  → 拿到 block_size、inodes_per_group
GroupDesc   → 拿到 inode table 的起始 block 号
Inode[ino]  → 拿到文件大小和 i_block[]
DataBlock   → 真正的文件内容
&lt;/code&gt;&lt;/pre&gt;&lt;p&gt;读目录（ls）就在最后一步多做一件事：把数据块里的 &lt;code&gt;ext2_dir_entry&lt;/code&gt; 链遍历一遍。&lt;/p&gt;
&lt;p&gt;每一步都需要从磁盘读若干个扇区——这正是上一章 ATA 驱动的用武之地。&lt;/p&gt;</description>
    </item>
    <item>
      <title>从零写OS（十二）：VFS，让系统调用不认识具体文件系统</title>
      <link>https://www.dafei.me/posts/os-12-vfs/</link>
      <pubDate>Wed, 06 May 2026 12:00:00 +0000</pubDate>
      <guid>https://www.dafei.me/posts/os-12-vfs/</guid>
      <description>&lt;p&gt;上一章我们写了一个 SimpleFS，可以创建文件、读写内容。但现在有个问题：&lt;code&gt;open()&lt;/code&gt; 系统调用直接调的是 &lt;code&gt;fs_create&lt;/code&gt;、&lt;code&gt;fs_read&lt;/code&gt; 这些 SimpleFS 专属函数。&lt;/p&gt;
&lt;p&gt;如果哪天要支持 FAT32，就得去改系统调用的代码。这显然不对。&lt;/p&gt;
&lt;p&gt;这一章加一层 &lt;strong&gt;VFS（Virtual File System，虚拟文件系统）&lt;/strong&gt;，把系统调用和具体文件系统隔开。&lt;/p&gt;
&lt;hr&gt;
&lt;h2 id=&#34;先把概念搞清楚&#34;&gt;先把概念搞清楚&lt;/h2&gt;
&lt;h3 id=&#34;间接层解耦&#34;&gt;间接层解耦&lt;/h3&gt;
&lt;p&gt;软件里有句老话：&lt;strong&gt;任何问题都可以通过加一层间接层解决&lt;/strong&gt;。VFS 就是这层间接层。&lt;/p&gt;
&lt;pre tabindex=&#34;0&#34;&gt;&lt;code&gt;用户进程
   ↓  open / read / write
  VFS（统一接口）
   ↓              ↓
SimpleFS         FAT32
&lt;/code&gt;&lt;/pre&gt;&lt;p&gt;系统调用只跟 VFS 说话，VFS 再转发给具体 FS。新增一种文件系统，只需要实现 VFS 要求的那几个函数，不动系统调用层。&lt;/p&gt;
&lt;h3 id=&#34;file_operationsc-语言的多态&#34;&gt;file_operations：C 语言的多态&lt;/h3&gt;
&lt;p&gt;VFS 要求每种文件系统提供一张&lt;strong&gt;函数指针表&lt;/strong&gt;：&lt;/p&gt;
&lt;div class=&#34;highlight&#34;&gt;&lt;pre tabindex=&#34;0&#34; style=&#34;color:#f8f8f2;background-color:#272822;-moz-tab-size:4;-o-tab-size:4;tab-size:4;-webkit-text-size-adjust:none;&#34;&gt;&lt;code class=&#34;language-c&#34; data-lang=&#34;c&#34;&gt;&lt;span style=&#34;display:flex;&#34;&gt;&lt;span&gt;&lt;span style=&#34;color:#66d9ef&#34;&gt;typedef&lt;/span&gt; &lt;span style=&#34;color:#66d9ef&#34;&gt;struct&lt;/span&gt; {
&lt;/span&gt;&lt;/span&gt;&lt;span style=&#34;display:flex;&#34;&gt;&lt;span&gt;    &lt;span style=&#34;color:#66d9ef&#34;&gt;int&lt;/span&gt; (&lt;span style=&#34;color:#f92672&#34;&gt;*&lt;/span&gt;read) (...);
&lt;/span&gt;&lt;/span&gt;&lt;span style=&#34;display:flex;&#34;&gt;&lt;span&gt;    &lt;span style=&#34;color:#66d9ef&#34;&gt;int&lt;/span&gt; (&lt;span style=&#34;color:#f92672&#34;&gt;*&lt;/span&gt;write)(...);
&lt;/span&gt;&lt;/span&gt;&lt;span style=&#34;display:flex;&#34;&gt;&lt;span&gt;    &lt;span style=&#34;color:#66d9ef&#34;&gt;uint32_t&lt;/span&gt; (&lt;span style=&#34;color:#f92672&#34;&gt;*&lt;/span&gt;lookup)(...);
&lt;/span&gt;&lt;/span&gt;&lt;span style=&#34;display:flex;&#34;&gt;&lt;span&gt;    &lt;span style=&#34;color:#66d9ef&#34;&gt;uint32_t&lt;/span&gt; (&lt;span style=&#34;color:#f92672&#34;&gt;*&lt;/span&gt;create)(...);
&lt;/span&gt;&lt;/span&gt;&lt;span style=&#34;display:flex;&#34;&gt;&lt;span&gt;} FileOps;
&lt;/span&gt;&lt;/span&gt;&lt;/code&gt;&lt;/pre&gt;&lt;/div&gt;&lt;p&gt;VFS 调 &lt;code&gt;fops-&amp;gt;read(...)&lt;/code&gt; 时，实际执行的是哪个函数，取决于挂载时注册的是哪张表。这是 C 语言实现&amp;quot;多态&amp;quot;的标准手法，Linux 内核里到处都是这个模式。&lt;/p&gt;
&lt;h3 id=&#34;vnodevfs-的通用-inode&#34;&gt;vnode：VFS 的通用 inode&lt;/h3&gt;
&lt;p&gt;具体 FS 有自己的 inode，VFS 层包一层 &lt;code&gt;vnode&lt;/code&gt;，里面放着具体 FS 的 inode 号和对应的函数指针表。对上层完全屏蔽了底层差异。&lt;/p&gt;</description>
    </item>
  </channel>
</rss>
