linux網絡編程:多播(組播)編程

什麼是多播    

      單播用於兩個主機之間的端對端通信,廣播用於一個主機對整個局域網上所有主機上的數據通信。單播和廣播是兩個極端,要麼對一個主機進行通信,要麼對整個局域網上的主機進行通信。實際情況下,經常需要對一組特定的主機進行通信,而不是整個局域網上的所有主機,這就是多播的用途。

      多播,也稱爲“組播”,將局域網中同一業務類型主機進行了邏輯上的分組,進行數據收發的時候其數據僅僅在同一分組中進行,其他的主機沒有加入此分組不能收發對應的數據。

       多播的地址是特定的,D類地址用於多播。D類IP地址就是多播IP地址,即224.0.0.0至239.255.255.255之間的IP地址,並被劃分爲局部連接多播地址、預留多播地址和管理權限多播地址3類:

       局部多播地址:在224.0.0.0~224.0.0.255之間,這是爲路由協議和其他用途保留的地址,路由器並不轉發屬於此範圍的IP包。

       預留多播地址:在224.0.1.0~238.255.255.255之間,可用於全球範圍(如Internet)或網絡協議。

       管理權限多播地址:在239.0.0.0~239.255.255.255之間,可供組織內部使用,類似於私有IP地址,不能用於Internet,可限制多播範圍。

屬於永久組的地址:

224.0.0.1   所有組播主機

224.0.0.2   所有組播路由器

224.0.0.4    DRMRP路由器

224.0.0.5    所有OSPF的路由器

224.0.0.6    OSPF指派路由器

224.0.0.9    RPIv2路由器

224.0.0.10  EIGRP路由器

224.0.0.13  PIM路由器

224.0.0.22   IGMPv3

224.0.0.25   RGMP

224.0.1.1      NTP網絡時間協議

IP到以太網地址映射 

        因爲以太網支持多種協議,所以要採取措施分配多播地址,避免衝突。IEEE管理以太網多播地址分配。IEEE把一塊以太網多播地址分給IANA以支持IP多播。塊的地址都以01:00:5e開頭。第25位爲0,低23位爲IPv4組播地址的低23位。IPv4組播地址與MAC地址的映射關係如圖所示:


       由於多播組號中的最高5bit在映射過程中被忽略,因此每個以太網多播地址對應的多播組是不唯一的。32個不同的多播組號被映射爲一個以太網地址。例如,多播地址
224.128.64.32(十六進制e0.80.40.20)和224.0.64.32(十六進制e0.00.40.20)都映射爲同一以太網地址01:00:5e:00:40:20。

多播主機

多播主機分爲三個級別:

    0級:主機不能發送或接收 IP 多播。

這種主機應該自動丟棄它收到的具有D類目的地址的分組。

    1級:主機能發送但不能接收 IP 多播。

在向某個 IP 多播組發送數據報之前,並不要求主機加入該組。多播數據報的發送方式與單播一樣,除了多播數據報的目的地址是 IP多播組之外。網絡驅動器必須能夠識別出這個地址,把在本地網絡上多播數據報。

    2級:主機能發送和接收 IP 多播。

爲了接收 IP 多播,主機必須能夠加入或離開多播組,而且必須支持IGMP,能夠在至少一個接口上交換組成員信息。多接口主機必須支持在它的接口的一個子網上的多播Net/3符合2級主機要求,可以完成多播路由器的工作。與單播IP選路一樣,我們假定所描述的系統是一個多播路由器,並加上了Net/3多播選路的程序。

linux多播編程

linux多播編程步驟:
      1>建立一個socket;
      2>設置多播的參數,例如超時時間TTL,本地迴環許可LOOP等
      3>加入多播組
      4>發送和接收數據
      5>從多播組離開

多播程序設計使用setsockopt()函數和getsockopt()函數來實現,組播的選項是IP層的。

getsockopt()/setsockopt()的選項

含    義

IP_MULTICAST_TTL

        設置多播組數據的TTL值

IP_ADD_MEMBERSHIP

        在指定接口上加入組播組

IP_DROP_MEMBERSHIP

        退出組播組

IP_MULTICAST_IF

        獲取默認接口或設置接口

IP_MULTICAST_LOOP

        禁止組播數據回送

1.選項IP_MULTICASE_TTL
選項IP_MULTICAST_TTL允許設置超時TTL,範圍爲0~255之間的任何值,例如:

unsigned char ttl=255;
setsockopt(s,IPPROTO_IP,IP_MULTICAST_TTL,&ttl,sizeof(ttl)); 


2.選項IP_MULTICAST_IF
選項IP_MULTICAST_IF用於設置組播默認網絡接口,會從給定的網絡接口發送,另一個網絡接口會忽略此數據。例如:

