TCP/IP学习(37)——L2如何设置包的协议类型

原文地址:TCP/IP学习(37)——L2如何设置包的协议类型 作者:GFree_Wind

本文的copyleft归gfree.wind@gmail.com所有,使用GPL发布,可以*拷贝,转载。但转载请保持文档的完整性,注明原作者及原链接,严禁用于任何商业用途。
作者:gfree.wind@gmail.com
博客:linuxfocus.blog.chinaunix.net
   

以太网的硬件地址长度为48 bits(6 字节),而L2数据帧有三种类型:单播,多播和广播,其中广播可看作多播的一种特殊情况。Bit 0用于表示多播还是单播,当bit 0为1时,为多播,为0时,表示单播。

Linux kernel使用eth_type_trans来判断数据帧的类型,及协议类型。
  1. __be16 eth_type_trans(struct sk_buff *skb, struct net_device *dev)
  2. {
  3.     struct ethhdr *eth;

  4.     skb->dev = dev;
  5.     skb_reset_mac_header(skb);
  6.     skb_pull_inline(skb, ETH_HLEN);
  7.     eth = eth_hdr(skb);

  8.     if (unlikely(is_multicast_ether_addr(eth->h_dest))) {
  9.         /* 如果是多播地址,即bit0为1*/

  1.         if (!compare_ether_addr_64bits(eth->h_dest, dev->broadcast))
  2.             skb->pkt_type = PACKET_BROADCAST; //与设备的广播地址相同,则帧为广播帧
  3.         else
  4.             skb->pkt_type = PACKET_MULTICAST; //与设备的广播地址不同,则帧为多播帧
  5.     }

  6.     /*
  7.      * This ALLMULTI check should be redundant by 1.4
  8.      * so don't forget to remove it.
  9.      *
  10.      * Seems, you forgot to remove it. All silly devices
  11.      * seems to set IFF_PROMISC.
  12.      */
     /* 
     这里为什么不检测IFF_PROMISC标志呢? 
     我怀疑是因为有的网卡不设置这个标志,依然可以收到不属于自己地址的数据包
     */
  1.     else if (1 /*dev->flags&IFF_PROMISC */ ) {
         
          /* 如果数据帧的目的地址不是网卡的地址,那么数据帧的类型为PACKET_OTHERHOST */
  1.         if (unlikely(compare_ether_addr_64bits(eth->h_dest, dev->dev_addr)))
  2.             skb->pkt_type = PACKET_OTHERHOST;

 /* 默认情况,skb->pkt_type为0,即PACKET_HOST,即数据帧是发给本主机的 */
  1.     }

     /* 下面开始判断L2协议 */

  1.     /*
  2.      * Some variants of DSA tagging don't have an ethertype field
  3.      * at all, so we check here whether one of those tagging
  4.      * variants has been configured on the receiving interface,
  5.      * and if so, set skb->protocol without looking at the packet.
  6.      */
     /*
     如上面的注释所说,当设备指定了DSA或者TRAILER,那么就不需要检查包,直接返回DSA或者TRAILER
     */
  1.     if (netdev_uses_dsa_tags(dev))
  2.         return htons(ETH_P_DSA);
  3.     if (netdev_uses_trailer_tags(dev))
  4.         return htons(ETH_P_TRAILER);

     /*
     当协议值大于136时,那么这个数据帧一定为ethernet frame 
     因为802.2和802.3的对应域为帧长,均要小于或等于1500,而ethernet frame的协议类型都大于等于1536.
     */
  1.     if (ntohs(eth->h_proto) >= 1536)
  2.         return eth->h_proto;

  3.     /*
  4.      * This is a magic hack to spot IPX packets. Older Novell breaks
  5.      * the protocol design and runs IPX over 802.3 without an 802.2 LLC
  6.      * layer. We look for FFFF which isn't a used 802.2 SSAP/DSAP. This
  7.      * won't work for fault tolerant netware but does for the rest.
  8.      */
     /*
     当IPX使用原始的802.3作为载体时,其头两个字节作为checksum,但是一般都设为0xffff。 
     */
  1.     if (skb->len >= 2 && *(unsigned short *)(skb->data) == 0xFFFF)
  2.         return htons(ETH_P_802_3);

  3.     /*
  4.      * Real 802.2 LLC
  5.      */
  6.     /* ok, 那么类型为802.2*/
  7.     return htons(ETH_P_802_2);
  8. }
这个函数逻辑上很简单,基本上都是由L2层协议所决定的。
上一篇:Qt使用自定义委托(QItemDelegate/QStyledItemDelegate)


下一篇:Qt-网易云音乐界面实现-7 消息中心实现,主要是QListWidget 自定义Item 和QTabwidget使用