35

私のアプリケーションはCentOS5.5で実行されています。私はデータを送信するためにrawソケットを使用しています:

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
}

このソケットを特定のネットワークインターフェイス(たとえばeth1)にバインドするにはどうすればよいですか?

4

3 に答える 3

43
const char *opt;
opt = "eth0";
const len = strnlen(opt, IFNAMSIZ);
if (len == IFNAMSIZ) {
    fprintf(stderr, "Too long iface name");
    return 1;
}
setsockopt(sd, SOL_SOCKET, SO_BINDTODEVICE, opt, len);

最初の行: 変数を設定します

2 行目: どのインターフェイスにバインドするかをプログラムに指示します。

行 3 ~ 5: インターフェイス名の長さを取得し、サイズが大きすぎないかどうかを確認します。

6 行目: socket のソケット オプションを設定sdし、 device にバインドしますopt

setsockopt プロトタイプ:

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

また、ヘッダー ファイルif.hsocket.hおよびstring.hヘッダー ファイルが含まれていることを確認してください。

于 2010-10-22T17:00:02.353 に答える
19

前述のように、正しいことは、 を使用し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));

また、セキュリティの観点から、ソケットをインターフェイスにバインドするのは良いことですがINADDR_ANY、リッスン IP アドレスとして使用するのは意味がありません。これを行うと、すべてのネットワーク インターフェイスの netstat でポートが開いているように見えます。

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

代わりに、使用されているインターフェイス (プライベート VLAN) に固有の IP アドレスを指定しました。これにより、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  
于 2012-06-07T05:37:50.227 に答える