2011-07-21 71 views
0

我正在編程一個多播環境場景。源將內容發送到多播地址239.0.1.1。在另一端客戶端想要加入這個多播地址。多播的實際工作是,一個想要加入多播組的客戶端發送一個加入請求給它所連接的路由器。我必須對接收到來自客戶端的多播加入請求的路由器操作進行編程,從多播地址獲取數據並將內容提供給客戶。我無法弄清楚如何做到這一點。對於單個用戶來說很好。但是,當有多個客戶端我使用的技術是不正確的......請幫我..一個代碼加入特定的地址和數據發送到一個單一的客戶端如下所示:在C中多播UDP

#include <sys/types.h> /* for type definitions */ 
#include <sys/socket.h> /* for socket API calls */ 
#include <netinet/in.h> /* for address structs */ 
#include <arpa/inet.h> /* for sockaddr_in */ 
#include <stdio.h>  /* for printf() and fprintf() */ 
#include <stdlib.h>  /* for atoi() */ 
#include <string.h>  /* for strlen() */ 
#include <unistd.h>  /* for close() */ 

#define MAX_LEN 4074 /* maximum receive string size */ 
#define MIN_PORT 1024 /* minimum port allowed */ 
#define MAX_PORT 65535 /* maximum port allowed */ 


int main(int argc, char *argv[]) 
{ 

    int sock;      /* socket descriptor */ 
    int flag_on = 1;    /* socket option flag */ 
    struct sockaddr_in mc_addr; /* socket address structure */ 
    char recv_str[MAX_LEN+1];  /* buffer to receive string */ 
    int recv_len;     /* length of string received */ 
    struct ip_mreq mc_req;  /* multicast request structure */ 
    char* mc_addr_str;   /* multicast IP address */ 
    unsigned short mc_port;  /* multicast port */ 
    //unsigned short mc_port1=1400;  /*multicast port */ 
    struct sockaddr_in from_addr; /* packet source */ 
    unsigned int from_len;  /* source addr length */ 
    // to send the data to the client 

    int sockfd, newsockfd, portno; 
    socklen_t clilen; 
    struct sockaddr_in serv_addr, cli_addr; 
    int n; 
    int cnt; 
    int no=4096,i; 
    char *buffer[no]; 

    int dest_sock; 
    struct sockaddr_in dest_addr; 
    char * dest_address="192.168.1.4"; 
    unsigned int dest_port=5000; 

    /* validate number of arguments */ 
    if (argc != 3) 
    { 
    fprintf(stderr, 
      "Usage: %s <Multicast IP> <Multicast Port>\n", 
      argv[0]); 
    exit(1); 
    } 

    mc_addr_str = argv[1];  /* arg 1: mlticast ip address */ 
    mc_port = atoi(argv[2]); /* arg 2: multicast port number */ 

    /* validate the port range */ 
    if ((mc_port < MIN_PORT) || (mc_port > MAX_PORT)) { 
    fprintf(stderr, "Invalid port number argument %d.\n", 
      mc_port); 
    fprintf(stderr, "Valid range is between %d and %d.\n", 
      MIN_PORT, MAX_PORT); 
    exit(1); 
    } 

    /* create socket to join multicast group on */ 
    if ((sock = socket(PF_INET, SOCK_DGRAM, IPPROTO_UDP)) < 0) { 
    perror("socket() failed"); 
    exit(1); 
    } 
    else 
    printf("Socket connection successful\n"); 


    /* set reuse port to on to allow multiple binds per host */ 
    if ((setsockopt(sock, SOL_SOCKET, SO_REUSEADDR, &flag_on, 
     sizeof(flag_on))) < 0) { 
    perror("setsockopt() failed"); 
    exit(1); 
    } 


    /* construct a multicast address structure */ 
    memset(&mc_addr, 0, sizeof(mc_addr)); 
    mc_addr.sin_family  = AF_INET; 
    mc_addr.sin_addr.s_addr = htonl(INADDR_ANY); 
    mc_addr.sin_port  = htons(mc_port); 


     /* bind to multicast address to socket */ 
    if ((bind(sock, (struct sockaddr *) &mc_addr, 
     sizeof(mc_addr))) < 0) { 
    perror("bind() failed"); 
    exit(1); 
    } 

    /* construct an IGMP join request structure */ 
    mc_req.imr_multiaddr.s_addr = inet_addr(mc_addr_str); 
    mc_req.imr_interface.s_addr = htonl(INADDR_ANY); 

    /* send an ADD MEMBERSHIP message via setsockopt */ 
    if ((setsockopt(sock, IPPROTO_IP, IP_ADD_MEMBERSHIP, 
     (void*) &mc_req, sizeof(mc_req))) < 0) { 
    perror("setsockopt() failed"); 
    exit(1); 
    } 
printf("setsocket successfull\n"); 

//for sending the data to the client 

//creating a socket to the client and waiting for the client to receive the data 


    if ((dest_sock = socket(PF_INET, SOCK_DGRAM, IPPROTO_UDP)) < 0) { 
    perror("socket() failed"); 
    exit(1); 
    } 
    else 
    printf("Socket connection successful\n"); 
    if ((setsockopt(dest_sock, SOL_SOCKET, SO_REUSEADDR, &flag_on, 
    sizeof(flag_on))) < 0) { 
    perror("setsockopt() failed"); 
    exit(1); 
    } 
    sockfd = socket(AF_INET, SOCK_STREAM, 0); 
    if (sockfd < 0) 
     error("ERROR opening socket"); 
    bzero((char *) &serv_addr, sizeof(serv_addr)); 
    portno = 5000;//atoi(argv[1]); 
    serv_addr.sin_family = AF_INET; 
    serv_addr.sin_addr.s_addr = INADDR_ANY; 
    serv_addr.sin_port = htons(portno); 
    if (bind(sockfd, (struct sockaddr *) &serv_addr, 
    //   sizeof(serv_addr)) < 0) 
    //   error("ERROR on binding"); 
    // listen(sockfd,5); 
    // clilen = sizeof(cli_addr); 
    // newsockfd = accept(sockfd, 
       //(struct sockaddr *) &cli_addr, 
    //   // &clilen); 
    //if (newsockfd < 0) 
    // error("ERROR on accept"); 
/*------------------------------------------------------------------------------------------------------*/ 

for(cnt=0;cnt<100;cnt++) {   /* loop forever */ 

    /* clear the receive buffers & structs */ 
    memset(recv_str, 0, sizeof(recv_str)); 
    from_len = sizeof(from_addr); 
    memset(&from_addr, 0, from_len); 

    /* block waiting to receive a packet */ 
    //buffer[cnt]=recvfrom(sock, recv_str, MAX_LEN, 0,(struct sockaddr*)&from_addr, &from_len); 
    // no++; 
    if ((recv_len = recvfrom(sock, recv_str, MAX_LEN, 0,(struct sockaddr*)&from_addr, &from_len)) < 0) 
    //if(strlen(buffer[cnt])<0) 
    { 
      perror("recvfrom() failed"); 
      exit(1); 
     } 

    /* output received string */ 
    // printf("Received %d bytes from %s: ", recv_len, inet_ntoa(from_addr.sin_addr)); 
    printf("\n"); 
    //printf("%s\n",recv_str); 
     //printf("\n"); 
    //buffer[i]=recv_str; 
    //printf("%s\n",buffer[i]); 
    //printf("Enter the writing mode\n"); 
    n = write(newsockfd,recv_str,sizeof(recv_str)); 
    i++; 
    //printf("%d\n",n); 
     if (n < 0) 
    { 
    error("ERROR writing to socket"); 
    exit(1); 
    } 
     i++; 

    //return 0; 

    } 
    close(newsockfd); 
    close(sockfd); 
    /* send a DROP MEMBERSHIP message via setsockopt */ 
    if ((setsockopt(sock, IPPROTO_IP, IP_DROP_MEMBERSHIP, 
     (void*) &mc_req, sizeof(mc_req))) < 0) { 
    perror("setsockopt() failed"); 

    exit(1); 
    } 
    close(sock); 

} 
+0

我認爲你在混合使用高於組播的pub-sub協議以及使用IGMP實現多播網絡時會感到困惑。 –

+0

@Steve對不起,我不明白你想說什麼...... – iwant2learn

+0

我認爲這是你的問題,首先閱讀關於IGMP的維基百科文章,然後閱讀關於發佈/訂閱的文章。 –

回答

0

它聽起來像你試圖實現一個多播路由器,如​​3210。

xorp的源代碼相當密集,但igmpproxy的功能與您之後的功能相似,沒有您不需要的其他功能堆。