用C語言socket RAW功能實現同一臺主機兩個網口間數據收發

1,本例子使用一臺PC,有兩個網口,OS爲CENTOS 6.8,然後使用兩條網線分別都連接到同一個交換機上,當數據經過交換機時,交換機的數據燈會閃爍,否則就沒有數據通過交換機。


2,開始測試,剛開始理所當然的想到的是TCP Server/TCP Client模式,建立socket

    int server_sockfd = socket(AF_INET,SOCK_STREAM, 0);
    struct sockaddr_in server_sockaddr;
    server_sockaddr.sin_family = AF_INET;
    server_sockaddr.sin_port = htons(MYPORT);
    server_sockaddr.sin_addr.s_addr = inet_addr(IP_PORT_0);

    if(bind(server_sockfd,(struct sockaddr *)&server_sockaddr,sizeof(server_sockaddr))==-1)
    {
        perror("bind");
        exit(1);
    }

    if(listen(server_sockfd,QUEUE) == -1)
    {
        perror("listen");
        exit(1);
    }


    char buffer[BUFFER_SIZE];
    struct sockaddr_in client_addr;
    socklen_t length = sizeof(client_addr);

    int conn = accept(server_sockfd, (struct sockaddr*)&client_addr, &length);
    if(conn<0)
    {
        perror("connect");
        exit(1);
    }

    while(1)
    {
        memset(buffer,0,sizeof(buffer));
        int len = recv(conn, buffer, sizeof(buffer),0);
        send(conn, buffer, len, 0);
    }
    close(conn);
    close(server_sockfd);
    return 0;
}


TCP Client端同樣寫相應代碼,編譯通過,先後執行server/client程序,顯示都有數據收發成功,但交換機的數據燈沒有任何顯示,於是明白了,系統默認直接從本地發送接收數據,並沒有從網卡端口實際發送出去,因爲TCP是基於IP協議之上,而IP協議是肯定走本地內部路由,因此使用TCP,UDP等IP之上的協議肯定不會成功。

int init_sock (char index[IFNAMSIZ])
{
  int fd = socket(PF_PACKET, SOCK_RAW, htons(ETH_P_ALL));  
  struct sockaddr_ll sll;                 
     struct ifreq ifr;                   
     strncpy(ifr.ifr_name, index , IFNAMSIZ);    
     ioctl(fd, SIOCGIFINDEX, &ifr);  
     bzero(&sll, sizeof(sll));  
     sll.sll_ifindex = ifr.ifr_ifindex;
     ioctl(fd, SIOCGIFHWADDR, &ifr);
     memcpy(sll.sll_addr,ifr.ifr_hwaddr.sa_data,6);
  sll.sll_family = AF_PACKET;
  sll.sll_protocol = htons(ETH_P_ALL);
  bind(fd,(struct sockaddr *)&sll,sizeof(sll));
  return fd;
}


使用如上RAW Socket初始化後,再發送接收,一切OK!!!,能夠接收到除了CRC之外的完整鏈路層幀,包括MAC地址,幀類型字段等,程序發送接收時,交換機的數據燈不停閃爍。



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