1#!/bin/bash 2# SPDX-License-Identifier: GPL-2.0 3 4ALL_TESTS="vlmc_control_test vlmc_querier_test vlmc_igmp_mld_version_test \ 5 vlmc_last_member_test vlmc_startup_query_test vlmc_membership_test \ 6 vlmc_querier_intvl_test vlmc_query_intvl_test vlmc_query_response_intvl_test \ 7 vlmc_router_port_test vlmc_filtering_test" 8NUM_NETIFS=4 9CHECK_TC="yes" 10TEST_GROUP="239.10.10.10" 11 12source lib.sh 13 14h1_create() 15{ 16 simple_if_init $h1 192.0.2.1/24 2001:db8:1::1/64 17 ip link add l $h1 $h1.10 up type vlan id 10 18} 19 20h1_destroy() 21{ 22 ip link del $h1.10 23 simple_if_fini $h1 192.0.2.1/24 2001:db8:1::1/64 24} 25 26h2_create() 27{ 28 simple_if_init $h2 192.0.2.2/24 2001:db8:1::2/64 29 ip link add l $h2 $h2.10 up type vlan id 10 30} 31 32h2_destroy() 33{ 34 ip link del $h2.10 35 simple_if_fini $h2 192.0.2.2/24 2001:db8:1::2/64 36} 37 38switch_create() 39{ 40 ip link add dev br0 type bridge mcast_snooping 1 mcast_querier 1 vlan_filtering 1 41 42 ip link set dev $swp1 master br0 43 ip link set dev $swp2 master br0 44 45 ip link set dev br0 up 46 ip link set dev $swp1 up 47 ip link set dev $swp2 up 48 49 tc qdisc add dev $swp1 clsact 50 tc qdisc add dev $swp2 clsact 51 52 bridge vlan add vid 10-11 dev $swp1 master 53 bridge vlan add vid 10-11 dev $swp2 master 54 55 ip link set dev br0 type bridge mcast_vlan_snooping 1 56 check_err $? "Could not enable global vlan multicast snooping" 57 log_test "Vlan multicast snooping enable" 58} 59 60switch_destroy() 61{ 62 tc qdisc del dev $swp2 clsact 63 tc qdisc del dev $swp1 clsact 64 65 ip link set dev $swp2 down 66 ip link set dev $swp1 down 67 68 ip link del dev br0 69} 70 71setup_prepare() 72{ 73 h1=${NETIFS[p1]} 74 swp1=${NETIFS[p2]} 75 76 swp2=${NETIFS[p3]} 77 h2=${NETIFS[p4]} 78 79 vrf_prepare 80 81 h1_create 82 h2_create 83 84 switch_create 85} 86 87cleanup() 88{ 89 pre_cleanup 90 91 switch_destroy 92 93 h2_destroy 94 h1_destroy 95 96 vrf_cleanup 97} 98 99vlmc_v2join_test() 100{ 101 local expect=$1 102 103 RET=0 104 ip address add dev $h2.10 $TEST_GROUP/32 autojoin 105 check_err $? "Could not join $TEST_GROUP" 106 107 sleep 5 108 bridge -j mdb show dev br0 | 109 jq -e ".[].mdb[] | select(.grp == \"$TEST_GROUP\" and .vid == 10)" &>/dev/null 110 if [ $expect -eq 0 ]; then 111 check_err $? "IGMPv2 report didn't create mdb entry for $TEST_GROUP" 112 else 113 check_fail $? "IGMPv2 report shouldn't have created mdb entry for $TEST_GROUP" 114 fi 115 116 # check if we need to cleanup 117 if [ $RET -eq 0 ]; then 118 ip address del dev $h2.10 $TEST_GROUP/32 2>&1 1>/dev/null 119 sleep 5 120 bridge -j mdb show dev br0 | 121 jq -e ".[].mdb[] | select(.grp == \"$TEST_GROUP\" and \ 122 .vid == 10)" &>/dev/null 123 check_fail $? "IGMPv2 leave didn't remove mdb entry for $TEST_GROUP" 124 fi 125} 126 127vlmc_control_test() 128{ 129 RET=0 130 local goutput=`bridge -j vlan global show` 131 echo -n $goutput | 132 jq -e ".[].vlans[] | select(.vlan == 10)" &>/dev/null 133 check_err $? "Could not find vlan 10's global options" 134 log_test "Vlan global options existence" 135 136 RET=0 137 echo -n $goutput | 138 jq -e ".[].vlans[] | select(.vlan == 10 and .mcast_snooping == 1) " &>/dev/null 139 check_err $? "Wrong default mcast_snooping global option value" 140 log_test "Vlan mcast_snooping global option default value" 141 142 RET=0 143 vlmc_v2join_test 0 144 bridge vlan global set vid 10 dev br0 mcast_snooping 0 145 check_err $? "Could not disable multicast snooping in vlan 10" 146 vlmc_v2join_test 1 147 log_test "Vlan 10 multicast snooping control" 148} 149 150# setup for general query counting 151vlmc_query_cnt_xstats() 152{ 153 local type=$1 154 local version=$2 155 local dev=$3 156 157 ip -j link xstats type bridge_slave dev $dev | \ 158 jq -e ".[].multicast.${type}_queries.tx_v${version}" 159} 160 161vlmc_query_cnt_setup() 162{ 163 local type=$1 164 local dev=$2 165 166 if [[ $type == "igmp" ]]; then 167 tc filter add dev $dev egress pref 10 prot 802.1Q \ 168 flower vlan_id 10 vlan_ethtype ipv4 dst_ip 224.0.0.1 ip_proto 2 \ 169 action pass 170 else 171 tc filter add dev $dev egress pref 10 prot 802.1Q \ 172 flower vlan_id 10 vlan_ethtype ipv6 dst_ip ff02::1 ip_proto icmpv6 \ 173 action pass 174 fi 175 176 ip link set dev br0 type bridge mcast_stats_enabled 1 177} 178 179vlmc_query_cnt_cleanup() 180{ 181 local dev=$1 182 183 ip link set dev br0 type bridge mcast_stats_enabled 0 184 tc filter del dev $dev egress pref 10 185} 186 187vlmc_check_query() 188{ 189 local type=$1 190 local version=$2 191 local dev=$3 192 local expect=$4 193 local time=$5 194 local ret=0 195 196 vlmc_query_cnt_setup $type $dev 197 198 local pre_tx_xstats=$(vlmc_query_cnt_xstats $type $version $dev) 199 bridge vlan global set vid 10 dev br0 mcast_snooping 1 mcast_querier 1 200 ret=$? 201 if [[ $ret -eq 0 ]]; then 202 sleep $time 203 204 local tcstats=$(tc_rule_stats_get $dev 10 egress) 205 local post_tx_xstats=$(vlmc_query_cnt_xstats $type $version $dev) 206 207 if [[ $tcstats != $expect || \ 208 $(($post_tx_xstats-$pre_tx_xstats)) != $expect || \ 209 $tcstats != $(($post_tx_xstats-$pre_tx_xstats)) ]]; then 210 ret=1 211 fi 212 fi 213 214 bridge vlan global set vid 10 dev br0 mcast_snooping 1 mcast_querier 0 215 vlmc_query_cnt_cleanup $dev 216 217 return $ret 218} 219 220vlmc_querier_test() 221{ 222 RET=0 223 local goutput=`bridge -j vlan global show` 224 echo -n $goutput | 225 jq -e ".[].vlans[] | select(.vlan == 10)" &>/dev/null 226 check_err $? "Could not find vlan 10's global options" 227 228 echo -n $goutput | 229 jq -e ".[].vlans[] | select(.vlan == 10 and .mcast_querier == 0) " &>/dev/null 230 check_err $? "Wrong default mcast_querier global vlan option value" 231 log_test "Vlan mcast_querier global option default value" 232 233 RET=0 234 bridge vlan global set vid 10 dev br0 mcast_snooping 1 mcast_querier 1 235 check_err $? "Could not enable querier in vlan 10" 236 log_test "Vlan 10 multicast querier enable" 237 bridge vlan global set vid 10 dev br0 mcast_snooping 1 mcast_querier 0 238 239 RET=0 240 vlmc_check_query igmp 2 $swp1 1 1 241 check_err $? "No vlan tagged IGMPv2 general query packets sent" 242 log_test "Vlan 10 tagged IGMPv2 general query sent" 243 244 RET=0 245 vlmc_check_query mld 1 $swp1 1 1 246 check_err $? "No vlan tagged MLD general query packets sent" 247 log_test "Vlan 10 tagged MLD general query sent" 248} 249 250vlmc_igmp_mld_version_test() 251{ 252 RET=0 253 local goutput=`bridge -j vlan global show` 254 echo -n $goutput | 255 jq -e ".[].vlans[] | select(.vlan == 10)" &>/dev/null 256 check_err $? "Could not find vlan 10's global options" 257 258 echo -n $goutput | 259 jq -e ".[].vlans[] | select(.vlan == 10 and .mcast_igmp_version == 2) " &>/dev/null 260 check_err $? "Wrong default mcast_igmp_version global vlan option value" 261 log_test "Vlan mcast_igmp_version global option default value" 262 263 RET=0 264 echo -n $goutput | 265 jq -e ".[].vlans[] | select(.vlan == 10 and .mcast_mld_version == 1) " &>/dev/null 266 check_err $? "Wrong default mcast_mld_version global vlan option value" 267 log_test "Vlan mcast_mld_version global option default value" 268 269 RET=0 270 bridge vlan global set vid 10 dev br0 mcast_snooping 1 mcast_igmp_version 3 271 check_err $? "Could not set mcast_igmp_version in vlan 10" 272 log_test "Vlan 10 mcast_igmp_version option changed to 3" 273 274 RET=0 275 vlmc_check_query igmp 3 $swp1 1 1 276 check_err $? "No vlan tagged IGMPv3 general query packets sent" 277 log_test "Vlan 10 tagged IGMPv3 general query sent" 278 279 RET=0 280 bridge vlan global set vid 10 dev br0 mcast_snooping 1 mcast_mld_version 2 281 check_err $? "Could not set mcast_mld_version in vlan 10" 282 log_test "Vlan 10 mcast_mld_version option changed to 2" 283 284 RET=0 285 vlmc_check_query mld 2 $swp1 1 1 286 check_err $? "No vlan tagged MLDv2 general query packets sent" 287 log_test "Vlan 10 tagged MLDv2 general query sent" 288 289 bridge vlan global set vid 10 dev br0 mcast_snooping 1 mcast_igmp_version 2 290 bridge vlan global set vid 10 dev br0 mcast_snooping 1 mcast_mld_version 1 291} 292 293vlmc_last_member_test() 294{ 295 RET=0 296 local goutput=`bridge -j vlan global show` 297 echo -n $goutput | 298 jq -e ".[].vlans[] | select(.vlan == 10)" &>/dev/null 299 check_err $? "Could not find vlan 10's global options" 300 301 echo -n $goutput | 302 jq -e ".[].vlans[] | select(.vlan == 10 and \ 303 .mcast_last_member_count == 2) " &>/dev/null 304 check_err $? "Wrong default mcast_last_member_count global vlan option value" 305 log_test "Vlan mcast_last_member_count global option default value" 306 307 RET=0 308 echo -n $goutput | 309 jq -e ".[].vlans[] | select(.vlan == 10 and \ 310 .mcast_last_member_interval == 100) " &>/dev/null 311 check_err $? "Wrong default mcast_last_member_interval global vlan option value" 312 log_test "Vlan mcast_last_member_interval global option default value" 313 314 RET=0 315 bridge vlan global set vid 10 dev br0 mcast_snooping 1 mcast_last_member_count 3 316 check_err $? "Could not set mcast_last_member_count in vlan 10" 317 log_test "Vlan 10 mcast_last_member_count option changed to 3" 318 bridge vlan global set vid 10 dev br0 mcast_snooping 1 mcast_last_member_count 2 319 320 RET=0 321 bridge vlan global set vid 10 dev br0 mcast_snooping 1 mcast_last_member_interval 200 322 check_err $? "Could not set mcast_last_member_interval in vlan 10" 323 log_test "Vlan 10 mcast_last_member_interval option changed to 200" 324 bridge vlan global set vid 10 dev br0 mcast_snooping 1 mcast_last_member_interval 100 325} 326 327vlmc_startup_query_test() 328{ 329 RET=0 330 local goutput=`bridge -j vlan global show` 331 echo -n $goutput | 332 jq -e ".[].vlans[] | select(.vlan == 10)" &>/dev/null 333 check_err $? "Could not find vlan 10's global options" 334 335 echo -n $goutput | 336 jq -e ".[].vlans[] | select(.vlan == 10 and \ 337 .mcast_startup_query_interval == 3125) " &>/dev/null 338 check_err $? "Wrong default mcast_startup_query_interval global vlan option value" 339 log_test "Vlan mcast_startup_query_interval global option default value" 340 341 RET=0 342 echo -n $goutput | 343 jq -e ".[].vlans[] | select(.vlan == 10 and \ 344 .mcast_startup_query_count == 2) " &>/dev/null 345 check_err $? "Wrong default mcast_startup_query_count global vlan option value" 346 log_test "Vlan mcast_startup_query_count global option default value" 347 348 RET=0 349 bridge vlan global set vid 10 dev br0 mcast_snooping 1 mcast_startup_query_interval 100 350 check_err $? "Could not set mcast_startup_query_interval in vlan 10" 351 vlmc_check_query igmp 2 $swp1 2 3 352 check_err $? "Wrong number of tagged IGMPv2 general queries sent" 353 log_test "Vlan 10 mcast_startup_query_interval option changed to 100" 354 355 RET=0 356 bridge vlan global set vid 10 dev br0 mcast_snooping 1 mcast_startup_query_count 3 357 check_err $? "Could not set mcast_startup_query_count in vlan 10" 358 vlmc_check_query igmp 2 $swp1 3 4 359 check_err $? "Wrong number of tagged IGMPv2 general queries sent" 360 log_test "Vlan 10 mcast_startup_query_count option changed to 3" 361 362 bridge vlan global set vid 10 dev br0 mcast_snooping 1 mcast_startup_query_interval 3125 363 bridge vlan global set vid 10 dev br0 mcast_snooping 1 mcast_startup_query_count 2 364} 365 366vlmc_membership_test() 367{ 368 RET=0 369 local goutput=`bridge -j vlan global show` 370 echo -n $goutput | 371 jq -e ".[].vlans[] | select(.vlan == 10)" &>/dev/null 372 check_err $? "Could not find vlan 10's global options" 373 374 echo -n $goutput | 375 jq -e ".[].vlans[] | select(.vlan == 10 and \ 376 .mcast_membership_interval == 26000) " &>/dev/null 377 check_err $? "Wrong default mcast_membership_interval global vlan option value" 378 log_test "Vlan mcast_membership_interval global option default value" 379 380 RET=0 381 bridge vlan global set vid 10 dev br0 mcast_snooping 1 mcast_membership_interval 200 382 check_err $? "Could not set mcast_membership_interval in vlan 10" 383 log_test "Vlan 10 mcast_membership_interval option changed to 200" 384 385 RET=0 386 vlmc_v2join_test 1 387 log_test "Vlan 10 mcast_membership_interval mdb entry expire" 388 389 bridge vlan global set vid 10 dev br0 mcast_snooping 1 mcast_membership_interval 26000 390} 391 392vlmc_querier_intvl_test() 393{ 394 RET=0 395 local goutput=`bridge -j vlan global show` 396 echo -n $goutput | 397 jq -e ".[].vlans[] | select(.vlan == 10)" &>/dev/null 398 check_err $? "Could not find vlan 10's global options" 399 400 echo -n $goutput | 401 jq -e ".[].vlans[] | select(.vlan == 10 and \ 402 .mcast_querier_interval == 25500) " &>/dev/null 403 check_err $? "Wrong default mcast_querier_interval global vlan option value" 404 log_test "Vlan mcast_querier_interval global option default value" 405 406 RET=0 407 bridge vlan global set vid 10 dev br0 mcast_snooping 1 mcast_querier_interval 100 408 check_err $? "Could not set mcast_querier_interval in vlan 10" 409 log_test "Vlan 10 mcast_querier_interval option changed to 100" 410 411 RET=0 412 ip link add dev br1 type bridge mcast_snooping 1 mcast_querier 1 vlan_filtering 1 \ 413 mcast_vlan_snooping 1 414 bridge vlan add vid 10 dev br1 self pvid untagged 415 ip link set dev $h1 master br1 416 ip link set dev br1 up 417 bridge vlan add vid 10 dev $h1 master 418 bridge vlan global set vid 10 dev br1 mcast_snooping 1 mcast_querier 1 419 sleep 2 420 ip link del dev br1 421 ip addr replace 2001:db8:1::1/64 dev $h1 422 vlmc_check_query igmp 2 $swp1 1 1 423 check_err $? "Wrong number of IGMPv2 general queries after querier interval" 424 log_test "Vlan 10 mcast_querier_interval expire after outside query" 425 426 bridge vlan global set vid 10 dev br0 mcast_snooping 1 mcast_querier_interval 25500 427} 428 429vlmc_query_intvl_test() 430{ 431 RET=0 432 local goutput=`bridge -j vlan global show` 433 echo -n $goutput | 434 jq -e ".[].vlans[] | select(.vlan == 10)" &>/dev/null 435 check_err $? "Could not find vlan 10's global options" 436 437 echo -n $goutput | 438 jq -e ".[].vlans[] | select(.vlan == 10 and \ 439 .mcast_query_interval == 12500) " &>/dev/null 440 check_err $? "Wrong default mcast_query_interval global vlan option value" 441 log_test "Vlan mcast_query_interval global option default value" 442 443 RET=0 444 bridge vlan global set vid 10 dev br0 mcast_snooping 1 mcast_startup_query_count 0 445 bridge vlan global set vid 10 dev br0 mcast_snooping 1 mcast_query_interval 200 446 check_err $? "Could not set mcast_query_interval in vlan 10" 447 # 1 is sent immediately, then 2 more in the next 5 seconds 448 vlmc_check_query igmp 2 $swp1 3 5 449 check_err $? "Wrong number of tagged IGMPv2 general queries sent" 450 log_test "Vlan 10 mcast_query_interval option changed to 200" 451 452 bridge vlan global set vid 10 dev br0 mcast_snooping 1 mcast_startup_query_count 2 453 bridge vlan global set vid 10 dev br0 mcast_snooping 1 mcast_query_interval 12500 454} 455 456vlmc_query_response_intvl_test() 457{ 458 RET=0 459 local goutput=`bridge -j vlan global show` 460 echo -n $goutput | 461 jq -e ".[].vlans[] | select(.vlan == 10)" &>/dev/null 462 check_err $? "Could not find vlan 10's global options" 463 464 echo -n $goutput | 465 jq -e ".[].vlans[] | select(.vlan == 10 and \ 466 .mcast_query_response_interval == 1000) " &>/dev/null 467 check_err $? "Wrong default mcast_query_response_interval global vlan option value" 468 log_test "Vlan mcast_query_response_interval global option default value" 469 470 RET=0 471 bridge vlan global set vid 10 dev br0 mcast_snooping 1 mcast_query_response_interval 200 472 check_err $? "Could not set mcast_query_response_interval in vlan 10" 473 log_test "Vlan 10 mcast_query_response_interval option changed to 200" 474 475 bridge vlan global set vid 10 dev br0 mcast_snooping 1 mcast_query_response_interval 1000 476} 477 478vlmc_router_port_test() 479{ 480 RET=0 481 local goutput=`bridge -j -d vlan show` 482 echo -n $goutput | 483 jq -e ".[] | select(.ifname == \"$swp1\" and \ 484 .vlans[].vlan == 10)" &>/dev/null 485 check_err $? "Could not find port vlan 10's options" 486 487 echo -n $goutput | 488 jq -e ".[] | select(.ifname == \"$swp1\" and \ 489 .vlans[].vlan == 10 and \ 490 .vlans[].mcast_router == 1)" &>/dev/null 491 check_err $? "Wrong default port mcast_router option value" 492 log_test "Port vlan 10 option mcast_router default value" 493 494 RET=0 495 bridge vlan set vid 10 dev $swp1 mcast_router 2 496 check_err $? "Could not set port vlan 10's mcast_router option" 497 log_test "Port vlan 10 mcast_router option changed to 2" 498 499 RET=0 500 tc filter add dev $swp1 egress pref 10 prot 802.1Q \ 501 flower vlan_id 10 vlan_ethtype ipv4 dst_ip 239.1.1.1 ip_proto udp action pass 502 tc filter add dev $swp2 egress pref 10 prot 802.1Q \ 503 flower vlan_id 10 vlan_ethtype ipv4 dst_ip 239.1.1.1 ip_proto udp action pass 504 bridge vlan set vid 10 dev $swp2 mcast_router 0 505 # we need to enable querier and disable query response interval to 506 # make sure packets are flooded only to router ports 507 bridge vlan global set vid 10 dev br0 mcast_snooping 1 mcast_querier 1 \ 508 mcast_query_response_interval 0 509 bridge vlan add vid 10 dev br0 self 510 sleep 1 511 mausezahn br0 -Q 10 -c 10 -p 128 -b 01:00:5e:01:01:01 -B 239.1.1.1 \ 512 -t udp "dp=1024" &>/dev/null 513 local swp1_tcstats=$(tc_rule_stats_get $swp1 10 egress) 514 if [[ $swp1_tcstats != 10 ]]; then 515 check_err 1 "Wrong number of vlan 10 multicast packets flooded" 516 fi 517 local swp2_tcstats=$(tc_rule_stats_get $swp2 10 egress) 518 check_err $swp2_tcstats "Vlan 10 multicast packets flooded to non-router port" 519 log_test "Flood unknown vlan multicast packets to router port only" 520 521 tc filter del dev $swp2 egress pref 10 522 tc filter del dev $swp1 egress pref 10 523 bridge vlan del vid 10 dev br0 self 524 bridge vlan global set vid 10 dev br0 mcast_snooping 1 mcast_query_response_interval 1000 525 bridge vlan set vid 10 dev $swp2 mcast_router 1 526 bridge vlan set vid 10 dev $swp1 mcast_router 1 527} 528 529vlmc_filtering_test() 530{ 531 RET=0 532 ip link set dev br0 type bridge vlan_filtering 0 533 ip -j -d link show dev br0 | \ 534 jq -e "select(.[0].linkinfo.info_data.mcast_vlan_snooping == 1)" &>/dev/null 535 check_fail $? "Vlan filtering is disabled but multicast vlan snooping is still enabled" 536 log_test "Disable multicast vlan snooping when vlan filtering is disabled" 537} 538 539trap cleanup EXIT 540 541setup_prepare 542setup_wait 543 544tests_run 545 546exit $EXIT_STATUS 547