# $NetBSD: t_bridge.sh,v 1.16 2016/11/25 08:51:16 ozaki-r Exp $ # # Copyright (c) 2014 The NetBSD Foundation, Inc. # All rights reserved. # # Redistribution and use in source and binary forms, with or without # modification, are permitted provided that the following conditions # are met: # 1. Redistributions of source code must retain the above copyright # notice, this list of conditions and the following disclaimer. # 2. Redistributions in binary form must reproduce the above copyright # notice, this list of conditions and the following disclaimer in the # documentation and/or other materials provided with the distribution. # # THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND CONTRIBUTORS # ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED # TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR # PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS # BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR # CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF # SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS # INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN # CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) # ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE # POSSIBILITY OF SUCH DAMAGE. # setglobal SOCK1 = 'unix://commsock1' setglobal SOCK2 = 'unix://commsock2' setglobal SOCK3 = 'unix://commsock3' setglobal IP1 = '10.0.0.1' setglobal IP2 = '10.0.0.2' setglobal IP61 = 'fc00::1' setglobal IP62 = 'fc00::2' setglobal IPBR1 = '10.0.0.11' setglobal IPBR2 = '10.0.0.12' setglobal IP6BR1 = 'fc00::11' setglobal IP6BR2 = 'fc00::12' setglobal DEBUG = $(DEBUG:-false) setglobal TIMEOUT = '5' atf_test_case bridge_ipv4 cleanup atf_test_case bridge_ipv6 cleanup atf_test_case bridge_rtable cleanup atf_test_case bridge_member_ipv4 cleanup atf_test_case bridge_member_ipv6 cleanup proc bridge_ipv4_head { atf_set "descr" "Does simple if_bridge tests" atf_set "require.progs" "rump_server" } proc bridge_ipv6_head { atf_set "descr" "Does simple if_bridge tests (IPv6)" atf_set "require.progs" "rump_server" } proc bridge_rtable_head { atf_set "descr" "Tests route table operations of if_bridge" atf_set "require.progs" "rump_server" } proc bridge_member_ipv4_head { atf_set "descr" "Tests if_bridge with members with an IP address" atf_set "require.progs" "rump_server" } proc bridge_member_ipv6_head { atf_set "descr" "Tests if_bridge with members with an IP address (IPv6)" atf_set "require.progs" "rump_server" } proc setup_endpoint { setglobal sock = $(1) setglobal addr = $(2) setglobal bus = $(3) setglobal mode = $(4) rump_server_add_iface $sock shmif0 $bus export RUMP_SERVER=$(sock) if test $mode = "ipv6" { atf_check -s exit:0 rump.ifconfig shmif0 inet6 $(addr) } else { atf_check -s exit:0 rump.ifconfig shmif0 inet $(addr) netmask 0xffffff00 } atf_check -s exit:0 rump.ifconfig shmif0 up $DEBUG && rump.ifconfig shmif0 } proc test_endpoint { setglobal sock = $(1) setglobal addr = $(2) setglobal bus = $(3) setglobal mode = $(4) export RUMP_SERVER=$(sock) atf_check -s exit:0 -o match:shmif0 rump.ifconfig if test $mode = "ipv6" { atf_check -s exit:0 -o ignore rump.ping6 -n -c 1 -X $TIMEOUT $(addr) } else { atf_check -s exit:0 -o ignore rump.ping -n -w $TIMEOUT -c 1 $(addr) } } proc test_setup { test_endpoint $SOCK1 $IP1 bus1 ipv4 test_endpoint $SOCK3 $IP2 bus2 ipv4 export RUMP_SERVER=$SOCK2 atf_check -s exit:0 -o match:shmif0 rump.ifconfig atf_check -s exit:0 -o match:shmif1 rump.ifconfig } proc test_setup6 { test_endpoint $SOCK1 $IP61 bus1 ipv6 test_endpoint $SOCK3 $IP62 bus2 ipv6 export RUMP_SERVER=$SOCK2 atf_check -s exit:0 -o match:shmif0 rump.ifconfig atf_check -s exit:0 -o match:shmif1 rump.ifconfig } proc setup_bridge_server { rump_server_add_iface $SOCK2 shmif0 bus1 rump_server_add_iface $SOCK2 shmif1 bus2 export RUMP_SERVER=$SOCK2 atf_check -s exit:0 rump.ifconfig shmif0 up atf_check -s exit:0 rump.ifconfig shmif1 up } proc setup { rump_server_start $SOCK1 bridge rump_server_start $SOCK2 bridge rump_server_start $SOCK3 bridge setup_endpoint $SOCK1 $IP1 bus1 ipv4 setup_endpoint $SOCK3 $IP2 bus2 ipv4 setup_bridge_server } proc setup6 { rump_server_start $SOCK1 netinet6 bridge rump_server_start $SOCK2 netinet6 bridge rump_server_start $SOCK3 netinet6 bridge setup_endpoint $SOCK1 $IP61 bus1 ipv6 setup_endpoint $SOCK3 $IP62 bus2 ipv6 setup_bridge_server } proc setup_bridge { export RUMP_SERVER=$SOCK2 atf_check -s exit:0 rump.ifconfig bridge0 create atf_check -s exit:0 rump.ifconfig bridge0 up export LD_PRELOAD=/usr/lib/librumphijack.so atf_check -s exit:0 /sbin/brconfig bridge0 add shmif0 atf_check -s exit:0 /sbin/brconfig bridge0 add shmif1 /sbin/brconfig bridge0 unset LD_PRELOAD rump.ifconfig shmif0 rump.ifconfig shmif1 } proc setup_member_ip { export RUMP_SERVER=$SOCK2 export LD_PRELOAD=/usr/lib/librumphijack.so atf_check -s exit:0 rump.ifconfig shmif0 $IPBR1/24 atf_check -s exit:0 rump.ifconfig shmif1 $IPBR2/24 atf_check -s exit:0 rump.ifconfig -w 10 /sbin/brconfig bridge0 unset LD_PRELOAD rump.ifconfig shmif0 rump.ifconfig shmif1 } proc setup_member_ip6 { export RUMP_SERVER=$SOCK2 export LD_PRELOAD=/usr/lib/librumphijack.so atf_check -s exit:0 rump.ifconfig shmif0 inet6 $IP6BR1 atf_check -s exit:0 rump.ifconfig shmif1 inet6 $IP6BR2 atf_check -s exit:0 rump.ifconfig -w 10 /sbin/brconfig bridge0 unset LD_PRELOAD rump.ifconfig shmif0 rump.ifconfig shmif1 } proc teardown_bridge { export RUMP_SERVER=$SOCK2 export LD_PRELOAD=/usr/lib/librumphijack.so /sbin/brconfig bridge0 atf_check -s exit:0 /sbin/brconfig bridge0 delete shmif0 atf_check -s exit:0 /sbin/brconfig bridge0 delete shmif1 /sbin/brconfig bridge0 unset LD_PRELOAD rump.ifconfig shmif0 rump.ifconfig shmif1 } proc test_setup_bridge { export RUMP_SERVER=$SOCK2 export LD_PRELOAD=/usr/lib/librumphijack.so atf_check -s exit:0 -o match:shmif0 /sbin/brconfig bridge0 atf_check -s exit:0 -o match:shmif1 /sbin/brconfig bridge0 /sbin/brconfig bridge0 unset LD_PRELOAD } proc down_up_interfaces { export RUMP_SERVER=$SOCK1 rump.ifconfig shmif0 down rump.ifconfig shmif0 up export RUMP_SERVER=$SOCK3 rump.ifconfig shmif0 down rump.ifconfig shmif0 up } proc test_ping_failure { export RUMP_SERVER=$SOCK1 atf_check -s not-exit:0 -o ignore rump.ping -q -n -w $TIMEOUT -c 1 $IP2 export RUMP_SERVER=$SOCK3 atf_check -s not-exit:0 -o ignore rump.ping -q -n -w $TIMEOUT -c 1 $IP1 } proc test_ping_success { export RUMP_SERVER=$SOCK1 rump.ifconfig -v shmif0 atf_check -s exit:0 -o ignore rump.ping -q -n -w $TIMEOUT -c 1 $IP2 rump.ifconfig -v shmif0 export RUMP_SERVER=$SOCK3 rump.ifconfig -v shmif0 atf_check -s exit:0 -o ignore rump.ping -q -n -w $TIMEOUT -c 1 $IP1 rump.ifconfig -v shmif0 } proc test_ping6_failure { export RUMP_SERVER=$SOCK1 atf_check -s not-exit:0 -o ignore rump.ping6 -q -n -c 1 -X $TIMEOUT $IP62 export RUMP_SERVER=$SOCK3 atf_check -s not-exit:0 -o ignore rump.ping6 -q -n -c 1 -X $TIMEOUT $IP61 } proc test_ping6_success { export RUMP_SERVER=$SOCK1 rump.ifconfig -v shmif0 atf_check -s exit:0 -o ignore rump.ping6 -q -n -c 1 -X $TIMEOUT $IP62 rump.ifconfig -v shmif0 export RUMP_SERVER=$SOCK3 rump.ifconfig -v shmif0 atf_check -s exit:0 -o ignore rump.ping6 -q -n -c 1 -X $TIMEOUT $IP61 rump.ifconfig -v shmif0 } proc test_ping_member { export RUMP_SERVER=$SOCK1 rump.ifconfig -v shmif0 atf_check -s exit:0 -o ignore rump.ping -q -n -w $TIMEOUT -c 1 $IPBR1 rump.ifconfig -v shmif0 # Test for PR#48104 atf_check -s exit:0 -o ignore rump.ping -q -n -w $TIMEOUT -c 1 $IPBR2 rump.ifconfig -v shmif0 export RUMP_SERVER=$SOCK3 rump.ifconfig -v shmif0 # Test for PR#48104 atf_check -s exit:0 -o ignore rump.ping -q -n -w $TIMEOUT -c 1 $IPBR1 rump.ifconfig -v shmif0 atf_check -s exit:0 -o ignore rump.ping -q -n -w $TIMEOUT -c 1 $IPBR2 rump.ifconfig -v shmif0 } proc test_ping6_member { export RUMP_SERVER=$SOCK1 rump.ifconfig -v shmif0 atf_check -s exit:0 -o ignore rump.ping6 -q -n -X $TIMEOUT -c 1 $IP6BR1 rump.ifconfig -v shmif0 # Test for PR#48104 atf_check -s exit:0 -o ignore rump.ping6 -q -n -X $TIMEOUT -c 1 $IP6BR2 rump.ifconfig -v shmif0 export RUMP_SERVER=$SOCK3 rump.ifconfig -v shmif0 # Test for PR#48104 atf_check -s exit:0 -o ignore rump.ping6 -q -n -X $TIMEOUT -c 1 $IP6BR1 rump.ifconfig -v shmif0 atf_check -s exit:0 -o ignore rump.ping6 -q -n -X $TIMEOUT -c 1 $IP6BR2 rump.ifconfig -v shmif0 } proc get_number_of_caches { export RUMP_SERVER=$SOCK2 export LD_PRELOAD=/usr/lib/librumphijack.so echo $shExpr('$(/sbin/brconfig bridge0 |grep -A 100 "Address cache" |wc -l) - 1') unset LD_PRELOAD } proc test_brconfig_maxaddr { setglobal addr1 = '', addr3 = '', n = '' # Get MAC addresses of the endpoints. setglobal addr1 = $[get_macaddr $SOCK1 shmif0] setglobal addr3 = $[get_macaddr $SOCK3 shmif0] # Refill the MAC addresses of the endpoints. export RUMP_SERVER=$SOCK1 atf_check -s exit:0 -o ignore rump.ping -n -w $TIMEOUT -c 1 $IP2 export RUMP_SERVER=$SOCK2 export LD_PRELOAD=/usr/lib/librumphijack.so /sbin/brconfig bridge0 atf_check -s exit:0 -o match:"$addr1 shmif0" /sbin/brconfig bridge0 atf_check -s exit:0 -o match:"$addr3 shmif1" /sbin/brconfig bridge0 # Check the default # of caches is 100 atf_check -s exit:0 -o match:"max cache: 100" /sbin/brconfig bridge0 # Test two MAC addresses are cached setglobal n = $[get_number_of_caches] atf_check_equal $n 2 # Limit # of caches to one atf_check -s exit:0 -o ignore /sbin/brconfig bridge0 maxaddr 1 atf_check -s exit:0 -o match:"max cache: 1" /sbin/brconfig bridge0 /sbin/brconfig bridge0 # Test just one address is cached setglobal n = $[get_number_of_caches] atf_check_equal $n 1 # Increase # of caches to two atf_check -s exit:0 -o ignore /sbin/brconfig bridge0 maxaddr 2 atf_check -s exit:0 -o match:"max cache: 2" /sbin/brconfig bridge0 unset LD_PRELOAD # Test we can cache two addresses again export RUMP_SERVER=$SOCK1 atf_check -s exit:0 -o ignore rump.ping -n -w $TIMEOUT -c 1 $IP2 export RUMP_SERVER=$SOCK2 export LD_PRELOAD=/usr/lib/librumphijack.so /sbin/brconfig bridge0 atf_check -s exit:0 -o match:"$addr1 shmif0" /sbin/brconfig bridge0 atf_check -s exit:0 -o match:"$addr3 shmif1" /sbin/brconfig bridge0 unset LD_PRELOAD } proc bridge_ipv4_body { setup test_setup # Enable once PR kern/49219 is fixed #test_ping_failure setup_bridge sleep 1 test_setup_bridge test_ping_success teardown_bridge test_ping_failure rump_server_destroy_ifaces } proc bridge_ipv6_body { setup6 test_setup6 test_ping6_failure setup_bridge sleep 1 test_setup_bridge test_ping6_success teardown_bridge test_ping6_failure rump_server_destroy_ifaces } proc bridge_rtable_body { setglobal addr1 = '', addr3 = '' setup setup_bridge # Get MAC addresses of the endpoints. setglobal addr1 = $[get_macaddr $SOCK1 shmif0] setglobal addr3 = $[get_macaddr $SOCK3 shmif0] # Confirm there is no MAC address caches. export RUMP_SERVER=$SOCK2 export LD_PRELOAD=/usr/lib/librumphijack.so $DEBUG && /sbin/brconfig bridge0 atf_check -s exit:0 -o not-match:"$addr1" /sbin/brconfig bridge0 atf_check -s exit:0 -o not-match:"$addr3" /sbin/brconfig bridge0 unset LD_PRELOAD # Make the bridge learn the MAC addresses of the endpoints. export RUMP_SERVER=$SOCK1 atf_check -s exit:0 -o ignore rump.ping -n -w $TIMEOUT -c 1 $IP2 unset RUMP_SERVER # Tests the addresses are in the cache. export RUMP_SERVER=$SOCK2 export LD_PRELOAD=/usr/lib/librumphijack.so $DEBUG && /sbin/brconfig bridge0 atf_check -s exit:0 -o match:"$addr1 shmif0" /sbin/brconfig bridge0 atf_check -s exit:0 -o match:"$addr3 shmif1" /sbin/brconfig bridge0 # Tests brconfig deladdr atf_check -s exit:0 -o ignore /sbin/brconfig bridge0 deladdr $addr1 atf_check -s exit:0 -o not-match:"$addr1 shmif0" /sbin/brconfig bridge0 atf_check -s exit:0 -o ignore /sbin/brconfig bridge0 deladdr $addr3 atf_check -s exit:0 -o not-match:"$addr3 shmif1" /sbin/brconfig bridge0 unset LD_PRELOAD # Refill the MAC addresses of the endpoints. export RUMP_SERVER=$SOCK1 atf_check -s exit:0 -o ignore rump.ping -n -w $TIMEOUT -c 1 $IP2 unset RUMP_SERVER export RUMP_SERVER=$SOCK2 export LD_PRELOAD=/usr/lib/librumphijack.so $DEBUG && /sbin/brconfig bridge0 atf_check -s exit:0 -o match:"$addr1 shmif0" /sbin/brconfig bridge0 atf_check -s exit:0 -o match:"$addr3 shmif1" /sbin/brconfig bridge0 # Tests brconfig flush. atf_check -s exit:0 -o ignore /sbin/brconfig bridge0 flush atf_check -s exit:0 -o not-match:"$addr1 shmif0" /sbin/brconfig bridge0 atf_check -s exit:0 -o not-match:"$addr3 shmif1" /sbin/brconfig bridge0 unset LD_PRELOAD # Tests brconfig timeout. export RUMP_SERVER=$SOCK2 export LD_PRELOAD=/usr/lib/librumphijack.so atf_check -s exit:0 -o match:"timeout: 1200" /sbin/brconfig bridge0 atf_check -s exit:0 -o ignore /sbin/brconfig bridge0 timeout 10 atf_check -s exit:0 -o match:"timeout: 10" /sbin/brconfig bridge0 unset LD_PRELOAD # Tests brconfig maxaddr. test_brconfig_maxaddr # TODO: brconfig static/flushall/discover/learn # TODO: cache expiration; it takes 5 minutes at least and we want to # wait here so long. Should we have a sysctl to change the period? rump_server_destroy_ifaces } proc bridge_member_ipv4_body { setup test_setup # Enable once PR kern/49219 is fixed #test_ping_failure setup_bridge sleep 1 test_setup_bridge test_ping_success setup_member_ip test_ping_member teardown_bridge test_ping_failure rump_server_destroy_ifaces } proc bridge_member_ipv6_body { setup6 test_setup6 test_ping6_failure setup_bridge sleep 1 test_setup_bridge test_ping6_success setup_member_ip6 test_ping6_member teardown_bridge test_ping6_failure rump_server_destroy_ifaces } proc bridge_ipv4_cleanup { $DEBUG && dump cleanup } proc bridge_ipv6_cleanup { $DEBUG && dump cleanup } proc bridge_rtable_cleanup { $DEBUG && dump cleanup } proc bridge_member_ipv4_cleanup { $DEBUG && dump cleanup } proc bridge_member_ipv6_cleanup { $DEBUG && dump cleanup } proc atf_init_test_cases { atf_add_test_case bridge_ipv4 atf_add_test_case bridge_ipv6 atf_add_test_case bridge_rtable atf_add_test_case bridge_member_ipv4 atf_add_test_case bridge_member_ipv6 }