一个关于组播的网络问题--暂没有时间确定真实原因

简介:
操作系统:Debian 3(内核2.6.8-2-686-smp)
网卡:四块网卡,驱动程序为e1000(e1000.ko)
应用程序:在其中一块或者多块网卡上启动heartbeat(向组播地址发包)
现象:heartbeat绑在ethx发送组播,然而ethx没有插网线,网卡口本身状态up,此时modprobe ip_conntrack,然后rmmod ip_conntrack,rmmod将会阻塞,通过ps查看它是running状态,发送SIGKILL给它也不行,此时只需要在ethx插上网线或者ifdown ethx,rmmod方可返回(此中ethx可以代表多个网卡口,若代表多个网卡,那么必须这几个网卡都插上线或者都ifdown了才可以);
分析过程:使用别的网卡是没有这个现象的,我的意思是说在别的不使用e1000驱动的网卡的机器上是没有这个问题的,另外不启动组播也是没有问题的,因此问题肯定在网卡驱动和组播。
内核代码情况:在ip_conntrack_core.c中有ip_conntrack_cleanup函数,肯定是在该函数中死循环了,因为/proc/net/ip_conntrack文件已经没有了,故而proc_net_remove("ip_conntrack")调用是成功的,按照下面的序列:
cleanup_proc:
    proc_net_remove("ip_conntrack");
 cleanup_init:
    ip_conntrack_cleanup();
 cleanup_nothing:
    return ret;
则肯定是ip_conntrack_cleanup的问题:
void ip_conntrack_cleanup(void)
{
    ip_ct_attach = NULL;
    synchronize_net();
 i_see_dead_people:
    ip_ct_selective_cleanup(kill_all, NULL);
    if (atomic_read(&ip_conntrack_count) != 0) {  //1
        schedule();
        goto i_see_dead_people;
    }
    while (atomic_read(&ip_conntrack_untracked.ct_general.use) > 1) //2
        schedule();
    kmem_cache_destroy(ip_conntrack_cachep);
    vfree(ip_conntrack_hash);
    nf_unregister_sockopt(&so_getorigdst);
}
该函数中有两个循环,一个是1处,一个是2处,经调试是1处死循环了,结果是ip_conntrack_count这个计数器一直是1,然而ip_ct_selective_cleanup中的:
while ((h = get_next_corpse(kill, data, &bucket)) != NULL)
其h返回却是NULL,进一步跟踪发现get_next_corpse是从ip_conntrack_hash中找数据的,而ip_conntrack_hash中的数据是在ip_confirm被加入的,也就是说在一个数据包必须离开内核的时候才会将其ip_conntrack加入到ip_conntrack_hash,然而ip_conntrack_count计数器却是在init_conntrack中被递增的,也就是在数据包刚进入内核的时候被递增的,这就在ip_conntrack_hash的更新和ip_conntrack_count的更新之间留下了一个时间段,不过这个时间段不要紧,内核并不保证ip_ct_selective_cleanup中的ip_conntrack_hash遍历的数目一定要和ip_conntrack_count完全一致,也就是说,一个ip_conntrack可以已经由于它的存在递增了ip_conntrack_count,但是它可以不加入ip_conntrack_hash哈希表,这样不会有任何问题,这是因为在ip_ct_selective_cleanup中遍历ip_conntrack_hash表,然后依次调用ip_conntrack_put来降之destroy掉,如果哈希表中没有的ip_conntrack则可以在别的内核路径调用ip_conntrack_put来destroy掉,进而在destroy中递减ip_conntrack_count这个计数器,最终肯定在ip_conntrack_cleanup中会得到计数器ip_conntrack_count成为0的信息,最终从i_see_dead_people这个不吉利的循环中出来。
     现在的问题就是为何在使用e1000驱动并且没有插网线发送组播的情况下(数据肯定发不出去),卸载ip_conntrack模块的时候没有任何内核路径调用ip_conntrack_put来释放这个游离的ip_conntrack,另外为何会有一个游离的ip_conntrack没有加入到ip_conntrack_hash哈希表却已经存在了,也即是它还没有准备从内核出去!此时,只要一插网线(在组播相关的所有网卡上插一下网线,哪怕插一下再拔下也行)或者ifdown所有的组播相关的网卡,一个ip_conntrack_put将被调用释放掉最后的那个游离ip_conntrack结构,递减掉最后的那个1,最终卸载ip_conntrack模块完成。

     目前没有时间来调试e1000的驱动,我觉得问题十有八九出在这个驱动里面,可能需要网卡工具对网卡设置一下什么的。



 本文转自 dog250 51CTO博客,原文链接:http://blog.51cto.com/dog250/1271145

相关文章
|
5月前
|
网络协议 Unix C语言
C语言 网络编程(十六)广播和组播
广播和组播是网络通信的重要方式。广播允许一台主机向子网内所有主机发送数据包,常用于局域网内的消息传播;组播则将数据包发送给特定的一组主机,适用于视频会议等应用场景。广播地址如 `192.168.1.255` 用于同一子网的所有主机。组播地址如 `224.0.0.0` 至 `239.255.255.255` 标识特定主机群。C语言示例展示了如何通过 UDP 实现广播和组播通信。此外,UNIX域套接字用于同一机器上进程间的高效通信。
295 14
|
9月前
|
监控 网络架构 负载均衡
|
9月前
|
前端开发 网络安全 网络架构
【华为数通HCIP | 网络工程师】821-BGP 组播高频题与解析(1)
【华为数通HCIP | 网络工程师】821-BGP 组播高频题与解析(1)
499 0
|
9月前
|
网络协议 算法 网络架构
计算机网络:网络层(IP组播)
计算机网络:网络层(IP组播)
120 0
|
9月前
|
缓存 算法 网络架构
【网络层】IP组播(多播)、硬件组播、IGMP、组播路由选择协议、移动IP、路由器详解、路由表和路由转发
【网络层】IP组播(多播)、硬件组播、IGMP、组播路由选择协议、移动IP、路由器详解、路由表和路由转发
196 0
|
网络协议
计算机网络 | 广播与组播
计算机网络 | 广播与组播
125 0
|
Cloud Native 块存储
阿里云最新产品手册——阿里云核心产品——块存储——飞天洛神3.0——云网络发展历程——云网络3.0时代——云原生组播能力
阿里云最新产品手册——阿里云核心产品——块存储——飞天洛神3.0——云网络发展历程——云网络3.0时代——云原生组播能力自制脑图
229 1
|
网络协议
Qt网络编程之搭建Udp通信【单播、组播、广播】
Qt网络编程之搭建Udp通信【单播、组播、广播】
937 0
|
网络协议 算法 网络架构
【计算机网络】网络层 : IP 组播 ( IP 数据报传输方式 | 组播 IP 地址 | 组播 MAC 地址 | IGMP 协议 | 组播路由选择协议 )
【计算机网络】网络层 : IP 组播 ( IP 数据报传输方式 | 组播 IP 地址 | 组播 MAC 地址 | IGMP 协议 | 组播路由选择协议 )
414 0
【计算机网络】网络层 : IP 组播 ( IP 数据报传输方式 | 组播 IP 地址 | 组播 MAC 地址 | IGMP 协议 | 组播路由选择协议 )

热门文章

最新文章