c - 如何正确使用 SO_RXQ_OVFL 检测 Linux 套接字上的 UDP 溢出?

标签 c linux sockets

我想使用Linux套接字选项SO_RXQ_OVFL来检测UDP溢出。此选项使用辅助消息来报告丢弃的数据包数量。来自男人:

SO_RXQ_OVFL (since Linux 2.6.33)
Indicates that an unsigned 32-bit value ancillary message (cmsg) should be     
attached to received skbs indicating the number of packets dropped by the   
socket between the last received packet and this received packet.

我的代码没有找到辅助消息。这是我所做的:

启动时,我创建套接字并指定 SO_RXQ_OVFL 套接字选项:

int dropmonitor_on = 1;
if ( setsockopt(udpSocket, SOL_SOCKET, SO_RXQ_OVFL, &dropmonitor_on, sizeof(dropmonitor_on) ) != 0 )
{
    perror("setsockopt SO_RXQ_OVFL not supported by your Linux Kernel");
}

然后我有一个接收函数,它调用 recvmsg 并查找辅助消息:

struct sockaddr_in src_addr;  // The source address will be assigned to here

struct iovec iov[1];
iov[0].iov_base=ap_rxBuffer;
iov[0].iov_len=a_maxSizeBytes-1;

int cmsg_len = CMSG_SPACE(sizeof(uint32_t));
char cmsg[CMSG_SPACE(sizeof(uint32_t))];
memset(cmsg,0,cmsg_len);

struct msghdr message;
memset(&message,0,sizeof(struct msghdr));
message.msg_name=&src_addr;
message.msg_namelen=sizeof(struct sockaddr_in);
message.msg_iov=iov;
message.msg_iovlen=1;
message.msg_control=cmsg;
message.msg_controllen=cmsg_len;

int receivedBytes = 0;

if ( (receivedBytes = recvmsg( a_socket, &message, 0 )) == SOCKET_ERROR )
{
    closeSocket(a_socket);
    fatal("recvmsg() failed");
}
else 
{
    // Reception successful so interrogate ancillary message to get number of dropped packets

    int udp_packets_dropped = 0;

    struct cmsghdr* p_cmsg;
    p_cmsg = CMSG_FIRSTHDR(&message);
    for (p_cmsg = CMSG_FIRSTHDR(&message); p_cmsg != NULL; p_cmsg = CMSG_NXTHDR(&message, p_cmsg))
    {
        if ((p_cmsg->cmsg_level == SOL_SOCKET) && (p_cmsg->cmsg_type == SO_RXQ_OVFL))
        {
            int* p_udp_packets_dropped = (int *) CMSG_DATA(p_cmsg);
            udp_packets_dropped = *p_udp_packets_dropped;
            cout << "UDP pkts dropped: " << udp_packets_dropped << endl;
            break;
        }
    }
    if (p_cmsg == NULL)
    {
        fatal("Error: p_cmsg == NULL");
    }
}

当我运行代码时,它会因以下 fatal error 而停止:

Error: p_cmsg == NULL

由上述代码生成,表示未找到辅助消息。奇怪的是,有时我确实收到了一条辅助消息,所以也许我留下了一些未初始化的东西。

我仔细检查了代码,但没有发现任何问题。如果您能提供一些帮助,我将不胜感激。

最佳答案

我也有这个问题

经过一番挖掘,我在内核源代码中发现了一些提示: https://github.com/torvalds/linux/blob/master/net/socket.c

static inline void sock_recv_drops(struct msghdr *msg, struct sock *sk,
                   struct sk_buff *skb)
{
    if (sock_flag(sk, SOCK_RXQ_OVFL) && skb && SOCK_SKB_CB(skb)->dropcount)
        put_cmsg(msg, SOL_SOCKET, SO_RXQ_OVFL,
            sizeof(__u32), &SOCK_SKB_CB(skb)->dropcount);
}

看看“if (... && SOCK_SKB_CB(skb)->dropcount)”,我认为如果dropcount为0,那么cmsg应该为空。您的代码应该是正确的。

关于c - 如何正确使用 SO_RXQ_OVFL 检测 Linux 套接字上的 UDP 溢出?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/40870873/

相关文章:

linux:每隔一秒执行一次线程

c - 动态增长的字符数组问题 C

创建线程

regex - 在文件中找到正则表达式的第一个匹配项,并打印它

python - 如何在 Python 中获取特定程序的当前 CPU、GPU 和 RAM 使用情况?

python - 找不到 _socket 模块

c - 判断输入是数字还是字符串然后打印回来

c - 有人知道BCD Utility头文件转换公式的好书或网站吗?

linux - 在 Linux 中从文件中打印最小值和最大值

java - 我如何知道服务器使用 Java 提供的默认密码套件