Linux组播发包命令 linux发送广播包( 二 )


linux 怎样加入一个多播组应用程序通过命令字IP_ADD_MEMBERSHIP把一个socket加入到一个多播组Linux组播发包命令,IP_ADD_MEMBERSHIP是一个IP层的命令字,其调用使用的参数是结构体struct ip_mreq,其定义如下Linux组播发包命令:
struct ip_mreq
{
struct in_addr imr_multiaddr;
struct in_addr imr_interface;
};
该结构体的两个成员分别用于指定所加入的多播组的组IP地址,和所要加入组的那个本地接口的IP地址 。该命令字没有源过滤的功能,它相当于实现IGMPv1的多播加入服务接口 。
ip_setsockopt实现了该命令字,它通过调用ip_mc_join_group把socket加入到多播组 。
表示socket的结构体struct inet_sock有一个成员mc_list,它是一个结构体struct ip_mc_socklist的指针,实际上一个该结构体的链表,该结构体的定义如下:
struct ip_mc_socklist
{
struct ip_mc_socklist*next;
struct ip_mreqnmulti;
unsigned intsfmode;
struct ip_sf_socklist*sflist;
};
next指向链表的下一个节点Linux组播发包命令;multi表示组信息,即在哪一个本地接口上,加入到哪一个多播组;sfmode是过滤模式,取值为
MCAST_INCLUDE或MCAST_EXCLUDE , 分别表示只接收sflist所列出的那些源的多播数据报,和不接收sflist所列出的那些源
的多播数据报;sflist是源列表,结构体struct ip_sf_socklist的定义如下:
struct ip_sf_socklist
{
unsigned intsl_max;
unsigned intsl_count;
__u32sl_addr[0];
};
sl_addr是源地址列表,sl_count应该是源地址列表中源地址的数量,sl_max应该是当前sl_addr数组的最大可容纳量(不确定) 。对
于通过调用IP_ADD_MEMBERSHIP加入的多播组,它会在struct inet_sock的mc_list的链表头添加如下一个节点:
struct ip_mc_socklist{
.next = 原来的链表头;
.multi = 所加入的多播组,和接口信息;
.sfmode = MCAST_EXCLUDE;
.sflist = NULL;即不排除任何源地址,也就是不存在源过滤 。
}
另外,一个socket所允许加入的多播组的最大数量也是有限制的 , mc_list中节点的数量不允许超过sysctl_igmp_max_memberships(缺省为20) 。
ip_mc_join_group还需要通过ip_mreq.imr_interface的指定值找到要加入多播组的那个接口,并为接口设置状态(即该接
口要加入哪个多播组,过滤哪些源 , 也就是为该接口增加一个组,如果要增加的组已存在 , 则增加该组的引用计数) 。代表网络设备接口的结构体struct
in_device有一个成员mc_list,这是一个结构体struct ip_mc_list的链表,该结构体的定义如下:
struct ip_mc_list
{
struct in_device*interface;
unsigned longmultiaddr;
struct ip_sf_list*sources;
struct ip_sf_list*tomb;
unsigned intsfmode;
unsigned longsfcount[2];
struct ip_mc_list*next;
struct timer_listtimer;
intusers;
atomic_trefcnt;
spinlock_tlock;
chartm_running;
charreporter;
charunsolicit_count;
charloaded;
unsigned chargsquery;
unsigned charcrcount;
};
interface指向网络设备接口,multicast即为加入的组的多播地址,users记录当前有几个socket在该接口上加入了该多播组 。
sfcount是一个有两个元素的数组 , 分别记录在该接口上加入多播组的socket的过滤模式为EXCLUDE和INCLUDE的数量 , sfmode为
该接口本身的过滤模式 。sources为源地址列表,该结构体具体内容稍后再分析 。timer为主动报告定时器 , 当一个接口(注意:不是socket)新
加入到一个多播组,需要向多播路由器发送一个igmp报告,以通知多播路由器需要向本地网络转发该组的数据报 。tm_running是一个标志,如果

推荐阅读