selftests/net: change shebang to bash to support "source"
[platform/kernel/linux-starfive.git] / tools / testing / selftests / net / fib_nexthops.sh
1 #!/bin/bash
2 # SPDX-License-Identifier: GPL-2.0
3 #
4 # ns: me               | ns: peer              | ns: remote
5 #   2001:db8:91::1     |       2001:db8:91::2  |
6 #   172.16.1.1         |       172.16.1.2      |
7 #            veth1 <---|---> veth2             |
8 #                      |              veth5 <--|--> veth6  172.16.101.1
9 #            veth3 <---|---> veth4             |           2001:db8:101::1
10 #   172.16.2.1         |       172.16.2.2      |
11 #   2001:db8:92::1     |       2001:db8:92::2  |
12 #
13 # This test is for checking IPv4 and IPv6 FIB behavior with nexthop
14 # objects. Device reference counts and network namespace cleanup tested
15 # by use of network namespace for peer.
16
17 ret=0
18 # Kselftest framework requirement - SKIP code is 4.
19 ksft_skip=4
20
21 # all tests in this script. Can be overridden with -t option
22 IPV4_TESTS="
23         ipv4_fcnal
24         ipv4_grp_fcnal
25         ipv4_res_grp_fcnal
26         ipv4_withv6_fcnal
27         ipv4_fcnal_runtime
28         ipv4_large_grp
29         ipv4_large_res_grp
30         ipv4_compat_mode
31         ipv4_fdb_grp_fcnal
32         ipv4_mpath_select
33         ipv4_torture
34         ipv4_res_torture
35 "
36
37 IPV6_TESTS="
38         ipv6_fcnal
39         ipv6_grp_fcnal
40         ipv6_res_grp_fcnal
41         ipv6_fcnal_runtime
42         ipv6_large_grp
43         ipv6_large_res_grp
44         ipv6_compat_mode
45         ipv6_fdb_grp_fcnal
46         ipv6_mpath_select
47         ipv6_torture
48         ipv6_res_torture
49 "
50
51 ALL_TESTS="
52         basic
53         basic_res
54         ${IPV4_TESTS}
55         ${IPV6_TESTS}
56 "
57 TESTS="${ALL_TESTS}"
58 VERBOSE=0
59 PAUSE_ON_FAIL=no
60 PAUSE=no
61 PING_TIMEOUT=5
62
63 nsid=100
64
65 ################################################################################
66 # utilities
67
68 log_test()
69 {
70         local rc=$1
71         local expected=$2
72         local msg="$3"
73
74         if [ ${rc} -eq ${expected} ]; then
75                 printf "TEST: %-60s  [ OK ]\n" "${msg}"
76                 nsuccess=$((nsuccess+1))
77         else
78                 ret=1
79                 nfail=$((nfail+1))
80                 printf "TEST: %-60s  [FAIL]\n" "${msg}"
81                 if [ "$VERBOSE" = "1" ]; then
82                         echo "    rc=$rc, expected $expected"
83                 fi
84
85                 if [ "${PAUSE_ON_FAIL}" = "yes" ]; then
86                 echo
87                         echo "hit enter to continue, 'q' to quit"
88                         read a
89                         [ "$a" = "q" ] && exit 1
90                 fi
91         fi
92
93         if [ "${PAUSE}" = "yes" ]; then
94                 echo
95                 echo "hit enter to continue, 'q' to quit"
96                 read a
97                 [ "$a" = "q" ] && exit 1
98         fi
99
100         [ "$VERBOSE" = "1" ] && echo
101 }
102
103 run_cmd()
104 {
105         local cmd="$1"
106         local out
107         local stderr="2>/dev/null"
108
109         if [ "$VERBOSE" = "1" ]; then
110                 printf "COMMAND: $cmd\n"
111                 stderr=
112         fi
113
114         out=$(eval $cmd $stderr)
115         rc=$?
116         if [ "$VERBOSE" = "1" -a -n "$out" ]; then
117                 echo "    $out"
118         fi
119
120         return $rc
121 }
122
123 get_linklocal()
124 {
125         local dev=$1
126         local ns
127         local addr
128
129         [ -n "$2" ] && ns="-netns $2"
130         addr=$(ip $ns -6 -br addr show dev ${dev} | \
131         awk '{
132                 for (i = 3; i <= NF; ++i) {
133                         if ($i ~ /^fe80/)
134                                 print $i
135                 }
136         }'
137         )
138         addr=${addr/\/*}
139
140         [ -z "$addr" ] && return 1
141
142         echo $addr
143
144         return 0
145 }
146
147 create_ns()
148 {
149         local n=${1}
150
151         ip netns del ${n} 2>/dev/null
152
153         set -e
154         ip netns add ${n}
155         ip netns set ${n} $((nsid++))
156         ip -netns ${n} addr add 127.0.0.1/8 dev lo
157         ip -netns ${n} link set lo up
158
159         ip netns exec ${n} sysctl -qw net.ipv4.ip_forward=1
160         ip netns exec ${n} sysctl -qw net.ipv4.fib_multipath_use_neigh=1
161         ip netns exec ${n} sysctl -qw net.ipv4.conf.default.ignore_routes_with_linkdown=1
162         ip netns exec ${n} sysctl -qw net.ipv6.conf.all.keep_addr_on_down=1
163         ip netns exec ${n} sysctl -qw net.ipv6.conf.all.forwarding=1
164         ip netns exec ${n} sysctl -qw net.ipv6.conf.default.forwarding=1
165         ip netns exec ${n} sysctl -qw net.ipv6.conf.default.ignore_routes_with_linkdown=1
166         ip netns exec ${n} sysctl -qw net.ipv6.conf.all.accept_dad=0
167         ip netns exec ${n} sysctl -qw net.ipv6.conf.default.accept_dad=0
168
169         set +e
170 }
171
172 setup()
173 {
174         cleanup
175
176         create_ns me
177         create_ns peer
178         create_ns remote
179
180         IP="ip -netns me"
181         BRIDGE="bridge -netns me"
182         set -e
183         $IP li add veth1 type veth peer name veth2
184         $IP li set veth1 up
185         $IP addr add 172.16.1.1/24 dev veth1
186         $IP -6 addr add 2001:db8:91::1/64 dev veth1 nodad
187
188         $IP li add veth3 type veth peer name veth4
189         $IP li set veth3 up
190         $IP addr add 172.16.2.1/24 dev veth3
191         $IP -6 addr add 2001:db8:92::1/64 dev veth3 nodad
192
193         $IP li set veth2 netns peer up
194         ip -netns peer addr add 172.16.1.2/24 dev veth2
195         ip -netns peer -6 addr add 2001:db8:91::2/64 dev veth2 nodad
196
197         $IP li set veth4 netns peer up
198         ip -netns peer addr add 172.16.2.2/24 dev veth4
199         ip -netns peer -6 addr add 2001:db8:92::2/64 dev veth4 nodad
200
201         ip -netns remote li add veth5 type veth peer name veth6
202         ip -netns remote li set veth5 up
203         ip -netns remote addr add dev veth5 172.16.101.1/24
204         ip -netns remote -6 addr add dev veth5 2001:db8:101::1/64 nodad
205         ip -netns remote ro add 172.16.0.0/22 via 172.16.101.2
206         ip -netns remote -6 ro add 2001:db8:90::/40 via 2001:db8:101::2
207
208         ip -netns remote li set veth6 netns peer up
209         ip -netns peer addr add dev veth6 172.16.101.2/24
210         ip -netns peer -6 addr add dev veth6 2001:db8:101::2/64 nodad
211         set +e
212 }
213
214 cleanup()
215 {
216         local ns
217
218         for ns in me peer remote; do
219                 ip netns del ${ns} 2>/dev/null
220         done
221 }
222
223 check_output()
224 {
225         local out="$1"
226         local expected="$2"
227         local rc=0
228
229         [ "${out}" = "${expected}" ] && return 0
230
231         if [ -z "${out}" ]; then
232                 if [ "$VERBOSE" = "1" ]; then
233                         printf "\nNo entry found\n"
234                         printf "Expected:\n"
235                         printf "    ${expected}\n"
236                 fi
237                 return 1
238         fi
239
240         out=$(echo ${out})
241         if [ "${out}" != "${expected}" ]; then
242                 rc=1
243                 if [ "${VERBOSE}" = "1" ]; then
244                         printf "    Unexpected entry. Have:\n"
245                         printf "        ${out}\n"
246                         printf "    Expected:\n"
247                         printf "        ${expected}\n\n"
248                 else
249                         echo "      WARNING: Unexpected route entry"
250                 fi
251         fi
252
253         return $rc
254 }
255
256 check_nexthop()
257 {
258         local nharg="$1"
259         local expected="$2"
260         local out
261
262         out=$($IP nexthop ls ${nharg} 2>/dev/null)
263
264         check_output "${out}" "${expected}"
265 }
266
267 check_nexthop_bucket()
268 {
269         local nharg="$1"
270         local expected="$2"
271         local out
272
273         # remove the idle time since we cannot match it
274         out=$($IP nexthop bucket ${nharg} \
275                 | sed s/idle_time\ [0-9.]*\ // 2>/dev/null)
276
277         check_output "${out}" "${expected}"
278 }
279
280 check_route()
281 {
282         local pfx="$1"
283         local expected="$2"
284         local out
285
286         out=$($IP route ls match ${pfx} 2>/dev/null)
287
288         check_output "${out}" "${expected}"
289 }
290
291 check_route6()
292 {
293         local pfx="$1"
294         local expected="$2"
295         local out
296
297         out=$($IP -6 route ls match ${pfx} 2>/dev/null | sed -e 's/pref medium//')
298
299         check_output "${out}" "${expected}"
300 }
301
302 check_large_grp()
303 {
304         local ipv=$1
305         local ecmp=$2
306         local grpnum=100
307         local nhidstart=100
308         local grpidstart=1000
309         local iter=0
310         local nhidstr=""
311         local grpidstr=""
312         local grpstr=""
313         local ipstr=""
314
315         if [ $ipv -eq 4 ]; then
316                 ipstr="172.16.1."
317         else
318                 ipstr="2001:db8:91::"
319         fi
320
321         #
322         # Create $grpnum groups with specified $ecmp and dump them
323         #
324
325         # create nexthops with different gateways
326         iter=2
327         while [ $iter -le $(($ecmp + 1)) ]
328         do
329                 nhidstr="$(($nhidstart + $iter))"
330                 run_cmd "$IP nexthop add id $nhidstr via $ipstr$iter dev veth1"
331                 check_nexthop "id $nhidstr" "id $nhidstr via $ipstr$iter dev veth1 scope link"
332
333                 if [ $iter -le $ecmp ]; then
334                         grpstr+="$nhidstr/"
335                 else
336                         grpstr+="$nhidstr"
337                 fi
338                 ((iter++))
339         done
340
341         # create duplicate large ecmp groups
342         iter=0
343         while [ $iter -le $grpnum ]
344         do
345                 grpidstr="$(($grpidstart + $iter))"
346                 run_cmd "$IP nexthop add id $grpidstr group $grpstr"
347                 check_nexthop "id $grpidstr" "id $grpidstr group $grpstr"
348                 ((iter++))
349         done
350
351         # dump large groups
352         run_cmd "$IP nexthop list"
353         log_test $? 0 "Dump large (x$ecmp) ecmp groups"
354 }
355
356 check_large_res_grp()
357 {
358         local ipv=$1
359         local buckets=$2
360         local ipstr=""
361
362         if [ $ipv -eq 4 ]; then
363                 ipstr="172.16.1.2"
364         else
365                 ipstr="2001:db8:91::2"
366         fi
367
368         # create a resilient group with $buckets buckets and dump them
369         run_cmd "$IP nexthop add id 100 via $ipstr dev veth1"
370         run_cmd "$IP nexthop add id 1000 group 100 type resilient buckets $buckets"
371         run_cmd "$IP nexthop bucket list"
372         log_test $? 0 "Dump large (x$buckets) nexthop buckets"
373 }
374
375 get_route_dev()
376 {
377         local pfx="$1"
378         local out
379
380         if out=$($IP -j route get "$pfx" | jq -re ".[0].dev"); then
381                 echo "$out"
382         fi
383 }
384
385 check_route_dev()
386 {
387         local pfx="$1"
388         local expected="$2"
389         local out
390
391         out=$(get_route_dev "$pfx")
392
393         check_output "$out" "$expected"
394 }
395
396 start_ip_monitor()
397 {
398         local mtype=$1
399
400         # start the monitor in the background
401         tmpfile=`mktemp /var/run/nexthoptestXXX`
402         mpid=`($IP monitor $mtype > $tmpfile & echo $!) 2>/dev/null`
403         sleep 0.2
404         echo "$mpid $tmpfile"
405 }
406
407 stop_ip_monitor()
408 {
409         local mpid=$1
410         local tmpfile=$2
411         local el=$3
412
413         # check the monitor results
414         kill $mpid
415         lines=`wc -l $tmpfile | cut "-d " -f1`
416         test $lines -eq $el
417         rc=$?
418         rm -rf $tmpfile
419
420         return $rc
421 }
422
423 check_nexthop_fdb_support()
424 {
425         $IP nexthop help 2>&1 | grep -q fdb
426         if [ $? -ne 0 ]; then
427                 echo "SKIP: iproute2 too old, missing fdb nexthop support"
428                 return $ksft_skip
429         fi
430 }
431
432 check_nexthop_res_support()
433 {
434         $IP nexthop help 2>&1 | grep -q resilient
435         if [ $? -ne 0 ]; then
436                 echo "SKIP: iproute2 too old, missing resilient nexthop group support"
437                 return $ksft_skip
438         fi
439 }
440
441 ipv6_fdb_grp_fcnal()
442 {
443         local rc
444
445         echo
446         echo "IPv6 fdb groups functional"
447         echo "--------------------------"
448
449         check_nexthop_fdb_support
450         if [ $? -eq $ksft_skip ]; then
451                 return $ksft_skip
452         fi
453
454         # create group with multiple nexthops
455         run_cmd "$IP nexthop add id 61 via 2001:db8:91::2 fdb"
456         run_cmd "$IP nexthop add id 62 via 2001:db8:91::3 fdb"
457         run_cmd "$IP nexthop add id 102 group 61/62 fdb"
458         check_nexthop "id 102" "id 102 group 61/62 fdb"
459         log_test $? 0 "Fdb Nexthop group with multiple nexthops"
460
461         ## get nexthop group
462         run_cmd "$IP nexthop get id 102"
463         check_nexthop "id 102" "id 102 group 61/62 fdb"
464         log_test $? 0 "Get Fdb nexthop group by id"
465
466         # fdb nexthop group can only contain fdb nexthops
467         run_cmd "$IP nexthop add id 63 via 2001:db8:91::4"
468         run_cmd "$IP nexthop add id 64 via 2001:db8:91::5"
469         run_cmd "$IP nexthop add id 103 group 63/64 fdb"
470         log_test $? 2 "Fdb Nexthop group with non-fdb nexthops"
471
472         # Non fdb nexthop group can not contain fdb nexthops
473         run_cmd "$IP nexthop add id 65 via 2001:db8:91::5 fdb"
474         run_cmd "$IP nexthop add id 66 via 2001:db8:91::6 fdb"
475         run_cmd "$IP nexthop add id 104 group 65/66"
476         log_test $? 2 "Non-Fdb Nexthop group with fdb nexthops"
477
478         # fdb nexthop cannot have blackhole
479         run_cmd "$IP nexthop add id 67 blackhole fdb"
480         log_test $? 2 "Fdb Nexthop with blackhole"
481
482         # fdb nexthop with oif
483         run_cmd "$IP nexthop add id 68 via 2001:db8:91::7 dev veth1 fdb"
484         log_test $? 2 "Fdb Nexthop with oif"
485
486         # fdb nexthop with onlink
487         run_cmd "$IP nexthop add id 68 via 2001:db8:91::7 onlink fdb"
488         log_test $? 2 "Fdb Nexthop with onlink"
489
490         # fdb nexthop with encap
491         run_cmd "$IP nexthop add id 69 encap mpls 101 via 2001:db8:91::8 dev veth1 fdb"
492         log_test $? 2 "Fdb Nexthop with encap"
493
494         run_cmd "$IP link add name vx10 type vxlan id 1010 local 2001:db8:91::9 remote 2001:db8:91::10 dstport 4789 nolearning noudpcsum tos inherit ttl 100"
495         run_cmd "$BRIDGE fdb add 02:02:00:00:00:13 dev vx10 nhid 102 self"
496         log_test $? 0 "Fdb mac add with nexthop group"
497
498         ## fdb nexthops can only reference nexthop groups and not nexthops
499         run_cmd "$BRIDGE fdb add 02:02:00:00:00:14 dev vx10 nhid 61 self"
500         log_test $? 255 "Fdb mac add with nexthop"
501
502         run_cmd "$IP -6 ro add 2001:db8:101::1/128 nhid 66"
503         log_test $? 2 "Route add with fdb nexthop"
504
505         run_cmd "$IP -6 ro add 2001:db8:101::1/128 nhid 103"
506         log_test $? 2 "Route add with fdb nexthop group"
507
508         run_cmd "$IP nexthop del id 61"
509         run_cmd "$BRIDGE fdb get to 02:02:00:00:00:13 dev vx10 self"
510         log_test $? 0 "Fdb entry after deleting a single nexthop"
511
512         run_cmd "$IP nexthop del id 102"
513         log_test $? 0 "Fdb nexthop delete"
514
515         run_cmd "$BRIDGE fdb get to 02:02:00:00:00:13 dev vx10 self"
516         log_test $? 254 "Fdb entry after deleting a nexthop group"
517
518         $IP link del dev vx10
519 }
520
521 ipv4_fdb_grp_fcnal()
522 {
523         local rc
524
525         echo
526         echo "IPv4 fdb groups functional"
527         echo "--------------------------"
528
529         check_nexthop_fdb_support
530         if [ $? -eq $ksft_skip ]; then
531                 return $ksft_skip
532         fi
533
534         # create group with multiple nexthops
535         run_cmd "$IP nexthop add id 12 via 172.16.1.2 fdb"
536         run_cmd "$IP nexthop add id 13 via 172.16.1.3 fdb"
537         run_cmd "$IP nexthop add id 102 group 12/13 fdb"
538         check_nexthop "id 102" "id 102 group 12/13 fdb"
539         log_test $? 0 "Fdb Nexthop group with multiple nexthops"
540
541         # get nexthop group
542         run_cmd "$IP nexthop get id 102"
543         check_nexthop "id 102" "id 102 group 12/13 fdb"
544         log_test $? 0 "Get Fdb nexthop group by id"
545
546         # fdb nexthop group can only contain fdb nexthops
547         run_cmd "$IP nexthop add id 14 via 172.16.1.2"
548         run_cmd "$IP nexthop add id 15 via 172.16.1.3"
549         run_cmd "$IP nexthop add id 103 group 14/15 fdb"
550         log_test $? 2 "Fdb Nexthop group with non-fdb nexthops"
551
552         # Non fdb nexthop group can not contain fdb nexthops
553         run_cmd "$IP nexthop add id 16 via 172.16.1.2 fdb"
554         run_cmd "$IP nexthop add id 17 via 172.16.1.3 fdb"
555         run_cmd "$IP nexthop add id 104 group 14/15"
556         log_test $? 2 "Non-Fdb Nexthop group with fdb nexthops"
557
558         # fdb nexthop cannot have blackhole
559         run_cmd "$IP nexthop add id 18 blackhole fdb"
560         log_test $? 2 "Fdb Nexthop with blackhole"
561
562         # fdb nexthop with oif
563         run_cmd "$IP nexthop add id 16 via 172.16.1.2 dev veth1 fdb"
564         log_test $? 2 "Fdb Nexthop with oif"
565
566         # fdb nexthop with onlink
567         run_cmd "$IP nexthop add id 16 via 172.16.1.2 onlink fdb"
568         log_test $? 2 "Fdb Nexthop with onlink"
569
570         # fdb nexthop with encap
571         run_cmd "$IP nexthop add id 17 encap mpls 101 via 172.16.1.2 dev veth1 fdb"
572         log_test $? 2 "Fdb Nexthop with encap"
573
574         run_cmd "$IP link add name vx10 type vxlan id 1010 local 10.0.0.1 remote 10.0.0.2 dstport 4789 nolearning noudpcsum tos inherit ttl 100"
575         run_cmd "$BRIDGE fdb add 02:02:00:00:00:13 dev vx10 nhid 102 self"
576         log_test $? 0 "Fdb mac add with nexthop group"
577
578         # fdb nexthops can only reference nexthop groups and not nexthops
579         run_cmd "$BRIDGE fdb add 02:02:00:00:00:14 dev vx10 nhid 12 self"
580         log_test $? 255 "Fdb mac add with nexthop"
581
582         run_cmd "$IP ro add 172.16.0.0/22 nhid 15"
583         log_test $? 2 "Route add with fdb nexthop"
584
585         run_cmd "$IP ro add 172.16.0.0/22 nhid 103"
586         log_test $? 2 "Route add with fdb nexthop group"
587
588         run_cmd "$IP nexthop del id 12"
589         run_cmd "$BRIDGE fdb get to 02:02:00:00:00:13 dev vx10 self"
590         log_test $? 0 "Fdb entry after deleting a single nexthop"
591
592         run_cmd "$IP nexthop del id 102"
593         log_test $? 0 "Fdb nexthop delete"
594
595         run_cmd "$BRIDGE fdb get to 02:02:00:00:00:13 dev vx10 self"
596         log_test $? 254 "Fdb entry after deleting a nexthop group"
597
598         $IP link del dev vx10
599 }
600
601 ipv4_mpath_select()
602 {
603         local rc dev match h addr
604
605         echo
606         echo "IPv4 multipath selection"
607         echo "------------------------"
608         if [ ! -x "$(command -v jq)" ]; then
609                 echo "SKIP: Could not run test; need jq tool"
610                 return $ksft_skip
611         fi
612
613         # Use status of existing neighbor entry when determining nexthop for
614         # multipath routes.
615         local -A gws
616         gws=([veth1]=172.16.1.2 [veth3]=172.16.2.2)
617         local -A other_dev
618         other_dev=([veth1]=veth3 [veth3]=veth1)
619
620         run_cmd "$IP nexthop add id 1 via ${gws["veth1"]} dev veth1"
621         run_cmd "$IP nexthop add id 2 via ${gws["veth3"]} dev veth3"
622         run_cmd "$IP nexthop add id 1001 group 1/2"
623         run_cmd "$IP ro add 172.16.101.0/24 nhid 1001"
624         rc=0
625         for dev in veth1 veth3; do
626                 match=0
627                 for h in {1..254}; do
628                         addr="172.16.101.$h"
629                         if [ "$(get_route_dev "$addr")" = "$dev" ]; then
630                                 match=1
631                                 break
632                         fi
633                 done
634                 if (( match == 0 )); then
635                         echo "SKIP: Did not find a route using device $dev"
636                         return $ksft_skip
637                 fi
638                 run_cmd "$IP neigh add ${gws[$dev]} dev $dev nud failed"
639                 if ! check_route_dev "$addr" "${other_dev[$dev]}"; then
640                         rc=1
641                         break
642                 fi
643                 run_cmd "$IP neigh del ${gws[$dev]} dev $dev"
644         done
645         log_test $rc 0 "Use valid neighbor during multipath selection"
646
647         run_cmd "$IP neigh add 172.16.1.2 dev veth1 nud incomplete"
648         run_cmd "$IP neigh add 172.16.2.2 dev veth3 nud incomplete"
649         run_cmd "$IP route get 172.16.101.1"
650         # if we did not crash, success
651         log_test $rc 0 "Multipath selection with no valid neighbor"
652 }
653
654 ipv6_mpath_select()
655 {
656         local rc dev match h addr
657
658         echo
659         echo "IPv6 multipath selection"
660         echo "------------------------"
661         if [ ! -x "$(command -v jq)" ]; then
662                 echo "SKIP: Could not run test; need jq tool"
663                 return $ksft_skip
664         fi
665
666         # Use status of existing neighbor entry when determining nexthop for
667         # multipath routes.
668         local -A gws
669         gws=([veth1]=2001:db8:91::2 [veth3]=2001:db8:92::2)
670         local -A other_dev
671         other_dev=([veth1]=veth3 [veth3]=veth1)
672
673         run_cmd "$IP nexthop add id 1 via ${gws["veth1"]} dev veth1"
674         run_cmd "$IP nexthop add id 2 via ${gws["veth3"]} dev veth3"
675         run_cmd "$IP nexthop add id 1001 group 1/2"
676         run_cmd "$IP ro add 2001:db8:101::/64 nhid 1001"
677         rc=0
678         for dev in veth1 veth3; do
679                 match=0
680                 for h in {1..65535}; do
681                         addr=$(printf "2001:db8:101::%x" $h)
682                         if [ "$(get_route_dev "$addr")" = "$dev" ]; then
683                                 match=1
684                                 break
685                         fi
686                 done
687                 if (( match == 0 )); then
688                         echo "SKIP: Did not find a route using device $dev"
689                         return $ksft_skip
690                 fi
691                 run_cmd "$IP neigh add ${gws[$dev]} dev $dev nud failed"
692                 if ! check_route_dev "$addr" "${other_dev[$dev]}"; then
693                         rc=1
694                         break
695                 fi
696                 run_cmd "$IP neigh del ${gws[$dev]} dev $dev"
697         done
698         log_test $rc 0 "Use valid neighbor during multipath selection"
699
700         run_cmd "$IP neigh add 2001:db8:91::2 dev veth1 nud incomplete"
701         run_cmd "$IP neigh add 2001:db8:92::2 dev veth3 nud incomplete"
702         run_cmd "$IP route get 2001:db8:101::1"
703         # if we did not crash, success
704         log_test $rc 0 "Multipath selection with no valid neighbor"
705 }
706
707 ################################################################################
708 # basic operations (add, delete, replace) on nexthops and nexthop groups
709 #
710 # IPv6
711
712 ipv6_fcnal()
713 {
714         local rc
715
716         echo
717         echo "IPv6"
718         echo "----------------------"
719
720         run_cmd "$IP nexthop add id 52 via 2001:db8:91::2 dev veth1"
721         rc=$?
722         log_test $rc 0 "Create nexthop with id, gw, dev"
723         if [ $rc -ne 0 ]; then
724                 echo "Basic IPv6 create fails; can not continue"
725                 return 1
726         fi
727
728         run_cmd "$IP nexthop get id 52"
729         log_test $? 0 "Get nexthop by id"
730         check_nexthop "id 52" "id 52 via 2001:db8:91::2 dev veth1 scope link"
731
732         run_cmd "$IP nexthop del id 52"
733         log_test $? 0 "Delete nexthop by id"
734         check_nexthop "id 52" ""
735
736         #
737         # gw, device spec
738         #
739         # gw validation, no device - fails since dev required
740         run_cmd "$IP nexthop add id 52 via 2001:db8:92::3"
741         log_test $? 2 "Create nexthop - gw only"
742
743         # gw is not reachable throught given dev
744         run_cmd "$IP nexthop add id 53 via 2001:db8:3::3 dev veth1"
745         log_test $? 2 "Create nexthop - invalid gw+dev combination"
746
747         # onlink arg overrides gw+dev lookup
748         run_cmd "$IP nexthop add id 53 via 2001:db8:3::3 dev veth1 onlink"
749         log_test $? 0 "Create nexthop - gw+dev and onlink"
750
751         # admin down should delete nexthops
752         set -e
753         run_cmd "$IP -6 nexthop add id 55 via 2001:db8:91::3 dev veth1"
754         run_cmd "$IP nexthop add id 56 via 2001:db8:91::4 dev veth1"
755         run_cmd "$IP nexthop add id 57 via 2001:db8:91::5 dev veth1"
756         run_cmd "$IP li set dev veth1 down"
757         set +e
758         check_nexthop "dev veth1" ""
759         log_test $? 0 "Nexthops removed on admin down"
760 }
761
762 ipv6_grp_refs()
763 {
764         if [ ! -x "$(command -v mausezahn)" ]; then
765                 echo "SKIP: Could not run test; need mausezahn tool"
766                 return
767         fi
768
769         run_cmd "$IP link set dev veth1 up"
770         run_cmd "$IP link add veth1.10 link veth1 up type vlan id 10"
771         run_cmd "$IP link add veth1.20 link veth1 up type vlan id 20"
772         run_cmd "$IP -6 addr add 2001:db8:91::1/64 dev veth1.10"
773         run_cmd "$IP -6 addr add 2001:db8:92::1/64 dev veth1.20"
774         run_cmd "$IP -6 neigh add 2001:db8:91::2 lladdr 00:11:22:33:44:55 dev veth1.10"
775         run_cmd "$IP -6 neigh add 2001:db8:92::2 lladdr 00:11:22:33:44:55 dev veth1.20"
776         run_cmd "$IP nexthop add id 100 via 2001:db8:91::2 dev veth1.10"
777         run_cmd "$IP nexthop add id 101 via 2001:db8:92::2 dev veth1.20"
778         run_cmd "$IP nexthop add id 102 group 100"
779         run_cmd "$IP route add 2001:db8:101::1/128 nhid 102"
780
781         # create per-cpu dsts through nh 100
782         run_cmd "ip netns exec me mausezahn -6 veth1.10 -B 2001:db8:101::1 -A 2001:db8:91::1 -c 5 -t tcp "dp=1-1023, flags=syn" >/dev/null 2>&1"
783
784         # remove nh 100 from the group to delete the route potentially leaving
785         # a stale per-cpu dst which holds a reference to the nexthop's net
786         # device and to the IPv6 route
787         run_cmd "$IP nexthop replace id 102 group 101"
788         run_cmd "$IP route del 2001:db8:101::1/128"
789
790         # add both nexthops to the group so a reference is taken on them
791         run_cmd "$IP nexthop replace id 102 group 100/101"
792
793         # if the bug described in commit "net: nexthop: release IPv6 per-cpu
794         # dsts when replacing a nexthop group" exists at this point we have
795         # an unlinked IPv6 route (but not freed due to stale dst) with a
796         # reference over the group so we delete the group which will again
797         # only unlink it due to the route reference
798         run_cmd "$IP nexthop del id 102"
799
800         # delete the nexthop with stale dst, since we have an unlinked
801         # group with a ref to it and an unlinked IPv6 route with ref to the
802         # group, the nh will only be unlinked and not freed so the stale dst
803         # remains forever and we get a net device refcount imbalance
804         run_cmd "$IP nexthop del id 100"
805
806         # if a reference was lost this command will hang because the net device
807         # cannot be removed
808         timeout -s KILL 5 ip netns exec me ip link del veth1.10 >/dev/null 2>&1
809
810         # we can't cleanup if the command is hung trying to delete the netdev
811         if [ $? -eq 137 ]; then
812                 return 1
813         fi
814
815         # cleanup
816         run_cmd "$IP link del veth1.20"
817         run_cmd "$IP nexthop flush"
818
819         return 0
820 }
821
822 ipv6_grp_fcnal()
823 {
824         local rc
825
826         echo
827         echo "IPv6 groups functional"
828         echo "----------------------"
829
830         # basic functionality: create a nexthop group, default weight
831         run_cmd "$IP nexthop add id 61 via 2001:db8:91::2 dev veth1"
832         run_cmd "$IP nexthop add id 101 group 61"
833         log_test $? 0 "Create nexthop group with single nexthop"
834
835         # get nexthop group
836         run_cmd "$IP nexthop get id 101"
837         log_test $? 0 "Get nexthop group by id"
838         check_nexthop "id 101" "id 101 group 61"
839
840         # delete nexthop group
841         run_cmd "$IP nexthop del id 101"
842         log_test $? 0 "Delete nexthop group by id"
843         check_nexthop "id 101" ""
844
845         $IP nexthop flush >/dev/null 2>&1
846         check_nexthop "id 101" ""
847
848         #
849         # create group with multiple nexthops - mix of gw and dev only
850         #
851         run_cmd "$IP nexthop add id 62 via 2001:db8:91::2 dev veth1"
852         run_cmd "$IP nexthop add id 63 via 2001:db8:91::3 dev veth1"
853         run_cmd "$IP nexthop add id 64 via 2001:db8:91::4 dev veth1"
854         run_cmd "$IP nexthop add id 65 dev veth1"
855         run_cmd "$IP nexthop add id 102 group 62/63/64/65"
856         log_test $? 0 "Nexthop group with multiple nexthops"
857         check_nexthop "id 102" "id 102 group 62/63/64/65"
858
859         # Delete nexthop in a group and group is updated
860         run_cmd "$IP nexthop del id 63"
861         check_nexthop "id 102" "id 102 group 62/64/65"
862         log_test $? 0 "Nexthop group updated when entry is deleted"
863
864         # create group with multiple weighted nexthops
865         run_cmd "$IP nexthop add id 63 via 2001:db8:91::3 dev veth1"
866         run_cmd "$IP nexthop add id 103 group 62/63,2/64,3/65,4"
867         log_test $? 0 "Nexthop group with weighted nexthops"
868         check_nexthop "id 103" "id 103 group 62/63,2/64,3/65,4"
869
870         # Delete nexthop in a weighted group and group is updated
871         run_cmd "$IP nexthop del id 63"
872         check_nexthop "id 103" "id 103 group 62/64,3/65,4"
873         log_test $? 0 "Weighted nexthop group updated when entry is deleted"
874
875         # admin down - nexthop is removed from group
876         run_cmd "$IP li set dev veth1 down"
877         check_nexthop "dev veth1" ""
878         log_test $? 0 "Nexthops in groups removed on admin down"
879
880         # expect groups to have been deleted as well
881         check_nexthop "" ""
882
883         run_cmd "$IP li set dev veth1 up"
884
885         $IP nexthop flush >/dev/null 2>&1
886
887         # group with nexthops using different devices
888         set -e
889         run_cmd "$IP nexthop add id 62 via 2001:db8:91::2 dev veth1"
890         run_cmd "$IP nexthop add id 63 via 2001:db8:91::3 dev veth1"
891         run_cmd "$IP nexthop add id 64 via 2001:db8:91::4 dev veth1"
892         run_cmd "$IP nexthop add id 65 via 2001:db8:91::5 dev veth1"
893
894         run_cmd "$IP nexthop add id 72 via 2001:db8:92::2 dev veth3"
895         run_cmd "$IP nexthop add id 73 via 2001:db8:92::3 dev veth3"
896         run_cmd "$IP nexthop add id 74 via 2001:db8:92::4 dev veth3"
897         run_cmd "$IP nexthop add id 75 via 2001:db8:92::5 dev veth3"
898         set +e
899
900         # multiple groups with same nexthop
901         run_cmd "$IP nexthop add id 104 group 62"
902         run_cmd "$IP nexthop add id 105 group 62"
903         check_nexthop "group" "id 104 group 62 id 105 group 62"
904         log_test $? 0 "Multiple groups with same nexthop"
905
906         run_cmd "$IP nexthop flush groups"
907         [ $? -ne 0 ] && return 1
908
909         # on admin down of veth1, it should be removed from the group
910         run_cmd "$IP nexthop add id 105 group 62/63/72/73/64"
911         run_cmd "$IP li set veth1 down"
912         check_nexthop "id 105" "id 105 group 72/73"
913         log_test $? 0 "Nexthops in group removed on admin down - mixed group"
914
915         run_cmd "$IP nexthop add id 106 group 105/74"
916         log_test $? 2 "Nexthop group can not have a group as an entry"
917
918         # a group can have a blackhole entry only if it is the only
919         # nexthop in the group. Needed for atomic replace with an
920         # actual nexthop group
921         run_cmd "$IP -6 nexthop add id 31 blackhole"
922         run_cmd "$IP nexthop add id 107 group 31"
923         log_test $? 0 "Nexthop group with a blackhole entry"
924
925         run_cmd "$IP nexthop add id 108 group 31/24"
926         log_test $? 2 "Nexthop group can not have a blackhole and another nexthop"
927
928         ipv6_grp_refs
929         log_test $? 0 "Nexthop group replace refcounts"
930 }
931
932 ipv6_res_grp_fcnal()
933 {
934         local rc
935
936         echo
937         echo "IPv6 resilient groups functional"
938         echo "--------------------------------"
939
940         check_nexthop_res_support
941         if [ $? -eq $ksft_skip ]; then
942                 return $ksft_skip
943         fi
944
945         #
946         # migration of nexthop buckets - equal weights
947         #
948         run_cmd "$IP nexthop add id 62 via 2001:db8:91::2 dev veth1"
949         run_cmd "$IP nexthop add id 63 via 2001:db8:91::3 dev veth1"
950         run_cmd "$IP nexthop add id 102 group 62/63 type resilient buckets 2 idle_timer 0"
951
952         run_cmd "$IP nexthop del id 63"
953         check_nexthop "id 102" \
954                 "id 102 group 62 type resilient buckets 2 idle_timer 0 unbalanced_timer 0 unbalanced_time 0"
955         log_test $? 0 "Nexthop group updated when entry is deleted"
956         check_nexthop_bucket "list id 102" \
957                 "id 102 index 0 nhid 62 id 102 index 1 nhid 62"
958         log_test $? 0 "Nexthop buckets updated when entry is deleted"
959
960         run_cmd "$IP nexthop add id 63 via 2001:db8:91::3 dev veth1"
961         run_cmd "$IP nexthop replace id 102 group 62/63 type resilient buckets 2 idle_timer 0"
962         check_nexthop "id 102" \
963                 "id 102 group 62/63 type resilient buckets 2 idle_timer 0 unbalanced_timer 0 unbalanced_time 0"
964         log_test $? 0 "Nexthop group updated after replace"
965         check_nexthop_bucket "list id 102" \
966                 "id 102 index 0 nhid 63 id 102 index 1 nhid 62"
967         log_test $? 0 "Nexthop buckets updated after replace"
968
969         $IP nexthop flush >/dev/null 2>&1
970
971         #
972         # migration of nexthop buckets - unequal weights
973         #
974         run_cmd "$IP nexthop add id 62 via 2001:db8:91::2 dev veth1"
975         run_cmd "$IP nexthop add id 63 via 2001:db8:91::3 dev veth1"
976         run_cmd "$IP nexthop add id 102 group 62,3/63,1 type resilient buckets 4 idle_timer 0"
977
978         run_cmd "$IP nexthop del id 63"
979         check_nexthop "id 102" \
980                 "id 102 group 62,3 type resilient buckets 4 idle_timer 0 unbalanced_timer 0 unbalanced_time 0"
981         log_test $? 0 "Nexthop group updated when entry is deleted - nECMP"
982         check_nexthop_bucket "list id 102" \
983                 "id 102 index 0 nhid 62 id 102 index 1 nhid 62 id 102 index 2 nhid 62 id 102 index 3 nhid 62"
984         log_test $? 0 "Nexthop buckets updated when entry is deleted - nECMP"
985
986         run_cmd "$IP nexthop add id 63 via 2001:db8:91::3 dev veth1"
987         run_cmd "$IP nexthop replace id 102 group 62,3/63,1 type resilient buckets 4 idle_timer 0"
988         check_nexthop "id 102" \
989                 "id 102 group 62,3/63 type resilient buckets 4 idle_timer 0 unbalanced_timer 0 unbalanced_time 0"
990         log_test $? 0 "Nexthop group updated after replace - nECMP"
991         check_nexthop_bucket "list id 102" \
992                 "id 102 index 0 nhid 63 id 102 index 1 nhid 62 id 102 index 2 nhid 62 id 102 index 3 nhid 62"
993         log_test $? 0 "Nexthop buckets updated after replace - nECMP"
994 }
995
996 ipv6_fcnal_runtime()
997 {
998         local rc
999
1000         echo
1001         echo "IPv6 functional runtime"
1002         echo "-----------------------"
1003
1004         #
1005         # IPv6 - the basics
1006         #
1007         run_cmd "$IP nexthop add id 81 via 2001:db8:91::2 dev veth1"
1008         run_cmd "$IP ro add 2001:db8:101::1/128 nhid 81"
1009         log_test $? 0 "Route add"
1010
1011         run_cmd "$IP ro delete 2001:db8:101::1/128 nhid 81"
1012         log_test $? 0 "Route delete"
1013
1014         run_cmd "$IP ro add 2001:db8:101::1/128 nhid 81"
1015         run_cmd "ip netns exec me ping -c1 -w$PING_TIMEOUT 2001:db8:101::1"
1016         log_test $? 0 "Ping with nexthop"
1017
1018         run_cmd "$IP nexthop add id 82 via 2001:db8:92::2 dev veth3"
1019         run_cmd "$IP nexthop add id 122 group 81/82"
1020         run_cmd "$IP ro replace 2001:db8:101::1/128 nhid 122"
1021         run_cmd "ip netns exec me ping -c1 -w$PING_TIMEOUT 2001:db8:101::1"
1022         log_test $? 0 "Ping - multipath"
1023
1024         #
1025         # IPv6 with blackhole nexthops
1026         #
1027         run_cmd "$IP -6 nexthop add id 83 blackhole"
1028         run_cmd "$IP ro replace 2001:db8:101::1/128 nhid 83"
1029         run_cmd "ip netns exec me ping -c1 -w$PING_TIMEOUT 2001:db8:101::1"
1030         log_test $? 2 "Ping - blackhole"
1031
1032         run_cmd "$IP nexthop replace id 83 via 2001:db8:91::2 dev veth1"
1033         run_cmd "ip netns exec me ping -c1 -w$PING_TIMEOUT 2001:db8:101::1"
1034         log_test $? 0 "Ping - blackhole replaced with gateway"
1035
1036         run_cmd "$IP -6 nexthop replace id 83 blackhole"
1037         run_cmd "ip netns exec me ping -c1 -w$PING_TIMEOUT 2001:db8:101::1"
1038         log_test $? 2 "Ping - gateway replaced by blackhole"
1039
1040         run_cmd "$IP ro replace 2001:db8:101::1/128 nhid 122"
1041         run_cmd "ip netns exec me ping -c1 -w$PING_TIMEOUT 2001:db8:101::1"
1042         if [ $? -eq 0 ]; then
1043                 run_cmd "$IP nexthop replace id 122 group 83"
1044                 run_cmd "ip netns exec me ping -c1 -w$PING_TIMEOUT 2001:db8:101::1"
1045                 log_test $? 2 "Ping - group with blackhole"
1046
1047                 run_cmd "$IP nexthop replace id 122 group 81/82"
1048                 run_cmd "ip netns exec me ping -c1 -w$PING_TIMEOUT 2001:db8:101::1"
1049                 log_test $? 0 "Ping - group blackhole replaced with gateways"
1050         else
1051                 log_test 2 0 "Ping - multipath failed"
1052         fi
1053
1054         #
1055         # device only and gw + dev only mix
1056         #
1057         run_cmd "$IP -6 nexthop add id 85 dev veth1"
1058         run_cmd "$IP ro replace 2001:db8:101::1/128 nhid 85"
1059         log_test $? 0 "IPv6 route with device only nexthop"
1060         check_route6 "2001:db8:101::1" "2001:db8:101::1 nhid 85 dev veth1 metric 1024"
1061
1062         run_cmd "$IP nexthop add id 123 group 81/85"
1063         run_cmd "$IP ro replace 2001:db8:101::1/128 nhid 123"
1064         log_test $? 0 "IPv6 multipath route with nexthop mix - dev only + gw"
1065         check_route6 "2001:db8:101::1" "2001:db8:101::1 nhid 123 metric 1024 nexthop via 2001:db8:91::2 dev veth1 weight 1 nexthop dev veth1 weight 1"
1066
1067         #
1068         # IPv6 route with v4 nexthop - not allowed
1069         #
1070         run_cmd "$IP ro delete 2001:db8:101::1/128"
1071         run_cmd "$IP nexthop add id 84 via 172.16.1.1 dev veth1"
1072         run_cmd "$IP ro add 2001:db8:101::1/128 nhid 84"
1073         log_test $? 2 "IPv6 route can not have a v4 gateway"
1074
1075         run_cmd "$IP ro replace 2001:db8:101::1/128 nhid 81"
1076         run_cmd "$IP nexthop replace id 81 via 172.16.1.1 dev veth1"
1077         log_test $? 2 "Nexthop replace - v6 route, v4 nexthop"
1078
1079         run_cmd "$IP ro replace 2001:db8:101::1/128 nhid 122"
1080         run_cmd "$IP nexthop replace id 81 via 172.16.1.1 dev veth1"
1081         log_test $? 2 "Nexthop replace of group entry - v6 route, v4 nexthop"
1082
1083         run_cmd "$IP nexthop add id 86 via 2001:db8:92::2 dev veth3"
1084         run_cmd "$IP nexthop add id 87 via 172.16.1.1 dev veth1"
1085         run_cmd "$IP nexthop add id 88 via 172.16.1.1 dev veth1"
1086         run_cmd "$IP nexthop add id 124 group 86/87/88"
1087         run_cmd "$IP ro replace 2001:db8:101::1/128 nhid 124"
1088         log_test $? 2 "IPv6 route can not have a group with v4 and v6 gateways"
1089
1090         run_cmd "$IP nexthop del id 88"
1091         run_cmd "$IP ro replace 2001:db8:101::1/128 nhid 124"
1092         log_test $? 2 "IPv6 route can not have a group with v4 and v6 gateways"
1093
1094         run_cmd "$IP nexthop del id 87"
1095         run_cmd "$IP ro replace 2001:db8:101::1/128 nhid 124"
1096         log_test $? 0 "IPv6 route using a group after removing v4 gateways"
1097
1098         run_cmd "$IP ro delete 2001:db8:101::1/128"
1099         run_cmd "$IP nexthop add id 87 via 172.16.1.1 dev veth1"
1100         run_cmd "$IP nexthop add id 88 via 172.16.1.1 dev veth1"
1101         run_cmd "$IP nexthop replace id 124 group 86/87/88"
1102         run_cmd "$IP ro replace 2001:db8:101::1/128 nhid 124"
1103         log_test $? 2 "IPv6 route can not have a group with v4 and v6 gateways"
1104
1105         run_cmd "$IP nexthop replace id 88 via 2001:db8:92::2 dev veth3"
1106         run_cmd "$IP ro replace 2001:db8:101::1/128 nhid 124"
1107         log_test $? 2 "IPv6 route can not have a group with v4 and v6 gateways"
1108
1109         run_cmd "$IP nexthop replace id 87 via 2001:db8:92::2 dev veth3"
1110         run_cmd "$IP ro replace 2001:db8:101::1/128 nhid 124"
1111         log_test $? 0 "IPv6 route using a group after replacing v4 gateways"
1112
1113         $IP nexthop flush >/dev/null 2>&1
1114
1115         #
1116         # weird IPv6 cases
1117         #
1118         run_cmd "$IP nexthop add id 86 via 2001:db8:91::2 dev veth1"
1119         run_cmd "$IP ro add 2001:db8:101::1/128 nhid 81"
1120
1121         # route can not use prefsrc with nexthops
1122         run_cmd "$IP ro add 2001:db8:101::2/128 nhid 86 from 2001:db8:91::1"
1123         log_test $? 2 "IPv6 route can not use src routing with external nexthop"
1124
1125         # check cleanup path on invalid metric
1126         run_cmd "$IP ro add 2001:db8:101::2/128 nhid 86 congctl lock foo"
1127         log_test $? 2 "IPv6 route with invalid metric"
1128
1129         # rpfilter and default route
1130         $IP nexthop flush >/dev/null 2>&1
1131         run_cmd "ip netns exec me ip6tables -t mangle -I PREROUTING 1 -m rpfilter --invert -j DROP"
1132         run_cmd "$IP nexthop add id 91 via 2001:db8:91::2 dev veth1"
1133         run_cmd "$IP nexthop add id 92 via 2001:db8:92::2 dev veth3"
1134         run_cmd "$IP nexthop add id 93 group 91/92"
1135         run_cmd "$IP -6 ro add default nhid 91"
1136         run_cmd "ip netns exec me ping -c1 -w$PING_TIMEOUT 2001:db8:101::1"
1137         log_test $? 0 "Nexthop with default route and rpfilter"
1138         run_cmd "$IP -6 ro replace default nhid 93"
1139         run_cmd "ip netns exec me ping -c1 -w$PING_TIMEOUT 2001:db8:101::1"
1140         log_test $? 0 "Nexthop with multipath default route and rpfilter"
1141
1142         # TO-DO:
1143         # existing route with old nexthop; append route with new nexthop
1144         # existing route with old nexthop; replace route with new
1145         # existing route with new nexthop; replace route with old
1146         # route with src address and using nexthop - not allowed
1147 }
1148
1149 ipv6_large_grp()
1150 {
1151         local ecmp=32
1152
1153         echo
1154         echo "IPv6 large groups (x$ecmp)"
1155         echo "---------------------"
1156
1157         check_large_grp 6 $ecmp
1158
1159         $IP nexthop flush >/dev/null 2>&1
1160 }
1161
1162 ipv6_large_res_grp()
1163 {
1164         echo
1165         echo "IPv6 large resilient group (128k buckets)"
1166         echo "-----------------------------------------"
1167
1168         check_nexthop_res_support
1169         if [ $? -eq $ksft_skip ]; then
1170                 return $ksft_skip
1171         fi
1172
1173         check_large_res_grp 6 $((128 * 1024))
1174
1175         $IP nexthop flush >/dev/null 2>&1
1176 }
1177
1178 ipv6_del_add_loop1()
1179 {
1180         while :; do
1181                 $IP nexthop del id 100
1182                 $IP nexthop add id 100 via 2001:db8:91::2 dev veth1
1183         done >/dev/null 2>&1
1184 }
1185
1186 ipv6_grp_replace_loop()
1187 {
1188         while :; do
1189                 $IP nexthop replace id 102 group 100/101
1190         done >/dev/null 2>&1
1191 }
1192
1193 ipv6_torture()
1194 {
1195         local pid1
1196         local pid2
1197         local pid3
1198         local pid4
1199         local pid5
1200
1201         echo
1202         echo "IPv6 runtime torture"
1203         echo "--------------------"
1204         if [ ! -x "$(command -v mausezahn)" ]; then
1205                 echo "SKIP: Could not run test; need mausezahn tool"
1206                 return
1207         fi
1208
1209         run_cmd "$IP nexthop add id 100 via 2001:db8:91::2 dev veth1"
1210         run_cmd "$IP nexthop add id 101 via 2001:db8:92::2 dev veth3"
1211         run_cmd "$IP nexthop add id 102 group 100/101"
1212         run_cmd "$IP route add 2001:db8:101::1 nhid 102"
1213         run_cmd "$IP route add 2001:db8:101::2 nhid 102"
1214
1215         ipv6_del_add_loop1 &
1216         pid1=$!
1217         ipv6_grp_replace_loop &
1218         pid2=$!
1219         ip netns exec me ping -f 2001:db8:101::1 >/dev/null 2>&1 &
1220         pid3=$!
1221         ip netns exec me ping -f 2001:db8:101::2 >/dev/null 2>&1 &
1222         pid4=$!
1223         ip netns exec me mausezahn -6 veth1 -B 2001:db8:101::2 -A 2001:db8:91::1 -c 0 -t tcp "dp=1-1023, flags=syn" >/dev/null 2>&1 &
1224         pid5=$!
1225
1226         sleep 300
1227         kill -9 $pid1 $pid2 $pid3 $pid4 $pid5
1228         wait $pid1 $pid2 $pid3 $pid4 $pid5 2>/dev/null
1229
1230         # if we did not crash, success
1231         log_test 0 0 "IPv6 torture test"
1232 }
1233
1234 ipv6_res_grp_replace_loop()
1235 {
1236         while :; do
1237                 $IP nexthop replace id 102 group 100/101 type resilient
1238         done >/dev/null 2>&1
1239 }
1240
1241 ipv6_res_torture()
1242 {
1243         local pid1
1244         local pid2
1245         local pid3
1246         local pid4
1247         local pid5
1248
1249         echo
1250         echo "IPv6 runtime resilient nexthop group torture"
1251         echo "--------------------------------------------"
1252
1253         check_nexthop_res_support
1254         if [ $? -eq $ksft_skip ]; then
1255                 return $ksft_skip
1256         fi
1257
1258         if [ ! -x "$(command -v mausezahn)" ]; then
1259                 echo "SKIP: Could not run test; need mausezahn tool"
1260                 return
1261         fi
1262
1263         run_cmd "$IP nexthop add id 100 via 2001:db8:91::2 dev veth1"
1264         run_cmd "$IP nexthop add id 101 via 2001:db8:92::2 dev veth3"
1265         run_cmd "$IP nexthop add id 102 group 100/101 type resilient buckets 512 idle_timer 0"
1266         run_cmd "$IP route add 2001:db8:101::1 nhid 102"
1267         run_cmd "$IP route add 2001:db8:101::2 nhid 102"
1268
1269         ipv6_del_add_loop1 &
1270         pid1=$!
1271         ipv6_res_grp_replace_loop &
1272         pid2=$!
1273         ip netns exec me ping -f 2001:db8:101::1 >/dev/null 2>&1 &
1274         pid3=$!
1275         ip netns exec me ping -f 2001:db8:101::2 >/dev/null 2>&1 &
1276         pid4=$!
1277         ip netns exec me mausezahn -6 veth1 \
1278                             -B 2001:db8:101::2 -A 2001:db8:91::1 -c 0 \
1279                             -t tcp "dp=1-1023, flags=syn" >/dev/null 2>&1 &
1280         pid5=$!
1281
1282         sleep 300
1283         kill -9 $pid1 $pid2 $pid3 $pid4 $pid5
1284         wait $pid1 $pid2 $pid3 $pid4 $pid5 2>/dev/null
1285
1286         # if we did not crash, success
1287         log_test 0 0 "IPv6 resilient nexthop group torture test"
1288 }
1289
1290 ipv4_fcnal()
1291 {
1292         local rc
1293
1294         echo
1295         echo "IPv4 functional"
1296         echo "----------------------"
1297
1298         #
1299         # basic IPv4 ops - add, get, delete
1300         #
1301         run_cmd "$IP nexthop add id 12 via 172.16.1.2 dev veth1"
1302         rc=$?
1303         log_test $rc 0 "Create nexthop with id, gw, dev"
1304         if [ $rc -ne 0 ]; then
1305                 echo "Basic IPv4 create fails; can not continue"
1306                 return 1
1307         fi
1308
1309         run_cmd "$IP nexthop get id 12"
1310         log_test $? 0 "Get nexthop by id"
1311         check_nexthop "id 12" "id 12 via 172.16.1.2 dev veth1 scope link"
1312
1313         run_cmd "$IP nexthop del id 12"
1314         log_test $? 0 "Delete nexthop by id"
1315         check_nexthop "id 52" ""
1316
1317         #
1318         # gw, device spec
1319         #
1320         # gw validation, no device - fails since dev is required
1321         run_cmd "$IP nexthop add id 12 via 172.16.2.3"
1322         log_test $? 2 "Create nexthop - gw only"
1323
1324         # gw not reachable through given dev
1325         run_cmd "$IP nexthop add id 13 via 172.16.3.2 dev veth1"
1326         log_test $? 2 "Create nexthop - invalid gw+dev combination"
1327
1328         # onlink flag overrides gw+dev lookup
1329         run_cmd "$IP nexthop add id 13 via 172.16.3.2 dev veth1 onlink"
1330         log_test $? 0 "Create nexthop - gw+dev and onlink"
1331
1332         # admin down should delete nexthops
1333         set -e
1334         run_cmd "$IP nexthop add id 15 via 172.16.1.3 dev veth1"
1335         run_cmd "$IP nexthop add id 16 via 172.16.1.4 dev veth1"
1336         run_cmd "$IP nexthop add id 17 via 172.16.1.5 dev veth1"
1337         run_cmd "$IP li set dev veth1 down"
1338         set +e
1339         check_nexthop "dev veth1" ""
1340         log_test $? 0 "Nexthops removed on admin down"
1341
1342         # nexthop route delete warning: route add with nhid and delete
1343         # using device
1344         run_cmd "$IP li set dev veth1 up"
1345         run_cmd "$IP nexthop add id 12 via 172.16.1.3 dev veth1"
1346         out1=`dmesg | grep "WARNING:.*fib_nh_match.*" | wc -l`
1347         run_cmd "$IP route add 172.16.101.1/32 nhid 12"
1348         run_cmd "$IP route delete 172.16.101.1/32 dev veth1"
1349         out2=`dmesg | grep "WARNING:.*fib_nh_match.*" | wc -l`
1350         [ $out1 -eq $out2 ]
1351         rc=$?
1352         log_test $rc 0 "Delete nexthop route warning"
1353         run_cmd "$IP route delete 172.16.101.1/32 nhid 12"
1354         run_cmd "$IP nexthop del id 12"
1355
1356         run_cmd "$IP nexthop add id 21 via 172.16.1.6 dev veth1"
1357         run_cmd "$IP ro add 172.16.101.0/24 nhid 21"
1358         run_cmd "$IP ro del 172.16.101.0/24 nexthop via 172.16.1.7 dev veth1 nexthop via 172.16.1.8 dev veth1"
1359         log_test $? 2 "Delete multipath route with only nh id based entry"
1360
1361         run_cmd "$IP nexthop add id 22 via 172.16.1.6 dev veth1"
1362         run_cmd "$IP ro add 172.16.102.0/24 nhid 22"
1363         run_cmd "$IP ro del 172.16.102.0/24 dev veth1"
1364         log_test $? 2 "Delete route when specifying only nexthop device"
1365
1366         run_cmd "$IP ro del 172.16.102.0/24 via 172.16.1.6"
1367         log_test $? 2 "Delete route when specifying only gateway"
1368
1369         run_cmd "$IP ro del 172.16.102.0/24"
1370         log_test $? 0 "Delete route when not specifying nexthop attributes"
1371 }
1372
1373 ipv4_grp_fcnal()
1374 {
1375         local rc
1376
1377         echo
1378         echo "IPv4 groups functional"
1379         echo "----------------------"
1380
1381         # basic functionality: create a nexthop group, default weight
1382         run_cmd "$IP nexthop add id 11 via 172.16.1.2 dev veth1"
1383         run_cmd "$IP nexthop add id 101 group 11"
1384         log_test $? 0 "Create nexthop group with single nexthop"
1385
1386         # get nexthop group
1387         run_cmd "$IP nexthop get id 101"
1388         log_test $? 0 "Get nexthop group by id"
1389         check_nexthop "id 101" "id 101 group 11"
1390
1391         # delete nexthop group
1392         run_cmd "$IP nexthop del id 101"
1393         log_test $? 0 "Delete nexthop group by id"
1394         check_nexthop "id 101" ""
1395
1396         $IP nexthop flush >/dev/null 2>&1
1397
1398         #
1399         # create group with multiple nexthops
1400         run_cmd "$IP nexthop add id 12 via 172.16.1.2 dev veth1"
1401         run_cmd "$IP nexthop add id 13 via 172.16.1.3 dev veth1"
1402         run_cmd "$IP nexthop add id 14 via 172.16.1.4 dev veth1"
1403         run_cmd "$IP nexthop add id 15 via 172.16.1.5 dev veth1"
1404         run_cmd "$IP nexthop add id 102 group 12/13/14/15"
1405         log_test $? 0 "Nexthop group with multiple nexthops"
1406         check_nexthop "id 102" "id 102 group 12/13/14/15"
1407
1408         # Delete nexthop in a group and group is updated
1409         run_cmd "$IP nexthop del id 13"
1410         check_nexthop "id 102" "id 102 group 12/14/15"
1411         log_test $? 0 "Nexthop group updated when entry is deleted"
1412
1413         # create group with multiple weighted nexthops
1414         run_cmd "$IP nexthop add id 13 via 172.16.1.3 dev veth1"
1415         run_cmd "$IP nexthop add id 103 group 12/13,2/14,3/15,4"
1416         log_test $? 0 "Nexthop group with weighted nexthops"
1417         check_nexthop "id 103" "id 103 group 12/13,2/14,3/15,4"
1418
1419         # Delete nexthop in a weighted group and group is updated
1420         run_cmd "$IP nexthop del id 13"
1421         check_nexthop "id 103" "id 103 group 12/14,3/15,4"
1422         log_test $? 0 "Weighted nexthop group updated when entry is deleted"
1423
1424         # admin down - nexthop is removed from group
1425         run_cmd "$IP li set dev veth1 down"
1426         check_nexthop "dev veth1" ""
1427         log_test $? 0 "Nexthops in groups removed on admin down"
1428
1429         # expect groups to have been deleted as well
1430         check_nexthop "" ""
1431
1432         run_cmd "$IP li set dev veth1 up"
1433
1434         $IP nexthop flush >/dev/null 2>&1
1435
1436         # group with nexthops using different devices
1437         set -e
1438         run_cmd "$IP nexthop add id 12 via 172.16.1.2 dev veth1"
1439         run_cmd "$IP nexthop add id 13 via 172.16.1.3 dev veth1"
1440         run_cmd "$IP nexthop add id 14 via 172.16.1.4 dev veth1"
1441         run_cmd "$IP nexthop add id 15 via 172.16.1.5 dev veth1"
1442
1443         run_cmd "$IP nexthop add id 22 via 172.16.2.2 dev veth3"
1444         run_cmd "$IP nexthop add id 23 via 172.16.2.3 dev veth3"
1445         run_cmd "$IP nexthop add id 24 via 172.16.2.4 dev veth3"
1446         run_cmd "$IP nexthop add id 25 via 172.16.2.5 dev veth3"
1447         set +e
1448
1449         # multiple groups with same nexthop
1450         run_cmd "$IP nexthop add id 104 group 12"
1451         run_cmd "$IP nexthop add id 105 group 12"
1452         check_nexthop "group" "id 104 group 12 id 105 group 12"
1453         log_test $? 0 "Multiple groups with same nexthop"
1454
1455         run_cmd "$IP nexthop flush groups"
1456         [ $? -ne 0 ] && return 1
1457
1458         # on admin down of veth1, it should be removed from the group
1459         run_cmd "$IP nexthop add id 105 group 12/13/22/23/14"
1460         run_cmd "$IP li set veth1 down"
1461         check_nexthop "id 105" "id 105 group 22/23"
1462         log_test $? 0 "Nexthops in group removed on admin down - mixed group"
1463
1464         run_cmd "$IP nexthop add id 106 group 105/24"
1465         log_test $? 2 "Nexthop group can not have a group as an entry"
1466
1467         # a group can have a blackhole entry only if it is the only
1468         # nexthop in the group. Needed for atomic replace with an
1469         # actual nexthop group
1470         run_cmd "$IP nexthop add id 31 blackhole"
1471         run_cmd "$IP nexthop add id 107 group 31"
1472         log_test $? 0 "Nexthop group with a blackhole entry"
1473
1474         run_cmd "$IP nexthop add id 108 group 31/24"
1475         log_test $? 2 "Nexthop group can not have a blackhole and another nexthop"
1476 }
1477
1478 ipv4_res_grp_fcnal()
1479 {
1480         local rc
1481
1482         echo
1483         echo "IPv4 resilient groups functional"
1484         echo "--------------------------------"
1485
1486         check_nexthop_res_support
1487         if [ $? -eq $ksft_skip ]; then
1488                 return $ksft_skip
1489         fi
1490
1491         #
1492         # migration of nexthop buckets - equal weights
1493         #
1494         run_cmd "$IP nexthop add id 12 via 172.16.1.2 dev veth1"
1495         run_cmd "$IP nexthop add id 13 via 172.16.1.3 dev veth1"
1496         run_cmd "$IP nexthop add id 102 group 12/13 type resilient buckets 2 idle_timer 0"
1497
1498         run_cmd "$IP nexthop del id 13"
1499         check_nexthop "id 102" \
1500                 "id 102 group 12 type resilient buckets 2 idle_timer 0 unbalanced_timer 0 unbalanced_time 0"
1501         log_test $? 0 "Nexthop group updated when entry is deleted"
1502         check_nexthop_bucket "list id 102" \
1503                 "id 102 index 0 nhid 12 id 102 index 1 nhid 12"
1504         log_test $? 0 "Nexthop buckets updated when entry is deleted"
1505
1506         run_cmd "$IP nexthop add id 13 via 172.16.1.3 dev veth1"
1507         run_cmd "$IP nexthop replace id 102 group 12/13 type resilient buckets 2 idle_timer 0"
1508         check_nexthop "id 102" \
1509                 "id 102 group 12/13 type resilient buckets 2 idle_timer 0 unbalanced_timer 0 unbalanced_time 0"
1510         log_test $? 0 "Nexthop group updated after replace"
1511         check_nexthop_bucket "list id 102" \
1512                 "id 102 index 0 nhid 13 id 102 index 1 nhid 12"
1513         log_test $? 0 "Nexthop buckets updated after replace"
1514
1515         $IP nexthop flush >/dev/null 2>&1
1516
1517         #
1518         # migration of nexthop buckets - unequal weights
1519         #
1520         run_cmd "$IP nexthop add id 12 via 172.16.1.2 dev veth1"
1521         run_cmd "$IP nexthop add id 13 via 172.16.1.3 dev veth1"
1522         run_cmd "$IP nexthop add id 102 group 12,3/13,1 type resilient buckets 4 idle_timer 0"
1523
1524         run_cmd "$IP nexthop del id 13"
1525         check_nexthop "id 102" \
1526                 "id 102 group 12,3 type resilient buckets 4 idle_timer 0 unbalanced_timer 0 unbalanced_time 0"
1527         log_test $? 0 "Nexthop group updated when entry is deleted - nECMP"
1528         check_nexthop_bucket "list id 102" \
1529                 "id 102 index 0 nhid 12 id 102 index 1 nhid 12 id 102 index 2 nhid 12 id 102 index 3 nhid 12"
1530         log_test $? 0 "Nexthop buckets updated when entry is deleted - nECMP"
1531
1532         run_cmd "$IP nexthop add id 13 via 172.16.1.3 dev veth1"
1533         run_cmd "$IP nexthop replace id 102 group 12,3/13,1 type resilient buckets 4 idle_timer 0"
1534         check_nexthop "id 102" \
1535                 "id 102 group 12,3/13 type resilient buckets 4 idle_timer 0 unbalanced_timer 0 unbalanced_time 0"
1536         log_test $? 0 "Nexthop group updated after replace - nECMP"
1537         check_nexthop_bucket "list id 102" \
1538                 "id 102 index 0 nhid 13 id 102 index 1 nhid 12 id 102 index 2 nhid 12 id 102 index 3 nhid 12"
1539         log_test $? 0 "Nexthop buckets updated after replace - nECMP"
1540 }
1541
1542 ipv4_withv6_fcnal()
1543 {
1544         local lladdr
1545
1546         set -e
1547         lladdr=$(get_linklocal veth2 peer)
1548         run_cmd "$IP nexthop add id 11 via ${lladdr} dev veth1"
1549         set +e
1550         run_cmd "$IP ro add 172.16.101.1/32 nhid 11"
1551         log_test $? 0 "IPv6 nexthop with IPv4 route"
1552         check_route "172.16.101.1" "172.16.101.1 nhid 11 via inet6 ${lladdr} dev veth1"
1553
1554         set -e
1555         run_cmd "$IP nexthop add id 12 via 172.16.1.2 dev veth1"
1556         run_cmd "$IP nexthop add id 101 group 11/12"
1557         set +e
1558         run_cmd "$IP ro replace 172.16.101.1/32 nhid 101"
1559         log_test $? 0 "IPv6 nexthop with IPv4 route"
1560
1561         check_route "172.16.101.1" "172.16.101.1 nhid 101 nexthop via inet6 ${lladdr} dev veth1 weight 1 nexthop via 172.16.1.2 dev veth1 weight 1"
1562
1563         run_cmd "$IP ro replace 172.16.101.1/32 via inet6 ${lladdr} dev veth1"
1564         log_test $? 0 "IPv4 route with IPv6 gateway"
1565         check_route "172.16.101.1" "172.16.101.1 via inet6 ${lladdr} dev veth1"
1566
1567         run_cmd "$IP ro replace 172.16.101.1/32 via inet6 2001:db8:50::1 dev veth1"
1568         log_test $? 2 "IPv4 route with invalid IPv6 gateway"
1569 }
1570
1571 ipv4_fcnal_runtime()
1572 {
1573         local lladdr
1574         local rc
1575
1576         echo
1577         echo "IPv4 functional runtime"
1578         echo "-----------------------"
1579
1580         run_cmd "$IP nexthop add id 21 via 172.16.1.2 dev veth1"
1581         run_cmd "$IP ro add 172.16.101.1/32 nhid 21"
1582         log_test $? 0 "Route add"
1583         check_route "172.16.101.1" "172.16.101.1 nhid 21 via 172.16.1.2 dev veth1"
1584
1585         run_cmd "$IP ro delete 172.16.101.1/32 nhid 21"
1586         log_test $? 0 "Route delete"
1587
1588         #
1589         # scope mismatch
1590         #
1591         run_cmd "$IP nexthop add id 22 via 172.16.1.2 dev veth1"
1592         run_cmd "$IP ro add 172.16.101.1/32 nhid 22 scope host"
1593         log_test $? 2 "Route add - scope conflict with nexthop"
1594
1595         run_cmd "$IP nexthop replace id 22 dev veth3"
1596         run_cmd "$IP ro add 172.16.101.1/32 nhid 22 scope host"
1597         run_cmd "$IP nexthop replace id 22 via 172.16.2.2 dev veth3"
1598         log_test $? 2 "Nexthop replace with invalid scope for existing route"
1599
1600         # check cleanup path on invalid metric
1601         run_cmd "$IP ro add 172.16.101.2/32 nhid 22 congctl lock foo"
1602         log_test $? 2 "IPv4 route with invalid metric"
1603
1604         #
1605         # add route with nexthop and check traffic
1606         #
1607         run_cmd "$IP nexthop replace id 21 via 172.16.1.2 dev veth1"
1608         run_cmd "$IP ro replace 172.16.101.1/32 nhid 21"
1609         run_cmd "ip netns exec me ping -c1 -w$PING_TIMEOUT 172.16.101.1"
1610         log_test $? 0 "Basic ping"
1611
1612         run_cmd "$IP nexthop replace id 22 via 172.16.2.2 dev veth3"
1613         run_cmd "$IP nexthop add id 122 group 21/22"
1614         run_cmd "$IP ro replace 172.16.101.1/32 nhid 122"
1615         run_cmd "ip netns exec me ping -c1 -w$PING_TIMEOUT 172.16.101.1"
1616         log_test $? 0 "Ping - multipath"
1617
1618         run_cmd "$IP ro delete 172.16.101.1/32 nhid 122"
1619
1620         #
1621         # multiple default routes
1622         # - tests fib_select_default
1623         run_cmd "$IP nexthop add id 501 via 172.16.1.2 dev veth1"
1624         run_cmd "$IP ro add default nhid 501"
1625         run_cmd "$IP ro add default via 172.16.1.3 dev veth1 metric 20"
1626         run_cmd "ip netns exec me ping -c1 -w$PING_TIMEOUT 172.16.101.1"
1627         log_test $? 0 "Ping - multiple default routes, nh first"
1628
1629         # flip the order
1630         run_cmd "$IP ro del default nhid 501"
1631         run_cmd "$IP ro del default via 172.16.1.3 dev veth1 metric 20"
1632         run_cmd "$IP ro add default via 172.16.1.2 dev veth1 metric 20"
1633         run_cmd "$IP nexthop replace id 501 via 172.16.1.3 dev veth1"
1634         run_cmd "$IP ro add default nhid 501 metric 20"
1635         run_cmd "ip netns exec me ping -c1 -w$PING_TIMEOUT 172.16.101.1"
1636         log_test $? 0 "Ping - multiple default routes, nh second"
1637
1638         run_cmd "$IP nexthop delete nhid 501"
1639         run_cmd "$IP ro del default"
1640
1641         #
1642         # IPv4 with blackhole nexthops
1643         #
1644         run_cmd "$IP nexthop add id 23 blackhole"
1645         run_cmd "$IP ro replace 172.16.101.1/32 nhid 23"
1646         run_cmd "ip netns exec me ping -c1 -w$PING_TIMEOUT 172.16.101.1"
1647         log_test $? 2 "Ping - blackhole"
1648
1649         run_cmd "$IP nexthop replace id 23 via 172.16.1.2 dev veth1"
1650         run_cmd "ip netns exec me ping -c1 -w$PING_TIMEOUT 172.16.101.1"
1651         log_test $? 0 "Ping - blackhole replaced with gateway"
1652
1653         run_cmd "$IP nexthop replace id 23 blackhole"
1654         run_cmd "ip netns exec me ping -c1 -w$PING_TIMEOUT 172.16.101.1"
1655         log_test $? 2 "Ping - gateway replaced by blackhole"
1656
1657         run_cmd "$IP ro replace 172.16.101.1/32 nhid 122"
1658         run_cmd "ip netns exec me ping -c1 -w$PING_TIMEOUT 172.16.101.1"
1659         if [ $? -eq 0 ]; then
1660                 run_cmd "$IP nexthop replace id 122 group 23"
1661                 run_cmd "ip netns exec me ping -c1 -w$PING_TIMEOUT 172.16.101.1"
1662                 log_test $? 2 "Ping - group with blackhole"
1663
1664                 run_cmd "$IP nexthop replace id 122 group 21/22"
1665                 run_cmd "ip netns exec me ping -c1 -w$PING_TIMEOUT 172.16.101.1"
1666                 log_test $? 0 "Ping - group blackhole replaced with gateways"
1667         else
1668                 log_test 2 0 "Ping - multipath failed"
1669         fi
1670
1671         #
1672         # device only and gw + dev only mix
1673         #
1674         run_cmd "$IP nexthop add id 85 dev veth1"
1675         run_cmd "$IP ro replace 172.16.101.1/32 nhid 85"
1676         log_test $? 0 "IPv4 route with device only nexthop"
1677         check_route "172.16.101.1" "172.16.101.1 nhid 85 dev veth1"
1678
1679         run_cmd "$IP nexthop add id 123 group 21/85"
1680         run_cmd "$IP ro replace 172.16.101.1/32 nhid 123"
1681         log_test $? 0 "IPv4 multipath route with nexthop mix - dev only + gw"
1682         check_route "172.16.101.1" "172.16.101.1 nhid 123 nexthop via 172.16.1.2 dev veth1 weight 1 nexthop dev veth1 weight 1"
1683
1684         #
1685         # IPv4 with IPv6
1686         #
1687         set -e
1688         lladdr=$(get_linklocal veth2 peer)
1689         run_cmd "$IP nexthop add id 24 via ${lladdr} dev veth1"
1690         set +e
1691         run_cmd "$IP ro replace 172.16.101.1/32 nhid 24"
1692         run_cmd "ip netns exec me ping -c1 -w$PING_TIMEOUT 172.16.101.1"
1693         log_test $? 0 "IPv6 nexthop with IPv4 route"
1694
1695         $IP neigh sh | grep -q "${lladdr} dev veth1"
1696         if [ $? -eq 1 ]; then
1697                 echo "    WARNING: Neigh entry missing for ${lladdr}"
1698                 $IP neigh sh | grep 'dev veth1'
1699         fi
1700
1701         $IP neigh sh | grep -q "172.16.101.1 dev eth1"
1702         if [ $? -eq 0 ]; then
1703                 echo "    WARNING: Neigh entry exists for 172.16.101.1"
1704                 $IP neigh sh | grep 'dev veth1'
1705         fi
1706
1707         set -e
1708         run_cmd "$IP nexthop add id 25 via 172.16.1.2 dev veth1"
1709         run_cmd "$IP nexthop add id 101 group 24/25"
1710         set +e
1711         run_cmd "$IP ro replace 172.16.101.1/32 nhid 101"
1712         log_test $? 0 "IPv4 route with mixed v4-v6 multipath route"
1713
1714         check_route "172.16.101.1" "172.16.101.1 nhid 101 nexthop via inet6 ${lladdr} dev veth1 weight 1 nexthop via 172.16.1.2 dev veth1 weight 1"
1715
1716         run_cmd "ip netns exec me ping -c1 -w$PING_TIMEOUT 172.16.101.1"
1717         log_test $? 0 "IPv6 nexthop with IPv4 route"
1718
1719         run_cmd "$IP ro replace 172.16.101.1/32 via inet6 ${lladdr} dev veth1"
1720         run_cmd "ip netns exec me ping -c1 -w$PING_TIMEOUT 172.16.101.1"
1721         log_test $? 0 "IPv4 route with IPv6 gateway"
1722
1723         $IP neigh sh | grep -q "${lladdr} dev veth1"
1724         if [ $? -eq 1 ]; then
1725                 echo "    WARNING: Neigh entry missing for ${lladdr}"
1726                 $IP neigh sh | grep 'dev veth1'
1727         fi
1728
1729         $IP neigh sh | grep -q "172.16.101.1 dev eth1"
1730         if [ $? -eq 0 ]; then
1731                 echo "    WARNING: Neigh entry exists for 172.16.101.1"
1732                 $IP neigh sh | grep 'dev veth1'
1733         fi
1734
1735         run_cmd "$IP ro del 172.16.101.1/32 via inet6 ${lladdr} dev veth1"
1736         run_cmd "$IP -4 ro add default via inet6 ${lladdr} dev veth1"
1737         run_cmd "ip netns exec me ping -c1 -w$PING_TIMEOUT 172.16.101.1"
1738         log_test $? 0 "IPv4 default route with IPv6 gateway"
1739
1740         #
1741         # MPLS as an example of LWT encap
1742         #
1743         run_cmd "$IP nexthop add id 51 encap mpls 101 via 172.16.1.2 dev veth1"
1744         log_test $? 0 "IPv4 route with MPLS encap"
1745         check_nexthop "id 51" "id 51 encap mpls 101 via 172.16.1.2 dev veth1 scope link"
1746         log_test $? 0 "IPv4 route with MPLS encap - check"
1747
1748         run_cmd "$IP nexthop add id 52 encap mpls 102 via inet6 2001:db8:91::2 dev veth1"
1749         log_test $? 0 "IPv4 route with MPLS encap and v6 gateway"
1750         check_nexthop "id 52" "id 52 encap mpls 102 via 2001:db8:91::2 dev veth1 scope link"
1751         log_test $? 0 "IPv4 route with MPLS encap, v6 gw - check"
1752 }
1753
1754 ipv4_large_grp()
1755 {
1756         local ecmp=32
1757
1758         echo
1759         echo "IPv4 large groups (x$ecmp)"
1760         echo "---------------------"
1761
1762         check_large_grp 4 $ecmp
1763
1764         $IP nexthop flush >/dev/null 2>&1
1765 }
1766
1767 ipv4_large_res_grp()
1768 {
1769         echo
1770         echo "IPv4 large resilient group (128k buckets)"
1771         echo "-----------------------------------------"
1772
1773         check_nexthop_res_support
1774         if [ $? -eq $ksft_skip ]; then
1775                 return $ksft_skip
1776         fi
1777
1778         check_large_res_grp 4 $((128 * 1024))
1779
1780         $IP nexthop flush >/dev/null 2>&1
1781 }
1782
1783 sysctl_nexthop_compat_mode_check()
1784 {
1785         local sysctlname="net.ipv4.nexthop_compat_mode"
1786         local lprefix=$1
1787
1788         IPE="ip netns exec me"
1789
1790         $IPE sysctl -q $sysctlname 2>&1 >/dev/null
1791         if [ $? -ne 0 ]; then
1792                 echo "SKIP: kernel lacks nexthop compat mode sysctl control"
1793                 return $ksft_skip
1794         fi
1795
1796         out=$($IPE sysctl $sysctlname 2>/dev/null)
1797         log_test $? 0 "$lprefix default nexthop compat mode check"
1798         check_output "${out}" "$sysctlname = 1"
1799 }
1800
1801 sysctl_nexthop_compat_mode_set()
1802 {
1803         local sysctlname="net.ipv4.nexthop_compat_mode"
1804         local mode=$1
1805         local lprefix=$2
1806
1807         IPE="ip netns exec me"
1808
1809         out=$($IPE sysctl -w $sysctlname=$mode)
1810         log_test $? 0 "$lprefix set compat mode - $mode"
1811         check_output "${out}" "net.ipv4.nexthop_compat_mode = $mode"
1812 }
1813
1814 ipv6_compat_mode()
1815 {
1816         local rc
1817
1818         echo
1819         echo "IPv6 nexthop api compat mode test"
1820         echo "--------------------------------"
1821
1822         sysctl_nexthop_compat_mode_check "IPv6"
1823         if [ $? -eq $ksft_skip ]; then
1824                 return $ksft_skip
1825         fi
1826
1827         run_cmd "$IP nexthop add id 62 via 2001:db8:91::2 dev veth1"
1828         run_cmd "$IP nexthop add id 63 via 2001:db8:91::3 dev veth1"
1829         run_cmd "$IP nexthop add id 122 group 62/63"
1830         ipmout=$(start_ip_monitor route)
1831
1832         run_cmd "$IP -6 ro add 2001:db8:101::1/128 nhid 122"
1833         # route add notification should contain expanded nexthops
1834         stop_ip_monitor $ipmout 3
1835         log_test $? 0 "IPv6 compat mode on - route add notification"
1836
1837         # route dump should contain expanded nexthops
1838         check_route6 "2001:db8:101::1" "2001:db8:101::1 nhid 122 metric 1024 nexthop via 2001:db8:91::2 dev veth1 weight 1 nexthop via 2001:db8:91::3 dev veth1 weight 1"
1839         log_test $? 0 "IPv6 compat mode on - route dump"
1840
1841         # change in nexthop group should generate route notification
1842         run_cmd "$IP nexthop add id 64 via 2001:db8:91::4 dev veth1"
1843         ipmout=$(start_ip_monitor route)
1844         run_cmd "$IP nexthop replace id 122 group 62/64"
1845         stop_ip_monitor $ipmout 3
1846
1847         log_test $? 0 "IPv6 compat mode on - nexthop change"
1848
1849         # set compat mode off
1850         sysctl_nexthop_compat_mode_set 0 "IPv6"
1851
1852         run_cmd "$IP -6 ro del 2001:db8:101::1/128 nhid 122"
1853
1854         run_cmd "$IP nexthop add id 62 via 2001:db8:91::2 dev veth1"
1855         run_cmd "$IP nexthop add id 63 via 2001:db8:91::3 dev veth1"
1856         run_cmd "$IP nexthop add id 122 group 62/63"
1857         ipmout=$(start_ip_monitor route)
1858
1859         run_cmd "$IP -6 ro add 2001:db8:101::1/128 nhid 122"
1860         # route add notification should not contain expanded nexthops
1861         stop_ip_monitor $ipmout 1
1862         log_test $? 0 "IPv6 compat mode off - route add notification"
1863
1864         # route dump should not contain expanded nexthops
1865         check_route6 "2001:db8:101::1" "2001:db8:101::1 nhid 122 metric 1024"
1866         log_test $? 0 "IPv6 compat mode off - route dump"
1867
1868         # change in nexthop group should not generate route notification
1869         run_cmd "$IP nexthop add id 64 via 2001:db8:91::4 dev veth1"
1870         ipmout=$(start_ip_monitor route)
1871         run_cmd "$IP nexthop replace id 122 group 62/64"
1872         stop_ip_monitor $ipmout 0
1873         log_test $? 0 "IPv6 compat mode off - nexthop change"
1874
1875         # nexthop delete should not generate route notification
1876         ipmout=$(start_ip_monitor route)
1877         run_cmd "$IP nexthop del id 122"
1878         stop_ip_monitor $ipmout 0
1879         log_test $? 0 "IPv6 compat mode off - nexthop delete"
1880
1881         # set compat mode back on
1882         sysctl_nexthop_compat_mode_set 1 "IPv6"
1883 }
1884
1885 ipv4_compat_mode()
1886 {
1887         local rc
1888
1889         echo
1890         echo "IPv4 nexthop api compat mode"
1891         echo "----------------------------"
1892
1893         sysctl_nexthop_compat_mode_check "IPv4"
1894         if [ $? -eq $ksft_skip ]; then
1895                 return $ksft_skip
1896         fi
1897
1898         run_cmd "$IP nexthop add id 21 via 172.16.1.2 dev veth1"
1899         run_cmd "$IP nexthop add id 22 via 172.16.1.2 dev veth1"
1900         run_cmd "$IP nexthop add id 122 group 21/22"
1901         ipmout=$(start_ip_monitor route)
1902
1903         run_cmd "$IP ro add 172.16.101.1/32 nhid 122"
1904         stop_ip_monitor $ipmout 3
1905
1906         # route add notification should contain expanded nexthops
1907         log_test $? 0 "IPv4 compat mode on - route add notification"
1908
1909         # route dump should contain expanded nexthops
1910         check_route "172.16.101.1" "172.16.101.1 nhid 122 nexthop via 172.16.1.2 dev veth1 weight 1 nexthop via 172.16.1.2 dev veth1 weight 1"
1911         log_test $? 0 "IPv4 compat mode on - route dump"
1912
1913         # change in nexthop group should generate route notification
1914         run_cmd "$IP nexthop add id 23 via 172.16.1.3 dev veth1"
1915         ipmout=$(start_ip_monitor route)
1916         run_cmd "$IP nexthop replace id 122 group 21/23"
1917         stop_ip_monitor $ipmout 3
1918         log_test $? 0 "IPv4 compat mode on - nexthop change"
1919
1920         sysctl_nexthop_compat_mode_set 0 "IPv4"
1921
1922         # cleanup
1923         run_cmd "$IP ro del 172.16.101.1/32 nhid 122"
1924
1925         ipmout=$(start_ip_monitor route)
1926         run_cmd "$IP ro add 172.16.101.1/32 nhid 122"
1927         stop_ip_monitor $ipmout 1
1928         # route add notification should not contain expanded nexthops
1929         log_test $? 0 "IPv4 compat mode off - route add notification"
1930
1931         # route dump should not contain expanded nexthops
1932         check_route "172.16.101.1" "172.16.101.1 nhid 122"
1933         log_test $? 0 "IPv4 compat mode off - route dump"
1934
1935         # change in nexthop group should not generate route notification
1936         ipmout=$(start_ip_monitor route)
1937         run_cmd "$IP nexthop replace id 122 group 21/22"
1938         stop_ip_monitor $ipmout 0
1939         log_test $? 0 "IPv4 compat mode off - nexthop change"
1940
1941         # nexthop delete should not generate route notification
1942         ipmout=$(start_ip_monitor route)
1943         run_cmd "$IP nexthop del id 122"
1944         stop_ip_monitor $ipmout 0
1945         log_test $? 0 "IPv4 compat mode off - nexthop delete"
1946
1947         sysctl_nexthop_compat_mode_set 1 "IPv4"
1948 }
1949
1950 ipv4_del_add_loop1()
1951 {
1952         while :; do
1953                 $IP nexthop del id 100
1954                 $IP nexthop add id 100 via 172.16.1.2 dev veth1
1955         done >/dev/null 2>&1
1956 }
1957
1958 ipv4_grp_replace_loop()
1959 {
1960         while :; do
1961                 $IP nexthop replace id 102 group 100/101
1962         done >/dev/null 2>&1
1963 }
1964
1965 ipv4_torture()
1966 {
1967         local pid1
1968         local pid2
1969         local pid3
1970         local pid4
1971         local pid5
1972
1973         echo
1974         echo "IPv4 runtime torture"
1975         echo "--------------------"
1976         if [ ! -x "$(command -v mausezahn)" ]; then
1977                 echo "SKIP: Could not run test; need mausezahn tool"
1978                 return
1979         fi
1980
1981         run_cmd "$IP nexthop add id 100 via 172.16.1.2 dev veth1"
1982         run_cmd "$IP nexthop add id 101 via 172.16.2.2 dev veth3"
1983         run_cmd "$IP nexthop add id 102 group 100/101"
1984         run_cmd "$IP route add 172.16.101.1 nhid 102"
1985         run_cmd "$IP route add 172.16.101.2 nhid 102"
1986
1987         ipv4_del_add_loop1 &
1988         pid1=$!
1989         ipv4_grp_replace_loop &
1990         pid2=$!
1991         ip netns exec me ping -f 172.16.101.1 >/dev/null 2>&1 &
1992         pid3=$!
1993         ip netns exec me ping -f 172.16.101.2 >/dev/null 2>&1 &
1994         pid4=$!
1995         ip netns exec me mausezahn veth1 -B 172.16.101.2 -A 172.16.1.1 -c 0 -t tcp "dp=1-1023, flags=syn" >/dev/null 2>&1 &
1996         pid5=$!
1997
1998         sleep 300
1999         kill -9 $pid1 $pid2 $pid3 $pid4 $pid5
2000         wait $pid1 $pid2 $pid3 $pid4 $pid5 2>/dev/null
2001
2002         # if we did not crash, success
2003         log_test 0 0 "IPv4 torture test"
2004 }
2005
2006 ipv4_res_grp_replace_loop()
2007 {
2008         while :; do
2009                 $IP nexthop replace id 102 group 100/101 type resilient
2010         done >/dev/null 2>&1
2011 }
2012
2013 ipv4_res_torture()
2014 {
2015         local pid1
2016         local pid2
2017         local pid3
2018         local pid4
2019         local pid5
2020
2021         echo
2022         echo "IPv4 runtime resilient nexthop group torture"
2023         echo "--------------------------------------------"
2024
2025         check_nexthop_res_support
2026         if [ $? -eq $ksft_skip ]; then
2027                 return $ksft_skip
2028         fi
2029
2030         if [ ! -x "$(command -v mausezahn)" ]; then
2031                 echo "SKIP: Could not run test; need mausezahn tool"
2032                 return
2033         fi
2034
2035         run_cmd "$IP nexthop add id 100 via 172.16.1.2 dev veth1"
2036         run_cmd "$IP nexthop add id 101 via 172.16.2.2 dev veth3"
2037         run_cmd "$IP nexthop add id 102 group 100/101 type resilient buckets 512 idle_timer 0"
2038         run_cmd "$IP route add 172.16.101.1 nhid 102"
2039         run_cmd "$IP route add 172.16.101.2 nhid 102"
2040
2041         ipv4_del_add_loop1 &
2042         pid1=$!
2043         ipv4_res_grp_replace_loop &
2044         pid2=$!
2045         ip netns exec me ping -f 172.16.101.1 >/dev/null 2>&1 &
2046         pid3=$!
2047         ip netns exec me ping -f 172.16.101.2 >/dev/null 2>&1 &
2048         pid4=$!
2049         ip netns exec me mausezahn veth1 \
2050                                 -B 172.16.101.2 -A 172.16.1.1 -c 0 \
2051                                 -t tcp "dp=1-1023, flags=syn" >/dev/null 2>&1 &
2052         pid5=$!
2053
2054         sleep 300
2055         kill -9 $pid1 $pid2 $pid3 $pid4 $pid5
2056         wait $pid1 $pid2 $pid3 $pid4 $pid5 2>/dev/null
2057
2058         # if we did not crash, success
2059         log_test 0 0 "IPv4 resilient nexthop group torture test"
2060 }
2061
2062 basic()
2063 {
2064         echo
2065         echo "Basic functional tests"
2066         echo "----------------------"
2067         run_cmd "$IP nexthop ls"
2068         log_test $? 0 "List with nothing defined"
2069
2070         run_cmd "$IP nexthop get id 1"
2071         log_test $? 2 "Nexthop get on non-existent id"
2072
2073         # attempt to create nh without a device or gw - fails
2074         run_cmd "$IP nexthop add id 1"
2075         log_test $? 2 "Nexthop with no device or gateway"
2076
2077         # attempt to create nh with down device - fails
2078         $IP li set veth1 down
2079         run_cmd "$IP nexthop add id 1 dev veth1"
2080         log_test $? 2 "Nexthop with down device"
2081
2082         # create nh with linkdown device - fails
2083         $IP li set veth1 up
2084         ip -netns peer li set veth2 down
2085         run_cmd "$IP nexthop add id 1 dev veth1"
2086         log_test $? 2 "Nexthop with device that is linkdown"
2087         ip -netns peer li set veth2 up
2088
2089         # device only
2090         run_cmd "$IP nexthop add id 1 dev veth1"
2091         log_test $? 0 "Nexthop with device only"
2092
2093         # create nh with duplicate id
2094         run_cmd "$IP nexthop add id 1 dev veth3"
2095         log_test $? 2 "Nexthop with duplicate id"
2096
2097         # blackhole nexthop
2098         run_cmd "$IP nexthop add id 2 blackhole"
2099         log_test $? 0 "Blackhole nexthop"
2100
2101         # blackhole nexthop can not have other specs
2102         run_cmd "$IP nexthop replace id 2 blackhole dev veth1"
2103         log_test $? 2 "Blackhole nexthop with other attributes"
2104
2105         # blackhole nexthop should not be affected by the state of the loopback
2106         # device
2107         run_cmd "$IP link set dev lo down"
2108         check_nexthop "id 2" "id 2 blackhole"
2109         log_test $? 0 "Blackhole nexthop with loopback device down"
2110
2111         run_cmd "$IP link set dev lo up"
2112
2113         # Dump should not loop endlessly when maximum nexthop ID is configured.
2114         run_cmd "$IP nexthop add id $((2**32-1)) blackhole"
2115         run_cmd "timeout 5 $IP nexthop"
2116         log_test $? 0 "Maximum nexthop ID dump"
2117
2118         #
2119         # groups
2120         #
2121
2122         run_cmd "$IP nexthop add id 101 group 1"
2123         log_test $? 0 "Create group"
2124
2125         run_cmd "$IP nexthop add id 102 group 2"
2126         log_test $? 0 "Create group with blackhole nexthop"
2127
2128         # multipath group can not have a blackhole as 1 path
2129         run_cmd "$IP nexthop add id 103 group 1/2"
2130         log_test $? 2 "Create multipath group where 1 path is a blackhole"
2131
2132         # multipath group can not have a member replaced by a blackhole
2133         run_cmd "$IP nexthop replace id 2 dev veth3"
2134         run_cmd "$IP nexthop replace id 102 group 1/2"
2135         run_cmd "$IP nexthop replace id 2 blackhole"
2136         log_test $? 2 "Multipath group can not have a member replaced by blackhole"
2137
2138         # attempt to create group with non-existent nexthop
2139         run_cmd "$IP nexthop add id 103 group 12"
2140         log_test $? 2 "Create group with non-existent nexthop"
2141
2142         # attempt to create group with same nexthop
2143         run_cmd "$IP nexthop add id 103 group 1/1"
2144         log_test $? 2 "Create group with same nexthop multiple times"
2145
2146         # replace nexthop with a group - fails
2147         run_cmd "$IP nexthop replace id 2 group 1"
2148         log_test $? 2 "Replace nexthop with nexthop group"
2149
2150         # replace nexthop group with a nexthop - fails
2151         run_cmd "$IP nexthop replace id 101 dev veth1"
2152         log_test $? 2 "Replace nexthop group with nexthop"
2153
2154         # nexthop group with other attributes fail
2155         run_cmd "$IP nexthop add id 104 group 1 dev veth1"
2156         log_test $? 2 "Nexthop group and device"
2157
2158         # Tests to ensure that flushing works as expected.
2159         run_cmd "$IP nexthop add id 105 blackhole proto 99"
2160         run_cmd "$IP nexthop add id 106 blackhole proto 100"
2161         run_cmd "$IP nexthop add id 107 blackhole proto 99"
2162         run_cmd "$IP nexthop flush proto 99"
2163         check_nexthop "id 105" ""
2164         check_nexthop "id 106" "id 106 blackhole proto 100"
2165         check_nexthop "id 107" ""
2166         run_cmd "$IP nexthop flush proto 100"
2167         check_nexthop "id 106" ""
2168
2169         run_cmd "$IP nexthop flush proto 100"
2170         log_test $? 0 "Test proto flush"
2171
2172         run_cmd "$IP nexthop add id 104 group 1 blackhole"
2173         log_test $? 2 "Nexthop group and blackhole"
2174
2175         $IP nexthop flush >/dev/null 2>&1
2176
2177         # Test to ensure that flushing with a multi-part nexthop dump works as
2178         # expected.
2179         local batch_file=$(mktemp)
2180
2181         for i in $(seq 1 $((64 * 1024))); do
2182                 echo "nexthop add id $i blackhole" >> $batch_file
2183         done
2184
2185         $IP -b $batch_file
2186         $IP nexthop flush >/dev/null 2>&1
2187         [[ $($IP nexthop | wc -l) -eq 0 ]]
2188         log_test $? 0 "Large scale nexthop flushing"
2189
2190         rm $batch_file
2191 }
2192
2193 check_nexthop_buckets_balance()
2194 {
2195         local nharg=$1; shift
2196         local ret
2197
2198         while (($# > 0)); do
2199                 local selector=$1; shift
2200                 local condition=$1; shift
2201                 local count
2202
2203                 count=$($IP -j nexthop bucket ${nharg} ${selector} | jq length)
2204                 (( $count $condition ))
2205                 ret=$?
2206                 if ((ret != 0)); then
2207                         return $ret
2208                 fi
2209         done
2210
2211         return 0
2212 }
2213
2214 basic_res()
2215 {
2216         echo
2217         echo "Basic resilient nexthop group functional tests"
2218         echo "----------------------------------------------"
2219
2220         check_nexthop_res_support
2221         if [ $? -eq $ksft_skip ]; then
2222                 return $ksft_skip
2223         fi
2224
2225         run_cmd "$IP nexthop add id 1 dev veth1"
2226
2227         #
2228         # resilient nexthop group addition
2229         #
2230
2231         run_cmd "$IP nexthop add id 101 group 1 type resilient buckets 8"
2232         log_test $? 0 "Add a nexthop group with default parameters"
2233
2234         run_cmd "$IP nexthop get id 101"
2235         check_nexthop "id 101" \
2236                 "id 101 group 1 type resilient buckets 8 idle_timer 120 unbalanced_timer 0 unbalanced_time 0"
2237         log_test $? 0 "Get a nexthop group with default parameters"
2238
2239         run_cmd "$IP nexthop add id 102 group 1 type resilient
2240                         buckets 4 idle_timer 100 unbalanced_timer 5"
2241         run_cmd "$IP nexthop get id 102"
2242         check_nexthop "id 102" \
2243                 "id 102 group 1 type resilient buckets 4 idle_timer 100 unbalanced_timer 5 unbalanced_time 0"
2244         log_test $? 0 "Get a nexthop group with non-default parameters"
2245
2246         run_cmd "$IP nexthop add id 103 group 1 type resilient buckets 0"
2247         log_test $? 2 "Add a nexthop group with 0 buckets"
2248
2249         #
2250         # resilient nexthop group replacement
2251         #
2252
2253         run_cmd "$IP nexthop replace id 101 group 1 type resilient
2254                         buckets 8 idle_timer 240 unbalanced_timer 80"
2255         log_test $? 0 "Replace nexthop group parameters"
2256         check_nexthop "id 101" \
2257                 "id 101 group 1 type resilient buckets 8 idle_timer 240 unbalanced_timer 80 unbalanced_time 0"
2258         log_test $? 0 "Get a nexthop group after replacing parameters"
2259
2260         run_cmd "$IP nexthop replace id 101 group 1 type resilient idle_timer 512"
2261         log_test $? 0 "Replace idle timer"
2262         check_nexthop "id 101" \
2263                 "id 101 group 1 type resilient buckets 8 idle_timer 512 unbalanced_timer 80 unbalanced_time 0"
2264         log_test $? 0 "Get a nexthop group after replacing idle timer"
2265
2266         run_cmd "$IP nexthop replace id 101 group 1 type resilient unbalanced_timer 256"
2267         log_test $? 0 "Replace unbalanced timer"
2268         check_nexthop "id 101" \
2269                 "id 101 group 1 type resilient buckets 8 idle_timer 512 unbalanced_timer 256 unbalanced_time 0"
2270         log_test $? 0 "Get a nexthop group after replacing unbalanced timer"
2271
2272         run_cmd "$IP nexthop replace id 101 group 1 type resilient"
2273         log_test $? 0 "Replace with no parameters"
2274         check_nexthop "id 101" \
2275                 "id 101 group 1 type resilient buckets 8 idle_timer 512 unbalanced_timer 256 unbalanced_time 0"
2276         log_test $? 0 "Get a nexthop group after replacing no parameters"
2277
2278         run_cmd "$IP nexthop replace id 101 group 1"
2279         log_test $? 2 "Replace nexthop group type - implicit"
2280
2281         run_cmd "$IP nexthop replace id 101 group 1 type mpath"
2282         log_test $? 2 "Replace nexthop group type - explicit"
2283
2284         run_cmd "$IP nexthop replace id 101 group 1 type resilient buckets 1024"
2285         log_test $? 2 "Replace number of nexthop buckets"
2286
2287         check_nexthop "id 101" \
2288                 "id 101 group 1 type resilient buckets 8 idle_timer 512 unbalanced_timer 256 unbalanced_time 0"
2289         log_test $? 0 "Get a nexthop group after replacing with invalid parameters"
2290
2291         #
2292         # resilient nexthop buckets dump
2293         #
2294
2295         $IP nexthop flush >/dev/null 2>&1
2296         run_cmd "$IP nexthop add id 1 dev veth1"
2297         run_cmd "$IP nexthop add id 2 dev veth3"
2298         run_cmd "$IP nexthop add id 101 group 1/2 type resilient buckets 4"
2299         run_cmd "$IP nexthop add id 201 group 1/2"
2300
2301         check_nexthop_bucket "" \
2302                 "id 101 index 0 nhid 2 id 101 index 1 nhid 2 id 101 index 2 nhid 1 id 101 index 3 nhid 1"
2303         log_test $? 0 "Dump all nexthop buckets"
2304
2305         check_nexthop_bucket "list id 101" \
2306                 "id 101 index 0 nhid 2 id 101 index 1 nhid 2 id 101 index 2 nhid 1 id 101 index 3 nhid 1"
2307         log_test $? 0 "Dump all nexthop buckets in a group"
2308
2309         sleep 0.1
2310         (( $($IP -j nexthop bucket list id 101 |
2311              jq '[.[] | select(.bucket.idle_time > 0 and
2312                                .bucket.idle_time < 2)] | length') == 4 ))
2313         log_test $? 0 "All nexthop buckets report a positive near-zero idle time"
2314
2315         check_nexthop_bucket "list dev veth1" \
2316                 "id 101 index 2 nhid 1 id 101 index 3 nhid 1"
2317         log_test $? 0 "Dump all nexthop buckets with a specific nexthop device"
2318
2319         check_nexthop_bucket "list nhid 2" \
2320                 "id 101 index 0 nhid 2 id 101 index 1 nhid 2"
2321         log_test $? 0 "Dump all nexthop buckets with a specific nexthop identifier"
2322
2323         run_cmd "$IP nexthop bucket list id 111"
2324         log_test $? 2 "Dump all nexthop buckets in a non-existent group"
2325
2326         run_cmd "$IP nexthop bucket list id 201"
2327         log_test $? 2 "Dump all nexthop buckets in a non-resilient group"
2328
2329         run_cmd "$IP nexthop bucket list dev bla"
2330         log_test $? 255 "Dump all nexthop buckets using a non-existent device"
2331
2332         run_cmd "$IP nexthop bucket list groups"
2333         log_test $? 255 "Dump all nexthop buckets with invalid 'groups' keyword"
2334
2335         run_cmd "$IP nexthop bucket list fdb"
2336         log_test $? 255 "Dump all nexthop buckets with invalid 'fdb' keyword"
2337
2338         # Dump should not loop endlessly when maximum nexthop ID is configured.
2339         run_cmd "$IP nexthop add id $((2**32-1)) group 1/2 type resilient buckets 4"
2340         run_cmd "timeout 5 $IP nexthop bucket"
2341         log_test $? 0 "Maximum nexthop ID dump"
2342
2343         #
2344         # resilient nexthop buckets get requests
2345         #
2346
2347         check_nexthop_bucket "get id 101 index 0" "id 101 index 0 nhid 2"
2348         log_test $? 0 "Get a valid nexthop bucket"
2349
2350         run_cmd "$IP nexthop bucket get id 101 index 999"
2351         log_test $? 2 "Get a nexthop bucket with valid group, but invalid index"
2352
2353         run_cmd "$IP nexthop bucket get id 201 index 0"
2354         log_test $? 2 "Get a nexthop bucket from a non-resilient group"
2355
2356         run_cmd "$IP nexthop bucket get id 999 index 0"
2357         log_test $? 2 "Get a nexthop bucket from a non-existent group"
2358
2359         #
2360         # tests for bucket migration
2361         #
2362
2363         $IP nexthop flush >/dev/null 2>&1
2364
2365         run_cmd "$IP nexthop add id 1 dev veth1"
2366         run_cmd "$IP nexthop add id 2 dev veth3"
2367         run_cmd "$IP nexthop add id 101
2368                         group 1/2 type resilient buckets 10
2369                         idle_timer 1 unbalanced_timer 20"
2370
2371         check_nexthop_buckets_balance "list id 101" \
2372                                       "nhid 1" "== 5" \
2373                                       "nhid 2" "== 5"
2374         log_test $? 0 "Initial bucket allocation"
2375
2376         run_cmd "$IP nexthop replace id 101
2377                         group 1,2/2,3 type resilient"
2378         check_nexthop_buckets_balance "list id 101" \
2379                                       "nhid 1" "== 4" \
2380                                       "nhid 2" "== 6"
2381         log_test $? 0 "Bucket allocation after replace"
2382
2383         # Check that increase in idle timer does not make buckets appear busy.
2384         run_cmd "$IP nexthop replace id 101
2385                         group 1,2/2,3 type resilient
2386                         idle_timer 10"
2387         run_cmd "$IP nexthop replace id 101
2388                         group 1/2 type resilient"
2389         check_nexthop_buckets_balance "list id 101" \
2390                                       "nhid 1" "== 5" \
2391                                       "nhid 2" "== 5"
2392         log_test $? 0 "Buckets migrated after idle timer change"
2393
2394         $IP nexthop flush >/dev/null 2>&1
2395 }
2396
2397 ################################################################################
2398 # usage
2399
2400 usage()
2401 {
2402         cat <<EOF
2403 usage: ${0##*/} OPTS
2404
2405         -t <test>   Test(s) to run (default: all)
2406                     (options: $ALL_TESTS)
2407         -4          IPv4 tests only
2408         -6          IPv6 tests only
2409         -p          Pause on fail
2410         -P          Pause after each test before cleanup
2411         -v          verbose mode (show commands and output)
2412         -w          Timeout for ping
2413
2414     Runtime test
2415         -n num      Number of nexthops to target
2416         -N          Use new style to install routes in DUT
2417
2418 done
2419 EOF
2420 }
2421
2422 ################################################################################
2423 # main
2424
2425 while getopts :t:pP46hvw: o
2426 do
2427         case $o in
2428                 t) TESTS=$OPTARG;;
2429                 4) TESTS=${IPV4_TESTS};;
2430                 6) TESTS=${IPV6_TESTS};;
2431                 p) PAUSE_ON_FAIL=yes;;
2432                 P) PAUSE=yes;;
2433                 v) VERBOSE=$(($VERBOSE + 1));;
2434                 w) PING_TIMEOUT=$OPTARG;;
2435                 h) usage; exit 0;;
2436                 *) usage; exit 1;;
2437         esac
2438 done
2439
2440 # make sure we don't pause twice
2441 [ "${PAUSE}" = "yes" ] && PAUSE_ON_FAIL=no
2442
2443 if [ "$(id -u)" -ne 0 ];then
2444         echo "SKIP: Need root privileges"
2445         exit $ksft_skip;
2446 fi
2447
2448 if [ ! -x "$(command -v ip)" ]; then
2449         echo "SKIP: Could not run test without ip tool"
2450         exit $ksft_skip
2451 fi
2452
2453 ip help 2>&1 | grep -q nexthop
2454 if [ $? -ne 0 ]; then
2455         echo "SKIP: iproute2 too old, missing nexthop command"
2456         exit $ksft_skip
2457 fi
2458
2459 out=$(ip nexthop ls 2>&1 | grep -q "Operation not supported")
2460 if [ $? -eq 0 ]; then
2461         echo "SKIP: kernel lacks nexthop support"
2462         exit $ksft_skip
2463 fi
2464
2465 for t in $TESTS
2466 do
2467         case $t in
2468         none) IP="ip -netns peer"; setup; exit 0;;
2469         *) setup; $t; cleanup;;
2470         esac
2471 done
2472
2473 if [ "$TESTS" != "none" ]; then
2474         printf "\nTests passed: %3d\n" ${nsuccess}
2475         printf "Tests failed: %3d\n"   ${nfail}
2476 fi
2477
2478 exit $ret