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

 

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

 

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

 

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

socket(PF_PACKET, SOCK_RAW, htons(ETH_P_ALL));

 

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

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接收数据;

recvfrom(sock_raw_receive, recv_buffer, sizeof(recv_buffer), 0, (struct sockaddr *)&sl_receive,  &addr_len);

 

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

 

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);
sendto(sock_raw_send, recv_buffer, recv_len, 0 , (struct sockaddr *)&sl_send, sizeof(sl_send));

 

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