socket通过多网卡收发数据

来源:互联网 发布:windows redmine 迁移 编辑:程序博客网 时间:2024/05/02 03:42

1.   通过bind机制,   socket必须要调用bind才能发送tcp包。 bind调用时需要一个ip地址。一般一台机器的多网口都要配置不同的ip地址(路由器除外,路由器是一个网桥设备,只是负责转发包,所以其它的端口一般没有ip地址)。

2.   通过ioctl来设置打开的socket.     (ioctl (fd, SIOCGIFINDEX, &ifr)==0;


TCP编程时不管是客户端还是服务器端,都要调用bind后才能连接/收发数据。

UDP在客户端时可以不调用bind而直接使用recvfrom/sendto来收发数据。 

在客户端一般是本地地址选择INETADDR_ANY, 表示所有网络接口。


代码如下:

#include <stdio.h>;
#include <string.h>;
#include <sys/socket.h>;
#include <netpacket/packet.h>;
#include <net/ethernet.h>;
#include <sys/ioctl.h>;
#include <net/if.h>;
#include <assert.h>;


int
main ()
{
        struct sockaddr_ll sll;
        int fd;
        struct ifreq ifr;
        char *dev;

        fd = socket (PF_PACKET, SOCK_RAW, htons(ETH_P_ALL));

        dev = "eth0";
        strncpy ((char *)ifr.ifr_name, dev, sizeof(ifr.ifr_name));
        assert (ioctl (fd, SIOCGIFINDEX, &ifr)==0);

        memset (&sll, 0, sizeof(sll));
        sll.sll_family = AF_PACKET;
        sll.sll_protocol = htons (ETH_P_ALL);
        sll.sll_ifindex = ifr.ifr_ifindex;

        assert (bind (fd, (struct sockaddr *)&sll, sizeof(sll))==0);

}





##############################################################




Windows网络编程总结(一)
作者:   Kendiv
出处:   CSDN.NET  

  关于bind:

  INADDR_ANY   的具体含义是,绑定到0.0.0.0。此时,对所有的地址都将是有效的,如果系统考虑冗余,采用多个网卡的话,那么使用此种bind,将在所有网卡上进行绑定。在这种情况下,你可以收到发送到所有有效地址上数据包。

  例如:

  SOCKADDR_IN   Local;

  Local.sin_addr.s_addr   =   htonl(INADDR_ANY);

  
  另外一种方式如下:

  SOCKADDR_IN   Local;

  hostent*   thisHost   =   gethostbyname( " ");

  char*   ip   =   inet_ntoa(*(struct   in_addr   *)*thisHost-> h_addr_list);

  Local.sin_addr.s_addr   =   inet_addr(ip);

  在这种方式下,将在系统中当前第一个可用地址上进行绑定。在多网卡的环境下,可能会出问题。

  
  最常见的方式:

  const   char   LocalIP[]   =   "192.168.0.100 ";

  SOCKADDR_IN   Local;

  Local.sin_addr.s_addr   =   inet_addr(LocalIP);

  bind(socket,   (LPSOCKADDR)&Local,   sizeof(SOCKADDR_IN)


  bind的安全问题:

  如果你在bind时,使用了INADDR_ANY那么,你将可以在所有有效的地址上进行监听,但是Socket有一个特性:可在同一端口上绑定多个Socket。

  让我们看看下面的情况:假设你的系统只有一个IP:192.168.0.100,你希望bind到4096端口。对于下面的两种bind,当数据包到达时,谁会接收到呢?

  Local.sin_addr.s_addr   =   htonl(INADDR_ANY);

  Local.sin_addr.s_addr   =   inet_addr(“192.168.0.100”);


  WinSocke库是这样处理的:谁绑定的最明确,谁获取数据包。显然,第二种bind将获取到达的数据包。如果避免这种情况呢?使用SO_EXECLUSINEADDRUSE选项。需要注意的是,此选项在Windows   NT   4   Service   Pack   4以后(包括SP4)才可以使用。

  示例代码:

  #ifndef   SO_EXECLUSINEADDRUSE

  #define   SO_EXECLUSINEADDRUSE   ((int)(~SO_REUSEADDR))

  #endif
  

  SOCKADDR_IN   Local;

  BOOL   val   =   TRUE;

  
  Local.   sin_family   =   AF_INET;

  Local.   sin_port   =   htons(4096);

  Local.sin_addr.s_addr   =   htonl(INADDR_ANY);

  
  setsocketopt(socket,

  SOL_SOCKET,

  SO_EXECLUSINEADDRUSE,

  (char*)&val,

  sizeof(val));

  
  bind(socket,   (LPSOCKADDR)&Local,   sizeof(SOCKADDR_IN)