应用raw socket构造UDP包不成功

使用raw socket构造UDP包不成功
抓不到包,请大家帮看看,先行谢过了。第一次接触raw socket编程,实在不知怎么搞了。
C/C++ code

#include <errno.h>
#include <sys/socket.h>
#include <arpa/inet.h>
#include <netinet/in.h>
#include <netinet/ip.h>
#include <netinet/udp.h>
#include <sys/time.h>
#include <string.h>

#include <iostream>

using namespace std;

const int LISTEN_PORT = 7800;

const int PACK_LEN = 8192;

struct psdhdr
{
    u_int32_t saddr;
    u_int32_t daddr;
    u_int8_t  zero;
    u_int8_t  protocol;
    u_int16_t len;
};


u_int16_t checksum(u_int16_t *buffer, int size);

int main(int argc, char *argv[])
{
    char payload[] = "hello";
    
    struct sockaddr_in sin, din;
    memset(&sin, 0, sizeof(struct sockaddr_in));
    memset(&din, 0, sizeof(struct sockaddr_in));
    sin.sin_family = AF_INET;
    din.sin_family = AF_INET;
    sin.sin_addr.s_addr = inet_addr("19.9.0.95");
    din.sin_addr.s_addr = inet_addr("18.8.6.66");
    sin.sin_port = htons(LISTEN_PORT);
    din.sin_port = htons(LISTEN_PORT);    

    const int listen_fd = socket(PF_INET, SOCK_RAW, IPPROTO_UDP);
    if (listen_fd < 0)
    {
        cout << "system error [ " << errno << ":" << strerror(errno) << "]"<< endl;
        return 1;
    }

    if (bind(listen_fd, (const struct sockaddr *)&sin, sizeof(sin)) < 0)
    {
        close(listen_fd);
        cout << "system error [ " << errno << ":" << strerror(errno) << "]"<< endl;
        return 1;
    }

    int on = 1;
    if (setsockopt(listen_fd, IPPROTO_IP, IP_HDRINCL, &on, sizeof(on)) < 0)
    {
        cout << "system error [ " << errno << ":" << strerror(errno) << "]" << endl;
        return 1;
    }

    char buf[PACK_LEN] = {0};
    struct iphdr *iphdr = (struct iphdr *) buf;
    struct psdhdr psd;
    struct udphdr *pudphdr = (struct udphdr *) (buf+sizeof(struct iphdr));
    
    memcpy(buf+sizeof(struct iphdr)+sizeof(struct udphdr), payload, strlen(payload));
    const int data_len = sizeof(struct iphdr)+sizeof(struct udphdr)+strlen(payload);
    iphdr->ihl = 5;
    iphdr->version = 4;
    iphdr->tos = 0;
    iphdr->tot_len = data_len;
    iphdr->id = htons(random());
    iphdr->frag_off = 0;
    iphdr->ttl = 0xff;
    iphdr->protocol = IPPROTO_UDP;
    iphdr->check = 0;
    iphdr->saddr = sin.sin_addr.s_addr;
    iphdr->daddr = din.sin_addr.s_addr;
    iphdr->check = checksum((u_int16_t *)buf, sizeof(struct iphdr));

    pudphdr->source = sin.sin_port;
    pudphdr->dest = din.sin_port;
    pudphdr->len = htons(sizeof(struct udphdr)+strlen(payload));
    pudphdr->check = 0;

    psd.saddr = iphdr->saddr;
    psd.daddr = iphdr->daddr;
    psd.zero = 0;
    psd.protocol = IPPROTO_UDP;
    psd.len = pudphdr->len;
    
    char tmp[sizeof(struct psdhdr) + ntohs(pudphdr->len)];
    struct psdhdr *ppsdhdr = (struct psdhdr *) tmp;
    ppsdhdr->saddr = iphdr->saddr;
    ppsdhdr->daddr = iphdr->daddr;
    ppsdhdr->zero = 0;
    ppsdhdr->protocol = IPPROTO_UDP;
    ppsdhdr->len = pudphdr->len;
    memcpy(tmp+sizeof(struct psdhdr), buf+sizeof(struct iphdr), ntohs(pudphdr->len));
    pudphdr->check = checksum((u_int16_t *)tmp, sizeof(tmp));
    
    for (int cnt = 1; cnt < 100; ++cnt)
    {
        if (sendto(listen_fd, buf, data_len, 0, (struct sockaddr *)&sin, sizeof(sin)) < 0)
        {
            cout << "system error [ " << errno << ":" << strerror(errno) << "]" << endl;
            cout << cnt << " failed" << endl;
        }
        else
        {
            cout << cnt << " success" << endl; 
        }
        sleep(2);
    }   

    cout << "Hello world!" << endl;
    return 0;
}

u_int16_t checksum(u_int16_t *buffer, int size)
{
    u_int64_t ret = 0;
    
    while (size > 1)
    {
        ret += *buffer++;
        size -= 2;
    }

    if (size == 1)
    {
        ret += *((u_int8_t *)buffer);
    }

    ret = (ret >> 16) + (ret & 0xffff);
    ret += (ret >> 16);

    return (u_int16_t)(~ret);
}