13

IN6ADDR_ANYソケットがまたはにバインドされていて、ソケットでメッセージを受信するINADDR_ANYなどの呼び出しを使用する場合。recvfrom()メッセージがどのインターフェイスから来たかを調べる方法はありますか?

IPv6 リンク スコープ メッセージの場合、from 引数のフィールドがインターフェイス ID に初期化されることを期待していましrecvfrom()scope_id。残念ながら0、私のテストプログラムではに設定されています。

この情報を見つける方法を知っている人はいますか?

4

5 に答える 5

10

dwc は正しいです。IPV6_PKTINFO は Linux 上の IPv6 で機能します。

さらに、IP_PKTINFO は IPv4 で機能します — 詳細はマンページ ip(7) で確認できます。

于 2009-03-03T20:15:12.010 に答える
5

送信元アドレス、宛先アドレス、およびインターフェイス アドレスを抽出する例を作成しました。簡潔にするために、エラー チェックは提供されません。この複製を参照してください:受信した UDP パケットの宛先アドレスを取得します

// sock is bound AF_INET socket, usually SOCK_DGRAM
// include struct in_pktinfo in the message "ancilliary" control data
setsockopt(sock, IPPROTO_IP, IP_PKTINFO, &opt, sizeof(opt));
// the control data is dumped here
char cmbuf[0x100];
// the remote/source sockaddr is put here
struct sockaddr_in peeraddr;
// if you want access to the data you need to init the msg_iovec fields
struct msghdr mh = {
    .msg_name = &peeraddr,
    .msg_namelen = sizeof(peeraddr),
    .msg_control = cmbuf,
    .msg_controllen = sizeof(cmbuf),
};
recvmsg(sock, &mh, 0);
for ( // iterate through all the control headers
    struct cmsghdr *cmsg = CMSG_FIRSTHDR(&mh);
    cmsg != NULL;
    cmsg = CMSG_NXTHDR(&mh, cmsg))
{
    // ignore the control headers that don't match what we want
    if (cmsg->cmsg_level != IPPROTO_IP ||
        cmsg->cmsg_type != IP_PKTINFO)
    {
        continue;
    }
    struct in_pktinfo *pi = CMSG_DATA(cmsg);
    // at this point, peeraddr is the source sockaddr
    // pi->ipi_spec_dst is the destination in_addr
    // pi->ipi_addr is the receiving interface in_addr
}
于 2011-03-15T08:33:24.317 に答える
4

各インターフェースへのバインドは別として、IPv4 自体を使用する方法については知りません。

IPv6 では、この欠点に対処するために IPV6_PKTINFO ソケット オプションが追加されました。このオプションを有効にすると、 astruct in6_pktinfoが補助データとして返されます。

于 2009-03-03T17:20:41.367 に答える
-1

C/C++ TCP/IP コーディングを行ってからしばらく経ちましたが、覚えている限り、すべてのメッセージ (または派生ソケット) で IP ヘッダー情報を取得できます。これらのヘッダーには、問い合わせているインターフェイスの IP となる受信アドレスが含まれている必要があります。

于 2009-03-02T22:16:28.887 に答える
-2

Glomekが提案したように、各インターフェースで個別のソケットを開く以外に、Windowsでこれを決定的に行う唯一の方法は、生のソケットを使用することです。

  SOCKET s = socket(AF_INET, SOCK_RAW, IPPROTO_IP);

このソケットからの各受信は、発信元アドレスと宛先アドレスの両方を含むIP パケットになります。私が取り組んでいるプログラムでは、SIO_RCVALL オプションを使用してソケットをプロミスキャス モードにする必要があります。これを行うということは、インターフェイスがネットワーク上で「見る」すべての IP パケットを取得することを意味します。アプリケーション用に明示的にパケットを抽出するには、IP および TCP/UDP ヘッダーのアドレスとポートを使用してデータをフィルタリングする必要があります。明らかに、それはおそらくあなたが興味を持っているよりも多くのオーバーヘッドです。私はこれを言うために言及しているだけです.無差別モードにしないで生のソケットを使用したことはありません. したがって、それを INADDR_ANY にバインドして、その時点から通常のソケットとして使用できるかどうかはわかりません。できるように思えます。私はそれを試したことがないだけです。

編集: Windows での raw ソケットに関する制限については、 この記事をお読みください。私のプロジェクトで直面した最大のハードルは、Windows 2000 以降で raw ソケットを開くには Administrators グループのメンバーでなければならないということでした。

于 2009-03-03T05:34:15.357 に答える