在项目中,经常有可能用到以太网的原始数据,就是链路层输出,不经过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不支持或者裸机使用的话就需要在以太网的接收中断里面处理。