C++ 套接字不接收传入数据包,在 Python 中工作

标签 c++ c linux sockets

我还没找到类似的问题,所以就在这里。

我有一个 Linux 盒子,通过 10GbE 电缆直接连接到另一台机器。另一端的机器正在将 8kB UDP 帧推送到我的 Linux 机器上的 NIC。我能够使用 iptraf 查看传入的 UDP 数据包,并确认它们到达预期的 NIC、IP 地址和端口号。

我可以使用 Python 和以下代码片段打开套接字并读取此数据:

import socket
ip = '192.168.4.5'
port = 60001
sock = socket.socket(socket.AF_INET, socket.SOCK_DGRAM)
sock.bind((ip, port))
data = sock.recv(8192)

除了Python太慢之外,这没有给我带来任何问题,而且无论我的循环有多紧,我最终都会丢失网络数据包。数据通过 10GbE 链路传入,并且正在全速运行。

因此,为了克服这个延迟问题,我决定转而用 C++/C 编写数据记录器。

#include <sys/types.h>
#include <stdlib.h>
#include <unistd.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <string.h>
#include <netdb.h>
#include <stdio.h>
#include <arpa/inet.h>
#include <iostream>

void error(const char *msg)
{
    perror(msg);
    exit(0);
}

int main(int argc, char *argv[])
{
int sock;
ssize_t n;
socklen_t length;
in_port_t port;
const char *ip_addr;
socklen_t fromlen;
struct sockaddr_in server;
struct sockaddr_in from;
char buf[8192];


ip_addr = "192.168.4.5"; // host to listen on
port = 60001; // port to listen on

std::cout << "ip address: " << ip_addr << std::endl;
std::cout << "port: " << port << std::endl;

// create inet socket
sock = socket(AF_INET, SOCK_DGRAM, 0);
if (sock < 0) error("Opening socket");
else std::cout << "socket opened.\n";

length = sizeof(server);
memset(&server, 0, length); // initialize with zeros

server.sin_family = AF_INET;
server.sin_addr.s_addr = inet_addr(ip_addr);
server.sin_port = port;

// bind to socket address and port
if (bind(sock,(struct sockaddr *)&server, length)<0)
   error("binding");
else std::cout << "binding successful\n";

// check binding status
struct sockaddr_in socket_address;
socklen_t socket_addr_len;

for (int i=0; i<10; i++) {

    std::cout << "iter: " << i << std::endl;
    n = recv(sock, buf, 9000, 0);

    if (n<0) error("recv");
    std::cout << "received a datagram: ";
    for (int k=0; k<10; k++){
        std::cout << buf[k];
    }
    std::cout << std::endl;

}
return 0;
}

上面的 C 程序永远卡在 recv() 上。如果我使其成为非阻塞,那么它会返回 EAGAIN,表明没有任何内容可读取。我尝试使用 getsockname 来确定套接字是否实际上绑定(bind)到了正确的地址;是的。

对我来说,当 Python 版本从套接字读取没有问题并且我可以直观地确认有数据传入时,我会遇到这个问题,这似乎很奇怪。我的猜测是我的 C 套接字设置有问题。

另一条可能相关的信息是我的 Linux 机器在同一子网上还有 1 个其他接口(interface)。

Linux 盒子上的接口(interface)

  • eth1:192.168.4.1
  • eth2:192.168.4.5

远程计算机上的接口(interface)

  • eth0:192.168.4.10

iptraf 的示例输出

eth2 上从 192.168.4.10:10000 到 192.168.4.5:60001 的 UDP(8220 字节)

如有任何指导,我们将不胜感激。

最佳答案

server.sin_port = htons(端口);

关于C++ 套接字不接收传入数据包,在 Python 中工作,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/42548653/

相关文章:

javascript - 当 JS 代码在应用程序构建时未知时从 QML 调用 JS 函数(帽子提示 SpiderMonkey)

c++ - 获取标题栏双击

c - 如何在 ext4 中查找文件名?

java - 无法在linux终端上运行java程序

linux - 可执行文件和共享库有什么区别

linux - 我的 Oracle Virtual Box 虚拟机无法启动的原因是什么?

c++ - 奇数重复符号错误

C++从文件阅读器中的另一个类调用set函数

c - 如何将链表的头节点地址存储在文件中并稍后检索

c - 从制表符分隔的文件中读取记录