我想实现这样一个功能用pcap抓包在回调函数中用可以处理底层数据的socket套接字发出去。现在发送IP包没问题但是发送MAC包貌似死循环发送了 部分代码如下:
这是我发送数据包的代码
int sockfd_MAC = socket(PF_PACKET, SOCK_RAW, htons(ETH_P_ALL)); // 创建Socket
void SendMACPacket(int sockfd, void* msg, size_t len)
{
struct sockaddr_ll addr;
memset( &addr, 0, sizeof(addr) );
addr.sll_family = AF_PACKET;
struct ifreq ifstruct;
strcpy(ifstruct.ifr_name, "eth0");
ioctl(sockfd, SIOCGIFINDEX, &ifstruct); //??I/O??
addr.sll_ifindex = ifstruct.ifr_ifindex;
addr.sll_protocol = htons(ETH_P_ALL);
memcpy(addr.sll_addr,((struct eth_hdr*)msg)->h_source,6);
sendto(sockfd, msg, len, 0, &addr, sizeof(struct sockaddr_ll));
}
pcap_loop(handle,packetCount,GenPcapPacket,NULL);
回调函数在这
int GenPcapPacket(u_char argument,const struct pcap_pkthdr pack,const u_char *content)
{
const u_char *buf;
buf = content;
SendMACPacket(sockfd_MAC, (void*)buf, pack->len);
return 0;
}
该回答引用ChatGPT
您的代码看起来没有明显的死循环,但是可能存在一些问题。首先,您的回调函数将从pcap_loop()函数中调用,其中pack参数是pcap捕获到的数据包的头部信息,content参数是指向数据包内容的指针。因此,您不需要从content中复制数据,只需要将它作为参数传递给SendMACPacket函数即可。
另外,在SendMACPacket函数中,您使用了memcpy将源MAC地址复制到addr.sll_addr中。但是,这是不正确的,因为它实际上应该是目的MAC地址,而不是源MAC地址。您需要将addr.sll_addr设置为目的MAC地址。
最后,在SendMACPacket函数中,您使用了sendto函数来发送数据包。但是,您没有检查sendto的返回值,这可能导致数据包未能成功发送。您应该检查sendto的返回值,并根据需要采取措施。
下面是一些更新的代码,以解决上述问题:
int sockfd_MAC = socket(PF_PACKET, SOCK_RAW, htons(ETH_P_ALL)); // 创建Socket
void SendMACPacket(int sockfd, const u_char* packet, size_t len)
{
struct sockaddr_ll addr;
memset(&addr, 0, sizeof(addr));
addr.sll_family = AF_PACKET;
struct ifreq ifstruct;
strcpy(ifstruct.ifr_name, "eth0");
ioctl(sockfd, SIOCGIFINDEX, &ifstruct);
addr.sll_ifindex = ifstruct.ifr_ifindex;
addr.sll_protocol = htons(ETH_P_ALL);
memcpy(addr.sll_addr, &packet[0], 6); // 设置目的MAC地址
int ret = sendto(sockfd, packet, len, 0, (struct sockaddr*)&addr, sizeof(addr));
if (ret == -1) {
perror("sendto error");
}
}
void GenPcapPacket(u_char* argument, const struct pcap_pkthdr* pack, const u_char* content)
{
SendMACPacket(sockfd_MAC, content, pack->len);
}
pcap_loop(handle, packetCount, GenPcapPacket, NULL);