如何使用C程序读取和打印来自接口的UDP数据包的十六进制流?

时间:2015-11-02 10:57:01

标签: c sockets udp

我有一台设备可以将UDP数据包连续发送到我的PC到端口6053。

我为UDP客户端写了一个C程序来接收这些数据包。

代码:

#include <stdio.h> 
#include <stdlib.h>
#include <string.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <unistd.h>
#include <fcntl.h>
#define MAX_BUF 2048
int main(int argc, char** argv)
{
        int sfd,x=1,port=1234;
        struct sockaddr_in serv;
        socklen_t len=sizeof(serv);
        char buf[MAX_BUF]="";

        if(argc!=3){
              printf("Usage: %s ipaddr port\n",argv[0]);
              return 1;
        }

        port=atoi(argv[2]);
        if(port<1 || port>65535){
            printf("Invalid port %d\nValid ports 1-65535\n",port);
            return 1;
        }

        sfd=socket(AF_INET,SOCK_DGRAM,0);
        if(sfd<0){
            perror("Socket\n");
            return 1;
        }

       if (inet_pton(AF_INET, argv[1], &(serv.sin_addr)) == 0){
          printf("Invalid IP\nUse ipv4 address only\n");
          return 1;
       } 

       serv.sin_family=AF_INET;
       serv.sin_port=htons(port);

      if((setsockopt(sfd,SOL_SOCKET,SO_REUSEADDR,&x,sizeof(x)))<0){
            perror("setsockopt:\n");
            return 1;
      } 

       while(1){
           if(recvfrom(sfd,buf,MAX_BUF,0,(struct sockaddr *)&serv,&len)<=0){
                 perror("Recv:\n");
                 close(sfd);
                 return 1;
           }
  //        printf("%.2X\n",buf);
            printf("%s\n",buf);
            memset(buf,0,MAX_BUF);
       }
       return 0;
}

现在,如果我编译并运行代码:

gcc -o udp file.c

./udp 192.168.20.105 6053

没有显示任何内容。

如果我使用gdb,我发现recvfrom()被阻止了。

我认为这是一个防火墙问题,所以我添加了6053打开,它没有用。

我禁用了防火墙,但它没有工作。

但是有些数据包进入我的界面,使用tcpdump

tcpdump -i eth0 port 6053

可能是什么问题?

1 个答案:

答案 0 :(得分:3)

代码错过了bind()到地址:端口,以便本地收听接收器。

传递给src_addr的{​​{1}}参数用于其他目的。

recv_from()”中的“ from ”并不是指从哪里接收(不是本地,也不是远程,也不是混合),而是告诉接收者收到的数据已发送到哪里。

来自Linux man-page to recv_from()

  

如果src_addr不为NULL,则底层协议提供          消息的源地址,源地址放在          src_addr指向的缓冲区。