struct in_addr addr;
setsockopt(s,IPPROTO_IP,IP_MULTICAST_IF,&addr,sizeof(addr));
參數addr是希望多播輸出接口的IP地址,使用INADDR_ANY地址回送到默認接口。

默認情況下,當本機發送組播數據到某個網絡接口時,在IP層,數據會回送到本地的迴環接口,選項IP_MULTICAST_LOOP用於控制數據是否回送到本地的迴環接口。例如:


unsigned char loop;
setsockopt(s,IPPROTO_IP,IP_MULTICAST_LOOP,&loop,sizeof(loop));參數loop設置爲0禁止回送,設置爲1允許回送。

3.選項IP_ADD_MEMBERSHIP和IP_DROP_MEMBERSHIP
加入或者退出一個多播組,通過選項IP_ADD_MEMBERSHIP和IP_DROP_MEMBER- SHIP,對一個結構struct ip_mreq類型的變量進行控制,struct ip_mreq原型如下:

struct ip_mreq          

    struct in_addr imn_multiaddr;      /*加入或者退出的廣播組IP地址*/ 
    struct in_addr imr_interface;        /*加入或者退出的網絡接口IP地址*/
};
選項IP_ADD_MEMBERSHIP用於加入某個多播組,之後就可以向這個多播組發送數據或者從多播組接收數據。此選項的值爲mreq結構,成員imn_multiaddr是需要加入的多播組IP地址,成員imr_interface是本機需要加入廣播組的網絡接口IP地址。例如:

struct ip_mreq mreq;
setsockopt(s,IPPROTO_IP,IP_ADD_MEMBERSHIP,&mreq,sizeof(mreq));


多播編程實例


服務器端
        下面是一個多播服務器的例子。多播服務器的程序設計很簡單,建立一個數據包套接字,選定多播的IP地址和端口,直接向此多播地址發送數據就可以了。多播服務器的程序設計,不需要服務器加入多播組,可以直接向某個多播組發送數據。

下面的例子持續向多播IP地址"224.0.0.100"的8888端口發送數據"BROADCAST TEST DATA",每發送一次間隔5s


/*
*broadcast_server.c - 多播服務程序
*/
#include<stdio.h>
#include<stdlib.h>  
#include<string.h>     
#include<unistd.h>      
#include<sys/types.h>    
#include<sys/socket.h>     
#include<arpa/inet.h>    
#include<netinet/in.h> 
 
#define MCAST_PORT         8888
#define MCAST_ADDR         "224.0.0.100"    /*一個局部連接多播地址,路由器不進行轉發*/
#define MCAST_DATA         "BROADCAST TEST DATA"            /*多播發送的數據*/
#define MCAST_INTERVAL     5                            /*發送間隔時間*/
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));     /*初始化IP多播地址爲0*/
    mcast_addr.sin_family = AF_INET;                /*設置協議族類行爲AF*/
    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;
        }      
       
        sleep(MCAST_INTERVAL);                          /*等待一段時間*/
    }
   
    return 0;
}


客戶端
多播組的IP地址爲224.0.0.88,端口爲8888,當客戶端接收到多播的數據後將打印出來。

        客戶端只有在加入多播組後才能接受多播組的數據,因此多播客戶端在接收多播組的數據之前需要先加入多播組,當接收完畢後要退出多播組。

/*
*broadcast_client.c - 多播的客戶端
*/
#include<stdio.h>
#include<stdlib.h>  
#include<string.h>     
#include<unistd.h>      
#include<sys/types.h>    
#include<sys/socket.h>     
#include<arpa/inet.h>    
#include<netinet/in.h>    
 
#define MCAST_PORT         8888
#define MCAST_ADDR         "224.0.0.100"     /*一個局部連接多播地址,路由器不進行轉發*/
#define MCAST_INTERVAL     5                        /*發送間隔時間*/
#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 = 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 = 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;
                                        /*循環接收多播組的消息,5次後退出*/
    for(times = 0;times<5;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;
}

補充:

1、TTL(生存時間值) 是IPv4包頭的一個8 bit字段。

        雖然TTL從字面上翻譯,是可以存活的時間,但實際上TTL是IP數據包在計算機網絡中可以轉發的最大跳數。TTL字段由IP數據包的發送者設置,在IP數據包從源到目的的整個轉發路徑上,每經過一個路由器,路由器都會修改這個TTL字段值,具體的做法是把該TTL的值減1,然後再將IP包轉發出去。如果在IP包到達目的IP之前,TTL減少爲0,路由器將會丟棄收到的TTL=0的IP包並向IP包的發送者發送 ICMP time exceeded消息。

TTL的最大值是255,TTL的一個推薦值是64。

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章