| #!/bin/bash |
| # SPDX-License-Identifier: GPL-2.0 |
| |
| lib_dir=$(dirname $0)/../../../net/forwarding |
| |
| ALL_TESTS=" |
| ipv4_route_addition_test |
| ipv4_route_deletion_test |
| ipv4_route_replacement_test |
| ipv4_route_offload_failed_test |
| ipv6_route_addition_test |
| ipv6_route_deletion_test |
| ipv6_route_replacement_test |
| ipv6_route_offload_failed_test |
| " |
| |
| NETDEVSIM_PATH=/sys/bus/netdevsim/ |
| DEV_ADDR=1337 |
| DEV=netdevsim${DEV_ADDR} |
| DEVLINK_DEV=netdevsim/${DEV} |
| SYSFS_NET_DIR=/sys/bus/netdevsim/devices/$DEV/net/ |
| DEBUGFS_DIR=/sys/kernel/debug/netdevsim/$DEV/ |
| NUM_NETIFS=0 |
| source $lib_dir/lib.sh |
| |
| check_rt_offload_failed() |
| { |
| local outfile=$1; shift |
| local line |
| |
| # Make sure that the first notification was emitted without |
| # RTM_F_OFFLOAD_FAILED flag and the second with RTM_F_OFFLOAD_FAILED |
| # flag |
| head -n 1 $outfile | grep -q "rt_offload_failed" |
| if [[ $? -eq 0 ]]; then |
| return 1 |
| fi |
| |
| head -n 2 $outfile | tail -n 1 | grep -q "rt_offload_failed" |
| } |
| |
| check_rt_trap() |
| { |
| local outfile=$1; shift |
| local line |
| |
| # Make sure that the first notification was emitted without RTM_F_TRAP |
| # flag and the second with RTM_F_TRAP flag |
| head -n 1 $outfile | grep -q "rt_trap" |
| if [[ $? -eq 0 ]]; then |
| return 1 |
| fi |
| |
| head -n 2 $outfile | tail -n 1 | grep -q "rt_trap" |
| } |
| |
| route_notify_check() |
| { |
| local outfile=$1; shift |
| local expected_num_lines=$1; shift |
| local offload_failed=${1:-0}; shift |
| |
| # check the monitor results |
| lines=`wc -l $outfile | cut "-d " -f1` |
| test $lines -eq $expected_num_lines |
| check_err $? "$expected_num_lines notifications were expected but $lines were received" |
| |
| if [[ $expected_num_lines -eq 1 ]]; then |
| return |
| fi |
| |
| if [[ $offload_failed -eq 0 ]]; then |
| check_rt_trap $outfile |
| check_err $? "Wrong RTM_F_TRAP flags in notifications" |
| else |
| check_rt_offload_failed $outfile |
| check_err $? "Wrong RTM_F_OFFLOAD_FAILED flags in notifications" |
| fi |
| } |
| |
| route_addition_check() |
| { |
| local ip=$1; shift |
| local notify=$1; shift |
| local route=$1; shift |
| local expected_num_notifications=$1; shift |
| local offload_failed=${1:-0}; shift |
| |
| ip netns exec testns1 sysctl -qw net.$ip.fib_notify_on_flag_change=$notify |
| |
| local outfile=$(mktemp) |
| |
| $IP monitor route &> $outfile & |
| sleep 1 |
| $IP route add $route dev dummy1 |
| sleep 1 |
| kill %% && wait %% &> /dev/null |
| |
| route_notify_check $outfile $expected_num_notifications $offload_failed |
| rm -f $outfile |
| |
| $IP route del $route dev dummy1 |
| } |
| |
| ipv4_route_addition_test() |
| { |
| RET=0 |
| |
| local ip="ipv4" |
| local route=192.0.2.0/24 |
| |
| # Make sure a single notification will be emitted for the programmed |
| # route. |
| local notify=0 |
| local expected_num_notifications=1 |
| # route_addition_check will assign value to RET. |
| route_addition_check $ip $notify $route $expected_num_notifications |
| |
| # Make sure two notifications will be emitted for the programmed route. |
| notify=1 |
| expected_num_notifications=2 |
| route_addition_check $ip $notify $route $expected_num_notifications |
| |
| # notify=2 means emit notifications only for failed route installation, |
| # make sure a single notification will be emitted for the programmed |
| # route. |
| notify=2 |
| expected_num_notifications=1 |
| route_addition_check $ip $notify $route $expected_num_notifications |
| |
| log_test "IPv4 route addition" |
| } |
| |
| route_deletion_check() |
| { |
| local ip=$1; shift |
| local notify=$1; shift |
| local route=$1; shift |
| local expected_num_notifications=$1; shift |
| |
| ip netns exec testns1 sysctl -qw net.$ip.fib_notify_on_flag_change=$notify |
| $IP route add $route dev dummy1 |
| sleep 1 |
| |
| local outfile=$(mktemp) |
| |
| $IP monitor route &> $outfile & |
| sleep 1 |
| $IP route del $route dev dummy1 |
| sleep 1 |
| kill %% && wait %% &> /dev/null |
| |
| route_notify_check $outfile $expected_num_notifications |
| rm -f $outfile |
| } |
| |
| ipv4_route_deletion_test() |
| { |
| RET=0 |
| |
| local ip="ipv4" |
| local route=192.0.2.0/24 |
| local expected_num_notifications=1 |
| |
| # Make sure a single notification will be emitted for the deleted route, |
| # regardless of fib_notify_on_flag_change value. |
| local notify=0 |
| # route_deletion_check will assign value to RET. |
| route_deletion_check $ip $notify $route $expected_num_notifications |
| |
| notify=1 |
| route_deletion_check $ip $notify $route $expected_num_notifications |
| |
| log_test "IPv4 route deletion" |
| } |
| |
| route_replacement_check() |
| { |
| local ip=$1; shift |
| local notify=$1; shift |
| local route=$1; shift |
| local expected_num_notifications=$1; shift |
| |
| ip netns exec testns1 sysctl -qw net.$ip.fib_notify_on_flag_change=$notify |
| $IP route add $route dev dummy1 |
| sleep 1 |
| |
| local outfile=$(mktemp) |
| |
| $IP monitor route &> $outfile & |
| sleep 1 |
| $IP route replace $route dev dummy2 |
| sleep 1 |
| kill %% && wait %% &> /dev/null |
| |
| route_notify_check $outfile $expected_num_notifications |
| rm -f $outfile |
| |
| $IP route del $route dev dummy2 |
| } |
| |
| ipv4_route_replacement_test() |
| { |
| RET=0 |
| |
| local ip="ipv4" |
| local route=192.0.2.0/24 |
| |
| $IP link add name dummy2 type dummy |
| $IP link set dev dummy2 up |
| |
| # Make sure a single notification will be emitted for the new route. |
| local notify=0 |
| local expected_num_notifications=1 |
| # route_replacement_check will assign value to RET. |
| route_replacement_check $ip $notify $route $expected_num_notifications |
| |
| # Make sure two notifications will be emitted for the new route. |
| notify=1 |
| expected_num_notifications=2 |
| route_replacement_check $ip $notify $route $expected_num_notifications |
| |
| # notify=2 means emit notifications only for failed route installation, |
| # make sure a single notification will be emitted for the new route. |
| notify=2 |
| expected_num_notifications=1 |
| route_replacement_check $ip $notify $route $expected_num_notifications |
| |
| $IP link del name dummy2 |
| |
| log_test "IPv4 route replacement" |
| } |
| |
| ipv4_route_offload_failed_test() |
| { |
| |
| RET=0 |
| |
| local ip="ipv4" |
| local route=192.0.2.0/24 |
| local offload_failed=1 |
| |
| echo "y"> $DEBUGFS_DIR/fib/fail_route_offload |
| check_err $? "Failed to setup route offload to fail" |
| |
| # Make sure a single notification will be emitted for the programmed |
| # route. |
| local notify=0 |
| local expected_num_notifications=1 |
| route_addition_check $ip $notify $route $expected_num_notifications \ |
| $offload_failed |
| |
| # Make sure two notifications will be emitted for the new route. |
| notify=1 |
| expected_num_notifications=2 |
| route_addition_check $ip $notify $route $expected_num_notifications \ |
| $offload_failed |
| |
| # notify=2 means emit notifications only for failed route installation, |
| # make sure two notifications will be emitted for the new route. |
| notify=2 |
| expected_num_notifications=2 |
| route_addition_check $ip $notify $route $expected_num_notifications \ |
| $offload_failed |
| |
| echo "n"> $DEBUGFS_DIR/fib/fail_route_offload |
| check_err $? "Failed to setup route offload not to fail" |
| |
| log_test "IPv4 route offload failed" |
| } |
| |
| ipv6_route_addition_test() |
| { |
| RET=0 |
| |
| local ip="ipv6" |
| local route=2001:db8:1::/64 |
| |
| # Make sure a single notification will be emitted for the programmed |
| # route. |
| local notify=0 |
| local expected_num_notifications=1 |
| route_addition_check $ip $notify $route $expected_num_notifications |
| |
| # Make sure two notifications will be emitted for the programmed route. |
| notify=1 |
| expected_num_notifications=2 |
| route_addition_check $ip $notify $route $expected_num_notifications |
| |
| # notify=2 means emit notifications only for failed route installation, |
| # make sure a single notification will be emitted for the programmed |
| # route. |
| notify=2 |
| expected_num_notifications=1 |
| route_addition_check $ip $notify $route $expected_num_notifications |
| |
| log_test "IPv6 route addition" |
| } |
| |
| ipv6_route_deletion_test() |
| { |
| RET=0 |
| |
| local ip="ipv6" |
| local route=2001:db8:1::/64 |
| local expected_num_notifications=1 |
| |
| # Make sure a single notification will be emitted for the deleted route, |
| # regardless of fib_notify_on_flag_change value. |
| local notify=0 |
| route_deletion_check $ip $notify $route $expected_num_notifications |
| |
| notify=1 |
| route_deletion_check $ip $notify $route $expected_num_notifications |
| |
| log_test "IPv6 route deletion" |
| } |
| |
| ipv6_route_replacement_test() |
| { |
| RET=0 |
| |
| local ip="ipv6" |
| local route=2001:db8:1::/64 |
| |
| $IP link add name dummy2 type dummy |
| $IP link set dev dummy2 up |
| |
| # Make sure a single notification will be emitted for the new route. |
| local notify=0 |
| local expected_num_notifications=1 |
| route_replacement_check $ip $notify $route $expected_num_notifications |
| |
| # Make sure two notifications will be emitted for the new route. |
| notify=1 |
| expected_num_notifications=2 |
| route_replacement_check $ip $notify $route $expected_num_notifications |
| |
| # notify=2 means emit notifications only for failed route installation, |
| # make sure a single notification will be emitted for the new route. |
| notify=2 |
| expected_num_notifications=1 |
| route_replacement_check $ip $notify $route $expected_num_notifications |
| |
| $IP link del name dummy2 |
| |
| log_test "IPv6 route replacement" |
| } |
| |
| ipv6_route_offload_failed_test() |
| { |
| |
| RET=0 |
| |
| local ip="ipv6" |
| local route=2001:db8:1::/64 |
| local offload_failed=1 |
| |
| echo "y"> $DEBUGFS_DIR/fib/fail_route_offload |
| check_err $? "Failed to setup route offload to fail" |
| |
| # Make sure a single notification will be emitted for the programmed |
| # route. |
| local notify=0 |
| local expected_num_notifications=1 |
| route_addition_check $ip $notify $route $expected_num_notifications \ |
| $offload_failed |
| |
| # Make sure two notifications will be emitted for the new route. |
| notify=1 |
| expected_num_notifications=2 |
| route_addition_check $ip $notify $route $expected_num_notifications \ |
| $offload_failed |
| |
| # notify=2 means emit notifications only for failed route installation, |
| # make sure two notifications will be emitted for the new route. |
| notify=2 |
| expected_num_notifications=2 |
| route_addition_check $ip $notify $route $expected_num_notifications \ |
| $offload_failed |
| |
| echo "n"> $DEBUGFS_DIR/fib/fail_route_offload |
| check_err $? "Failed to setup route offload not to fail" |
| |
| log_test "IPv6 route offload failed" |
| } |
| |
| setup_prepare() |
| { |
| modprobe netdevsim &> /dev/null |
| echo "$DEV_ADDR 1" > ${NETDEVSIM_PATH}/new_device |
| while [ ! -d $SYSFS_NET_DIR ] ; do :; done |
| |
| ip netns add testns1 |
| |
| if [ $? -ne 0 ]; then |
| echo "Failed to add netns \"testns1\"" |
| exit 1 |
| fi |
| |
| devlink dev reload $DEVLINK_DEV netns testns1 |
| |
| if [ $? -ne 0 ]; then |
| echo "Failed to reload into netns \"testns1\"" |
| exit 1 |
| fi |
| |
| IP="ip -n testns1" |
| |
| $IP link add name dummy1 type dummy |
| $IP link set dev dummy1 up |
| } |
| |
| cleanup() |
| { |
| pre_cleanup |
| |
| $IP link del name dummy1 |
| ip netns del testns1 |
| echo "$DEV_ADDR" > ${NETDEVSIM_PATH}/del_device |
| modprobe -r netdevsim &> /dev/null |
| } |
| |
| trap cleanup EXIT |
| |
| setup_prepare |
| |
| tests_run |
| |
| exit $EXIT_STATUS |