ARM使用ETH链路层原始数据的方法

2022-05-11 08:29:25 浏览数 (1)

在项目中,经常有可能用到以太网的原始数据,就是链路层输出,不经过TCPIP,UDP这类协议解析的数据。一般称呼这种为RAW数据。

主要分两类,一类是在LINUX下如何截取使用数据,一类是在MCU下如何截取使用数据。

无论哪一类首先需要使网卡进入混杂模式。

在linux下,首先建立一个接收所有数据的socket

代码语言:javascript复制
socket(PF_PACKET, SOCK_RAW, htons(ETH_P_ALL));

对于多个网卡的需要先绑定网卡

代码语言:javascript复制
memset(&sl_receive, 0x00, sizeof(sl_receive));
  memset(&ifr_receive, 0x00, sizeof(ifr_receive));
  strncpy(ifr_receive.ifr_name, "eth1", sizeof(ifr_receive.ifr_name));
  if(ioctl(sock_raw_receive, SIOCGIFINDEX, &ifr_receive)!=0);
  {
    perror("ioctl");
  }

然后便可以从这个socetk接收数据;

代码语言:javascript复制
recvfrom(sock_raw_receive, recv_buffer, sizeof(recv_buffer), 0, (struct sockaddr *)&sl_receive,  &addr_len);

发送也一样,建立socket,绑定,然后发送

代码语言:javascript复制
sock_raw_send = socket(PF_PACKET, SOCK_RAW, htons(ETH_P_ALL));
  if (-1 == sock_raw_send)
  {
    printf("socekt error.n");
  }
  else
  {
    printf("ok.n");
  }
  memset(&sl_send, 0x00, sizeof(sl_send));
  memset(&ifr_send, 0x00, sizeof(ifr_send));
  strncpy(ifr_send.ifr_name, "eth0", sizeof(ifr_send.ifr_name));
  if(ioctl(sock_raw_send, SIOCGIFINDEX, &ifr_send)!=0);
代码语言:javascript复制
sendto(sock_raw_send, recv_buffer, recv_len, 0 , (struct sockaddr *)&sl_send, sizeof(sl_send));

MCU的,如果使用RTOS支持并且有相应的库函数可以直接使用,那直接调用就可以了。如果RTOS不支持或者裸机使用的话就需要在以太网的接收中断里面处理。

0 人点赞