Linux および IGMPv3 でのマルチキャスト参加
-
03-07-2019 - |
質問
厄介な問題に遭遇してしまいました。マルチキャスト UDP トラフィックを受信する C++ プログラムを作成しています。私たちはアプリケーションを別のネットワーク環境に移行する過程にあり、運用チームはアプリケーションからの IGMPv3 メンバーシップのアナウンスをサポートするように要求しました。初期調査では、Linux 2.6 カーネルが IGMPv3 をサポートしていることが示されています。したがって、tcpdump を実行すると次の出力トレースが表示されることに困惑しています。
[rtv@myhost]$ sudo /usr/sbin/tcpdump -i eth1.22 igmp
tcpdump: listening on eth1.22
00:20:09.007094 switch-a.stage > ALL-SYSTEMS.MCAST.NET: igmp query v2 [max resp time 20] [ttl 1]
00:20:09.241946 10.129.22.236 > 232.0.1.10: igmp v2 report 232.0.1.10 (DF) [tos 0xc0] [ttl 1]
00:20:10.472159 10.129.22.236 > 236.0.1.101: igmp v2 report 236.0.1.101 (DF) [tos 0xc0] [ttl 1]
44 packets received by filter
私の理解では、ファイル /proc/sys/net/ipv4/conf/eth1.22/force_igmp_version; にゼロ以外の値を指定することで、カーネルに IGMP の下位バージョンの使用を強制できるということです。ただし、ファイルの値がゼロの構成であることは確認しました。
私たちのアプリケーションは、次のコードを使用してマルチキャスト グループに参加しています。
... joinMulticast(in_addr mcast_addr, in_addr interface_addr)
{
struct ip_mreq mcast_req;
mcast_req.imr_multiaddr.s_addr = mcast_addr;
mcast_req.imr_interface.s_addr = interface_addr;
int err = setsockopt(fFileDesc, IPPROTO_IP, IP_ADD_MEMBERSHIP,
(char*)&theMulti, sizeof(theMulti));
// handle errors etc.
// ...
}
IGMPv3 を強制するためにソース プログラムに何か追加する必要があるのでしょうか?
解決
注意すべきことがいくつかあります。
1 つ目は、(私が理解しているところによると) /proc/sys/net/ipv4/conf/eth1.22/force_igmp_version
0 は「v3 を使用する」という意味ではなく、実際には「自動」に設定されます。igmp v3 を強制的に使用するには、これを 3 に設定できると思います。
ただし、もう 1 つ注意すべき点は、igmp スタックの動作はそれが置かれている環境によって決まるということです。Linux ボックスが上流の igmp ルーターから igmp v2 メンバーシップ クエリを受信している場合、デフォルトの Linux の動作 (igmp v3 rfc で義務付けられている) はレポートに igmp v2 のみを使用すると思います。
設定するとわかりますが、 /proc/sys/net/ipv4/conf/eth1.22/force_igmp_version
0 に設定すると、この動作が使用されます。