- r - 以节省内存的方式增长 data.frame
- ruby-on-rails - ruby/ruby on rails 内存泄漏检测
- android - 无法解析导入android.support.v7.app
- UNIX 域套接字与共享内存(映射文件)
首先,我是套接字编程的新手,所以我可能遗漏了一些明显的东西。
目标:
我正在尝试编写一个小型实用程序来手工制作 ARP 帧并通过网络发送它们,然后监听回复。我正在使用 AF_PACKET、SOCK_DGRAM,因此内核会处理以太网报头/报尾,但我在发送和接收时可以原始访问帧的数据部分。
该应用程序本质上是诊断性的,因此现有的 API 和用户空间工具不适合。我试图在广播域中找到一个(或多个)设备来响应 ARP 请求,但我根本不关心填充系统 ARP 缓存或类似的东西。
问题:
我希望用户能够通过名称指定接口(interface),或者改用适当的默认值。到目前为止,如果我指定接口(interface),一切都可以完美运行……但如果我不指定,我不确定如何最好地处理 bind() 和 sendto()。
详细信息:
这是相关代码的代表性示例。假设 *cfg 指向一个预先分配的结构,其中 cfg->interface.name == "eth0":
int i;
int sock;
void *buf;
struct my_config_options *cfg;
struct ifreq ifr;
struct sockaddr_ll addr;
struct pollfd pfd;
// Packet buffer
buf = malloc(BUF_LEN);
if (buf == NULL) {
return 0;
}
// Open a socket
sock = socket(AF_PACKET, SOCK_DGRAM, 0);
if (sock_fd == -1) {
return 0;
}
// Get interface index
strncpy(ifr.ifr_name, cfg->interface.name, IFNAMSIZ);
if (ioctl(sock, SIOCGIFINDEX, &ifr) == -1) {
return 0;
}
cfg->interface.idx = ifr.ifr_ifindex;
// Get source MAC address
if (ioctl(sock, SIOCGIFHWADDR, &ifr) == -1) {
return 0;
}
for (i = 0; i < 6; i++) {
cfg->interface.mac[i] = ifr.ifr_hwaddr.sa_data[i];
}
// Get source IP address
if (ioctl(sock, SIOCGIFADDR, &ifr) == -1) {
return 0;
}
cfg->interface.ip = ((struct sockaddr_in *)(&ifr.ifr_addr))->sin_addr.s_addr;
// Bind socket to interface
memset(&addr, 0x00, sizeof(addr));
addr.sll_family = AF_PACKET;
addr.sll_protocol = htons(ETH_P_ARP);
addr.sll_ifindex = cfg->interface.idx;
if (bind(sock, (struct sockaddr*)&addr, sizeof(addr)) == -1) {
close(sock);
return 0;
}
...
// Send an ARP request
memset(&addr, 0x00, sizeof(addr));
addr.sll_family = AF_PACKET;
addr.sll_protocol = htons(ETH_P_ARP);
addr.sll_ifindex = cfg->interface.idx;
addr.sll_halen = ETH_ALEN;
for (i = 0; i < 6; i++) {
addr.sll_addr[i] = 0xff;
}
if (sendto(sock, buf, BUF_LEN, 0, (struct sockaddr *)&addr, sizeof(addr)) != BUF_LEN) {
close(sock);
return 0;
}
while (1) {
pfd.fd = sock;
pfd.events = POLLIN;
pfd.revents = 0;
i = poll(&pfd, 1, cfg->delay);
// Error
if (i < 0) {
close(sock);
return 0;
}
// Timed out
if (i == 0) {
break;
}
// Read buffered packet
i = sizeof(addr);
i = recvfrom(sock, buf, BUF_LEN, MSG_DONTWAIT, (struct sockaddr*)&addr, &i);
if (i < 0) {
close(sock);
return 0;
}
// Handle received data
...
}
到目前为止一切顺利。
但是,如果我没有接口(interface)名称或接口(interface)索引,我不知道该告诉 bind() 什么。当涉及到数据包模式时,手册页的细节相当稀疏。显然,我可以为所有接口(interface)绑定(bind)到“0.0.0.0”,但我不在此处的 IP 层工作——我有一个 sockaddr_ll 结构。
有没有好的方法告诉 bind() 选择合适的接口(interface)?这在第 2 层是否是一个合理的请求?
可能的解决方案:
1) 选择第一个 UP 而不是 loopback 的接口(interface)。
2) 更进一步检查分配的地址。
不过,我不知道这两者是否真的合意。我有点希望代码在这里“做正确的事”——也就是说,用户所期望的。因此,如果对任何给定工具如何选择导出接口(interface)有一个常规顺序,这就是我想要做的。
我怀疑许多工具都依赖路由表来选择最佳接口(interface),但这不太适合这里。通常,ARP 流量绑定(bind)到您所在的子网。在这种情况下,用户可能正在寻找配置错误的设备,或同一广播域中并行网络上的设备。 (也就是说:如果一切正常,他们就不需要这个工具。所以所有的赌注都没有了。)
现在,我不能 100% 确定其他主机将如何处理来自不在其子网中的 IP 的 ARP 请求,但这是另一天的问题。
TLDR:
如果您不能依赖路由表,是否有一种确定性的方法可以在具有多个主机的主机上选择“正确的导出接口(interface)”(如果存在这样的东西)?
最佳答案
首先,正如您已经注意到的,您没有在 IP 层上工作 - 但是随后,您需要一个原始套接字,因此使用以下参数打开它:AF_PACKET、SOCK_RAW、IPPROTO_RAW
。
要获取所有接口(interface)的列表,请参阅 here .您仍然需要标准来选择合适的标准。我找到了 sending 的代码示例和 receiving原始以太网帧,包括从命令行参数中选择接口(interface)。我没有验证它们是否有效,但您可以看看。
如果您只需区分接口(interface)是有线还是无线就足够了,请参阅 here .
最后只是一个提示:如果您的程序仍然无法运行,您可能没有适当的权限 (CAP_NET_RAW),如所述 here .
关于c - 在 C 中使用数据包模式时如何绑定(bind)到默认接口(interface)?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/32878555/
我正在尝试在我的代码库中为我正在编写的游戏服务器更多地使用接口(interface),并了解高级概念以及何时应该使用接口(interface)(我认为)。在我的例子中,我使用它们将我的包相互分离,并使
我有一个名为 Widget 的接口(interface),它在我的整个项目中都在使用。但是,它也用作名为 Widget 的组件的 Prop 。 处理此问题的最佳方法是什么?我应该更改我的 Widget
有一个接口(interface)可以是多个接口(interface)之一 interface a {x:string} interface b {y:string} interface c {z:st
我遇到了一种情况,我需要调用第三方服务来获取一些信息。这些服务对于不同的客户可能会有所不同。我的界面中有一个身份验证功能,如下所示。 interface IServiceProvider { bool
在我的例子中,“RequestHandlerProxy”是一个结构,其字段为接口(interface)“IAdapter”,接口(interface)有可能被调用的方法,该方法的输入为结构“Reque
我有一个接口(interface)Interface1,它已由类A实现,并且设置了一些私有(private)变量值,并且我将类A的对象发送到下一个接受输入作为Interface2的类。那么我怎样才能将
假设我有这样的类和接口(interface)结构: interface IService {} interface IEmailService : IService { Task SendAs
有人知道我在哪里可以找到 XML-RPC 接口(interface)的定义(在 OpenERP 7 中)?我想知道创建或获取对象需要哪些参数和对象属性。每个元素的 XML 示例也将非常有帮助。 最佳答
最近,我一直在阅读有关接口(interface)是抽象的错误概念的文章。一篇这样的帖子是http://blog.ploeh.dk/2010/12/02/InterfacesAreNotAbstract
如果我有一个由第三方实现的现有 IInterface 后代,并且我想添加辅助例程,Delphi 是否提供了任何简单的方法来实现此目的,而无需手动重定向每个接口(interface)方法?也就是说,给定
我正在尝试将 Article 数组分配给我的 Mongoose 文档,但 Typescript 似乎不喜欢这样,我不知道为什么它显示此警告/错误,表明它不可分配. 我的 Mongoose 模式和接口(
我有两个接口(interface): public interface IController { void doSomething(IEntity thing); } public inte
是否可以创建一个扩展 Serializable 接口(interface)的接口(interface)? 如果是,那么扩展接口(interface)的行为是否会像 Serilizable 接口(int
我试图在两个存储之间创建一个中间层,它从存储 A 中获取数据,将其转换为相应类型的存储 B,然后存储它。由于我需要转换大约 50-100 种类型,我希望使用 map[string]func 并根据 s
我正在处理一个要求,其中我收到一个 JSON 对象,其中包含一个日期值作为字符串。我的任务是将 Date 对象存储在数据库中。 这种东西: {"start_date": "2019-05-29", "
我们的方法的目标是为我们现有的 DAO 和模型类引入接口(interface)。模型类由各种类型的资源 ID 标识,资源 ID 不仅仅是随机数,还带有语义和行为。因此,我们必须用对象而不是原始类型来表
Collection 接口(interface)有多个方法。 List 接口(interface)扩展了 Collection 接口(interface)。它声明与 Collection 接口(int
我有一个 Java 服务器应用程序,它使用 Jackson 使用反射 API 对 DTO 进行一般序列化。例如对于这个 DTO 接口(interface): package com.acme.libr
如果我在 Kotlin 中有一个接口(interface): interface KotlinInterface { val id: String } 我可以这样实现: class MyCla
我知道Java中所有访问修饰符之间的区别。然而,有人问了我一个非常有趣的问题,我很难找到答案:Java 中的 private 接口(interface)和 public 接口(interface)有什
我是一名优秀的程序员,十分优秀!