<?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/%E9%A9%B1%E5%8A%A8/</link>
    <description>Recent content in 驱动 on 大飞的博客</description>
    <generator>Hugo</generator>
    <language>zh-cn</language>
    <lastBuildDate>Wed, 06 May 2026 14:00:00 +0000</lastBuildDate>
    <atom:link href="https://www.dafei.me/tags/%E9%A9%B1%E5%8A%A8/index.xml" rel="self" type="application/rss+xml" />
    <item>
      <title>从零写OS（十四）：ATA 驱动，让内核能读磁盘</title>
      <link>https://www.dafei.me/posts/os-14-ata/</link>
      <pubDate>Wed, 06 May 2026 14:00:00 +0000</pubDate>
      <guid>https://www.dafei.me/posts/os-14-ata/</guid>
      <description>&lt;p&gt;前面的文件系统都是存在内存里的——重启数据就没了。要读真实磁盘，得先搞清楚操作系统怎么和磁盘&amp;quot;说话&amp;quot;。这一章做 ATA 磁盘驱动。&lt;/p&gt;
&lt;hr&gt;
&lt;h2 id=&#34;磁盘和内核怎么通信&#34;&gt;磁盘和内核怎么通信&lt;/h2&gt;
&lt;p&gt;硬盘插在主板上，操作系统通过 &lt;strong&gt;ATA 协议&lt;/strong&gt;和它通信。ATA 协议规定了一组固定的 x86 I/O 端口，内核用 &lt;code&gt;in&lt;/code&gt;/&lt;code&gt;out&lt;/code&gt; 指令直接操作这些端口，就能控制磁盘。&lt;/p&gt;
&lt;p&gt;这叫 &lt;strong&gt;PIO 模式&lt;/strong&gt;（Programmed I/O）——CPU 亲自一个字搬一个字地读数据。慢，但实现只需要几十行代码，是学习的最佳起点。&lt;/p&gt;
&lt;p&gt;真实生产内核用 DMA（磁盘直接写内存，CPU 不搬数据），那是后话。&lt;/p&gt;
&lt;hr&gt;
&lt;h2 id=&#34;磁盘的最小单位扇区&#34;&gt;磁盘的最小单位：扇区&lt;/h2&gt;
&lt;p&gt;磁盘被切成 512 字节的&lt;strong&gt;扇区&lt;/strong&gt;，每个扇区有一个编号，叫 &lt;strong&gt;LBA&lt;/strong&gt;（Logical Block Address），从 0 开始数。&lt;/p&gt;
&lt;pre tabindex=&#34;0&#34;&gt;&lt;code&gt;LBA=0 → 前 512 字节（Boot Sector）
LBA=1 → 512~1023 字节
LBA=2 → 1024~1535 字节（ext2 Superblock 就在这里）
...
&lt;/code&gt;&lt;/pre&gt;&lt;p&gt;要读文件系统里偏移 1024 字节的内容，就是读 &lt;code&gt;LBA = 1024 / 512 = 2&lt;/code&gt;。&lt;/p&gt;
&lt;hr&gt;
&lt;h2 id=&#34;ata-端口&#34;&gt;ATA 端口&lt;/h2&gt;
&lt;p&gt;ATA 协议设计于 1980 年代，把控制磁盘的所有操作映射到一组固定的 I/O 端口号上——这是当时 PC 硬件的惯例，如今这些端口号已经写死在无数设备里，成了不能改的&amp;quot;历史遗产&amp;quot;。&lt;/p&gt;
&lt;p&gt;Primary ATA 控制器的端口：&lt;/p&gt;
&lt;table&gt;
  &lt;thead&gt;
      &lt;tr&gt;
          &lt;th&gt;端口&lt;/th&gt;
          &lt;th&gt;用途&lt;/th&gt;
      &lt;/tr&gt;
  &lt;/thead&gt;
  &lt;tbody&gt;
      &lt;tr&gt;
          &lt;td&gt;&lt;code&gt;0x1F0&lt;/code&gt;&lt;/td&gt;
          &lt;td&gt;数据（读写 16-bit）&lt;/td&gt;
      &lt;/tr&gt;
      &lt;tr&gt;
          &lt;td&gt;&lt;code&gt;0x1F2&lt;/code&gt;&lt;/td&gt;
          &lt;td&gt;要读几个扇区&lt;/td&gt;
      &lt;/tr&gt;
      &lt;tr&gt;
          &lt;td&gt;&lt;code&gt;0x1F3&lt;/code&gt;&lt;/td&gt;
          &lt;td&gt;LBA[7:0]&lt;/td&gt;
      &lt;/tr&gt;
      &lt;tr&gt;
          &lt;td&gt;&lt;code&gt;0x1F4&lt;/code&gt;&lt;/td&gt;
          &lt;td&gt;LBA[15:8]&lt;/td&gt;
      &lt;/tr&gt;
      &lt;tr&gt;
          &lt;td&gt;&lt;code&gt;0x1F5&lt;/code&gt;&lt;/td&gt;
          &lt;td&gt;LBA[23:16]&lt;/td&gt;
      &lt;/tr&gt;
      &lt;tr&gt;
          &lt;td&gt;&lt;code&gt;0x1F6&lt;/code&gt;&lt;/td&gt;
          &lt;td&gt;选盘 + LBA[27:24]&lt;/td&gt;
      &lt;/tr&gt;
      &lt;tr&gt;
          &lt;td&gt;&lt;code&gt;0x1F7&lt;/code&gt;&lt;/td&gt;
          &lt;td&gt;命令（写）/ 状态（读）&lt;/td&gt;
      &lt;/tr&gt;
  &lt;/tbody&gt;
&lt;/table&gt;
&lt;p&gt;状态寄存器的两个关键位：&lt;/p&gt;</description>
    </item>
  </channel>
</rss>
