<?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>SIGPIPE on 大飞的博客</title>
    <link>https://www.dafei.me/tags/sigpipe/</link>
    <description>Recent content in SIGPIPE on 大飞的博客</description>
    <generator>Hugo</generator>
    <language>zh-cn</language>
    <lastBuildDate>Mon, 08 Jun 2026 10:00:00 +0000</lastBuildDate>
    <atom:link href="https://www.dafei.me/tags/sigpipe/index.xml" rel="self" type="application/rss+xml" />
    <item>
      <title>从零写OS（五十四）：完整信号系统——sigaction、mask 与 sigreturn</title>
      <link>https://www.dafei.me/posts/os-54-signal/</link>
      <pubDate>Mon, 08 Jun 2026 10:00:00 +0000</pubDate>
      <guid>https://www.dafei.me/posts/os-54-signal/</guid>
      <description>&lt;p&gt;跑 busybox sh 时发现一个诡异的问题：&lt;code&gt;Ctrl+C&lt;/code&gt; 能杀掉进程，但 handler 执行完之后程序直接崩了——寄存器全乱了。调查下去，发现 ch53 的信号系统有三个根本缺陷。&lt;/p&gt;
&lt;h2 id=&#34;问题一寄存器没有保存&#34;&gt;问题一：寄存器没有保存&lt;/h2&gt;
&lt;p&gt;ch53 的 &lt;code&gt;signal_dispatch&lt;/code&gt; 只保存了 &lt;code&gt;user_rip&lt;/code&gt; 和 &lt;code&gt;user_rsp&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;uint64_t&lt;/span&gt; new_rsp &lt;span style=&#34;color:#f92672&#34;&gt;=&lt;/span&gt; &lt;span style=&#34;color:#f92672&#34;&gt;*&lt;/span&gt;user_rsp &lt;span style=&#34;color:#f92672&#34;&gt;-&lt;/span&gt; &lt;span style=&#34;color:#ae81ff&#34;&gt;8&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;(&lt;span style=&#34;color:#66d9ef&#34;&gt;uint64_t&lt;/span&gt; &lt;span style=&#34;color:#f92672&#34;&gt;*&lt;/span&gt;)(phys &lt;span style=&#34;color:#f92672&#34;&gt;+&lt;/span&gt; ...) &lt;span style=&#34;color:#f92672&#34;&gt;=&lt;/span&gt; &lt;span style=&#34;color:#f92672&#34;&gt;*&lt;/span&gt;user_rip;  &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:#f92672&#34;&gt;*&lt;/span&gt;user_rsp &lt;span style=&#34;color:#f92672&#34;&gt;=&lt;/span&gt; new_rsp;
&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;user_rip &lt;span style=&#34;color:#f92672&#34;&gt;=&lt;/span&gt; handler;
&lt;/span&gt;&lt;/span&gt;&lt;/code&gt;&lt;/pre&gt;&lt;/div&gt;&lt;p&gt;handler 执行期间会用到 &lt;code&gt;r15&lt;/code&gt;, &lt;code&gt;r14&lt;/code&gt;, &amp;hellip;, &lt;code&gt;rbx&lt;/code&gt;, &lt;code&gt;rbp&lt;/code&gt;——这些全都没有保存。handler 返回时弹出的 rip 确实是原来的位置，但所有调用者保存的寄存器都被 handler 破坏了，接下来的代码跑的是垃圾数据。&lt;/p&gt;
&lt;h2 id=&#34;问题二sigaction-是空-stub&#34;&gt;问题二：sigaction 是空 stub&lt;/h2&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;case&lt;/span&gt; SYS_SIGACTION:
&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;case&lt;/span&gt; SYS_SIGPROCMASK:
&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;case&lt;/span&gt; SYS_SIGRETURN:
&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;0&lt;/span&gt;;   &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;musl 的 &lt;code&gt;signal()&lt;/code&gt; 底层调用 &lt;code&gt;sigaction&lt;/code&gt;，返回 0 让它以为设置成功了，实际上 handler 根本没有注册进去。&lt;/p&gt;
&lt;h2 id=&#34;问题三没有-signal-mask&#34;&gt;问题三：没有 signal mask&lt;/h2&gt;
&lt;p&gt;handler 执行期间，如果同一个信号再次到来，会再次触发 dispatch，handler 递归执行，栈很快就溢出了。&lt;/p&gt;
&lt;hr&gt;
&lt;h2 id=&#34;解法signal_frame_t--真正的-sigaction&#34;&gt;解法：signal_frame_t + 真正的 sigaction&lt;/h2&gt;
&lt;h3 id=&#34;核心数据结构&#34;&gt;核心数据结构&lt;/h3&gt;
&lt;p&gt;把所有寄存器打包成一个结构体，压到用户栈：&lt;/p&gt;</description>
    </item>
  </channel>
</rss>
