在項(xiàng)目中,經(jīng)常有可能用到以太網(wǎng)的原始數(shù)據(jù),就是鏈路層輸出,不經(jīng)過TCPIP,UDP這類協(xié)議解析的數(shù)據(jù)。一般稱呼這種為RAW數(shù)據(jù)。
主要分兩類,一類是在LINUX下如何截取使用數(shù)據(jù),一類是在MCU下如何截取使用數(shù)據(jù)。
無論哪一類首先需要使網(wǎng)卡進(jìn)入混雜模式。
在linux下,首先建立一個(gè)接收所有數(shù)據(jù)的socket
socket(PF_PACKET, SOCK_RAW, htons(ETH_P_ALL));
對于多個(gè)網(wǎng)卡的需要先綁定網(wǎng)卡
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");
}
然后便可以從這個(gè)socetk接收數(shù)據(jù);
recvfrom(sock_raw_receive, recv_buffer, sizeof(recv_buffer), 0, (struct sockaddr *)&sl_receive, &addr_len);
發(fā)送也一樣,建立socket,綁定,然后發(fā)送
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支持并且有相應(yīng)的庫函數(shù)可以直接使用,那直接調(diào)用就可以了。如果RTOS不支持或者裸機(jī)使用的話就需要在以太網(wǎng)的接收中斷里面處理。