<?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/%E8%A1%8C%E7%BC%93%E5%86%B2/</link>
    <description>Recent content in 行缓冲 on 大飞的博客</description>
    <generator>Hugo</generator>
    <language>zh-cn</language>
    <lastBuildDate>Fri, 22 May 2026 04:00:00 +0000</lastBuildDate>
    <atom:link href="https://www.dafei.me/tags/%E8%A1%8C%E7%BC%93%E5%86%B2/index.xml" rel="self" type="application/rss+xml" />
    <item>
      <title>从零写OS（三十三）：阻塞式 TTY —— read 不再忙等</title>
      <link>https://www.dafei.me/posts/os-33-blocking-tty/</link>
      <pubDate>Fri, 22 May 2026 04:00:00 +0000</pubDate>
      <guid>https://www.dafei.me/posts/os-33-blocking-tty/</guid>
      <description>&lt;p&gt;上一章 busybox sh 成功显示了 &lt;code&gt;/ #&lt;/code&gt; 提示符，但 shell 拿不到任何输入——因为 &lt;code&gt;tty_read&lt;/code&gt; 是忙轮询的，没有字符就直接返回 0，shell 以为收到 EOF，立刻退出。这一章实现真正的阻塞式 TTY，让 shell 能等待用户输入。&lt;/p&gt;
&lt;h2 id=&#34;之前的问题&#34;&gt;之前的问题&lt;/h2&gt;
&lt;p&gt;之前的 &lt;code&gt;tty_read&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;int&lt;/span&gt; &lt;span style=&#34;color:#a6e22e&#34;&gt;tty_read&lt;/span&gt;(&lt;span style=&#34;color:#66d9ef&#34;&gt;char&lt;/span&gt; &lt;span style=&#34;color:#f92672&#34;&gt;*&lt;/span&gt;buf, &lt;span style=&#34;color:#66d9ef&#34;&gt;int&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:#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;if&lt;/span&gt; (&lt;span style=&#34;color:#f92672&#34;&gt;!&lt;/span&gt;(&lt;span style=&#34;color:#a6e22e&#34;&gt;inb&lt;/span&gt;(&lt;span style=&#34;color:#ae81ff&#34;&gt;0x3F8&lt;/span&gt; &lt;span style=&#34;color:#f92672&#34;&gt;+&lt;/span&gt; &lt;span style=&#34;color:#ae81ff&#34;&gt;5&lt;/span&gt;) &lt;span style=&#34;color:#f92672&#34;&gt;&amp;amp;&lt;/span&gt; &lt;span style=&#34;color:#ae81ff&#34;&gt;1&lt;/span&gt;)) &lt;span style=&#34;color:#66d9ef&#34;&gt;return&lt;/span&gt; &lt;span style=&#34;color:#ae81ff&#34;&gt;0&lt;/span&gt;;   &lt;span style=&#34;color:#75715e&#34;&gt;// 没数据就返回 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:#f92672&#34;&gt;*&lt;/span&gt;buf &lt;span style=&#34;color:#f92672&#34;&gt;=&lt;/span&gt; &lt;span style=&#34;color:#a6e22e&#34;&gt;inb&lt;/span&gt;(&lt;span style=&#34;color:#ae81ff&#34;&gt;0x3F8&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;return&lt;/span&gt; &lt;span style=&#34;color:#ae81ff&#34;&gt;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;p&gt;这有两个问题：&lt;/p&gt;
&lt;ol&gt;
&lt;li&gt;没有输入时返回 0，上层程序（shell）以为是 EOF&lt;/li&gt;
&lt;li&gt;如果上层在循环里调这个，CPU 100% 占用&lt;/li&gt;
&lt;/ol&gt;
&lt;p&gt;正确做法：没有输入时&lt;strong&gt;挂起进程&lt;/strong&gt;，等串口中断来了再唤醒。&lt;/p&gt;
&lt;h2 id=&#34;串口中断irq4&#34;&gt;串口中断（IRQ4）&lt;/h2&gt;
&lt;p&gt;COM1 对应 IRQ4，接在 PIC 主片的 IR4 引脚。要用串口中断，需要两步：&lt;/p&gt;
&lt;p&gt;&lt;strong&gt;1. 在 PIC 上 unmask IRQ4&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:#75715e&#34;&gt;// PIC 主片 IMR 寄存器：0 表示开放，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;uint8_t&lt;/span&gt; mask &lt;span style=&#34;color:#f92672&#34;&gt;=&lt;/span&gt; &lt;span style=&#34;color:#a6e22e&#34;&gt;inb&lt;/span&gt;(&lt;span style=&#34;color:#ae81ff&#34;&gt;0x21&lt;/span&gt;);
&lt;/span&gt;&lt;/span&gt;&lt;span style=&#34;display:flex;&#34;&gt;&lt;span&gt;mask &lt;span style=&#34;color:#f92672&#34;&gt;&amp;amp;=&lt;/span&gt; &lt;span style=&#34;color:#f92672&#34;&gt;~&lt;/span&gt;(&lt;span style=&#34;color:#ae81ff&#34;&gt;1&lt;/span&gt; &lt;span style=&#34;color:#f92672&#34;&gt;&amp;lt;&amp;lt;&lt;/span&gt; &lt;span style=&#34;color:#ae81ff&#34;&gt;4&lt;/span&gt;);   &lt;span style=&#34;color:#75715e&#34;&gt;// 清除 bit4，开放 IRQ4
&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;outb&lt;/span&gt;(&lt;span style=&#34;color:#ae81ff&#34;&gt;0x21&lt;/span&gt;, mask);
&lt;/span&gt;&lt;/span&gt;&lt;/code&gt;&lt;/pre&gt;&lt;/div&gt;&lt;p&gt;&lt;strong&gt;2. 开启串口的接收中断&lt;/strong&gt;：&lt;/p&gt;</description>
    </item>
  </channel>
</rss>
