selftests/net: change shebang to bash to support "source"
[platform/kernel/linux-starfive.git] / tools / testing / selftests / net / test_vxlan_nolocalbypass.sh
1 #!/bin/bash
2 # SPDX-License-Identifier: GPL-2.0
3
4 # This test is for checking the [no]localbypass VXLAN device option. The test
5 # configures two VXLAN devices in the same network namespace and a tc filter on
6 # the loopback device that drops encapsulated packets. The test sends packets
7 # from the first VXLAN device and verifies that by default these packets are
8 # received by the second VXLAN device. The test then enables the nolocalbypass
9 # option and verifies that packets are no longer received by the second VXLAN
10 # device.
11
12 ret=0
13 # Kselftest framework requirement - SKIP code is 4.
14 ksft_skip=4
15
16 TESTS="
17         nolocalbypass
18 "
19 VERBOSE=0
20 PAUSE_ON_FAIL=no
21 PAUSE=no
22
23 ################################################################################
24 # Utilities
25
26 log_test()
27 {
28         local rc=$1
29         local expected=$2
30         local msg="$3"
31
32         if [ ${rc} -eq ${expected} ]; then
33                 printf "TEST: %-60s  [ OK ]\n" "${msg}"
34                 nsuccess=$((nsuccess+1))
35         else
36                 ret=1
37                 nfail=$((nfail+1))
38                 printf "TEST: %-60s  [FAIL]\n" "${msg}"
39                 if [ "$VERBOSE" = "1" ]; then
40                         echo "    rc=$rc, expected $expected"
41                 fi
42
43                 if [ "${PAUSE_ON_FAIL}" = "yes" ]; then
44                 echo
45                         echo "hit enter to continue, 'q' to quit"
46                         read a
47                         [ "$a" = "q" ] && exit 1
48                 fi
49         fi
50
51         if [ "${PAUSE}" = "yes" ]; then
52                 echo
53                 echo "hit enter to continue, 'q' to quit"
54                 read a
55                 [ "$a" = "q" ] && exit 1
56         fi
57
58         [ "$VERBOSE" = "1" ] && echo
59 }
60
61 run_cmd()
62 {
63         local cmd="$1"
64         local out
65         local stderr="2>/dev/null"
66
67         if [ "$VERBOSE" = "1" ]; then
68                 printf "COMMAND: $cmd\n"
69                 stderr=
70         fi
71
72         out=$(eval $cmd $stderr)
73         rc=$?
74         if [ "$VERBOSE" = "1" -a -n "$out" ]; then
75                 echo "    $out"
76         fi
77
78         return $rc
79 }
80
81 tc_check_packets()
82 {
83         local ns=$1; shift
84         local id=$1; shift
85         local handle=$1; shift
86         local count=$1; shift
87         local pkts
88
89         sleep 0.1
90         pkts=$(tc -n $ns -j -s filter show $id \
91                 | jq ".[] | select(.options.handle == $handle) | \
92                 .options.actions[0].stats.packets")
93         [[ $pkts == $count ]]
94 }
95
96 ################################################################################
97 # Setup
98
99 setup()
100 {
101         ip netns add ns1
102
103         ip -n ns1 link set dev lo up
104         ip -n ns1 address add 192.0.2.1/32 dev lo
105         ip -n ns1 address add 198.51.100.1/32 dev lo
106
107         ip -n ns1 link add name vx0 up type vxlan id 100 local 198.51.100.1 \
108                 dstport 4789 nolearning
109         ip -n ns1 link add name vx1 up type vxlan id 100 dstport 4790
110 }
111
112 cleanup()
113 {
114         ip netns del ns1 &> /dev/null
115 }
116
117 ################################################################################
118 # Tests
119
120 nolocalbypass()
121 {
122         local smac=00:01:02:03:04:05
123         local dmac=00:0a:0b:0c:0d:0e
124
125         run_cmd "bridge -n ns1 fdb add $dmac dev vx0 self static dst 192.0.2.1 port 4790"
126
127         run_cmd "tc -n ns1 qdisc add dev vx1 clsact"
128         run_cmd "tc -n ns1 filter add dev vx1 ingress pref 1 handle 101 proto all flower src_mac $smac dst_mac $dmac action pass"
129
130         run_cmd "tc -n ns1 qdisc add dev lo clsact"
131         run_cmd "tc -n ns1 filter add dev lo ingress pref 1 handle 101 proto ip flower ip_proto udp dst_port 4790 action drop"
132
133         run_cmd "ip -n ns1 -d -j link show dev vx0 | jq -e '.[][\"linkinfo\"][\"info_data\"][\"localbypass\"] == true'"
134         log_test $? 0 "localbypass enabled"
135
136         run_cmd "ip netns exec ns1 mausezahn vx0 -a $smac -b $dmac -c 1 -p 100 -q"
137
138         tc_check_packets "ns1" "dev vx1 ingress" 101 1
139         log_test $? 0 "Packet received by local VXLAN device - localbypass"
140
141         run_cmd "ip -n ns1 link set dev vx0 type vxlan nolocalbypass"
142
143         run_cmd "ip -n ns1 -d -j link show dev vx0 | jq -e '.[][\"linkinfo\"][\"info_data\"][\"localbypass\"] == false'"
144         log_test $? 0 "localbypass disabled"
145
146         run_cmd "ip netns exec ns1 mausezahn vx0 -a $smac -b $dmac -c 1 -p 100 -q"
147
148         tc_check_packets "ns1" "dev vx1 ingress" 101 1
149         log_test $? 0 "Packet not received by local VXLAN device - nolocalbypass"
150
151         run_cmd "ip -n ns1 link set dev vx0 type vxlan localbypass"
152
153         run_cmd "ip -n ns1 -d -j link show dev vx0 | jq -e '.[][\"linkinfo\"][\"info_data\"][\"localbypass\"] == true'"
154         log_test $? 0 "localbypass enabled"
155
156         run_cmd "ip netns exec ns1 mausezahn vx0 -a $smac -b $dmac -c 1 -p 100 -q"
157
158         tc_check_packets "ns1" "dev vx1 ingress" 101 2
159         log_test $? 0 "Packet received by local VXLAN device - localbypass"
160 }
161
162 ################################################################################
163 # Usage
164
165 usage()
166 {
167         cat <<EOF
168 usage: ${0##*/} OPTS
169
170         -t <test>   Test(s) to run (default: all)
171                     (options: $TESTS)
172         -p          Pause on fail
173         -P          Pause after each test before cleanup
174         -v          Verbose mode (show commands and output)
175 EOF
176 }
177
178 ################################################################################
179 # Main
180
181 trap cleanup EXIT
182
183 while getopts ":t:pPvh" opt; do
184         case $opt in
185                 t) TESTS=$OPTARG ;;
186                 p) PAUSE_ON_FAIL=yes;;
187                 P) PAUSE=yes;;
188                 v) VERBOSE=$(($VERBOSE + 1));;
189                 h) usage; exit 0;;
190                 *) usage; exit 1;;
191         esac
192 done
193
194 # Make sure we don't pause twice.
195 [ "${PAUSE}" = "yes" ] && PAUSE_ON_FAIL=no
196
197 if [ "$(id -u)" -ne 0 ];then
198         echo "SKIP: Need root privileges"
199         exit $ksft_skip;
200 fi
201
202 if [ ! -x "$(command -v ip)" ]; then
203         echo "SKIP: Could not run test without ip tool"
204         exit $ksft_skip
205 fi
206
207 if [ ! -x "$(command -v bridge)" ]; then
208         echo "SKIP: Could not run test without bridge tool"
209         exit $ksft_skip
210 fi
211
212 if [ ! -x "$(command -v mausezahn)" ]; then
213         echo "SKIP: Could not run test without mausezahn tool"
214         exit $ksft_skip
215 fi
216
217 if [ ! -x "$(command -v jq)" ]; then
218         echo "SKIP: Could not run test without jq tool"
219         exit $ksft_skip
220 fi
221
222 ip link help vxlan 2>&1 | grep -q "localbypass"
223 if [ $? -ne 0 ]; then
224         echo "SKIP: iproute2 ip too old, missing VXLAN nolocalbypass support"
225         exit $ksft_skip
226 fi
227
228 cleanup
229
230 for t in $TESTS
231 do
232         setup; $t; cleanup;
233 done
234
235 if [ "$TESTS" != "none" ]; then
236         printf "\nTests passed: %3d\n" ${nsuccess}
237         printf "Tests failed: %3d\n"   ${nfail}
238 fi
239
240 exit $ret