Warning: file_get_contents(/data/phpspider/zhask/data//catemap/4/c/56.json): failed to open stream: No such file or directory in /data/phpspider/zhask/libs/function.php on line 167

Warning: Invalid argument supplied for foreach() in /data/phpspider/zhask/libs/tag.function.php on line 1116

Notice: Undefined index: in /data/phpspider/zhask/libs/function.php on line 180

Warning: array_chunk() expects parameter 1 to be array, null given in /data/phpspider/zhask/libs/function.php on line 181
C 如何将原始套接字绑定到特定接口_C_Linux_Sockets_Network Programming_Raw Sockets - Fatal编程技术网

C 如何将原始套接字绑定到特定接口

C 如何将原始套接字绑定到特定接口,c,linux,sockets,network-programming,raw-sockets,C,Linux,Sockets,Network Programming,Raw Sockets,我的应用程序正在CentOS 5.5上运行。 我正在使用原始套接字发送数据: sd = socket(AF_INET, SOCK_RAW, IPPROTO_RAW); if (sd < 0) { // Error } const int opt_on = 1; rc = setsockopt(m_SocketDescriptor, IPPROTO_IP, IP_HDRINCL, &opt_on, sizeof(opt_on)); if (rc < 0) { close

我的应用程序正在CentOS 5.5上运行。 我正在使用原始套接字发送数据:

sd = socket(AF_INET, SOCK_RAW, IPPROTO_RAW);
if (sd < 0) {
  // Error
}
const int opt_on = 1;
rc = setsockopt(m_SocketDescriptor, IPPROTO_IP, IP_HDRINCL, &opt_on, sizeof(opt_on));
if (rc < 0) {
  close(sd);
  // Error
}
struct sockaddr_in sin;
memset(&sin, 0, sizeof(sin));
sin.sin_family = AF_INET;
sin.sin_addr.s_addr = my_ip_address;

if (sendto(m_SocketDescriptor, DataBuffer, (size_t)TotalSize, 0, (struct sockaddr *)&sin, sizeof(struct sockaddr)) < 0)  {
  close(sd);
  // Error
}
sd=socket(AF_INET、SOCK_RAW、IPPROTO_RAW);
如果(sd<0){
//错误
}
常数int opt_on=1;
rc=setsockopt(m_SocketDescriptor、IPPROTO_IP、IP_HDRINCL和opt_on、sizeof(opt_on));
if(rc<0){
关闭(sd);
//错误
}
sin中的结构sockaddr_;
memset(&sin,0,sizeof(sin));
sin.sin_family=AF_INET;
sin.sin\u addr.s\u addr=我的ip地址;
if(发送到(m_SocketDescriptor,DataBuffer,(size_t)TotalSize,0,(struct sockaddr*)和sin,sizeof(struct sockaddr))<0){
关闭(sd);
//错误
}
如何将此套接字绑定到特定的网络接口(例如eth1)

第一行:设置变量

第二行:告诉程序绑定到哪个接口

第3-5行:获取接口名称的长度,并检查其大小是否过大

六行:设置套接字
sd
的套接字选项,绑定到设备
opt

setsockopt原型:

int setsockopt(int s, int level, int optname, const void *optval, socklen_t optlen);

另外,请确保包含
if.h
socket.h
string.h
头文件

,如前所述,正确的做法是使用
struct ifreq
指定接口名称。这是我的代码示例

#define SERVERPORT 5555
...
struct ifreq ifr;


/* Create the socket */
sd = socket(AF_INET, SOCK_STREAM, 0);
if (sd < 0) 
{
    printf("Error in socket() creation - %s", strerror(errno));
}

/* Bind to eth1 interface only - this is a private VLAN */
memset(&ifr, 0, sizeof(ifr));
snprintf(ifr.ifr_name, sizeof(ifr.ifr_name), "eth1");
if ((rc = setsockopt(sd, SOL_SOCKET, SO_BINDTODEVICE, (void *)&ifr, sizeof(ifr))) < 0)
{
    perror("Server-setsockopt() error for SO_BINDTODEVICE");
    printf("%s\n", strerror(errno));
    close(sd);
    exit(-1);
}

/* bind to an address */
memset(&serveraddr, 0x00, sizeof(struct sockaddr_in));
serveraddr.sin_family = AF_INET;
serveraddr.sin_port = htons(SERVERPORT);
serveraddr.sin_addr.s_addr = inet_addr("9.1.2.3");

int rc = bind(sd, (struct sockaddr *)&serveraddr, sizeof(serveraddr));
相反,我指定了一个特定于所使用接口的IP地址(专用VLAN)。这也修复了netstat输出:

Proto Recv-Q Send-Q Local Address    Foreign Address    State     User Inode      PID/Program name
tcp   0      0      9.1.2.3:5555     0.0.0.0:*          LISTEN    0    210898     26996/myserver  

将套接字绑定到特定的接口IP地址

int bind_using_iface_ip(int fd, char *ipaddr, uint16_t port)
{
    struct sockaddr_in localaddr = {0};
    localaddr.sin_family    = AF_INET;
    localaddr.sin_port  = htons(port);
    localaddr.sin_addr.s_addr = inet_addr(ipaddr);
    return bind(fd, (struct sockaddr*) &localaddr, sizeof(struct sockaddr_in));
}

将套接字绑定到特定接口名称

int bind_using_iface_name(int fd, char *iface_name)
{
    return setsockopt(fd, SOL_SOCKET, SO_BINDTODEVICE, iface_name, strlen(iface_name))
}

bind\u使用\u iface\u ip
中,应该传递绑定到任何端口
0
的消息。而且,如果
fd
是原始套接字,则需要将端口作为
0
传递。这种绑定机制适用于所有类型的套接字,如raw、dgram和stream。

为什么要这样做?除非您确定您的计算机将具有名为预定义名称的接口,否则您的程序将失去可移植性。对于嵌入式设备,不需要可移植性。我有6个以太网端口,我需要使用特定的接口发送数据,它可以工作,但有一个小的修改:ifreq接口;memset(&Interface,0,sizeof(Interface));strncpy(Interface.ifr_ifrn.ifrn_name,“eth1”,IFNAMSIZ);if(setsockopt(sd,SOL_SOCKET,SO_BINDTODEVICE,&Interface,sizeof(Interface))<0{close(sd);//Error}SO_BINDTODEVICE仅在以root用户身份运行时有效,对吗?(至少在Linux上)“struct ifreq”的第一个元素是接口的char[]名称。您还可以指定IFNAMSIZ。因此,这应该是相同的(就像在“eth0”中添加任何随机缓冲区一样),根据我在CentOS 7上的经验,这不适用于原始套接字。相反,你需要使用
bind()
,而不是这里显示的
setsockopt()
方法。@JasonR:你的评论缩短了我和我同事的沮丧时间。(我们终于从CentOS 6加速到了CentOS 7,我们一直在绞尽脑汁想为什么一些遗留代码不起作用。)谢谢!在哪里指定使用
struct ifreq
?我的手册页上说“传递的选项是一个可变长度的以null结尾的接口名称字符串”。您建议的问题(对于SOCK\u流套接字,INADDR\u ANY)在使用SOCK\u RAW的原始海报上没有出现。@Bryan这两种方法都应该有效,至少在Linux上是如此。根据netdevice(7),
ifr\u name
字符串是
struct ifreq
的第一个成员,因此指向
struct ifreq
char
的指针的作用相同。这是正确的答案。我们应该绑定到ip地址。不连接到接口?
int bind_using_iface_ip(int fd, char *ipaddr, uint16_t port)
{
    struct sockaddr_in localaddr = {0};
    localaddr.sin_family    = AF_INET;
    localaddr.sin_port  = htons(port);
    localaddr.sin_addr.s_addr = inet_addr(ipaddr);
    return bind(fd, (struct sockaddr*) &localaddr, sizeof(struct sockaddr_in));
}
int bind_using_iface_name(int fd, char *iface_name)
{
    return setsockopt(fd, SOL_SOCKET, SO_BINDTODEVICE, iface_name, strlen(iface_name))
}