<?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%87%8D%E5%AE%9A%E5%90%91/</link>
    <description>Recent content in 重定向 on 大飞的博客</description>
    <generator>Hugo</generator>
    <language>zh-cn</language>
    <lastBuildDate>Thu, 14 May 2026 02:00:00 +0000</lastBuildDate>
    <atom:link href="https://www.dafei.me/tags/%E9%87%8D%E5%AE%9A%E5%90%91/index.xml" rel="self" type="application/rss+xml" />
    <item>
      <title>从零写OS（二十二）：管道 pipe 与 dup2</title>
      <link>https://www.dafei.me/posts/os-22-pipe/</link>
      <pubDate>Thu, 14 May 2026 02:00:00 +0000</pubDate>
      <guid>https://www.dafei.me/posts/os-22-pipe/</guid>
      <description>&lt;p&gt;上一章实现了文件描述符，进程可以用 &lt;code&gt;open/read/close&lt;/code&gt; 访问 ext2 文件。但两个进程之间还没有办法通信。shell pipeline &lt;code&gt;cmd1 | cmd2&lt;/code&gt; 的本质是：cmd1 的输出直接流进 cmd2 的输入，中间不落磁盘。&lt;/p&gt;
&lt;p&gt;这一章实现 &lt;strong&gt;pipe&lt;/strong&gt; 和 &lt;strong&gt;dup2&lt;/strong&gt;，打通进程间通信的最小路径。&lt;/p&gt;
&lt;hr&gt;
&lt;h2 id=&#34;pipe-是什么&#34;&gt;pipe 是什么&lt;/h2&gt;
&lt;p&gt;pipe 是内核里的一块环形字节缓冲区。系统调用 &lt;code&gt;pipe(fds)&lt;/code&gt; 返回两个文件描述符：&lt;/p&gt;
&lt;pre tabindex=&#34;0&#34;&gt;&lt;code&gt;fds[0] = 读端    fds[1] = 写端

写进程 → fds[1] → [内核 ring buffer 256字节] → fds[0] → 读进程
&lt;/code&gt;&lt;/pre&gt;&lt;p&gt;和普通文件一样，管道也走 fd → VFS → 底层数据 这三层，只不过底层数据不是磁盘，而是内核里的 &lt;code&gt;pipe_t&lt;/code&gt;。&lt;/p&gt;
&lt;hr&gt;
&lt;h2 id=&#34;dup2-是什么&#34;&gt;dup2 是什么&lt;/h2&gt;
&lt;p&gt;&lt;code&gt;dup2(oldfd, newfd)&lt;/code&gt; 让 &lt;code&gt;newfd&lt;/code&gt; 指向和 &lt;code&gt;oldfd&lt;/code&gt; 同一个内核文件描述符，如果 &lt;code&gt;newfd&lt;/code&gt; 已经打开则先关掉它。&lt;/p&gt;
&lt;p&gt;shell pipeline 的核心操作就是 dup2 + exec：&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;// 子进程（cmd1），stdout → pipe 写端
&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;dup2&lt;/span&gt;(fds[&lt;span style=&#34;color:#ae81ff&#34;&gt;1&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 style=&#34;color:#a6e22e&#34;&gt;close&lt;/span&gt;(fds[&lt;span style=&#34;color:#ae81ff&#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:#a6e22e&#34;&gt;close&lt;/span&gt;(fds[&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 style=&#34;color:#a6e22e&#34;&gt;exec&lt;/span&gt;(&lt;span style=&#34;color:#e6db74&#34;&gt;&amp;#34;cmd1&amp;#34;&lt;/span&gt;);    &lt;span style=&#34;color:#75715e&#34;&gt;// cmd1 以为自己在写 stdout，实际写进了管道
&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:#75715e&#34;&gt;// 父进程（cmd2），stdin → pipe 读端
&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;dup2&lt;/span&gt;(fds[&lt;span style=&#34;color:#ae81ff&#34;&gt;0&lt;/span&gt;], &lt;span style=&#34;color:#ae81ff&#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:#a6e22e&#34;&gt;close&lt;/span&gt;(fds[&lt;span style=&#34;color:#ae81ff&#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:#a6e22e&#34;&gt;close&lt;/span&gt;(fds[&lt;span style=&#34;color:#ae81ff&#34;&gt;1&lt;/span&gt;]);   &lt;span style=&#34;color:#75715e&#34;&gt;// 关掉写端，否则 cmd2 永远等不到 EOF
&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;exec&lt;/span&gt;(&lt;span style=&#34;color:#e6db74&#34;&gt;&amp;#34;cmd2&amp;#34;&lt;/span&gt;);
&lt;/span&gt;&lt;/span&gt;&lt;/code&gt;&lt;/pre&gt;&lt;/div&gt;&lt;hr&gt;
&lt;h2 id=&#34;pipe-的-eof-语义&#34;&gt;pipe 的 EOF 语义&lt;/h2&gt;
&lt;p&gt;&lt;code&gt;read&lt;/code&gt; 管道什么时候返回 0（EOF）？&lt;/p&gt;</description>
    </item>
  </channel>
</rss>
