我一直没有find类似的问题,所以在这里。
我有一个通过10GbE电缆直接连接到另一台机器的Linux机器。 另一端的机器将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之外,这不会给我带来任何问题,而且速度太慢,无论我做多less循环,最终都会丢失networking数据包。 数据通过10GbE链路传输,并全速运行。
因此,解决这个延迟问题,我决定用C ++ / C编写一个数据logging器。
#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
来确定套接字是否实际绑定到正确的地址; 它是。
对于我来说,当Python版本从套接字读取时没有任何问题,并且我可以直观地确认有数据传入的时候,我会遇到这个问题,这似乎很奇怪。 我的猜测是,我的套接字在C中有一个问题
另一点可能相关的信息是我的Linux机器在同一个子网上有一个其他接口。
UDP (8220 bytes) from 192.168.4.10:10000 to 192.168.4.5:60001 on eth2
任何指导将不胜感激。
server.sin_port = htons (port);