Linux内核分析 - 网络[五]:网桥

上传人:M****1 文档编号:507565147 上传时间:2022-09-17 格式:DOC 页数:8 大小:157.50KB
返回 下载 相关 举报
Linux内核分析 - 网络[五]:网桥_第1页
第1页 / 共8页
Linux内核分析 - 网络[五]:网桥_第2页
第2页 / 共8页
Linux内核分析 - 网络[五]:网桥_第3页
第3页 / 共8页
Linux内核分析 - 网络[五]:网桥_第4页
第4页 / 共8页
Linux内核分析 - 网络[五]:网桥_第5页
第5页 / 共8页
点击查看更多>>
资源描述

《Linux内核分析 - 网络[五]:网桥》由会员分享,可在线阅读,更多相关《Linux内核分析 - 网络[五]:网桥(8页珍藏版)》请在金锄头文库上搜索。

1、 看完了路由表,重新回到netif_receive_skb ()函数,在提交给上层协议处理前,会执行下面一句,这就是网桥的相关操作,也是这篇要讲解的内容。view plaincopy to clipboardprint?1. skb=handle_bridge(skb,&pt_prev,&ret,orig_dev);skb = handle_bridge(skb, &pt_prev, &ret, orig_dev); 网桥可以简单理解为交换机,以下图为例,一台linux机器可以看作网桥和路由的结合,网桥将物理上的两个局域网LAN1、LAN2当作一个局域网处理,路由连接了两个子网1.0和2.0。

2、从eth0和eth1网卡收到的报文在Bridge模块中会被处理成是由Bridge收到的,因此Bridge也相当于一个虚拟网卡。STP五种状态 DISABLED BLOCKING LISTENING LEARNING FORWARDING创建新的网桥br_add_bridge netbridgebr_if.c当使用SIOCBRADDBR调用ioctl时,会创建新的网桥br_add_bridge。 首先是创建新的网桥:view plaincopy to clipboardprint?1. dev=new_bridge_dev(net,name);dev = new_bridge_dev(net,

3、name); 然后设置dev-dev.type为br_type,而br_type是个全局变量,只初始化了一个名字变量view plaincopy to clipboardprint?1. SET_NETDEV_DEVTYPE(dev,&br_type);2. staticstructdevice_typebr_type=3. .name=bridge,4. ;SET_NETDEV_DEVTYPE(dev, &br_type);static struct device_type br_type = .name = bridge,; 然后注册新创建的设备dev,网桥就相当一个虚拟网卡设备,注册过的

4、设备用ifconfig就可查看到:view plaincopy to clipboardprint?1. ret=register_netdevice(dev);ret = register_netdevice(dev); 最后在sysfs文件系统中也创建相应项,便于查看和管理:view plaincopy to clipboardprint?1. ret=br_sysfs_addbr(dev);ret = br_sysfs_addbr(dev);将端口加入网桥br_add_if() netbridgebr_if.c当使用SIOCBRADDIF调用ioctl时,会向网卡加入新的端口br_add

5、_if。 创建新的net_bridge_port p,会从br-port_list中分配一个未用的port_no,p-br会指向br,p-state设为BR_STATE_DISABLED。这里的p实际代表的就是网卡设备。view plaincopy to clipboardprint?1. p=new_nbp(br,dev);p = new_nbp(br, dev); 将新创建的p加入CAM表中,CAM表是用来记录mac地址与物理端口的对应关系;而刚刚创建了p,因此也要加入CAM表中,并且该表项应是local的关系如下图,可以看到,CAM表在实现中作为net_bridge的hash表,以add

6、r作为hash值,链入net_bridge_fdb_entry,再由它的dst指向net_bridge_port。view plaincopy to clipboardprint?1. err=br_fdb_insert(br,p,dev-dev_addr);err = br_fdb_insert(br, p, dev-dev_addr); 设备的br_port指向p。这里要明白的是,net_bridge可以看作全局量,是网桥,而net_bridge_port则是与网卡相对应的端口,因此每个设备dev有个指针br_port指向该端口。view plaincopy to clipboardpri

