Linux 二层协议架构组织
本文主要讲解了Linux 二层协议架构组织,使用的内核的版本是2.6.32.27
为了方便理解,本文采用整体流程图加伪代码的方式从内核高层面上梳理了Linux 二层协议架构组织,希望可以对大家有所帮助。阅读本文章假设大家对C语言有了一定的了解
Linux中1层2层标准化及在Linux系统中的实现位置如下所示
1层和2a层(MAC)层体现了不同网络介质的区别,在Linux的设备驱动中实现;2b层(LLC)则对上提供了统一的接口,在Linux的内核中实现该部分
Linux在网络架构中的活动形式如下
1-4层是在内核中进行的,交付应用程序处理的数据报文是在用户空间进行的,具体的情参考我的前两篇博客
对于LLC层的协议控制信息及扩展信息如下
所有的协议报文,都通过链路层的协议指示标明三层需要哪个协议实例处理,通过eth_type_trans() 就可以知道了,我们分析一下源代码
/*某一MAC协议类型的全部网络设备都是使用相同的---XX_type_trans, * 如 tr_type_trans() --- 令牌环王 * fddi_type_trans() --- FDDI网络 */ __be16 eth_type_trans(struct sk_buff *skb, struct net_device *dev) { struct ethhdr *eth; unsigned char *rawp; skb->dev = dev; skb_reset_mac_header(skb); /*获取第二层数据包的报头*/ skb_pull(skb, ETH_HLEN); eth = eth_hdr(skb); /*在skb->pkt_type中得到识别和注册 * PACKET_BROADCAST:广播包 * PACKET_MULTICAST:2层组播地址 * PACKET_HOST:发送给本机的报文 * PACKET_OTHERHOST: 发送给其他主机的报文,这里主机应该处于混杂模式中 */ if (unlikely(is_multicast_ether_addr(eth->h_dest))) { if (!compare_ether_addr_64bits(eth->h_dest, dev->broadcast)) skb->pkt_type = PACKET_BROADCAST; else skb->pkt_type = PACKET_MULTICAST; } else if (1 /*dev->flags&IFF_PROMISC */ ) { if (unlikely(compare_ether_addr_64bits(eth->h_dest, dev->dev_addr))) skb->pkt_type = PACKET_OTHERHOST; } ; /*如果长度大于1536(最大帧长度),那么他是802.3兼容以太网卡,协议字段在eth->h_proto中标识*/ if (ntohs(eth->h_proto) >= 1536) return eth->h_proto; rawp = skb->data; if (*(unsigned short *)rawp == 0xFFFF) return htons(ETH_P_802_3); /* * Real 802.2 LLC, * 802.2的协议头返回后,被当做三层报文处理, * 协议处理被散列在ptype_base中,协议处理函数为p8022_rcv() * 对应的三层协议可以通过 register_8022_client() 将自己的处理函数注册到p8022_rcv()中 * 如果 register_8022_client(0xAA, snap_rcv)就将SNAP的三层处理函数注册进去了 */ return htons(ETH_P_802_2); }
通过上面的文章描述,我们清楚了LInux是如何组织二层协议处理的,希望大家批评指正