2

我坚持发送原始以太网帧。

[编辑]
我发现了一些错误。
1. 在套接字调用中它必须是 AF_PACKET。
2. AF_PACKET 没有 SOCK_PACKET 选项,但是 SOCK_DGRAM 和 SOCK_RAW

使用 SOCK_DGRAM,wireshark 捕获格式错误的 LLC 数据包。
但是使用 SOCK_RAW 没有错误消息,也没有捕获数据包。

我没有发现真正出了什么问题。
[/编辑]

编码:

if ((ethernet_socket = socket(AF_PACKET, SOCK_RAW, htons(ETH_P_ALL))) == -1)
    cout << "Ethernet Socket: "<< strerror(errno) << endl;

struct sockaddr_ll socket_address;
socket_address.sll_family   = PF_PACKET;
socket_address.sll_protocol = htons(ETH_P_IP);
socket_address.sll_ifindex=if_nametoindex("eth0");
socket_address.sll_hatype   = 1; // ARPHRD_ETHER
socket_address.sll_pkttype  = PACKET_OTHERHOST;
socket_address.sll_halen    = ETH_ALEN;
memcpy(socket_address.sll_addr,dest_mac_addr,ETH_ALEN);

int send_result = 0;

char *opt=(char*)malloc(4*sizeof(char));
strcpy(opt,"eth0");

if(setsockopt(ethernet_socket, SOL_SOCKET, SO_BINDTODEVICE, opt, 4)==-1)
    cout << "Could not bind socket to device: " << strerror(errno) << endl;

if ((send_result  
= sendto(ethernet_socket, &buffer, sizeof(buffer), 0,  
    (struct sockaddr*)&socket_address, sizeof(socket_address)))==-1){
        cout << "sendto error: "<< strerror(errno) << endl;
        return send_result;
}

关于CK

4

2 回答 2

2

我找到了解决方案这是如何工作的。我看了一下 Linux 程序 PackEth 的源代码。

以下代码对我有用。

if ((ethernet_socket = socket(PF_PACKET, SOCK_RAW, htons(ETH_P_ALL))) == -1)
    cout << "Ethernet Socket: "<< strerror(errno) << endl;  //errorhandling

memset(&ifr, 0, sizeof(ifr));
strncpy (ifr.ifr_name, "eth0", sizeof(ifr.ifr_name) - 1);
ifr.ifr_name[sizeof(ifr.ifr_name)-1] = '\0';


if (ioctl(ethernet_socket, SIOCGIFINDEX, &ifr) == -1) {
    cout << "No such interface:"<< strerror(errno) << endl;
    close(ethernet_socket);
}

ioctl(ethernet_socket, SIOCGIFFLAGS, &ifr);
if ( (ifr.ifr_flags & 0x1) == 0) {
    cout << "Interface is down: "<< strerror(errno) << endl;
    close(ethernet_socket);
}

ioctl(ethernet_socket, SIOCGIFINDEX, &ifr);


memset(&socket_address, 0, sizeof (socket_address));
socket_address.sll_family    = AF_PACKET;
socket_address.sll_ifindex   = ifr.ifr_ifindex;
socket_address.sll_protocol  = htons(ETH_P_ALL);

if ((raw_send = sendto(ethernet_socket, buffer, size_payload+14,0,(struct     sockaddr*)     
                           &socket_address, sizeof(socket_address)))==-1){
    cout << "sendto error: "<< strerror(errno) << endl;
    return raw_send;
}
于 2012-11-21T08:13:34.547 回答
0
$ man 7 packet

“SOCK_RAW 数据包传入和传出设备驱动程序,数据包数据没有任何变化。接收数据包时,地址仍会被解析并以标准 sockaddr_ll 地址结构传递。 传输数据包时,用户提供的缓冲区应包含物理层标头。然后,该数据包未经修改地排队到由目标地址定义的接口的网络驱动程序中。一些设备驱动程序总是添加其他标头。

SOCK_DGRAM 在稍高的级别上运行。在将数据包传递给用户之前,物理头被删除。通过 SOCK_DGRAM 数据包套接字发送的数据包在排队之前会根据 sockaddr_ll 目标地址中的信息获得合适的物理层标头。”

于 2012-10-22T12:02:05.633 回答