PMTUD By UDP

通过UDP探测MTU,并实现udp echo server
// Description: UDP echo server.
// g++ udp_echo_server.cc -o udp_echo_server
#include <iostream>
#include <cstring>
#include <arpa/inet.h>
#include <unistd.h>

#define PORT                12345
#define BUFFER_SIZE         4096
#define IPV4_HEADER_SIZE    20
#define UDP_HEADER_SIZE     8

int main() {
    int sockfd;
    char buffer[BUFFER_SIZE];
    struct sockaddr_in servaddr, cliaddr;
    socklen_t len;

    // 创建UDP套接字
    sockfd = socket(AF_INET, SOCK_DGRAM, 0);
    if (sockfd < 0) {
        std::cerr << "Socket creation failed" << std::endl;
        return 1;
    }

    // 设置不分片选项
    int dontfrag = IP_PMTUDISC_DO;
    if (setsockopt(sockfd, IPPROTO_IP, IP_MTU_DISCOVER, &dontfrag, sizeof(dontfrag)) < 0) {
        std::cerr << "Failed to set IP_MTU_DISCOVER option" << std::endl;
        close(sockfd);
        return 1;
    }

    // 清空地址结构
    memset(&servaddr, 0, sizeof(servaddr));
    memset(&cliaddr, 0, sizeof(cliaddr));

    // 填充服务器信息
    servaddr.sin_family = AF_INET;
    servaddr.sin_addr.s_addr = htonl(INADDR_ANY);
    servaddr.sin_port = htons(PORT);

    // 绑定套接字到指定端口
    if (bind(sockfd, (const struct sockaddr *)&servaddr, sizeof(servaddr)) < 0) {
        std::cerr << "Bind failed" << std::endl;
        close(sockfd);
        return 1;
    }

    std::cout << "UDP echo server started, listening on port " << PORT << "..." << std::endl;

    while (true) {
        len = sizeof(cliaddr);
        // 接收数据
        int n = recvfrom(sockfd, buffer, BUFFER_SIZE, 0, (struct sockaddr *)&cliaddr, &len);
        if (n > 0) {
            buffer[n] = '\0';
            char host_buffer[INET_ADDRSTRLEN];
            inet_ntop(AF_INET, &cliaddr.sin_addr, host_buffer, INET_ADDRSTRLEN);
            printf("RX from [%s:%u] -> %d B, mtu = %d\n", host_buffer, ntohs(cliaddr.sin_port), n, n + IPV4_HEADER_SIZE + UDP_HEADER_SIZE);

            // 发送回显数据
            sendto(sockfd, buffer, n, 0, (const struct sockaddr *)&cliaddr, len);
        } else {
            perror("recvfrom");
        }
    }

    close(sockfd);
    return 0;
}
// Description: UDP echo client with path MTU discovery.
// g++ udp_echo_client.cc -o udp_echo_client
#include <iostream>
#include <cstring>
#include <arpa/inet.h>
#include <netdb.h>
#include <unistd.h>

#define SERVER_PORT 12345
#define BUFFER_SIZE 65536

#define IPV4_HEADER_SIZE    20
#define UDP_HEADER_SIZE     8

int discover_path_mtu(const struct sockaddr_in &servaddr)
{
    int sockfd;
    char probe_packet[BUFFER_SIZE];

    int32_t mtu_lbound, mtu_current, mtu_ubound, mtu_best;
    mtu_best = 68;
    mtu_lbound = 68;
    mtu_ubound = 65535;

    struct sockaddr_in remote_host;

    // 创建UDP套接字
    sockfd = socket(AF_INET, SOCK_DGRAM, 0);
    if (sockfd < 0) {
        std::cerr << "Socket creation failed" << std::endl;
        return -1;
    }

    // 设置不分片选项
    int dontfrag = IP_PMTUDISC_DO;
    if (setsockopt(sockfd, IPPROTO_IP, IP_MTU_DISCOVER, &dontfrag, sizeof(dontfrag)) < 0) {
        std::cerr << "Failed to set IP_MTU_DISCOVER option" << std::endl;
        close(sockfd);
        return -1;
    }

    struct timeval timeout;
    timeout.tv_sec = 0;
    timeout.tv_usec = 500000; // 500ms
    if (setsockopt(sockfd, SOL_SOCKET, SO_RCVTIMEO, (char *)&timeout, sizeof(timeout)) != 0) {
        perror("Error in setsockopt(timeout/udp)");
        return -1;
    }

    while (mtu_lbound <= mtu_ubound) {
        mtu_current = (mtu_lbound + mtu_ubound) / 2;
        memset(probe_packet, 'a', mtu_current - IPV4_HEADER_SIZE - UDP_HEADER_SIZE); // 减去IPv4头(20字节)和UDP头(8字节)
        probe_packet[mtu_current - IPV4_HEADER_SIZE - UDP_HEADER_SIZE] = '\0';

        // 发送探测数据包
        int bytes = sendto(sockfd, probe_packet, mtu_current - IPV4_HEADER_SIZE - UDP_HEADER_SIZE, 0, (const struct sockaddr *)&servaddr, sizeof(servaddr));
        if (bytes < 0) {
            if (errno == EMSGSIZE) {
                printf("packet(%d) [%d, %d] too big for local interface\n", mtu_current, mtu_lbound, mtu_ubound);
                mtu_ubound = mtu_current - 1; // update range
                continue;
            }

            perror("Error in sendto()");
            return -1;
        }

        char buffer[BUFFER_SIZE] = {0};
        socklen_t len = sizeof(remote_host);
        bytes = recvfrom(sockfd, buffer, BUFFER_SIZE, 0, (struct sockaddr *)&remote_host, &len);
        if (bytes < 0) {
            // timeout
            if (errno == EAGAIN || errno == EWOULDBLOCK) {
                printf("timeout\n");
                mtu_ubound = mtu_current - 1;
                continue;
            }

            perror("Error in recvfrom()");
            return -1;
        } else if (bytes > 0) {
            printf("mtu = %d [%d, %d]\n", bytes + IPV4_HEADER_SIZE + UDP_HEADER_SIZE, mtu_lbound, mtu_ubound);
            if (strncmp(probe_packet, buffer, bytes) == 0) {
                mtu_lbound = mtu_current + 1;
            } else {
                mtu_ubound = mtu_current - 1;
            }

            if (mtu_current > mtu_best) {
                mtu_best = mtu_current;
            }
        } else {
            printf("recvfrom() returned 0\n");
            return -1;
        }
    }

    close(sockfd);
    std::cout << "The MTU of the path is: " << mtu_best << " B" << std::endl;
    return mtu_best;
}