7、nt?1. rcu_assign_pointer(dev-br_port,p);rcu_assign_pointer(dev-br_port, p); 将新创建的net_bridge_port加入br的链表port_list中,在创建新的net_bridge_port时,会分配一个未用的port_no,而这个port_no就是根据br-port_list中的已经添加的net_bridge_port来找到未用的port_no的具体如下图。view plaincopy to clipboardprint?1. list_add_rcu(&p-list,&br-port_list);list_add

8、_rcu(&p-list, &br-port_list); 重新计算网桥的ID,这里根据br-port_list链表中的net_bridge_port的最小的addr来作为网桥的ID。view plaincopy to clipboardprint?1. br_stp_recalculate_bridge_id(br);br_stp_recalculate_bridge_id(br); 网卡设备的删除br_del_bridge()与端口的移除add_del_if()与添加差不多,不再详述。熟悉了网桥的创建与添加,再来看下网桥是如何工作的。 当收到数据包,通过netif_receive_skb(

9、)-handle_bridge()处理网桥:view plaincopy to clipboardprint?1. staticinlinestructsk_buff*handle_bridge(structsk_buff*skb,2. structpacket_type*pt_prev,int*ret,3. structnet_device*orig_dev)4. 5. structnet_bridge_port*port;6. 7. if(skb-pkt_type=PACKET_LOOPBACK|8. (port=rcu_dereference(skb-dev-br_port)=NULL)

10、9. returnskb;10. 11. if(*pt_prev)12. *ret=deliver_skb(skb,*pt_prev,orig_dev);13. *pt_prev=NULL;14. 15. 16. returnbr_handle_frame_hook(port,skb);17. static inline struct sk_buff *handle_bridge(struct sk_buff *skb, struct packet_type *pt_prev, int *ret, struct net_device *orig_dev) struct net_bridge_p

11、ort *port; if (skb-pkt_type = PACKET_LOOPBACK | (port = rcu_dereference(skb-dev-br_port) = NULL) return skb; if (*pt_prev) *ret = deliver_skb(skb, *pt_prev, orig_dev); *pt_prev = NULL; return br_handle_frame_hook(port, skb); 1.如果报文来自lo设备,或者dev-br_port为空(skb-dev是收到报文的网卡设备,而在向网桥添加端口时,dev-br_port被赋予了创建

12、的与网卡相对应的端口p),此时不需要网桥处理,直接返回报文; 2.如果报文匹配了之前的ptype_all中的协议,则pt_prev不为空,此时要先进行ptype_all中协议的处理,再进行网桥的处理; 3.br_handle_frame_hook是网桥处理钩子函数,在br_init() netbridgebr.c中 br_handle_frame_hook = br_handle_frame; br_handle_frame() netbridgebr_input.c是真正的网桥处理函数, 下面进入br_handle_frame()开始网桥部分的处理: 与前面802.1q讲的一样,首先检查us

13、ers来决定是否复制报文:view plaincopy to clipboardprint?1. skb=skb_share_check(skb,GFP_ATOMIC);skb = skb_share_check(skb, GFP_ATOMIC); 如果报文的目的地址是01:80:c2:00:00:0X,则是发往STP的多播地址,此时调用br_handle_local_finish()来完成报文的进一步处理:view plaincopy to clipboardprint?1. if(unlikely(is_link_local(dest)2. 3. if(NF_HOOK(PF_BRIDGE,

14、NF_BR_LOCAL_IN,skb,skb-dev,4. NULL,br_handle_local_finish)5. returnNULL;/*frameconsumedbyfilter*/6. else7. returnskb;8. if (unlikely(is_link_local(dest)if (NF_HOOK(PF_BRIDGE, NF_BR_LOCAL_IN, skb, skb-dev, NULL, br_handle_local_finish) return NULL; /* frame consumed by filter */ else return skb; 而br_handle_local_finish()所做

展开阅读全文
相关资源
相关搜索

当前位置:首页 > 幼儿/小学教育 > 小学课件

电脑版 |金锄头文库版权所有
经营许可证:蜀ICP备13022795号 | 川公网安备 51140202000112号