<?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>DNS on 大飞的博客</title>
    <link>https://www.dafei.me/tags/dns/</link>
    <description>Recent content in DNS on 大飞的博客</description>
    <generator>Hugo</generator>
    <language>zh-cn</language>
    <lastBuildDate>Tue, 02 Jun 2026 10:00:00 +0000</lastBuildDate>
    <atom:link href="https://www.dafei.me/tags/dns/index.xml" rel="self" type="application/rss+xml" />
    <item>
      <title>从零写OS（四十四）：UDP Socket &#43; DNS 解析</title>
      <link>https://www.dafei.me/posts/os-44-udp-dns/</link>
      <pubDate>Tue, 02 Jun 2026 10:00:00 +0000</pubDate>
      <guid>https://www.dafei.me/posts/os-44-udp-dns/</guid>
      <description>&lt;p&gt;上一章（四十三）把 TCP 重传做好了。这一章加两个新功能：&lt;strong&gt;UDP socket 用户空间接口&lt;/strong&gt; 和 &lt;strong&gt;DNS 解析 syscall&lt;/strong&gt;，让用户程序可以用域名来连接服务器。&lt;/p&gt;
&lt;h2 id=&#34;目标&#34;&gt;目标&lt;/h2&gt;
&lt;p&gt;内核里其实早就有 &lt;code&gt;udp_send&lt;/code&gt; 和 &lt;code&gt;dns_resolve&lt;/code&gt; 这两个函数了，但用户程序用不到，因为没有对应的 syscall。这章要做的就是把这两个内核能力&amp;quot;打通&amp;quot;到用户空间：&lt;/p&gt;
&lt;ul&gt;
&lt;li&gt;&lt;code&gt;socket(AF_INET, SOCK_DGRAM, 0)&lt;/code&gt; → 分配一个 UDP socket，返回 fd 200+&lt;/li&gt;
&lt;li&gt;&lt;code&gt;sendto&lt;/code&gt; / &lt;code&gt;recvfrom&lt;/code&gt; → 通过 UDP socket 收发数据&lt;/li&gt;
&lt;li&gt;&lt;code&gt;SYS_DNS_RESOLVE&lt;/code&gt;（自定义 syscall 500）→ 通过域名查 IP&lt;/li&gt;
&lt;/ul&gt;
&lt;h2 id=&#34;udp-socket-设计&#34;&gt;UDP Socket 设计&lt;/h2&gt;
&lt;p&gt;UDP 比 TCP 简单很多：无连接、无状态机、无重传。核心是一个&lt;strong&gt;接收队列&lt;/strong&gt;：内核收到 UDP 包时，把它放进对应 socket 的队列，用户程序再来取。&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;uint32_t&lt;/span&gt; src_ip;
&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;uint16_t&lt;/span&gt; src_port;
&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;uint16_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;uint8_t&lt;/span&gt;  data[&lt;span style=&#34;color:#ae81ff&#34;&gt;512&lt;/span&gt;];      &lt;span style=&#34;color:#75715e&#34;&gt;// 每个包最多 512 字节
&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;udp_pkt_entry_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;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;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;uint16_t&lt;/span&gt;         local_port;
&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;udp_pkt_entry_t&lt;/span&gt;  queue[&lt;span style=&#34;color:#ae81ff&#34;&gt;4&lt;/span&gt;];  &lt;span style=&#34;color:#75715e&#34;&gt;// 最多暂存 4 个包
&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;uint32_t&lt;/span&gt;         qhead, qtail;
&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;udp_sock_t&lt;/span&gt;;
&lt;/span&gt;&lt;/span&gt;&lt;/code&gt;&lt;/pre&gt;&lt;/div&gt;&lt;p&gt;队列用无限增长的 &lt;code&gt;qhead/qtail&lt;/code&gt; 计数（不是环形下标），访问时用 &lt;code&gt;% UDP_PKT_MAX&lt;/code&gt;，满了就丢包：&lt;/p&gt;</description>
    </item>
  </channel>
</rss>
