我正在尝试编写一个函数,该函数采用字节流(包括以太网标头,也许还包括封装在以太网数据包中的上层协议)并将其通过特定接口在网络上发送出去。
这是我的代码的摘要:
// create socket
int s = socket(PF_PACKET, SOCK_RAW, htons(ETH_P_ALL));
if (s < 0) {
// error handling
}
// set up to just receive/send packets on one interface (stored in the variable iface_name e.g. eth0)
struct ifreq ifr;
bzero(&ifr, sizeof(struct ifreq));
strncpy(ifr.ifr_ifrn.ifrn_name, iface_name, IFNAMSIZ);
if (ioctl(s, SIOCGIFINDEX, &ifr) < 0) {
// error handling
}
// set up socaddr for write
struct sockaddr sa;
sa.sa_family = PF_PACKET;
sa.sa_data = htons(ETH_P_ALL);
// write to wire
// buf has type char* and len has type int
// buf contains ethernet header, followed by payload (e.g. ARP packet or IP packet)
if ( sendto(s, buf, len, 0, &sa, sizeof(struct sockaddr) ) < 0 ) {
perror("sendto");
// more error handling
}
我得到错误
sendto: Invalid argument
我该如何解决这个错误?
我最初的猜测是,这在某种程度上是由sa
论点引起的,因为所有其他的都是相当标准的,没有什么可出错的。我可以用这个例子sockaddr_ll
中的类型参数替换它,但这意味着从中提取标题信息,这似乎有点毫无意义,因为它已经在那里,准备好了。一定会有更好的办法?封装、取消封装、重新封装、发送似乎有太多不必要的步骤。我正在为一些预先存在的代码编写底层接口,因此我无法调整输入以使其不包含数据链路层标头。buf