int main(int argc, char* argv[])
{
    int opt;
    const char *server_address = "127.0.0.1";
    int32_t port = SERVER_PORT;
    while ((opt = getopt(argc, argv, "s:p:")) != -1) {
        switch (opt) {
        case 's':
            server_address = optarg;
            break;
        case 'p':
            port = atoi(optarg);
            break;
        default: /* '?' */
            fprintf(stderr, "Usage: %s -s server_address -p port\n", argv[0]);
            return 0;
        }
    }

    // 解析服务器地址
    struct hostent* host = gethostbyname(server_address);
    if (host == nullptr) {
        std::cerr << "Failed to resolve hostname: " << server_address << std::endl;
        return 1;
    }

    int sockfd;
    char buffer[BUFFER_SIZE];
    struct sockaddr_in servaddr;

    // 创建UDP套接字
    sockfd = socket(AF_INET, SOCK_DGRAM, 0);
    if (sockfd < 0) {
        std::cerr << "Socket creation failed" << std::endl;
        return 1;
    }

    // 设置不分片选项
    int dontfrag = IP_PMTUDISC_DO;
    if (setsockopt(sockfd, IPPROTO_IP, IP_MTU_DISCOVER, &dontfrag, sizeof(dontfrag)) < 0) {
        std::cerr << "Failed to set IP_MTU_DISCOVER option" << std::endl;
        close(sockfd);
        return -1;
    }

    // 清空服务器地址结构
    memset(&servaddr, 0, sizeof(servaddr));

    // 填充服务器信息
    servaddr.sin_family = AF_INET;
    servaddr.sin_port = htons(port);
    memcpy(&servaddr.sin_addr, host->h_addr, host->h_length);

    // 探测路径MTU
    int mtu = discover_path_mtu(servaddr);
    if (mtu <= 0) {
        std::cerr << "Path MTU detection failed" << std::endl;
    }

    while (true) {
        std::cout << "Please enter the data to be sent:";
        std::string message;
        std::getline(std::cin, message);
        if (message.empty()) {
            break;
        }

        // 发送数据
        sendto(sockfd, message.c_str(), message.size(), 0, (const struct sockaddr *)&servaddr, sizeof(servaddr));

        // 接收回显数据
        socklen_t len = sizeof(servaddr);
        int n = recvfrom(sockfd, buffer, BUFFER_SIZE, 0, (struct sockaddr *)&servaddr, &len);
        buffer[n] = '\0';
        std::cout << "RX:" << buffer << std::endl;
    }

    close(sockfd);
    return 0;
}
<think>嗯,用户这次问的是LWIP中实现PMTUD的方法,这是个非常专业的网络协议栈技术问题。从ta直接提到RFC 1191和ip_id字段这些细节来看,应该是有一定嵌入式网络开发经验的工程师。 上次对话历史只看到系统级指令,没有实际上下文,不过用户特意强调要参考站内引用,说明ta可能已经看过一些零散资料但需要系统整合。ta的核心诉求应该是在资源受限的嵌入式环境中实现路径MTU发现功能,重点关注实现步骤和避坑指南。 PMTUD在LWIP中确实是个容易踩坑的点,需要从协议原理和代码实现两个层面回应。用户没提具体版本,但lwip-2.1.0后的实现差异较大,决定以最新lwip-2.2.0为例。关键点在于: 第一要说明如何启用PMTUD功能组(用户可能不知道需要编译开关) 第二要解释DF标志位联动机制(很多开发者会遗漏ip_id初始化) 第三必须强调重传定时器配置(这是实际项目中最常见的故障点) 考虑到用户可能遇到的典型问题,特意加入"常见问题"章节。比如分片重组缓存不足会导致PMTUD假失败,这个bug在社区论坛里反复出现。示例代码特意选择udp_sendto场景,因为相比tcp,udpPMTUD实现更暴露细节。 最后扩展的RFC链接不是随便选的,RFC 4821对于理解PLPMTUD特别重要——很多物联网设备用UDP传输,必须依赖这个增强方案。用户如果做DTLS加密传输肯定会遇到这个问题。 </think>在LWIP中实现PMTUD(Path MTU Discovery)需要遵循RFC 1191规范,通过动态探测路径最大传输单元以避免IP分片。以下是关键实现步骤和代码示例: --- ### **一、LWIP PMTUD核心实现步骤** 1. **启用PMTUD功能** 在 `lwipopts.h` 中启用配置: ```c #define LWIP_PLUGIN_PMTU 1 // 启用PMTUD #define IP_DF 0x4000 // 设置DF标志位 ``` 2. **设置DF标志位** 在发送IP数据包时强制设置DF(Don't Fragment)标志: ```c // 在ip_output()函数中添加 if (netif_mtu_larger_than_mss(netif)) { iphdr->_flags |= PP_HTONS(IP_DF); // 设置DF位 } ``` 3. **处理ICMP Fragmentation Needed** 在 `icmp.c` 中解析ICMPv4类型3代码4(Fragmentation Needed)错误: ```c case ICMP_DUR_FRAG: // Type=3, Code=4 u16_t received_mtu = lwip_htons(*(u16_t*)icmp_payload); if (received_mtu < netif->mtu) { netif->mtu = received_mtu; // 更新路径MTU } break; ``` 4. **实现超时重传机制** 当PMTUD探测失败时,启动指数退避重传: ```c #define PMTU_RETRY_TIMEOUT 5000 // 5秒重传间隔 sys_timeout(PMTU_RETRY_TIMEOUT, pmtu_retry_callback, arg); ``` --- ### **二、关键代码示例** ```c // 在 ip4.c 中修改数据包发送逻辑 err_t ip4_output_if(struct pbuf *p, const ip4_addr_t *src, const ip4_addr_t *dest) { // ... 其他代码 ... if (ip_current_netif() && (netif_mtu_larger_than_mss(ip_current_netif()))) { iphdr->_flags |= PP_HTONS(IP_DF); // 强制设置DF位 } // ... 发送数据包 ... } // 在 icmp_input() 中处理错误 void icmp_input(struct pbuf *p, struct netif *inp) { // ... 解析ICMP头部 ... if (type == ICMP_DUR && code == ICMP_DUR_FRAG) { u16_t new_mtu = lwip_htons(*(u16_t*)(icmphdr + 1)); if (new_mtu < netif->mtu) { netif->mtu = new_mtu; // 更新MTU LWIP_DEBUGF(PMTU_DEBUG, ("Path MTU updated to %d\n", new_mtu)); } } } ``` --- ### **三、注意事项** 1. **MTU缓存管理** 建议为每个目标IP维护MTU缓存表(推荐哈希表实现),避免重复探测: ```c struct pmtu_entry { ip4_addr_t dest; u16_t mtu; u32_t timestamp; }; ``` 2. **分片处理** 当收到`ICMP_FRAG_NEEDED`时,若应用层支持,应使用`TCP_MSS`选项协商较小分段: ```c #define TCP_MSS (netif->mtu - IP_HLEN - TCP_HLEN) ``` 3. **IPv6支持** 若需支持IPv6(RFC 8201),需额外处理ICMPv6 Packet Too Big消息(Type=2)。 --- ### **四、常见问题** 1. **PMTUD失效** - 检查路由器是否过滤ICMP消息(需放行Type=3 Code=4) - 确保DF标志位正确设置(Wireshark抓包验证) 2. **性能优化** - 设置MTU探测上限(默认1500),避免微小MTU导致性能下降 - 使用`SO_PMTU_DISCOVER`套接字选项(需LwIP应用层支持) > 参考实现:LwIP官方源码 `src/core/ipv4/ip4.c` 和 `src/core/ipv4/icmp.c` [^lwip-src] --- ### **相关问题** 1. PMTUD失效时如何触发TCP回退到分段传输? 2. 在嵌入式系统中如何优化PMTUD的存储开销? 3. LWIP如何处理IPv6路径MTU发现(PLPMTUD)? 4. 如何通过Wireshark诊断PMTUD协议交互问题? [^lwip-src]: LwIP Source Code Repository, https://savannah.nongnu.org/projects/lwip/
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值