Linux kernel mirror (for testing)
git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
kernel
os
linux
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
12ret=0
13# Kselftest framework requirement - SKIP code is 4.
14ksft_skip=4
15
16TESTS="
17 nolocalbypass
18"
19VERBOSE=0
20PAUSE_ON_FAIL=no
21PAUSE=no
22
23################################################################################
24# Utilities
25
26log_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
61run_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
81tc_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
99setup()
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
112cleanup()
113{
114 ip netns del ns1 &> /dev/null
115}
116
117################################################################################
118# Tests
119
120nolocalbypass()
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
165usage()
166{
167 cat <<EOF
168usage: ${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)
175EOF
176}
177
178################################################################################
179# Main
180
181trap cleanup EXIT
182
183while 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
192done
193
194# Make sure we don't pause twice.
195[ "${PAUSE}" = "yes" ] && PAUSE_ON_FAIL=no
196
197if [ "$(id -u)" -ne 0 ];then
198 echo "SKIP: Need root privileges"
199 exit $ksft_skip;
200fi
201
202if [ ! -x "$(command -v ip)" ]; then
203 echo "SKIP: Could not run test without ip tool"
204 exit $ksft_skip
205fi
206
207if [ ! -x "$(command -v bridge)" ]; then
208 echo "SKIP: Could not run test without bridge tool"
209 exit $ksft_skip
210fi
211
212if [ ! -x "$(command -v mausezahn)" ]; then
213 echo "SKIP: Could not run test without mausezahn tool"
214 exit $ksft_skip
215fi
216
217if [ ! -x "$(command -v jq)" ]; then
218 echo "SKIP: Could not run test without jq tool"
219 exit $ksft_skip
220fi
221
222ip link help vxlan 2>&1 | grep -q "localbypass"
223if [ $? -ne 0 ]; then
224 echo "SKIP: iproute2 ip too old, missing VXLAN nolocalbypass support"
225 exit $ksft_skip
226fi
227
228cleanup
229
230for t in $TESTS
231do
232 setup; $t; cleanup;
233done
234
235if [ "$TESTS" != "none" ]; then
236 printf "\nTests passed: %3d\n" ${nsuccess}
237 printf "Tests failed: %3d\n" ${nfail}
238fi
239
240exit $ret