linux 组播

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 <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <netinet/in.h>


#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 <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <netinet/in.h>

#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组的消息);

 我理解的是一般也不需要这么用,一个设备上两个网络设备接收一样的消息也没必要;


 

猜你喜欢

转载自blog.csdn.net/chuxuezhe_158/article/details/83587205