<?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>字符设备 on 大飞的博客</title>
    <link>https://www.dafei.me/tags/%E5%AD%97%E7%AC%A6%E8%AE%BE%E5%A4%87/</link>
    <description>Recent content in 字符设备 on 大飞的博客</description>
    <generator>Hugo</generator>
    <language>zh-cn</language>
    <lastBuildDate>Fri, 22 May 2026 07:00:00 +0000</lastBuildDate>
    <atom:link href="https://www.dafei.me/tags/%E5%AD%97%E7%AC%A6%E8%AE%BE%E5%A4%87/index.xml" rel="self" type="application/rss+xml" />
    <item>
      <title>从零写OS（三十六）：ext2 写操作 —— 文件创建、引用计数与 tty 字符设备</title>
      <link>https://www.dafei.me/posts/os-36-ext2-write/</link>
      <pubDate>Fri, 22 May 2026 07:00:00 +0000</pubDate>
      <guid>https://www.dafei.me/posts/os-36-ext2-write/</guid>
      <description>&lt;p&gt;前几章文件系统都是只读的。这一章实现 ext2 的写路径，让 &lt;code&gt;echo hello &amp;gt; /tmp/a.txt &amp;amp;&amp;amp; cat /tmp/a.txt&lt;/code&gt; 能正常工作，且重启后文件仍然存在。过程中还修了 fd 引用计数和 tty 字符设备两个问题。&lt;/p&gt;
&lt;h2 id=&#34;ext2-磁盘结构回顾&#34;&gt;ext2 磁盘结构回顾&lt;/h2&gt;
&lt;pre tabindex=&#34;0&#34;&gt;&lt;code&gt;块组 0:
  [超级块][组描述符][block bitmap][inode bitmap][inode table][数据块...]
&lt;/code&gt;&lt;/pre&gt;&lt;p&gt;关键字段：&lt;/p&gt;
&lt;ul&gt;
&lt;li&gt;&lt;code&gt;sb-&amp;gt;s_free_inodes_count&lt;/code&gt; / &lt;code&gt;gd-&amp;gt;bg_free_inodes_count&lt;/code&gt;：空闲 inode 计数&lt;/li&gt;
&lt;li&gt;&lt;code&gt;sb-&amp;gt;s_free_blocks_count&lt;/code&gt; / &lt;code&gt;gd-&amp;gt;bg_free_blocks_count&lt;/code&gt;：空闲 block 计数&lt;/li&gt;
&lt;li&gt;&lt;code&gt;gd-&amp;gt;bg_inode_bitmap&lt;/code&gt;：inode bitmap 所在块号&lt;/li&gt;
&lt;li&gt;&lt;code&gt;gd-&amp;gt;bg_block_bitmap&lt;/code&gt;：block bitmap 所在块号&lt;/li&gt;
&lt;li&gt;&lt;code&gt;gd-&amp;gt;bg_inode_table&lt;/code&gt;：inode table 起始块号&lt;/li&gt;
&lt;/ul&gt;
&lt;p&gt;inode 号从 1 开始；前 10 个是系统保留的（root=2，lost+found=11）。&lt;/p&gt;
&lt;h2 id=&#34;alloc_inode--alloc_block&#34;&gt;alloc_inode / alloc_block&lt;/h2&gt;
&lt;h3 id=&#34;alloc_inode&#34;&gt;alloc_inode&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;uint32_t&lt;/span&gt; &lt;span style=&#34;color:#a6e22e&#34;&gt;alloc_inode&lt;/span&gt;(&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:#75715e&#34;&gt;// 1. 读 inode bitmap 块
&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;// 2. 找第一个为 0 的位（bit i → inode i+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:#75715e&#34;&gt;// 3. 置位，写回 bitmap
&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;// 4. 更新超级块和组描述符的 free_inodes_count
&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;// 5. 返回 inode 号（从 1 开始）
&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;/code&gt;&lt;/pre&gt;&lt;/div&gt;&lt;h3 id=&#34;alloc_block&#34;&gt;alloc_block&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;uint32_t&lt;/span&gt; &lt;span style=&#34;color:#a6e22e&#34;&gt;alloc_block&lt;/span&gt;(&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:#75715e&#34;&gt;// 1. 读 block bitmap 块
&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;// 2. 找第一个为 0 的位，跳过 block 0（保留）
&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;// 3. 置位，写回 bitmap
&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;// 4. 更新超级块和组描述符的 free_blocks_count
&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;// 5. 清零新分配的块（避免垃圾数据）
&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;// 6. 返回块号
&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;/code&gt;&lt;/pre&gt;&lt;/div&gt;&lt;p&gt;注意：block bitmap 的 bit 0 对应 block 0，必须跳过（block 0 不存在）。实际第一个可分配的块由文件系统布局决定（本项目中是 610）。&lt;/p&gt;</description>
    </item>
    <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>
  </channel>
</rss>
