10

如何在 POSIXy C++ 中接收第 2 层数据包?数据包只有 src 和 dst MAC 地址、类型/长度和自定义格式的数据。它们不是 TCP 或 UDP 或 IP 或 IGMP 或 ARP 或其他任何东西——它们是硬件人员给我的自制格式。

socket(AF_PACKET, SOCK_RAW, IPPROTO_RAW)永远不会从它的recvfrom().

我可以很好地发送,无论我向网络堆栈抛出什么选项,我都无法接收。

(平台是 VxWorks,但我可以翻译 POSIX 或 Linux 或其他...)

接收代码(当前化身):

 int s;

 if ((s = socket(AF_PACKET, SOCK_RAW, IPPROTO_RAW)) < 0) {
  printf("socket create error.");
      return -1;
 }

   struct ifreq          _ifr;   
   strncpy(_ifr.ifr_name, "lltemac0", strlen("lltemac0"));
   ioctl(s, IP_SIOCGIFINDEX, &_ifr);

   struct sockaddr_ll _sockAttrib;
   memset(&_sockAttrib, 0, sizeof(_sockAttrib));
   _sockAttrib.sll_len      = sizeof(_sockAttrib);
   _sockAttrib.sll_family   = AF_PACKET;
   _sockAttrib.sll_protocol = IFT_ETHER;
   _sockAttrib.sll_ifindex  = _ifr.ifr_ifindex;
   _sockAttrib.sll_hatype   = 0xFFFF;
   _sockAttrib.sll_pkttype  = PACKET_HOST;
   _sockAttrib.sll_halen    = 6;
   _sockAttrib.sll_addr[0]  = 0x00;
   _sockAttrib.sll_addr[1]  = 0x02;
   _sockAttrib.sll_addr[2]  = 0x03;
   _sockAttrib.sll_addr[3]  = 0x12;
   _sockAttrib.sll_addr[4]  = 0x34;
   _sockAttrib.sll_addr[5]  = 0x56;
   int _sockAttribLen = sizeof(_sockAttrib);


 char packet[64];
 memset(packet, 0, sizeof(packet));

   if (recvfrom(s, (char *)packet, sizeof(packet), 0,
                (struct sockaddr *)&_sockAttrib, &_sockAttribLen) < 0)
   {
      printf("packet receive error.");
   }

   // code never reaches here
4

5 回答 5

4

我认为这样做的方法是编写自己的网络服务,该服务绑定到 VxWorks 网络堆栈中的 MUX 层。这在 VxWorks Network Programmer's Guide 中有很好的记录,我已经做过很多次了。

自定义网络服务可以配置为查看使用 MUX_PROTO_SNARF 服务类型的网络接口上接收的所有第 2 层数据包,这就是 Wind River 自己的 WDB 协议的工作方式,或具有特定协议类型的数据包。

还可以通过编写位于网络服务和套接字 API 之间的自定义套接字后端来向自定义网络服务添加套接字接口。如果您愿意在网络服务中进行应用程序处理,则不需要这样做。

您还没有说您使用的是哪个版本的 VxWorks,但我认为上述内容适用于 VxWorks 5.5.x 和 6.x

于 2010-08-20T09:14:20.223 回答
3

您是否尝试将套接字协议设置htons(ETH_P_ALL)为中规定的packet(7)?您所做的与 IP 没有太大关系(尽管IPPROTO_RAW可能是一些通配符值,不知道)

于 2010-08-19T22:46:31.873 回答
1

我认为这将是一个比你预期的更难解决的问题。鉴于它根本不是 IP(或者显然任何其他协议都可以识别),我认为您无法完全使用用户级代码解决您的问题。在 Linux 上,我认为您需要编写自己的与设备无关的接口驱动程序(可能使用NAPI)。让它在 VxWorks 下运行几乎肯定不是一件容易的事(更像是从头开始完全重写,而不是大多数人认为的移植)。

于 2010-08-19T23:05:09.003 回答
0

您是否尝试过通过 Wireshark 确认数据包实际上是从另一端发送的?

此外,对于调试,请询问您的硬件人员是否有调试引脚(您可以连接到逻辑分析仪),他们可以在收到数据包时断言。只是为了确保硬件可以正常接收数据包。

于 2010-08-19T23:47:28.667 回答
0

首先,您需要将协议指定为 ETH_P_ALL,以便您的接口获取所有数据包。将您的套接字设置为混杂模式。然后在执行接收之前将您的 RAW 套接字绑定到接口。

于 2016-03-30T15:11:19.493 回答