1、Linux下组播 IGMP(Internet Group Managerment Protocol)---- Internet组管理协议,是因特网协议家族中的一个组播协议;
2、除了组播(又称多播),还有单播和广播;具体的定义和区别网上能百度到;
3、常用的组播地址如下:
224.0.1.0~238.255.255.255为用户可用的组播地址(临时组地址),全网范围内有效。
239.0.0.0~239.255.255.255为本地管理组播地址,仅在特定的本地范围内有效。
4、组播发送程序,server.c:
/*
*broadcast_server.c - 多播服务程序
*/
#include
#include
#include
#include
#define MCAST_PORT 8888
#define MCAST_ADDR "224.1.1.100" /*一个局部连接多播地址,路由器不进行转发*/
#define MCAST_DATA "Broadcast test data"
#define MCAST_INTERVAL 2 /*发送间隔时间*/
int main(int argc, char*argv)
{
int s;
struct sockaddr_in mcast_addr;
s = socket(AF_INET, SOCK_DGRAM, 0); /*建立套接字*/
if (s == -1)
{
perror("socket()");
return -1;
}
memset(&mcast_addr, 0, sizeof(mcast_addr));
mcast_addr.sin_family = AF_INET; //类型
mcast_addr.sin_addr.s_addr = inet_addr(MCAST_ADDR); //IP地址
mcast_addr.sin_port = htons(MCAST_PORT); //端口
/*向多播地址发送数据*/
while(1) {
int n = sendto(s, /*套接字描述符*/
MCAST_DATA, /*数据*/
sizeof(MCAST_DATA), /*长度*/
0,
(struct sockaddr*)&mcast_addr,
sizeof(mcast_addr)) ;
if( n < 0)
{
perror("sendto()");
return -2;
}
printf("send ...\n"); //***debug
sleep(MCAST_INTERVAL); /*等待一段时间*/
}
return 0;
}
5、组播接收段程序,client.c :
/*
多播客户端在接收多播组的数据之前需要先加入多播组,当接收完毕后要退出多播组。*/
/*
*broadcast_client.c - 多播的客户端
*/
#include
#include
#include
#include
#define MCAST_PORT 8888
#define MCAST_ADDR "224.1.1.100" /*一个局部连接多播地址,路由器不进行转发*/
#define MCAST_INTERVAL 2 /*发送间隔时间*/
#define BUFF_SIZE 256 /*接收缓冲区大小*/
int main(int argc, char*argv[])
{
int s; /*套接字文件描述符*/
struct sockaddr_in local_addr; /*本地地址*/
int err = -1;
s = socket(AF_INET, SOCK_DGRAM, 0); /*建立套接字*/
if (s == -1)
{
perror("socket()");
return -1;
}
/*初始化地址*/
memset(&local_addr, 0, sizeof(local_addr));
local_addr.sin_family = AF_INET;
local_addr.sin_addr.s_addr = inet_addr(MCAST_ADDR); //如果使用默认的htonl(INADDR_ANY);也可以,如果网络中有多个组播组,这里可以指定
local_addr.sin_port = htons(MCAST_PORT); // 组播服务器的端口
/*绑定socket*/
err = bind(s,(struct sockaddr*)&local_addr, sizeof(local_addr)) ;
if(err < 0)
{
perror("bind()");
return -2;
}
/*设置回环许可*/
int loop = 1;
err = setsockopt(s,IPPROTO_IP, IP_MULTICAST_LOOP,&loop, sizeof(loop));
if(err < 0)
{
perror("setsockopt():IP_MULTICAST_LOOP");
return -3;
}
struct ip_mreq mreq; /*加入多播组*/
mreq.imr_multiaddr.s_addr = inet_addr(MCAST_ADDR); /*多播地址*/
mreq.imr_interface.s_addr = inet_addr("192.168.1.251"); /*这里可以是默认的接口htonl(INADDR_ANY); ,在多网卡时,可以选择要加入组播组的网卡*/
/*将本机加入多播组*/
err = setsockopt(s, IPPROTO_IP, IP_ADD_MEMBERSHIP,&mreq, sizeof(mreq)); //将本机加入组播组
if (err < 0)
{
perror("setsockopt():IP_ADD_MEMBERSHIP");
return -4;
}
int times = 0;
int addr_len = 0;
char buff[BUFF_SIZE];
int n = 0;
/*循环接收多播组的消息,100次后退出*/
for(times = 0;times<100;times++)
{
addr_len = sizeof(local_addr);
memset(buff, 0, BUFF_SIZE); /*清空接收缓冲区*/
/*接收数据*/
n = recvfrom(s, buff, BUFF_SIZE, 0,(struct sockaddr*)&local_addr,&addr_len);
if( n== -1)
{
perror("recvfrom()");
}
printf("Recv %dst message from server: %s\n", times, buff);
sleep(MCAST_INTERVAL);
}
/*退出多播组*/
err = setsockopt(s, IPPROTO_IP, IP_DROP_MEMBERSHIP,&mreq, sizeof(mreq)); //退出组播组
close(s);
return 0;
}
编译:
gcc server.c -o server
gcc client.c -o client
6、server端如果没有配置路由,请添加路由 route add -net 224.0.0.0 netmask 240.0.0.0 dev eth0
否则数据无法发送到组播地址;
7、server和client可以在不同网段的PC上运行,如果client端收不到数据,请运行tcpdump看是否有收到组播数据,如果没有,请检查网线,如果有,但是client程序却收不到数据(我的fedora18就出现了这个情况),这时可能是linux 反向过滤造成的;
在/etc/sysctl.conf下
把 net.ipv4.conf.all.rp_filter和 net.ipv4.conf.default.rp_filter设为0即可
net.ipv4.conf.default.rp_filter = 0
net.ipv4.conf.all.rp_filter = 0
reboot重启即可;
8、client可以运行在多个PC设备上以接收同一个server端的消息;
9、但是如果一台PC上有两个网卡设备eth0(192.168.0.251)和eth1(192.168.1.251),这两个设备不能同时运行client接收同一个组的数据,但是可以接收不同组的数据(比如eth0接收224.1.1.100组的消息,eth1接收224.1.1.101组的消息);
我理解的是一般也不需要这么用,一个设备上两个网络设备接收一样的消息也没必要;