xref: /linux/tools/testing/selftests/net/mptcp/mptcp_join.sh (revision 0e50474fa514822e9d990874e554bf8043a201d7)
1#!/bin/bash
2# SPDX-License-Identifier: GPL-2.0
3
4# Double quotes to prevent globbing and word splitting is recommended in new
5# code but we accept it, especially because there were too many before having
6# address all other issues detected by shellcheck.
7#shellcheck disable=SC2086
8
9# ShellCheck incorrectly believes that most of the code here is unreachable
10# because it's invoked by variable name, see how the "tests" array is used
11#shellcheck disable=SC2317,SC2329
12
13. "$(dirname "${0}")/mptcp_lib.sh"
14
15ret=0
16sin=""
17sinfail=""
18sout=""
19cin=""
20cinfail=""
21cinsent=""
22tmpfile=""
23cout=""
24err=""
25capout=""
26cappid=""
27ns1=""
28ns2=""
29iptables="iptables"
30ip6tables="ip6tables"
31timeout_poll=30
32timeout_test=$((timeout_poll * 2 + 1))
33capture=false
34checksum=false
35check_invert=0
36validate_checksum=false
37init=0
38evts_ns1=""
39evts_ns2=""
40evts_ns1_pid=0
41evts_ns2_pid=0
42last_test_failed=0
43last_test_skipped=0
44last_test_ignored=1
45
46declare -A all_tests
47declare -a only_tests_ids
48declare -a only_tests_names
49declare -A failed_tests
50MPTCP_LIB_TEST_FORMAT="%03u %s\n"
51TEST_NAME=""
52nr_blank=6
53
54# These var are used only in some tests, make sure they are not already set
55unset FAILING_LINKS
56unset test_linkfail
57unset addr_nr_ns1
58unset addr_nr_ns2
59unset cestab_ns1
60unset cestab_ns2
61unset sflags
62unset fastclose
63unset fullmesh
64unset speed
65unset bind_addr
66unset join_syn_rej
67unset join_csum_ns1
68unset join_csum_ns2
69unset join_fail_nr
70unset join_rst_nr
71unset join_infi_nr
72unset join_corrupted_pkts
73unset join_syn_tx
74unset join_create_err
75unset join_bind_err
76unset join_connect_err
77
78unset fb_ns1
79unset fb_ns2
80unset fb_infinite_map_tx
81unset fb_dss_corruption
82unset fb_simult_conn
83unset fb_mpc_passive
84unset fb_mpc_active
85unset fb_mpc_data
86unset fb_md5_sig
87unset fb_dss
88
89# generated using "nfbpf_compile '(ip && (ip[54] & 0xf0) == 0x30) ||
90#				  (ip6 && (ip6[74] & 0xf0) == 0x30)'"
91CBPF_MPTCP_SUBOPTION_ADD_ADDR="14,
92			       48 0 0 0,
93			       84 0 0 240,
94			       21 0 3 64,
95			       48 0 0 54,
96			       84 0 0 240,
97			       21 6 7 48,
98			       48 0 0 0,
99			       84 0 0 240,
100			       21 0 4 96,
101			       48 0 0 74,
102			       84 0 0 240,
103			       21 0 1 48,
104			       6 0 0 65535,
105			       6 0 0 0"
106
107init_partial()
108{
109	capout=$(mktemp)
110
111	mptcp_lib_ns_init ns1 ns2
112
113	local netns
114	for netns in "$ns1" "$ns2"; do
115		ip netns exec $netns sysctl -q net.mptcp.pm_type=0 2>/dev/null || true
116		if $checksum; then
117			ip netns exec $netns sysctl -q net.mptcp.checksum_enabled=1
118		fi
119	done
120
121	check_invert=0
122	validate_checksum=$checksum
123
124	#  ns1         ns2
125	# ns1eth1    ns2eth1
126	# ns1eth2    ns2eth2
127	# ns1eth3    ns2eth3
128	# ns1eth4    ns2eth4
129
130	local i
131	for i in $(seq 1 4); do
132		ip link add ns1eth$i netns "$ns1" type veth peer name ns2eth$i netns "$ns2"
133		ip -net "$ns1" addr add 10.0.$i.1/24 dev ns1eth$i
134		ip -net "$ns1" addr add dead:beef:$i::1/64 dev ns1eth$i nodad
135		ip -net "$ns1" link set ns1eth$i up
136
137		ip -net "$ns2" addr add 10.0.$i.2/24 dev ns2eth$i
138		ip -net "$ns2" addr add dead:beef:$i::2/64 dev ns2eth$i nodad
139		ip -net "$ns2" link set ns2eth$i up
140
141		# let $ns2 reach any $ns1 address from any interface
142		ip -net "$ns2" route add default via 10.0.$i.1 dev ns2eth$i metric 10$i
143		ip -net "$ns2" route add default via dead:beef:$i::1 dev ns2eth$i metric 10$i
144	done
145}
146
147init_shapers()
148{
149	local i
150	for i in $(seq 1 4); do
151		tc -n $ns1 qdisc add dev ns1eth$i root netem rate 20mbit delay 1ms
152		tc -n $ns2 qdisc add dev ns2eth$i root netem rate 20mbit delay 1ms
153	done
154}
155
156cleanup_partial()
157{
158	rm -f "$capout"
159
160	mptcp_lib_ns_exit "${ns1}" "${ns2}"
161}
162
163init() {
164	init=1
165
166	mptcp_lib_check_mptcp
167	mptcp_lib_check_kallsyms
168	mptcp_lib_check_tools ip tc ss "${iptables}" "${ip6tables}"
169
170	sin=$(mktemp)
171	sout=$(mktemp)
172	cin=$(mktemp)
173	cinsent=$(mktemp)
174	cout=$(mktemp)
175	err=$(mktemp)
176	evts_ns1=$(mktemp)
177	evts_ns2=$(mktemp)
178
179	trap cleanup EXIT
180
181	make_file "$cin" "client" 1 >/dev/null
182	make_file "$sin" "server" 1 >/dev/null
183}
184
185cleanup()
186{
187	rm -f "$cin" "$cout" "$sinfail"
188	rm -f "$sin" "$sout" "$cinsent" "$cinfail"
189	rm -f "$tmpfile"
190	rm -rf $evts_ns1 $evts_ns2
191	rm -f "$err"
192	cleanup_partial
193}
194
195print_check()
196{
197	printf "%-${nr_blank}s%-36s" " " "${*}"
198}
199
200print_info()
201{
202	# It can be empty, no need to print anything then
203	[ -z "${1}" ] && return
204
205	mptcp_lib_print_info "      Info: ${*}"
206}
207
208print_ok()
209{
210	mptcp_lib_pr_ok "${@}"
211}
212
213print_fail()
214{
215	mptcp_lib_pr_fail "${@}"
216}
217
218print_skip()
219{
220	mptcp_lib_pr_skip "${@}"
221}
222
223# $1: check name; $2: rc
224print_results()
225{
226	local check="${1}"
227	local rc=${2}
228
229	print_check "${check}"
230	if [ ${rc} = ${KSFT_PASS} ]; then
231		print_ok
232	elif [ ${rc} = ${KSFT_SKIP} ]; then
233		print_skip
234	else
235		fail_test "see above"
236	fi
237}
238
239# [ $1: fail msg ]
240mark_as_skipped()
241{
242	local msg="${1:-"Feature not supported"}"
243
244	mptcp_lib_fail_if_expected_feature "${msg}"
245
246	print_check "${msg}"
247	print_skip
248
249	last_test_skipped=1
250}
251
252# $@: condition
253continue_if()
254{
255	if ! "${@}"; then
256		mark_as_skipped
257		return 1
258	fi
259}
260
261skip_test()
262{
263	if [ "${#only_tests_ids[@]}" -eq 0 ] && [ "${#only_tests_names[@]}" -eq 0 ]; then
264		return 1
265	fi
266
267	local i
268	for i in "${only_tests_ids[@]}"; do
269		if [ "$((MPTCP_LIB_TEST_COUNTER+1))" -eq "${i}" ]; then
270			return 1
271		fi
272	done
273	for i in "${only_tests_names[@]}"; do
274		if [ "${TEST_NAME}" = "${i}" ]; then
275			return 1
276		fi
277	done
278
279	return 0
280}
281
282append_prev_results()
283{
284	if [ ${last_test_failed} -eq 1 ]; then
285		mptcp_lib_result_fail "${TEST_NAME}"
286	elif [ ${last_test_skipped} -eq 1 ]; then
287		mptcp_lib_result_skip "${TEST_NAME}"
288	elif [ ${last_test_ignored} -ne 1 ]; then
289		mptcp_lib_result_pass "${TEST_NAME}"
290	fi
291
292	last_test_failed=0
293	last_test_skipped=0
294	last_test_ignored=0
295}
296
297# $1: test name
298reset()
299{
300	append_prev_results
301
302	TEST_NAME="${1}"
303
304	MPTCP_LIB_SUBTEST_FLAKY=0 # reset if modified
305
306	if skip_test; then
307		MPTCP_LIB_TEST_COUNTER=$((MPTCP_LIB_TEST_COUNTER+1))
308		last_test_ignored=1
309		return 1
310	fi
311
312	mptcp_lib_print_title "${TEST_NAME}"
313
314	if [ "${init}" != "1" ]; then
315		init
316	else
317		cleanup_partial
318	fi
319
320	init_partial
321
322	return 0
323}
324
325# $1: test name ; $2: counter to check
326reset_check_counter()
327{
328	reset "${1}" || return 1
329
330	local counter="${2}"
331
332	if ! nstat -asz "${counter}" | grep -wq "${counter}"; then
333		mark_as_skipped "counter '${counter}' is not available"
334		return 1
335	fi
336}
337
338# $1: test name
339reset_with_cookies()
340{
341	reset "${1}" || return 1
342
343	local netns
344	for netns in "$ns1" "$ns2"; do
345		ip netns exec $netns sysctl -q net.ipv4.tcp_syncookies=2
346	done
347}
348
349# $1: test name
350reset_with_add_addr_timeout()
351{
352	local ip="${2:-4}"
353	local tables
354
355	reset "${1}" || return 1
356
357	tables="${iptables}"
358	if [ $ip -eq 6 ]; then
359		tables="${ip6tables}"
360	fi
361
362	# set a maximum, to avoid too long timeout with exponential backoff
363	ip netns exec $ns1 sysctl -q net.mptcp.add_addr_timeout=1
364
365	if ! ip netns exec $ns2 $tables -A OUTPUT -p tcp \
366			-m tcp --tcp-option 30 \
367			-m bpf --bytecode \
368			"$CBPF_MPTCP_SUBOPTION_ADD_ADDR" \
369			-j DROP; then
370		mark_as_skipped "unable to set the 'add addr' rule"
371		return 1
372	fi
373}
374
375# $1: test name
376reset_with_checksum()
377{
378	local ns1_enable=$1
379	local ns2_enable=$2
380
381	reset "checksum test ${ns1_enable} ${ns2_enable}" || return 1
382
383	ip netns exec $ns1 sysctl -q net.mptcp.checksum_enabled=$ns1_enable
384	ip netns exec $ns2 sysctl -q net.mptcp.checksum_enabled=$ns2_enable
385
386	validate_checksum=true
387}
388
389reset_with_allow_join_id0()
390{
391	local ns1_enable=$2
392	local ns2_enable=$3
393
394	reset "${1}" || return 1
395
396	ip netns exec $ns1 sysctl -q net.mptcp.allow_join_initial_addr_port=$ns1_enable
397	ip netns exec $ns2 sysctl -q net.mptcp.allow_join_initial_addr_port=$ns2_enable
398}
399
400# Modify TCP payload without corrupting the TCP packet
401#
402# This rule inverts a 8-bit word at byte offset 148 for the 2nd TCP ACK packets
403# carrying enough data.
404# Once it is done, the TCP Checksum field is updated so the packet is still
405# considered as valid at the TCP level.
406# Because the MPTCP checksum, covering the TCP options and data, has not been
407# updated, the modification will be detected and an MP_FAIL will be emitted:
408# what we want to validate here without corrupting "random" MPTCP options.
409#
410# To avoid having tc producing this pr_info() message for each TCP ACK packets
411# not carrying enough data:
412#
413#     tc action pedit offset 162 out of bounds
414#
415# Netfilter is used to mark packets with enough data.
416setup_fail_rules()
417{
418	check_invert=1
419	validate_checksum=true
420	local i="$1"
421	local ip="${2:-4}"
422	local tables
423
424	tables="${iptables}"
425	if [ $ip -eq 6 ]; then
426		tables="${ip6tables}"
427	fi
428
429	ip netns exec $ns2 $tables \
430		-t mangle \
431		-A OUTPUT \
432		-o ns2eth$i \
433		-p tcp \
434		-m length --length 150:9999 \
435		-m statistic --mode nth --packet 1 --every 99999 \
436		-j MARK --set-mark 42 || return ${KSFT_SKIP}
437
438	tc -n $ns2 qdisc add dev ns2eth$i clsact || return ${KSFT_SKIP}
439	tc -n $ns2 filter add dev ns2eth$i egress \
440		protocol ip prio 1000 \
441		handle 42 fw \
442		action pedit munge offset 148 u8 invert \
443		pipe csum tcp \
444		index 100 || return ${KSFT_SKIP}
445}
446
447reset_with_fail()
448{
449	reset_check_counter "${1}" "MPTcpExtInfiniteMapTx" || return 1
450	shift
451
452	ip netns exec $ns1 sysctl -q net.mptcp.checksum_enabled=1
453	ip netns exec $ns2 sysctl -q net.mptcp.checksum_enabled=1
454
455	local rc=0
456	setup_fail_rules "${@}" || rc=$?
457
458	if [ ${rc} -eq ${KSFT_SKIP} ]; then
459		mark_as_skipped "unable to set the 'fail' rules"
460		return 1
461	fi
462}
463
464start_events()
465{
466	mptcp_lib_events "${ns1}" "${evts_ns1}" evts_ns1_pid
467	mptcp_lib_events "${ns2}" "${evts_ns2}" evts_ns2_pid
468}
469
470reset_with_events()
471{
472	reset "${1}" || return 1
473
474	start_events
475}
476
477reset_with_tcp_filter()
478{
479	reset "${1}" || return 1
480	shift
481
482	local ns="${!1}"
483	local src="${2}"
484	local target="${3}"
485	local chain="${4:-INPUT}"
486
487	if ! ip netns exec "${ns}" ${iptables} \
488			-A "${chain}" \
489			-s "${src}" \
490			-p tcp \
491			-j "${target}"; then
492		mark_as_skipped "unable to set the filter rules"
493		return 1
494	fi
495}
496
497# $1: err msg
498fail_test()
499{
500	if ! mptcp_lib_subtest_is_flaky; then
501		ret=${KSFT_FAIL}
502	fi
503
504	if [ ${#} -gt 0 ]; then
505		print_fail "${@}"
506	fi
507
508	# just in case a test is marked twice as failed
509	if [ ${last_test_failed} -eq 0 ]; then
510		failed_tests[${MPTCP_LIB_TEST_COUNTER}]="${TEST_NAME}"
511		dump_stats
512		last_test_failed=1
513	fi
514}
515
516get_failed_tests_ids()
517{
518	# sorted
519	local i
520	for i in "${!failed_tests[@]}"; do
521		echo "${i}"
522	done | sort -n
523}
524
525check_transfer()
526{
527	local in=$1
528	local out=$2
529	local what=$3
530	local bytes=$4
531	local i a b
532
533	local line
534	if [ -n "$bytes" ]; then
535		local out_size
536		# when truncating we must check the size explicitly
537		out_size=$(wc -c $out | awk '{print $1}')
538		if [ $out_size -ne $bytes ]; then
539			fail_test "$what output file has wrong size ($out_size, $bytes)"
540			return 1
541		fi
542
543		# note: BusyBox's "cmp" command doesn't support --bytes
544		tmpfile=$(mktemp)
545		head --bytes="$bytes" "$in" > "$tmpfile"
546		mv "$tmpfile" "$in"
547		head --bytes="$bytes" "$out" > "$tmpfile"
548		mv "$tmpfile" "$out"
549		tmpfile=""
550	fi
551	cmp -l "$in" "$out" | while read -r i a b; do
552		local sum=$((0${a} + 0${b}))
553		if [ $check_invert -eq 0 ] || [ $sum -ne $((0xff)) ]; then
554			fail_test "$what does not match (in, out):"
555			mptcp_lib_print_file_err "$in"
556			mptcp_lib_print_file_err "$out"
557
558			return 1
559		else
560			print_info "$what has inverted byte at ${i}"
561		fi
562	done
563
564	return 0
565}
566
567do_ping()
568{
569	local listener_ns="$1"
570	local connector_ns="$2"
571	local connect_addr="$3"
572
573	if ! ip netns exec ${connector_ns} ping -q -c 1 $connect_addr >/dev/null; then
574		fail_test "$listener_ns -> $connect_addr connectivity"
575	fi
576}
577
578link_failure()
579{
580	local ns="$1"
581
582	if [ -z "$FAILING_LINKS" ]; then
583		l=$((RANDOM%4))
584		FAILING_LINKS=$((l+1))
585	fi
586
587	local l
588	for l in $FAILING_LINKS; do
589		local veth="ns1eth$l"
590		ip -net "$ns" link set "$veth" down
591	done
592}
593
594rm_addr_count()
595{
596	mptcp_lib_get_counter "${1}" "MPTcpExtRmAddr"
597}
598
599# $1: ns, $2: old rm_addr counter in $ns
600wait_rm_addr()
601{
602	local ns="${1}"
603	local old_cnt="${2}"
604	local cnt
605
606	local i
607	for i in $(seq 10); do
608		cnt=$(rm_addr_count ${ns})
609		[ "$cnt" = "${old_cnt}" ] || break
610		sleep 0.1
611	done
612}
613
614rm_sf_count()
615{
616	mptcp_lib_get_counter "${1}" "MPTcpExtRmSubflow"
617}
618
619# $1: ns, $2: old rm_sf counter in $ns
620wait_rm_sf()
621{
622	local ns="${1}"
623	local old_cnt="${2}"
624	local cnt
625
626	local i
627	for i in $(seq 10); do
628		cnt=$(rm_sf_count ${ns})
629		[ "$cnt" = "${old_cnt}" ] || break
630		sleep 0.1
631	done
632}
633
634wait_mpj()
635{
636	local ns="${1}"
637	local cnt old_cnt
638
639	old_cnt=$(mptcp_lib_get_counter ${ns} "MPTcpExtMPJoinAckRx")
640
641	local i
642	for i in $(seq 10); do
643		cnt=$(mptcp_lib_get_counter ${ns} "MPTcpExtMPJoinAckRx")
644		[ "$cnt" = "${old_cnt}" ] || break
645		sleep 0.1
646	done
647}
648
649wait_ll_ready()
650{
651	local ns="${1}"
652
653	local i
654	for i in $(seq 50); do
655		ip -n "${ns}" -6 addr show scope link | grep "inet6 fe80" |
656			grep -qw "tentative" || break
657		sleep 0.1
658	done
659}
660
661get_ll_addr()
662{
663	local ns="${1}"
664	local iface="${2}"
665
666	ip -n "${ns}" -6 addr show dev "${iface}" scope link |
667		grep "inet6 fe80" | sed 's#.*\(fe80::.*\)/.*#\1#'
668}
669
670kill_events_pids()
671{
672	mptcp_lib_kill_wait $evts_ns1_pid
673	evts_ns1_pid=0
674	mptcp_lib_kill_wait $evts_ns2_pid
675	evts_ns2_pid=0
676}
677
678pm_nl_set_limits()
679{
680	mptcp_lib_pm_nl_set_limits "${@}"
681}
682
683pm_nl_add_endpoint()
684{
685	mptcp_lib_pm_nl_add_endpoint "${@}"
686}
687
688pm_nl_del_endpoint()
689{
690	mptcp_lib_pm_nl_del_endpoint "${@}"
691}
692
693pm_nl_flush_endpoint()
694{
695	mptcp_lib_pm_nl_flush_endpoint "${@}"
696}
697
698pm_nl_show_endpoints()
699{
700	mptcp_lib_pm_nl_show_endpoints "${@}"
701}
702
703pm_nl_change_endpoint()
704{
705	mptcp_lib_pm_nl_change_endpoint "${@}"
706}
707
708pm_nl_check_endpoint()
709{
710	local msg="$1"
711	local ns=$2
712	local addr=$3
713	local flags dev id port
714
715	print_check "${msg}"
716
717	shift 3
718	while [ -n "$1" ]; do
719		case "${1}" in
720		"flags" | "dev" | "id" | "port")
721			eval "${1}"="${2}"
722			shift
723			;;
724		*)
725			;;
726		esac
727
728		shift
729	done
730
731	if [ -z "${id}" ]; then
732		fail_test "bad test - missing endpoint id"
733		return
734	fi
735
736	check_output "mptcp_lib_pm_nl_get_endpoint ${ns} ${id}" \
737		"$(mptcp_lib_pm_nl_format_endpoints \
738			"${id},${addr},${flags//","/" "},${dev},${port}")"
739}
740
741pm_nl_set_endpoint()
742{
743	local listener_ns="$1"
744	local connector_ns="$2"
745	local connect_addr="$3"
746
747	local addr_nr_ns1=${addr_nr_ns1:-0}
748	local addr_nr_ns2=${addr_nr_ns2:-0}
749	local sflags=${sflags:-""}
750	local fullmesh=${fullmesh:-""}
751
752	local flags="subflow"
753	if [ -n "${fullmesh}" ]; then
754		flags="${flags},fullmesh"
755		addr_nr_ns2=${fullmesh}
756	fi
757
758	# let the mptcp subflow be established in background before
759	# do endpoint manipulation
760	if [ $addr_nr_ns1 != "0" ] || [ $addr_nr_ns2 != "0" ]; then
761		sleep 1
762	fi
763
764	if [ $addr_nr_ns1 -gt 0 ]; then
765		local counter=2
766		local add_nr_ns1=${addr_nr_ns1}
767		local id=10
768		while [ $add_nr_ns1 -gt 0 ]; do
769			local addr
770			if mptcp_lib_is_v6 "${connect_addr}"; then
771				addr="dead:beef:$counter::1"
772			else
773				addr="10.0.$counter.1"
774			fi
775			pm_nl_add_endpoint $ns1 $addr flags signal
776			counter=$((counter + 1))
777			add_nr_ns1=$((add_nr_ns1 - 1))
778			id=$((id + 1))
779		done
780	elif [ $addr_nr_ns1 -lt 0 ]; then
781		local rm_nr_ns1=$((-addr_nr_ns1))
782		if [ $rm_nr_ns1 -lt 8 ]; then
783			local counter=0
784			local line
785			pm_nl_show_endpoints ${listener_ns} | while read -r line; do
786				# shellcheck disable=SC2206 # we do want to split per word
787				local arr=($line)
788				local nr=0
789
790				local i
791				for i in "${arr[@]}"; do
792					if [ $i = "id" ]; then
793						if [ $counter -eq $rm_nr_ns1 ]; then
794							break
795						fi
796						id=${arr[$nr+1]}
797						rm_addr=$(rm_addr_count ${connector_ns})
798						pm_nl_del_endpoint ${listener_ns} $id
799						wait_rm_addr ${connector_ns} ${rm_addr}
800						counter=$((counter + 1))
801					fi
802					nr=$((nr + 1))
803				done
804			done
805		elif [ $rm_nr_ns1 -eq 8 ]; then
806			pm_nl_flush_endpoint ${listener_ns}
807		elif [ $rm_nr_ns1 -eq 9 ]; then
808			pm_nl_del_endpoint ${listener_ns} 0 ${connect_addr}
809		fi
810	fi
811
812	# if newly added endpoints must be deleted, give the background msk
813	# some time to created them
814	[ $addr_nr_ns1 -gt 0 ] && [ $addr_nr_ns2 -lt 0 ] && sleep 1
815
816	if [ $addr_nr_ns2 -gt 0 ]; then
817		local add_nr_ns2=${addr_nr_ns2}
818		local counter=3
819		local id=20
820		while [ $add_nr_ns2 -gt 0 ]; do
821			local addr
822			if mptcp_lib_is_v6 "${connect_addr}"; then
823				addr="dead:beef:$counter::2"
824			else
825				addr="10.0.$counter.2"
826			fi
827			pm_nl_add_endpoint $ns2 $addr flags $flags
828			counter=$((counter + 1))
829			add_nr_ns2=$((add_nr_ns2 - 1))
830			id=$((id + 1))
831		done
832	elif [ $addr_nr_ns2 -lt 0 ]; then
833		local rm_nr_ns2=$((-addr_nr_ns2))
834		if [ $rm_nr_ns2 -lt 8 ]; then
835			local counter=0
836			local line
837			pm_nl_show_endpoints ${connector_ns} | while read -r line; do
838				# shellcheck disable=SC2206 # we do want to split per word
839				local arr=($line)
840				local nr=0
841
842				local i
843				for i in "${arr[@]}"; do
844					if [ $i = "id" ]; then
845						if [ $counter -eq $rm_nr_ns2 ]; then
846							break
847						fi
848						local id rm_addr
849						# rm_addr are serialized, allow the previous one to
850						# complete
851						id=${arr[$nr+1]}
852						rm_addr=$(rm_addr_count ${listener_ns})
853						pm_nl_del_endpoint ${connector_ns} $id
854						wait_rm_addr ${listener_ns} ${rm_addr}
855						counter=$((counter + 1))
856					fi
857					nr=$((nr + 1))
858				done
859			done
860		elif [ $rm_nr_ns2 -eq 8 ]; then
861			pm_nl_flush_endpoint ${connector_ns}
862		elif [ $rm_nr_ns2 -eq 9 ]; then
863			local addr
864			if mptcp_lib_is_v6 "${connect_addr}"; then
865				addr="dead:beef:1::2"
866			else
867				addr="10.0.1.2"
868			fi
869			pm_nl_del_endpoint ${connector_ns} 0 $addr
870		fi
871	fi
872
873	if [ -n "${sflags}" ]; then
874		sleep 1
875
876		local netns
877		for netns in "$ns1" "$ns2"; do
878			local line
879			pm_nl_show_endpoints $netns | while read -r line; do
880				# shellcheck disable=SC2206 # we do want to split per word
881				local arr=($line)
882				local nr=0
883				local id
884
885				local i
886				for i in "${arr[@]}"; do
887					if [ $i = "id" ]; then
888						id=${arr[$nr+1]}
889					fi
890					nr=$((nr + 1))
891				done
892				pm_nl_change_endpoint $netns $id $sflags
893			done
894		done
895	fi
896}
897
898chk_cestab_nr()
899{
900	local ns=$1
901	local cestab=$2
902	local count
903
904	print_check "currently established: $cestab"
905	count=$(mptcp_lib_get_counter ${ns} "MPTcpExtMPCurrEstab")
906	if [ -z "$count" ]; then
907		print_skip
908	elif [ "$count" != "$cestab" ]; then
909		fail_test "got $count current establish[s] expected $cestab"
910	else
911		print_ok
912	fi
913}
914
915# $1 namespace 1, $2 namespace 2
916check_cestab()
917{
918	if [ -n "${cestab_ns1}" ]; then
919		chk_cestab_nr ${1} ${cestab_ns1}
920	fi
921	if [ -n "${cestab_ns2}" ]; then
922		chk_cestab_nr ${2} ${cestab_ns2}
923	fi
924}
925
926cond_start_capture()
927{
928	local ns="$1"
929
930	:> "$capout"
931
932	if $capture; then
933		local capuser capfile
934		if [ -z $SUDO_USER ]; then
935			capuser=""
936		else
937			capuser="-Z $SUDO_USER"
938		fi
939
940		capfile=$(printf "mp_join-%02u-%s.pcap" "$MPTCP_LIB_TEST_COUNTER" "$ns")
941
942		echo "Capturing traffic for test $MPTCP_LIB_TEST_COUNTER into $capfile"
943		ip netns exec "$ns" tcpdump -i any -s 65535 -B 32768 $capuser -w "$capfile" > "$capout" 2>&1 &
944		cappid=$!
945
946		sleep 1
947	fi
948}
949
950cond_stop_capture()
951{
952	if $capture; then
953		sleep 1
954		kill $cappid
955		cat "$capout"
956	fi
957}
958
959get_port()
960{
961	echo "$((10000 + MPTCP_LIB_TEST_COUNTER - 1))"
962}
963
964do_transfer()
965{
966	local listener_ns="$1"
967	local connector_ns="$2"
968	local cl_proto="$3"
969	local srv_proto="$4"
970	local connect_addr="$5"
971	local port
972
973	local FAILING_LINKS=${FAILING_LINKS:-""}
974	local fastclose=${fastclose:-""}
975	local speed=${speed:-"fast"}
976	local bind_addr=${bind_addr:-"::"}
977	local listener_in="${sin}"
978	local connector_in="${cin}"
979	port=$(get_port)
980
981	:> "$cout"
982	:> "$sout"
983
984	cond_start_capture ${listener_ns}
985
986	NSTAT_HISTORY=/tmp/${listener_ns}.nstat ip netns exec ${listener_ns} \
987		nstat -n
988	NSTAT_HISTORY=/tmp/${connector_ns}.nstat ip netns exec ${connector_ns} \
989		nstat -n
990
991	local extra_args
992	if [ $speed = "fast" ]; then
993		extra_args="-j"
994	elif [ $speed = "slow" ]; then
995		extra_args="-r 50"
996	elif [ $speed -gt 0 ]; then
997		extra_args="-r ${speed}"
998	fi
999
1000	local extra_cl_args=""
1001	local extra_srv_args=""
1002	local trunc_size=""
1003	if [ -n "${fastclose}" ]; then
1004		if [ ${test_linkfail} -le 1 ]; then
1005			fail_test "fastclose tests need test_linkfail argument"
1006			return 1
1007		fi
1008
1009		# disconnect
1010		trunc_size=${test_linkfail}
1011		local side=${fastclose}
1012
1013		if [ ${side} = "client" ]; then
1014			extra_cl_args="-f ${test_linkfail}"
1015			extra_srv_args="-f -1"
1016		elif [ ${side} = "server" ]; then
1017			extra_srv_args="-f ${test_linkfail}"
1018			extra_cl_args="-f -1"
1019		else
1020			fail_test "wrong/unknown fastclose spec ${side}"
1021			return 1
1022		fi
1023	fi
1024
1025	extra_srv_args="$extra_args $extra_srv_args"
1026	if [ "$test_linkfail" -gt 1 ];then
1027		listener_in="${sinfail}"
1028	fi
1029	timeout ${timeout_test} \
1030		ip netns exec ${listener_ns} \
1031			./mptcp_connect -t ${timeout_poll} -l -p ${port} -s ${srv_proto} \
1032				${extra_srv_args} "${bind_addr}" < "${listener_in}" > "${sout}" &
1033	local spid=$!
1034
1035	mptcp_lib_wait_local_port_listen "${listener_ns}" "${port}"
1036
1037	extra_cl_args="$extra_args $extra_cl_args"
1038	if [ "$test_linkfail" -eq 0 ];then
1039		timeout ${timeout_test} \
1040			ip netns exec ${connector_ns} \
1041				./mptcp_connect -t ${timeout_poll} -p $port -s ${cl_proto} \
1042					$extra_cl_args $connect_addr < "$cin" > "$cout" &
1043	elif [ "$test_linkfail" -eq 1 ] || [ "$test_linkfail" -eq 2 ];then
1044		connector_in="${cinsent}"
1045		( cat "$cinfail" ; sleep 2; link_failure $listener_ns ; cat "$cinfail" ) | \
1046			tee "$cinsent" | \
1047			timeout ${timeout_test} \
1048				ip netns exec ${connector_ns} \
1049					./mptcp_connect -t ${timeout_poll} -p $port -s ${cl_proto} \
1050						$extra_cl_args $connect_addr > "$cout" &
1051	else
1052		connector_in="${cinsent}"
1053		tee "$cinsent" < "$cinfail" | \
1054			timeout ${timeout_test} \
1055				ip netns exec ${connector_ns} \
1056					./mptcp_connect -t ${timeout_poll} -p $port -s ${cl_proto} \
1057						$extra_cl_args $connect_addr > "$cout" &
1058	fi
1059	local cpid=$!
1060
1061	pm_nl_set_endpoint $listener_ns $connector_ns $connect_addr
1062	check_cestab $listener_ns $connector_ns
1063
1064	wait $cpid
1065	local retc=$?
1066	wait $spid
1067	local rets=$?
1068
1069	cond_stop_capture
1070
1071	NSTAT_HISTORY=/tmp/${listener_ns}.nstat ip netns exec ${listener_ns} \
1072		nstat | grep Tcp > /tmp/${listener_ns}.out
1073	NSTAT_HISTORY=/tmp/${connector_ns}.nstat ip netns exec ${connector_ns} \
1074		nstat | grep Tcp > /tmp/${connector_ns}.out
1075
1076	if [ ${rets} -ne 0 ] || [ ${retc} -ne 0 ]; then
1077		fail_test "client exit code $retc, server $rets"
1078		mptcp_lib_pr_err_stats "${listener_ns}" "${connector_ns}" "${port}" \
1079			"/tmp/${listener_ns}.out" "/tmp/${connector_ns}.out"
1080		return 1
1081	fi
1082
1083	check_transfer $listener_in $cout "file received by client" $trunc_size
1084	retc=$?
1085	check_transfer $connector_in $sout "file received by server" $trunc_size
1086	rets=$?
1087
1088	[ $retc -eq 0 ] && [ $rets -eq 0 ]
1089}
1090
1091make_file()
1092{
1093	local name=$1
1094	local who=$2
1095	local size=$3
1096
1097	mptcp_lib_make_file $name 1024 $size
1098
1099	print_info "Test file (size $size KB) for $who"
1100}
1101
1102run_tests()
1103{
1104	local listener_ns="$1"
1105	local connector_ns="$2"
1106	local connect_addr="$3"
1107
1108	local size
1109	local test_linkfail=${test_linkfail:-0}
1110
1111	# The values above 2 are reused to make test files
1112	# with the given sizes (KB)
1113	if [ "$test_linkfail" -gt 2 ]; then
1114		size=$test_linkfail
1115
1116		if [ -z "$cinfail" ]; then
1117			cinfail=$(mktemp)
1118		fi
1119		make_file "$cinfail" "client" $size
1120	# create the input file for the failure test when
1121	# the first failure test run
1122	elif [ "$test_linkfail" -ne 0 ] && [ -z "$cinfail" ]; then
1123		# the client file must be considerably larger
1124		# of the maximum expected cwin value, or the
1125		# link utilization will be not predicable
1126		size=$((RANDOM%2))
1127		size=$((size+1))
1128		size=$((size*8192))
1129		size=$((size + ( RANDOM % 8192) ))
1130
1131		cinfail=$(mktemp)
1132		make_file "$cinfail" "client" $size
1133	fi
1134
1135	if [ "$test_linkfail" -gt 2 ]; then
1136		size=$test_linkfail
1137
1138		if [ -z "$sinfail" ]; then
1139			sinfail=$(mktemp)
1140		fi
1141		make_file "$sinfail" "server" $size
1142	elif [ "$test_linkfail" -eq 2 ] && [ -z "$sinfail" ]; then
1143		size=$((RANDOM%16))
1144		size=$((size+1))
1145		size=$((size*2048))
1146
1147		sinfail=$(mktemp)
1148		make_file "$sinfail" "server" $size
1149	fi
1150
1151	do_transfer ${listener_ns} ${connector_ns} MPTCP MPTCP ${connect_addr}
1152}
1153
1154dump_stats()
1155{
1156	echo Server ns stats
1157	ip netns exec $ns1 nstat -as | grep Tcp
1158	echo Client ns stats
1159	ip netns exec $ns2 nstat -as | grep Tcp
1160}
1161
1162chk_csum_nr()
1163{
1164	local csum_ns1=${1:-0}
1165	local csum_ns2=${2:-0}
1166	local count
1167	local extra_msg=""
1168	local allow_multi_errors_ns1=0
1169	local allow_multi_errors_ns2=0
1170
1171	if [[ "${csum_ns1}" = "+"* ]]; then
1172		allow_multi_errors_ns1=1
1173		csum_ns1=${csum_ns1:1}
1174	fi
1175	if [[ "${csum_ns2}" = "+"* ]]; then
1176		allow_multi_errors_ns2=1
1177		csum_ns2=${csum_ns2:1}
1178	fi
1179
1180	print_check "checksum server"
1181	count=$(mptcp_lib_get_counter ${ns1} "MPTcpExtDataCsumErr")
1182	if [ -n "$count" ] && [ "$count" != "$csum_ns1" ]; then
1183		extra_msg+=" ns1=$count"
1184	fi
1185	if [ -z "$count" ]; then
1186		print_skip
1187	elif { [ "$count" != $csum_ns1 ] && [ $allow_multi_errors_ns1 -eq 0 ]; } ||
1188	     { [ "$count" -lt $csum_ns1 ] && [ $allow_multi_errors_ns1 -eq 1 ]; }; then
1189		fail_test "got $count data checksum error[s] expected $csum_ns1"
1190	else
1191		print_ok
1192	fi
1193
1194	print_check "checksum client"
1195	count=$(mptcp_lib_get_counter ${ns2} "MPTcpExtDataCsumErr")
1196	if [ -n "$count" ] && [ "$count" != "$csum_ns2" ]; then
1197		extra_msg+=" ns2=$count"
1198	fi
1199	if [ -z "$count" ]; then
1200		print_skip
1201	elif { [ "$count" != $csum_ns2 ] && [ $allow_multi_errors_ns2 -eq 0 ]; } ||
1202	     { [ "$count" -lt $csum_ns2 ] && [ $allow_multi_errors_ns2 -eq 1 ]; }; then
1203		fail_test "got $count data checksum error[s] expected $csum_ns2"
1204	else
1205		print_ok
1206	fi
1207
1208	print_info "$extra_msg"
1209}
1210
1211chk_fail_nr()
1212{
1213	local fail_tx=$1
1214	local fail_rx=$2
1215	local ns_invert=${3:-""}
1216	local count
1217	local ns_tx=$ns1
1218	local ns_rx=$ns2
1219	local tx="server"
1220	local rx="client"
1221	local extra_msg=""
1222	local allow_tx_lost=0
1223	local allow_rx_lost=0
1224
1225	if [[ $ns_invert = "invert" ]]; then
1226		ns_tx=$ns2
1227		ns_rx=$ns1
1228		tx="client"
1229		rx="server"
1230	fi
1231
1232	if [[ "${fail_tx}" = "-"* ]]; then
1233		allow_tx_lost=1
1234		fail_tx=${fail_tx:1}
1235	fi
1236	if [[ "${fail_rx}" = "-"* ]]; then
1237		allow_rx_lost=1
1238		fail_rx=${fail_rx:1}
1239	fi
1240
1241	print_check "fail tx ${tx}"
1242	count=$(mptcp_lib_get_counter ${ns_tx} "MPTcpExtMPFailTx")
1243	if [ -n "$count" ] && [ "$count" != "$fail_tx" ]; then
1244		extra_msg+=" tx=$count"
1245	fi
1246	if [ -z "$count" ]; then
1247		print_skip
1248	elif { [ "$count" != "$fail_tx" ] && [ $allow_tx_lost -eq 0 ]; } ||
1249	     { [ "$count" -gt "$fail_tx" ] && [ $allow_tx_lost -eq 1 ]; }; then
1250		fail_test "got $count MP_FAIL[s] TX expected $fail_tx"
1251	else
1252		print_ok
1253	fi
1254
1255	print_check "fail rx ${rx}"
1256	count=$(mptcp_lib_get_counter ${ns_rx} "MPTcpExtMPFailRx")
1257	if [ -n "$count" ] && [ "$count" != "$fail_rx" ]; then
1258		extra_msg+=" rx=$count"
1259	fi
1260	if [ -z "$count" ]; then
1261		print_skip
1262	elif { [ "$count" != "$fail_rx" ] && [ $allow_rx_lost -eq 0 ]; } ||
1263	     { [ "$count" -gt "$fail_rx" ] && [ $allow_rx_lost -eq 1 ]; }; then
1264		fail_test "got $count MP_FAIL[s] RX expected $fail_rx"
1265	else
1266		print_ok
1267	fi
1268
1269	print_info "$extra_msg"
1270}
1271
1272chk_fclose_nr()
1273{
1274	local fclose_tx=$1
1275	local fclose_rx=$2
1276	local ns_invert=$3
1277	local count
1278	local ns_tx=$ns2
1279	local ns_rx=$ns1
1280	local tx="client"
1281	local rx="server"
1282
1283	if [[ $ns_invert = "invert" ]]; then
1284		ns_tx=$ns1
1285		ns_rx=$ns2
1286		tx="server"
1287		rx="client"
1288	fi
1289
1290	print_check "fast close tx ${tx}"
1291	count=$(mptcp_lib_get_counter ${ns_tx} "MPTcpExtMPFastcloseTx")
1292	if [ -z "$count" ]; then
1293		print_skip
1294	elif [ "$count" != "$fclose_tx" ]; then
1295		fail_test "got $count MP_FASTCLOSE[s] TX expected $fclose_tx"
1296	else
1297		print_ok
1298	fi
1299
1300	print_check "fast close rx ${rx}"
1301	count=$(mptcp_lib_get_counter ${ns_rx} "MPTcpExtMPFastcloseRx")
1302	if [ -z "$count" ]; then
1303		print_skip
1304	elif [ "$count" != "$fclose_rx" ]; then
1305		fail_test "got $count MP_FASTCLOSE[s] RX expected $fclose_rx"
1306	else
1307		print_ok
1308	fi
1309}
1310
1311chk_rst_nr()
1312{
1313	local rst_tx=$1
1314	local rst_rx=$2
1315	local ns_invert=${3:-""}
1316	local count
1317	local ns_tx=$ns1
1318	local ns_rx=$ns2
1319	local tx="server"
1320	local rx="client"
1321
1322	if [[ $ns_invert = "invert" ]]; then
1323		ns_tx=$ns2
1324		ns_rx=$ns1
1325		tx="client"
1326		rx="server"
1327	fi
1328
1329	print_check "reset tx ${tx}"
1330	count=$(mptcp_lib_get_counter ${ns_tx} "MPTcpExtMPRstTx")
1331	if [ -z "$count" ]; then
1332		print_skip
1333	# accept more rst than expected except if we don't expect any
1334	elif { [ $rst_tx -ne 0 ] && [ $count -lt $rst_tx ]; } ||
1335	     { [ $rst_tx -eq 0 ] && [ $count -ne 0 ]; }; then
1336		fail_test "got $count MP_RST[s] TX expected $rst_tx"
1337	else
1338		print_ok
1339	fi
1340
1341	print_check "reset rx ${rx}"
1342	count=$(mptcp_lib_get_counter ${ns_rx} "MPTcpExtMPRstRx")
1343	if [ -z "$count" ]; then
1344		print_skip
1345	# accept more rst than expected except if we don't expect any
1346	elif { [ $rst_rx -ne 0 ] && [ $count -lt $rst_rx ]; } ||
1347	     { [ $rst_rx -eq 0 ] && [ $count -ne 0 ]; }; then
1348		fail_test "got $count MP_RST[s] RX expected $rst_rx"
1349	else
1350		print_ok
1351	fi
1352}
1353
1354chk_infi_nr()
1355{
1356	local infi_tx=$1
1357	local infi_rx=$2
1358	local count
1359
1360	print_check "infi tx client"
1361	count=$(mptcp_lib_get_counter ${ns2} "MPTcpExtInfiniteMapTx")
1362	if [ -z "$count" ]; then
1363		print_skip
1364	elif [ "$count" != "$infi_tx" ]; then
1365		fail_test "got $count infinite map[s] TX expected $infi_tx"
1366	else
1367		print_ok
1368	fi
1369
1370	print_check "infi rx server"
1371	count=$(mptcp_lib_get_counter ${ns1} "MPTcpExtInfiniteMapRx")
1372	if [ -z "$count" ]; then
1373		print_skip
1374	elif [ "$count" != "$infi_rx" ]; then
1375		fail_test "got $count infinite map[s] RX expected $infi_rx"
1376	else
1377		print_ok
1378	fi
1379}
1380
1381chk_join_tx_nr()
1382{
1383	local syn_tx=${join_syn_tx:-0}
1384	local create=${join_create_err:-0}
1385	local bind=${join_bind_err:-0}
1386	local connect=${join_connect_err:-0}
1387	local rc=${KSFT_PASS}
1388	local count
1389
1390	count=$(mptcp_lib_get_counter ${ns2} "MPTcpExtMPJoinSynTx")
1391	if [ -z "$count" ]; then
1392		rc=${KSFT_SKIP}
1393	elif [ "$count" != "$syn_tx" ]; then
1394		rc=${KSFT_FAIL}
1395		print_check "syn tx"
1396		fail_test "got $count JOIN[s] syn tx expected $syn_tx"
1397	fi
1398
1399	count=$(mptcp_lib_get_counter ${ns2} "MPTcpExtMPJoinSynTxCreatSkErr")
1400	if [ -z "$count" ]; then
1401		rc=${KSFT_SKIP}
1402	elif [ "$count" != "$create" ]; then
1403		rc=${KSFT_FAIL}
1404		print_check "syn tx create socket error"
1405		fail_test "got $count JOIN[s] syn tx create socket error expected $create"
1406	fi
1407
1408	count=$(mptcp_lib_get_counter ${ns2} "MPTcpExtMPJoinSynTxBindErr")
1409	if [ -z "$count" ]; then
1410		rc=${KSFT_SKIP}
1411	elif [ "$count" != "$bind" ]; then
1412		rc=${KSFT_FAIL}
1413		print_check "syn tx bind error"
1414		fail_test "got $count JOIN[s] syn tx bind error expected $bind"
1415	fi
1416
1417	count=$(mptcp_lib_get_counter ${ns2} "MPTcpExtMPJoinSynTxConnectErr")
1418	if [ -z "$count" ]; then
1419		rc=${KSFT_SKIP}
1420	elif [ "$count" != "$connect" ]; then
1421		rc=${KSFT_FAIL}
1422		print_check "syn tx connect error"
1423		fail_test "got $count JOIN[s] syn tx connect error expected $connect"
1424	fi
1425
1426	print_results "join Tx" ${rc}
1427}
1428
1429chk_fallback_nr()
1430{
1431	local infinite_map_tx=${fb_infinite_map_tx:-0}
1432	local dss_corruption=${fb_dss_corruption:-0}
1433	local simult_conn=${fb_simult_conn:-0}
1434	local mpc_passive=${fb_mpc_passive:-0}
1435	local mpc_active=${fb_mpc_active:-0}
1436	local mpc_data=${fb_mpc_data:-0}
1437	local md5_sig=${fb_md5_sig:-0}
1438	local dss=${fb_dss:-0}
1439	local rc=${KSFT_PASS}
1440	local ns=$1
1441	local count
1442
1443	count=$(mptcp_lib_get_counter ${!ns} "MPTcpExtInfiniteMapTx")
1444	if [ -z "$count" ]; then
1445		rc=${KSFT_SKIP}
1446	elif [ "$count" != "$infinite_map_tx" ]; then
1447		rc=${KSFT_FAIL}
1448		print_check "$ns infinite map tx fallback"
1449		fail_test "got $count infinite map tx fallback[s] in $ns expected $infinite_map_tx"
1450	fi
1451
1452	count=$(mptcp_lib_get_counter ${!ns} "MPTcpExtDSSCorruptionFallback")
1453	if [ -z "$count" ]; then
1454		rc=${KSFT_SKIP}
1455	elif [ "$count" != "$dss_corruption" ]; then
1456		rc=${KSFT_FAIL}
1457		print_check "$ns dss corruption fallback"
1458		fail_test "got $count dss corruption fallback[s] in $ns expected $dss_corruption"
1459	fi
1460
1461	count=$(mptcp_lib_get_counter ${!ns} "MPTcpExtSimultConnectFallback")
1462	if [ -z "$count" ]; then
1463		rc=${KSFT_SKIP}
1464	elif [ "$count" != "$simult_conn" ]; then
1465		rc=${KSFT_FAIL}
1466		print_check "$ns simult conn fallback"
1467		fail_test "got $count simult conn fallback[s] in $ns expected $simult_conn"
1468	fi
1469
1470	count=$(mptcp_lib_get_counter ${!ns} "MPTcpExtMPCapableFallbackACK")
1471	if [ -z "$count" ]; then
1472		rc=${KSFT_SKIP}
1473	elif [ "$count" != "$mpc_passive" ]; then
1474		rc=${KSFT_FAIL}
1475		print_check "$ns mpc passive fallback"
1476		fail_test "got $count mpc passive fallback[s] in $ns expected $mpc_passive"
1477	fi
1478
1479	count=$(mptcp_lib_get_counter ${!ns} "MPTcpExtMPCapableFallbackSYNACK")
1480	if [ -z "$count" ]; then
1481		rc=${KSFT_SKIP}
1482	elif [ "$count" != "$mpc_active" ]; then
1483		rc=${KSFT_FAIL}
1484		print_check "$ns mpc active fallback"
1485		fail_test "got $count mpc active fallback[s] in $ns expected $mpc_active"
1486	fi
1487
1488	count=$(mptcp_lib_get_counter ${!ns} "MPTcpExtMPCapableDataFallback")
1489	if [ -z "$count" ]; then
1490		rc=${KSFT_SKIP}
1491	elif [ "$count" != "$mpc_data" ]; then
1492		rc=${KSFT_FAIL}
1493		print_check "$ns mpc data fallback"
1494		fail_test "got $count mpc data fallback[s] in $ns expected $mpc_data"
1495	fi
1496
1497	count=$(mptcp_lib_get_counter ${!ns} "MPTcpExtMD5SigFallback")
1498	if [ -z "$count" ]; then
1499		rc=${KSFT_SKIP}
1500	elif [ "$count" != "$md5_sig" ]; then
1501		rc=${KSFT_FAIL}
1502		print_check "$ns MD5 Sig fallback"
1503		fail_test "got $count MD5 Sig fallback[s] in $ns expected $md5_sig"
1504	fi
1505
1506	count=$(mptcp_lib_get_counter ${!ns} "MPTcpExtDssFallback")
1507	if [ -z "$count" ]; then
1508		rc=${KSFT_SKIP}
1509	elif [ "$count" != "$dss" ]; then
1510		rc=${KSFT_FAIL}
1511		print_check "$ns dss fallback"
1512		fail_test "got $count dss fallback[s] in $ns expected $dss"
1513	fi
1514
1515	return $rc
1516}
1517
1518chk_fallback_nr_all()
1519{
1520	local netns=("ns1" "ns2")
1521	local fb_ns=("fb_ns1" "fb_ns2")
1522	local rc=${KSFT_PASS}
1523
1524	for i in 0 1; do
1525		if [ -n "${!fb_ns[i]}" ]; then
1526			eval "${!fb_ns[i]}" \
1527				chk_fallback_nr ${netns[i]} || rc=${?}
1528		else
1529			chk_fallback_nr ${netns[i]} || rc=${?}
1530		fi
1531	done
1532
1533	if [ "${rc}" != "${KSFT_PASS}" ]; then
1534		print_results "fallback" ${rc}
1535	fi
1536}
1537
1538chk_join_nr()
1539{
1540	local syn_nr=$1
1541	local syn_ack_nr=$2
1542	local ack_nr=$3
1543	local syn_rej=${join_syn_rej:-0}
1544	local csum_ns1=${join_csum_ns1:-0}
1545	local csum_ns2=${join_csum_ns2:-0}
1546	local fail_nr=${join_fail_nr:-0}
1547	local rst_nr=${join_rst_nr:-0}
1548	local infi_nr=${join_infi_nr:-0}
1549	local corrupted_pkts=${join_corrupted_pkts:-0}
1550	local rc=${KSFT_PASS}
1551	local count
1552	local with_cookie
1553
1554	if [ "${corrupted_pkts}" -gt 0 ]; then
1555		print_info "${corrupted_pkts} corrupted pkts"
1556	fi
1557
1558	count=$(mptcp_lib_get_counter ${ns1} "MPTcpExtMPJoinSynRx")
1559	if [ -z "$count" ]; then
1560		rc=${KSFT_SKIP}
1561	elif [ "$count" != "$syn_nr" ]; then
1562		rc=${KSFT_FAIL}
1563		print_check "syn rx"
1564		fail_test "got $count JOIN[s] syn rx expected $syn_nr"
1565	fi
1566
1567	with_cookie=$(ip netns exec $ns2 sysctl -n net.ipv4.tcp_syncookies)
1568	count=$(mptcp_lib_get_counter ${ns2} "MPTcpExtMPJoinSynAckRx")
1569	if [ -z "$count" ]; then
1570		rc=${KSFT_SKIP}
1571	elif [ "$count" != "$syn_ack_nr" ]; then
1572		# simult connections exceeding the limit with cookie enabled could go up to
1573		# synack validation as the conn limit can be enforced reliably only after
1574		# the subflow creation
1575		if [ "$with_cookie" != 2 ] || [ "$count" -le "$syn_ack_nr" ] || [ "$count" -gt "$syn_nr" ]; then
1576			rc=${KSFT_FAIL}
1577			print_check "synack rx"
1578			fail_test "got $count JOIN[s] synack rx expected $syn_ack_nr"
1579		fi
1580	fi
1581
1582	count=$(mptcp_lib_get_counter ${ns2} "MPTcpExtMPJoinSynAckHMacFailure")
1583	if [ -z "$count" ]; then
1584		rc=${KSFT_SKIP}
1585	elif [ "$count" != "0" ]; then
1586		rc=${KSFT_FAIL}
1587		print_check "synack HMAC"
1588		fail_test "got $count JOIN[s] synack HMAC failure expected 0"
1589	fi
1590
1591	count=$(mptcp_lib_get_counter ${ns1} "MPTcpExtMPJoinAckRx")
1592	if [ -z "$count" ]; then
1593		rc=${KSFT_SKIP}
1594	elif [ "$count" != "$ack_nr" ]; then
1595		rc=${KSFT_FAIL}
1596		print_check "ack rx"
1597		fail_test "got $count JOIN[s] ack rx expected $ack_nr"
1598	fi
1599
1600	count=$(mptcp_lib_get_counter ${ns1} "MPTcpExtMPJoinAckHMacFailure")
1601	if [ -z "$count" ]; then
1602		rc=${KSFT_SKIP}
1603	elif [ "$count" != "0" ]; then
1604		rc=${KSFT_FAIL}
1605		print_check "ack HMAC"
1606		fail_test "got $count JOIN[s] ack HMAC failure expected 0"
1607	fi
1608
1609	count=$(mptcp_lib_get_counter ${ns1} "MPTcpExtMPJoinRejected")
1610	if [ -z "$count" ]; then
1611		rc=${KSFT_SKIP}
1612	elif [ "$count" != "$syn_rej" ]; then
1613		rc=${KSFT_FAIL}
1614		print_check "syn rejected"
1615		fail_test "got $count JOIN[s] syn rejected expected $syn_rej"
1616	fi
1617
1618	print_results "join Rx" ${rc}
1619
1620	join_syn_tx="${join_syn_tx:-${syn_nr}}" \
1621		chk_join_tx_nr
1622
1623	chk_fallback_nr_all
1624
1625	if $validate_checksum; then
1626		chk_csum_nr $csum_ns1 $csum_ns2
1627		chk_fail_nr $fail_nr $fail_nr
1628		chk_rst_nr $rst_nr $rst_nr
1629		chk_infi_nr $infi_nr $infi_nr
1630	fi
1631}
1632
1633# a negative value for 'stale_max' means no upper bound:
1634# for bidirectional transfer, if one peer sleep for a while
1635# - as these tests do - we can have a quite high number of
1636# stale/recover conversions, proportional to
1637# sleep duration/ MPTCP-level RTX interval.
1638chk_stale_nr()
1639{
1640	local ns=$1
1641	local stale_min=$2
1642	local stale_max=$3
1643	local stale_delta=$4
1644	local dump_stats
1645	local stale_nr
1646	local recover_nr
1647
1648	print_check "stale"
1649
1650	stale_nr=$(mptcp_lib_get_counter ${ns} "MPTcpExtSubflowStale")
1651	recover_nr=$(mptcp_lib_get_counter ${ns} "MPTcpExtSubflowRecover")
1652	if [ -z "$stale_nr" ] || [ -z "$recover_nr" ]; then
1653		print_skip
1654	elif [ $stale_nr -lt $stale_min ] ||
1655	   { [ $stale_max -gt 0 ] && [ $stale_nr -gt $stale_max ]; } ||
1656	   [ $((stale_nr - recover_nr)) -ne $stale_delta ]; then
1657		fail_test "got $stale_nr stale[s] $recover_nr recover[s], " \
1658		     " expected stale in range [$stale_min..$stale_max]," \
1659		     " stale-recover delta $stale_delta"
1660		dump_stats=1
1661	else
1662		print_ok
1663	fi
1664
1665	if [ "${dump_stats}" = 1 ]; then
1666		echo $ns stats
1667		ip netns exec $ns ip -s link show
1668		ip netns exec $ns nstat -as | grep MPTcp
1669	fi
1670}
1671
1672chk_add_nr()
1673{
1674	local add_nr=$1
1675	local echo_nr=$2
1676	local port_nr=${3:-0}
1677	local ns_invert=${4:-""}
1678	local syn_nr=$port_nr
1679	local syn_ack_nr=$port_nr
1680	local ack_nr=$port_nr
1681	local mis_syn_nr=0
1682	local mis_ack_nr=0
1683	local ns_tx=$ns1
1684	local ns_rx=$ns2
1685	local tx=""
1686	local rx=""
1687	local count
1688
1689	if [[ $ns_invert = "invert" ]]; then
1690		ns_tx=$ns2
1691		ns_rx=$ns1
1692		tx=" client"
1693		rx=" server"
1694	fi
1695
1696	print_check "add addr rx${rx}"
1697	count=$(mptcp_lib_get_counter ${ns_rx} "MPTcpExtAddAddr")
1698	if [ -z "$count" ]; then
1699		print_skip
1700	# Tolerate more ADD_ADDR then expected (if any), due to retransmissions
1701	elif [ "$count" != "$add_nr" ] &&
1702	     { [ "$add_nr" -eq 0 ] || [ "$count" -lt "$add_nr" ]; }; then
1703		fail_test "got $count ADD_ADDR[s] expected $add_nr"
1704	else
1705		print_ok
1706	fi
1707
1708	print_check "add addr echo rx${tx}"
1709	count=$(mptcp_lib_get_counter ${ns_tx} "MPTcpExtEchoAdd")
1710	if [ -z "$count" ]; then
1711		print_skip
1712	elif [ "$count" != "$echo_nr" ]; then
1713		fail_test "got $count ADD_ADDR echo[s] expected $echo_nr"
1714	else
1715		print_ok
1716	fi
1717
1718	if [ $port_nr -gt 0 ]; then
1719		print_check "add addr rx with port${rx}"
1720		count=$(mptcp_lib_get_counter ${ns_rx} "MPTcpExtPortAdd")
1721		if [ -z "$count" ]; then
1722			print_skip
1723		elif [ "$count" != "$port_nr" ]; then
1724			fail_test "got $count ADD_ADDR[s] with a port-number expected $port_nr"
1725		else
1726			print_ok
1727		fi
1728
1729		print_check "syn rx port${tx}"
1730		count=$(mptcp_lib_get_counter ${ns_tx} "MPTcpExtMPJoinPortSynRx")
1731		if [ -z "$count" ]; then
1732			print_skip
1733		elif [ "$count" != "$syn_nr" ]; then
1734			fail_test "got $count JOIN[s] syn with a different \
1735				   port-number expected $syn_nr"
1736		else
1737			print_ok
1738		fi
1739
1740		print_check "synack rx port${rx}"
1741		count=$(mptcp_lib_get_counter ${ns_rx} "MPTcpExtMPJoinPortSynAckRx")
1742		if [ -z "$count" ]; then
1743			print_skip
1744		elif [ "$count" != "$syn_ack_nr" ]; then
1745			fail_test "got $count JOIN[s] synack with a different \
1746				   port-number expected $syn_ack_nr"
1747		else
1748			print_ok
1749		fi
1750
1751		print_check "ack rx port${tx}"
1752		count=$(mptcp_lib_get_counter ${ns_tx} "MPTcpExtMPJoinPortAckRx")
1753		if [ -z "$count" ]; then
1754			print_skip
1755		elif [ "$count" != "$ack_nr" ]; then
1756			fail_test "got $count JOIN[s] ack with a different \
1757				   port-number expected $ack_nr"
1758		else
1759			print_ok
1760		fi
1761
1762		print_check "syn rx port mismatch${tx}"
1763		count=$(mptcp_lib_get_counter ${ns_tx} "MPTcpExtMismatchPortSynRx")
1764		if [ -z "$count" ]; then
1765			print_skip
1766		elif [ "$count" != "$mis_syn_nr" ]; then
1767			fail_test "got $count JOIN[s] syn with a mismatched \
1768				   port-number expected $mis_syn_nr"
1769		else
1770			print_ok
1771		fi
1772
1773		print_check "ack rx port mismatch${tx}"
1774		count=$(mptcp_lib_get_counter ${ns_tx} "MPTcpExtMismatchPortAckRx")
1775		if [ -z "$count" ]; then
1776			print_skip
1777		elif [ "$count" != "$mis_ack_nr" ]; then
1778			fail_test "got $count JOIN[s] ack with a mismatched \
1779				   port-number expected $mis_ack_nr"
1780		else
1781			print_ok
1782		fi
1783	fi
1784}
1785
1786chk_add_tx_nr()
1787{
1788	local add_tx_nr=$1
1789	local echo_tx_nr=$2
1790	local count
1791
1792	print_check "add addr tx"
1793	count=$(mptcp_lib_get_counter ${ns1} "MPTcpExtAddAddrTx")
1794	if [ -z "$count" ]; then
1795		print_skip
1796	# Tolerate more ADD_ADDR then expected (if any), due to retransmissions
1797	elif [ "$count" != "$add_tx_nr" ] &&
1798	     { [ "$add_tx_nr" -eq 0 ] || [ "$count" -lt "$add_tx_nr" ]; }; then
1799		fail_test "got $count ADD_ADDR[s] TX, expected $add_tx_nr"
1800	else
1801		print_ok
1802	fi
1803
1804	print_check "add addr echo tx"
1805	count=$(mptcp_lib_get_counter ${ns2} "MPTcpExtEchoAddTx")
1806	if [ -z "$count" ]; then
1807		print_skip
1808	elif [ "$count" != "$echo_tx_nr" ]; then
1809		fail_test "got $count ADD_ADDR echo[s] TX, expected $echo_tx_nr"
1810	else
1811		print_ok
1812	fi
1813}
1814
1815chk_rm_nr()
1816{
1817	local rm_addr_nr=$1
1818	local rm_subflow_nr=$2
1819	local invert
1820	local simult
1821	local count
1822	local addr_ns=$ns1
1823	local subflow_ns=$ns2
1824	local addr="server"
1825	local subflow="client"
1826	local extra_msg=""
1827
1828	shift 2
1829	while [ -n "$1" ]; do
1830		[ "$1" = "invert" ] && invert=true
1831		[ "$1" = "simult" ] && simult=true
1832		shift
1833	done
1834
1835	if [ "$invert" = "true" ]; then
1836		addr_ns=$ns2
1837		subflow_ns=$ns1
1838		addr="client"
1839		subflow="server"
1840	fi
1841
1842	print_check "rm addr rx ${addr}"
1843	count=$(mptcp_lib_get_counter ${addr_ns} "MPTcpExtRmAddr")
1844	if [ -z "$count" ]; then
1845		print_skip
1846	elif [ "$count" != "$rm_addr_nr" ]; then
1847		fail_test "got $count RM_ADDR[s] expected $rm_addr_nr"
1848	else
1849		print_ok
1850	fi
1851
1852	print_check "rm subflow ${subflow}"
1853	count=$(mptcp_lib_get_counter ${subflow_ns} "MPTcpExtRmSubflow")
1854	if [ -z "$count" ]; then
1855		print_skip
1856	elif [ -n "$simult" ]; then
1857		local cnt suffix
1858
1859		cnt=$(mptcp_lib_get_counter ${addr_ns} "MPTcpExtRmSubflow")
1860
1861		# in case of simult flush, the subflow removal count on each side is
1862		# unreliable
1863		count=$((count + cnt))
1864		if [ "$count" != "$rm_subflow_nr" ]; then
1865			suffix="$count in [$rm_subflow_nr:$((rm_subflow_nr*2))]"
1866			extra_msg="simult"
1867		fi
1868		if [ $count -ge "$rm_subflow_nr" ] && \
1869		   [ "$count" -le "$((rm_subflow_nr *2 ))" ]; then
1870			print_ok "$suffix"
1871		else
1872			fail_test "got $count RM_SUBFLOW[s] expected in range [$rm_subflow_nr:$((rm_subflow_nr*2))]"
1873		fi
1874	elif [ "$count" != "$rm_subflow_nr" ]; then
1875		fail_test "got $count RM_SUBFLOW[s] expected $rm_subflow_nr"
1876	else
1877		print_ok
1878	fi
1879
1880	print_info "$extra_msg"
1881}
1882
1883chk_rm_tx_nr()
1884{
1885	local rm_addr_tx_nr=$1
1886
1887	print_check "rm addr tx client"
1888	count=$(mptcp_lib_get_counter ${ns2} "MPTcpExtRmAddrTx")
1889	if [ -z "$count" ]; then
1890		print_skip
1891	elif [ "$count" != "$rm_addr_tx_nr" ]; then
1892		fail_test "got $count RM_ADDR[s] expected $rm_addr_tx_nr"
1893	else
1894		print_ok
1895	fi
1896}
1897
1898chk_prio_nr()
1899{
1900	local mp_prio_nr_tx=$1
1901	local mp_prio_nr_rx=$2
1902	local mpj_syn=$3
1903	local mpj_syn_ack=$4
1904	local count
1905
1906	print_check "mp_prio tx server"
1907	count=$(mptcp_lib_get_counter ${ns1} "MPTcpExtMPPrioTx")
1908	if [ -z "$count" ]; then
1909		print_skip
1910	elif [ "$count" != "$mp_prio_nr_tx" ]; then
1911		fail_test "got $count MP_PRIO[s] TX expected $mp_prio_nr_tx"
1912	else
1913		print_ok
1914	fi
1915
1916	print_check "mp_prio rx client"
1917	count=$(mptcp_lib_get_counter ${ns1} "MPTcpExtMPPrioRx")
1918	if [ -z "$count" ]; then
1919		print_skip
1920	elif [ "$count" != "$mp_prio_nr_rx" ]; then
1921		fail_test "got $count MP_PRIO[s] RX expected $mp_prio_nr_rx"
1922	else
1923		print_ok
1924	fi
1925
1926	print_check "syn backup"
1927	count=$(mptcp_lib_get_counter ${ns1} "MPTcpExtMPJoinSynBackupRx")
1928	if [ -z "$count" ]; then
1929		print_skip
1930	elif [ "$count" != "$mpj_syn" ]; then
1931		fail_test "got $count JOIN[s] syn with Backup expected $mpj_syn"
1932	else
1933		print_ok
1934	fi
1935
1936	print_check "synack backup"
1937	count=$(mptcp_lib_get_counter ${ns2} "MPTcpExtMPJoinSynAckBackupRx")
1938	if [ -z "$count" ]; then
1939		print_skip
1940	elif [ "$count" != "$mpj_syn_ack" ]; then
1941		fail_test "got $count JOIN[s] synack with Backup expected $mpj_syn_ack"
1942	else
1943		print_ok
1944	fi
1945}
1946
1947chk_subflow_nr()
1948{
1949	local msg="$1"
1950	local subflow_nr=$2
1951	local cnt1
1952	local cnt2
1953	local dump_stats
1954
1955	print_check "${msg}"
1956
1957	cnt1=$(ss -N $ns1 -tOni | grep -c token)
1958	cnt2=$(ss -N $ns2 -tOni | grep -c token)
1959	if [ "$cnt1" != "$subflow_nr" ] || [ "$cnt2" != "$subflow_nr" ]; then
1960		fail_test "got $cnt1:$cnt2 subflows expected $subflow_nr"
1961		dump_stats=1
1962	else
1963		print_ok
1964	fi
1965
1966	if [ "${dump_stats}" = 1 ]; then
1967		ss -N $ns1 -tOni
1968		ss -N $ns1 -tOni | grep token
1969		ip -n $ns1 mptcp endpoint
1970	fi
1971}
1972
1973chk_mptcp_info()
1974{
1975	local info1=$1
1976	local exp1=$2
1977	local info2=$3
1978	local exp2=$4
1979	local cnt1
1980	local cnt2
1981	local dump_stats
1982
1983	print_check "mptcp_info ${info1:0:15}=$exp1:$exp2"
1984
1985	cnt1=$(ss -N $ns1 -inmHM | mptcp_lib_get_info_value "$info1" "$info1")
1986	cnt2=$(ss -N $ns2 -inmHM | mptcp_lib_get_info_value "$info2" "$info2")
1987	# 'ss' only display active connections and counters that are not 0.
1988	[ -z "$cnt1" ] && cnt1=0
1989	[ -z "$cnt2" ] && cnt2=0
1990
1991	if [ "$cnt1" != "$exp1" ] || [ "$cnt2" != "$exp2" ]; then
1992		fail_test "got $cnt1:$cnt2 $info1:$info2 expected $exp1:$exp2"
1993		dump_stats=1
1994	else
1995		print_ok
1996	fi
1997
1998	if [ "$dump_stats" = 1 ]; then
1999		ss -N $ns1 -inmHM
2000		ss -N $ns2 -inmHM
2001	fi
2002}
2003
2004# $1: subflows in ns1 ; $2: subflows in ns2
2005# number of all subflows, including the initial subflow.
2006chk_subflows_total()
2007{
2008	local cnt1
2009	local cnt2
2010	local info="subflows_total"
2011	local dump_stats
2012
2013	# if subflows_total counter is supported, use it:
2014	if [ -n "$(ss -N $ns1 -inmHM | mptcp_lib_get_info_value $info $info)" ]; then
2015		chk_mptcp_info $info $1 $info $2
2016		return
2017	fi
2018
2019	print_check "$info $1:$2"
2020
2021	# if not, count the TCP connections that are in fact MPTCP subflows
2022	cnt1=$(ss -N $ns1 -ti state established state syn-sent state syn-recv |
2023	       grep -c tcp-ulp-mptcp)
2024	cnt2=$(ss -N $ns2 -ti state established state syn-sent state syn-recv |
2025	       grep -c tcp-ulp-mptcp)
2026
2027	if [ "$1" != "$cnt1" ] || [ "$2" != "$cnt2" ]; then
2028		fail_test "got subflows $cnt1:$cnt2 expected $1:$2"
2029		dump_stats=1
2030	else
2031		print_ok
2032	fi
2033
2034	if [ "$dump_stats" = 1 ]; then
2035		ss -N $ns1 -ti
2036		ss -N $ns2 -ti
2037	fi
2038}
2039
2040chk_link_usage()
2041{
2042	local ns=$1
2043	local link=$2
2044	local out=$3
2045	local expected_rate=$4
2046
2047	local tx_link tx_total
2048	tx_link=$(ip netns exec $ns cat /sys/class/net/$link/statistics/tx_bytes)
2049	tx_total=$(stat --format=%s $out)
2050	local tx_rate=$((tx_link * 100 / tx_total))
2051	local tolerance=5
2052
2053	print_check "link usage"
2054	if [ $tx_rate -lt $((expected_rate - tolerance)) ] || \
2055	   [ $tx_rate -gt $((expected_rate + tolerance)) ]; then
2056		fail_test "got $tx_rate% usage, expected $expected_rate%"
2057	else
2058		print_ok
2059	fi
2060}
2061
2062wait_attempt_fail()
2063{
2064	local timeout_ms=$((timeout_poll * 1000))
2065	local time=0
2066	local ns=$1
2067
2068	while [ $time -lt $timeout_ms ]; do
2069		local cnt
2070
2071		cnt=$(mptcp_lib_get_counter ${ns} "TcpAttemptFails")
2072
2073		[ "$cnt" = 1 ] && return 1
2074		time=$((time + 100))
2075		sleep 0.1
2076	done
2077	return 1
2078}
2079
2080set_userspace_pm()
2081{
2082	local ns=$1
2083
2084	ip netns exec $ns sysctl -q net.mptcp.pm_type=1
2085}
2086
2087subflows_tests()
2088{
2089	if reset "no JOIN"; then
2090		run_tests $ns1 $ns2 10.0.1.1
2091		chk_join_nr 0 0 0
2092	fi
2093
2094	# subflow limited by client
2095	if reset "single subflow, limited by client"; then
2096		pm_nl_set_limits $ns1 0 0
2097		pm_nl_set_limits $ns2 0 0
2098		pm_nl_add_endpoint $ns2 10.0.3.2 flags subflow
2099		run_tests $ns1 $ns2 10.0.1.1
2100		chk_join_nr 0 0 0
2101	fi
2102
2103	# subflow limited by server
2104	if reset "single subflow, limited by server"; then
2105		pm_nl_set_limits $ns1 0 0
2106		pm_nl_set_limits $ns2 0 1
2107		pm_nl_add_endpoint $ns2 10.0.3.2 flags subflow
2108		run_tests $ns1 $ns2 10.0.1.1
2109		join_syn_rej=1 \
2110			chk_join_nr 1 1 0
2111	fi
2112
2113	# subflow
2114	if reset "single subflow"; then
2115		pm_nl_set_limits $ns1 0 1
2116		pm_nl_set_limits $ns2 0 1
2117		pm_nl_add_endpoint $ns2 10.0.3.2 flags subflow
2118		run_tests $ns1 $ns2 10.0.1.1
2119		chk_join_nr 1 1 1
2120	fi
2121
2122	# multiple subflows
2123	if reset "multiple subflows"; then
2124		pm_nl_set_limits $ns1 0 2
2125		pm_nl_set_limits $ns2 0 2
2126		pm_nl_add_endpoint $ns2 10.0.3.2 flags subflow
2127		pm_nl_add_endpoint $ns2 10.0.2.2 flags subflow
2128		run_tests $ns1 $ns2 10.0.1.1
2129		chk_join_nr 2 2 2
2130	fi
2131
2132	# multiple subflows limited by server
2133	if reset "multiple subflows, limited by server"; then
2134		pm_nl_set_limits $ns1 0 1
2135		pm_nl_set_limits $ns2 0 2
2136		pm_nl_add_endpoint $ns2 10.0.3.2 flags subflow
2137		pm_nl_add_endpoint $ns2 10.0.2.2 flags subflow
2138		run_tests $ns1 $ns2 10.0.1.1
2139		join_syn_rej=1 \
2140			chk_join_nr 2 2 1
2141	fi
2142
2143	# single subflow, dev
2144	if reset "single subflow, dev"; then
2145		pm_nl_set_limits $ns1 0 1
2146		pm_nl_set_limits $ns2 0 1
2147		pm_nl_add_endpoint $ns2 10.0.3.2 flags subflow dev ns2eth3
2148		run_tests $ns1 $ns2 10.0.1.1
2149		chk_join_nr 1 1 1
2150	fi
2151}
2152
2153subflows_error_tests()
2154{
2155	# If a single subflow is configured, and matches the MPC src
2156	# address, no additional subflow should be created
2157	if reset "no MPC reuse with single endpoint"; then
2158		pm_nl_set_limits $ns1 0 1
2159		pm_nl_set_limits $ns2 0 1
2160		pm_nl_add_endpoint $ns2 10.0.1.2 flags subflow
2161		pm_nl_add_endpoint $ns2 10.0.12.2 flags subflow
2162		speed=slow \
2163			run_tests $ns1 $ns2 10.0.1.1
2164		join_bind_err=1 \
2165			chk_join_nr 0 0 0
2166	fi
2167
2168	# multiple subflows, with subflow creation error
2169	if reset_with_tcp_filter "multi subflows, with failing subflow" ns1 10.0.3.2 REJECT &&
2170	   continue_if mptcp_lib_kallsyms_has "mptcp_pm_subflow_check_next$"; then
2171		pm_nl_set_limits $ns1 0 2
2172		pm_nl_set_limits $ns2 0 2
2173		pm_nl_add_endpoint $ns2 10.0.3.2 flags subflow
2174		pm_nl_add_endpoint $ns2 10.0.2.2 flags subflow
2175		speed=slow \
2176			run_tests $ns1 $ns2 10.0.1.1
2177		join_syn_tx=2 \
2178			chk_join_nr 1 1 1
2179	fi
2180
2181	# multiple subflows, with subflow timeout on MPJ
2182	if reset_with_tcp_filter "multi subflows, with subflow timeout" ns1 10.0.3.2 DROP &&
2183	   continue_if mptcp_lib_kallsyms_has "mptcp_pm_subflow_check_next$"; then
2184		pm_nl_set_limits $ns1 0 2
2185		pm_nl_set_limits $ns2 0 2
2186		pm_nl_add_endpoint $ns2 10.0.3.2 flags subflow
2187		pm_nl_add_endpoint $ns2 10.0.2.2 flags subflow
2188		speed=slow \
2189			run_tests $ns1 $ns2 10.0.1.1
2190		join_syn_tx=2 \
2191			chk_join_nr 1 1 1
2192	fi
2193
2194	# multiple subflows, check that the endpoint corresponding to
2195	# closed subflow (due to reset) is not reused if additional
2196	# subflows are added later
2197	if reset_with_tcp_filter "multi subflows, fair usage on close" ns1 10.0.3.2 REJECT &&
2198	   continue_if mptcp_lib_kallsyms_has "mptcp_pm_subflow_check_next$"; then
2199		pm_nl_set_limits $ns1 0 1
2200		pm_nl_set_limits $ns2 0 1
2201		pm_nl_add_endpoint $ns2 10.0.3.2 flags subflow
2202		speed=slow \
2203			run_tests $ns1 $ns2 10.0.1.1 &
2204
2205		# mpj subflow will be in TW after the reset
2206		wait_attempt_fail $ns2
2207		pm_nl_add_endpoint $ns2 10.0.2.2 flags subflow
2208		wait
2209
2210		# additional subflow could be created only if the PM select
2211		# the later endpoint, skipping the already used one
2212		join_syn_tx=2 \
2213			chk_join_nr 1 1 1
2214	fi
2215}
2216
2217signal_address_tests()
2218{
2219	# add_address, unused
2220	if reset "unused signal address"; then
2221		pm_nl_add_endpoint $ns1 10.0.2.1 flags signal
2222		run_tests $ns1 $ns2 10.0.1.1
2223		chk_join_nr 0 0 0
2224		chk_add_tx_nr 1 1
2225		chk_add_nr 1 1
2226	fi
2227
2228	# accept and use add_addr
2229	if reset "signal address"; then
2230		pm_nl_set_limits $ns1 0 1
2231		pm_nl_set_limits $ns2 1 1
2232		pm_nl_add_endpoint $ns1 10.0.2.1 flags signal
2233		run_tests $ns1 $ns2 10.0.1.1
2234		chk_join_nr 1 1 1
2235		chk_add_nr 1 1
2236	fi
2237
2238	# accept and use add_addr with an additional subflow
2239	# note: signal address in server ns and local addresses in client ns must
2240	# belong to different subnets or one of the listed local address could be
2241	# used for 'add_addr' subflow
2242	if reset "subflow and signal"; then
2243		pm_nl_add_endpoint $ns1 10.0.2.1 flags signal
2244		pm_nl_set_limits $ns1 0 2
2245		pm_nl_set_limits $ns2 1 2
2246		pm_nl_add_endpoint $ns2 10.0.3.2 flags subflow
2247		run_tests $ns1 $ns2 10.0.1.1
2248		chk_join_nr 2 2 2
2249		chk_add_nr 1 1
2250	fi
2251
2252	# uncommon: subflow and signal flags on the same endpoint
2253	# or because the user wrongly picked both, but still expects the client
2254	# to create additional subflows
2255	if reset "subflow and signal together"; then
2256		pm_nl_set_limits $ns1 0 2
2257		pm_nl_set_limits $ns2 0 2
2258		pm_nl_add_endpoint $ns2 10.0.3.2 flags signal,subflow
2259		run_tests $ns1 $ns2 10.0.1.1
2260		chk_join_nr 1 1 1
2261		chk_add_nr 1 1 0 invert  # only initiated by ns2
2262		chk_add_nr 0 0 0         # none initiated by ns1
2263		chk_rst_nr 0 0 invert    # no RST sent by the client
2264		chk_rst_nr 0 0           # no RST sent by the server
2265	fi
2266
2267	# accept and use add_addr with additional subflows
2268	if reset "multiple subflows and signal"; then
2269		pm_nl_set_limits $ns1 0 3
2270		pm_nl_add_endpoint $ns1 10.0.2.1 flags signal
2271		pm_nl_set_limits $ns2 1 3
2272		pm_nl_add_endpoint $ns2 10.0.3.2 flags subflow
2273		pm_nl_add_endpoint $ns2 10.0.4.2 flags subflow
2274		run_tests $ns1 $ns2 10.0.1.1
2275		chk_join_nr 3 3 3
2276		chk_add_nr 1 1
2277	fi
2278
2279	# signal addresses
2280	if reset "signal addresses"; then
2281		pm_nl_set_limits $ns1 3 3
2282		pm_nl_add_endpoint $ns1 10.0.2.1 flags signal
2283		pm_nl_add_endpoint $ns1 10.0.3.1 flags signal
2284		pm_nl_add_endpoint $ns1 10.0.4.1 flags signal
2285		pm_nl_set_limits $ns2 3 3
2286		speed=slow \
2287			run_tests $ns1 $ns2 10.0.1.1
2288		chk_join_nr 3 3 3
2289		chk_add_nr 3 3
2290	fi
2291
2292	# signal invalid addresses
2293	if reset "signal invalid addresses"; then
2294		pm_nl_set_limits $ns1 3 3
2295		pm_nl_add_endpoint $ns1 10.0.12.1 flags signal
2296		pm_nl_add_endpoint $ns1 10.0.3.1 flags signal
2297		pm_nl_add_endpoint $ns1 10.0.14.1 flags signal
2298		pm_nl_set_limits $ns2 3 3
2299		speed=slow \
2300			run_tests $ns1 $ns2 10.0.1.1
2301		join_syn_tx=3 \
2302			chk_join_nr 1 1 1
2303		chk_add_nr 3 3
2304	fi
2305
2306	# signal addresses race test
2307	if reset "signal addresses race test"; then
2308		pm_nl_set_limits $ns1 4 4
2309		pm_nl_set_limits $ns2 4 4
2310		pm_nl_add_endpoint $ns1 10.0.1.1 flags signal
2311		pm_nl_add_endpoint $ns1 10.0.2.1 flags signal
2312		pm_nl_add_endpoint $ns1 10.0.3.1 flags signal
2313		pm_nl_add_endpoint $ns1 10.0.4.1 flags signal
2314		pm_nl_add_endpoint $ns2 10.0.1.2 flags signal
2315		pm_nl_add_endpoint $ns2 10.0.2.2 flags signal
2316		pm_nl_add_endpoint $ns2 10.0.3.2 flags signal
2317		pm_nl_add_endpoint $ns2 10.0.4.2 flags signal
2318
2319		# the peer could possibly miss some addr notification, allow retransmission
2320		ip netns exec $ns1 sysctl -q net.mptcp.add_addr_timeout=1
2321		speed=slow \
2322			run_tests $ns1 $ns2 10.0.1.1
2323
2324		# It is not directly linked to the commit introducing this
2325		# symbol but for the parent one which is linked anyway.
2326		if ! mptcp_lib_kallsyms_has "mptcp_pm_subflow_check_next$"; then
2327			chk_join_nr 3 3 2
2328			chk_add_nr 4 4
2329		else
2330			chk_join_nr 3 3 3
2331			# the server will not signal the address terminating
2332			# the MPC subflow
2333			chk_add_nr 3 3
2334		fi
2335	fi
2336}
2337
2338laminar_endp_tests()
2339{
2340	# no laminar endpoints: routing rules are used
2341	if reset_with_tcp_filter "without a laminar endpoint" ns1 10.0.2.2 REJECT &&
2342	   continue_if mptcp_lib_kallsyms_has "mptcp_pm_get_endp_laminar_max$"; then
2343		pm_nl_set_limits $ns1 0 2
2344		pm_nl_set_limits $ns2 2 2
2345		pm_nl_add_endpoint $ns1 10.0.2.1 flags signal
2346		run_tests $ns1 $ns2 10.0.1.1
2347		join_syn_tx=1 \
2348			chk_join_nr 0 0 0
2349		chk_add_nr 1 1
2350	fi
2351
2352	# laminar endpoints: this endpoint is used
2353	if reset_with_tcp_filter "with a laminar endpoint" ns1 10.0.2.2 REJECT &&
2354	   continue_if mptcp_lib_kallsyms_has "mptcp_pm_get_endp_laminar_max$"; then
2355		pm_nl_set_limits $ns1 0 2
2356		pm_nl_set_limits $ns2 2 2
2357		pm_nl_add_endpoint $ns1 10.0.2.1 flags signal
2358		pm_nl_add_endpoint $ns2 10.0.3.2 flags laminar
2359		run_tests $ns1 $ns2 10.0.1.1
2360		chk_join_nr 1 1 1
2361		chk_add_nr 1 1
2362	fi
2363
2364	# laminar endpoints: these endpoints are used
2365	if reset_with_tcp_filter "with multiple laminar endpoints" ns1 10.0.2.2 REJECT &&
2366	   continue_if mptcp_lib_kallsyms_has "mptcp_pm_get_endp_laminar_max$"; then
2367		pm_nl_set_limits $ns1 0 2
2368		pm_nl_set_limits $ns2 2 2
2369		pm_nl_add_endpoint $ns1 10.0.2.1 flags signal
2370		pm_nl_add_endpoint $ns1 10.0.3.1 flags signal
2371		pm_nl_add_endpoint $ns2 dead:beef:3::2 flags laminar
2372		pm_nl_add_endpoint $ns2 10.0.3.2 flags laminar
2373		pm_nl_add_endpoint $ns2 10.0.4.2 flags laminar
2374		run_tests $ns1 $ns2 10.0.1.1
2375		chk_join_nr 2 2 2
2376		chk_add_nr 2 2
2377	fi
2378
2379	# laminar endpoints: only one endpoint is used
2380	if reset_with_tcp_filter "single laminar endpoint" ns1 10.0.2.2 REJECT &&
2381	   continue_if mptcp_lib_kallsyms_has "mptcp_pm_get_endp_laminar_max$"; then
2382		pm_nl_set_limits $ns1 0 2
2383		pm_nl_set_limits $ns2 2 2
2384		pm_nl_add_endpoint $ns1 10.0.2.1 flags signal
2385		pm_nl_add_endpoint $ns1 10.0.3.1 flags signal
2386		pm_nl_add_endpoint $ns2 10.0.3.2 flags laminar
2387		run_tests $ns1 $ns2 10.0.1.1
2388		chk_join_nr 1 1 1
2389		chk_add_nr 2 2
2390	fi
2391
2392	# laminar endpoints: subflow and laminar flags
2393	if reset_with_tcp_filter "sublow + laminar endpoints" ns1 10.0.2.2 REJECT &&
2394	   continue_if mptcp_lib_kallsyms_has "mptcp_pm_get_endp_laminar_max$"; then
2395		pm_nl_set_limits $ns1 0 4
2396		pm_nl_set_limits $ns2 2 4
2397		pm_nl_add_endpoint $ns1 10.0.2.1 flags signal
2398		pm_nl_add_endpoint $ns2 10.0.1.2 flags subflow,laminar
2399		pm_nl_add_endpoint $ns2 10.0.3.2 flags subflow,laminar
2400		run_tests $ns1 $ns2 10.0.1.1
2401		chk_join_nr 1 1 1
2402		chk_add_nr 1 1
2403	fi
2404}
2405
2406link_failure_tests()
2407{
2408	# accept and use add_addr with additional subflows and link loss
2409	if reset "multiple flows, signal, link failure"; then
2410		# without any b/w limit each veth could spool the packets and get
2411		# them acked at xmit time, so that the corresponding subflow will
2412		# have almost always no outstanding pkts, the scheduler will pick
2413		# always the first subflow and we will have hard time testing
2414		# active backup and link switch-over.
2415		# Let's set some arbitrary (low) virtual link limits.
2416		init_shapers
2417		pm_nl_set_limits $ns1 0 3
2418		pm_nl_add_endpoint $ns1 10.0.2.1 dev ns1eth2 flags signal
2419		pm_nl_set_limits $ns2 1 3
2420		pm_nl_add_endpoint $ns2 10.0.3.2 dev ns2eth3 flags subflow
2421		pm_nl_add_endpoint $ns2 10.0.4.2 dev ns2eth4 flags subflow
2422		test_linkfail=1 \
2423			run_tests $ns1 $ns2 10.0.1.1
2424		chk_join_nr 3 3 3
2425		chk_add_nr 1 1
2426		chk_stale_nr $ns2 1 5 1
2427	fi
2428
2429	# accept and use add_addr with additional subflows and link loss
2430	# for bidirectional transfer
2431	if reset "multi flows, signal, bidi, link fail"; then
2432		init_shapers
2433		pm_nl_set_limits $ns1 0 3
2434		pm_nl_add_endpoint $ns1 10.0.2.1 dev ns1eth2 flags signal
2435		pm_nl_set_limits $ns2 1 3
2436		pm_nl_add_endpoint $ns2 10.0.3.2 dev ns2eth3 flags subflow
2437		pm_nl_add_endpoint $ns2 10.0.4.2 dev ns2eth4 flags subflow
2438		test_linkfail=2 \
2439			run_tests $ns1 $ns2 10.0.1.1
2440		chk_join_nr 3 3 3
2441		chk_add_nr 1 1
2442		chk_stale_nr $ns2 1 -1 1
2443	fi
2444
2445	# 2 subflows plus 1 backup subflow with a lossy link, backup
2446	# will never be used
2447	if reset "backup subflow unused, link failure"; then
2448		init_shapers
2449		pm_nl_set_limits $ns1 0 2
2450		pm_nl_add_endpoint $ns1 10.0.2.1 dev ns1eth2 flags signal
2451		pm_nl_set_limits $ns2 1 2
2452		pm_nl_add_endpoint $ns2 10.0.3.2 dev ns2eth3 flags subflow,backup
2453		FAILING_LINKS="1" test_linkfail=1 \
2454			run_tests $ns1 $ns2 10.0.1.1
2455		chk_join_nr 2 2 2
2456		chk_add_nr 1 1
2457		chk_link_usage $ns2 ns2eth3 $cinsent 0
2458	fi
2459
2460	# 2 lossy links after half transfer, backup will get half of
2461	# the traffic
2462	if reset "backup flow used, multi links fail"; then
2463		init_shapers
2464		pm_nl_set_limits $ns1 0 2
2465		pm_nl_add_endpoint $ns1 10.0.2.1 dev ns1eth2 flags signal
2466		pm_nl_set_limits $ns2 1 2
2467		pm_nl_add_endpoint $ns2 10.0.3.2 dev ns2eth3 flags subflow,backup
2468		FAILING_LINKS="1 2" test_linkfail=1 \
2469			run_tests $ns1 $ns2 10.0.1.1
2470		chk_join_nr 2 2 2
2471		chk_add_nr 1 1
2472		chk_stale_nr $ns2 2 4 2
2473		chk_link_usage $ns2 ns2eth3 $cinsent 50
2474	fi
2475
2476	# use a backup subflow with the first subflow on a lossy link
2477	# for bidirectional transfer
2478	if reset "backup flow used, bidi, link failure"; then
2479		init_shapers
2480		pm_nl_set_limits $ns1 0 2
2481		pm_nl_add_endpoint $ns1 10.0.2.1 dev ns1eth2 flags signal
2482		pm_nl_set_limits $ns2 1 3
2483		pm_nl_add_endpoint $ns2 10.0.3.2 dev ns2eth3 flags subflow,backup
2484		FAILING_LINKS="1 2" test_linkfail=2 \
2485			run_tests $ns1 $ns2 10.0.1.1
2486		chk_join_nr 2 2 2
2487		chk_add_nr 1 1
2488		chk_stale_nr $ns2 1 -1 2
2489		chk_link_usage $ns2 ns2eth3 $cinsent 50
2490	fi
2491}
2492
2493add_addr_timeout_tests()
2494{
2495	# add_addr timeout
2496	if reset_with_add_addr_timeout "signal address, ADD_ADDR timeout"; then
2497		pm_nl_set_limits $ns1 0 1
2498		pm_nl_set_limits $ns2 1 1
2499		pm_nl_add_endpoint $ns1 10.0.2.1 flags signal
2500		speed=slow \
2501			run_tests $ns1 $ns2 10.0.1.1
2502		chk_join_nr 1 1 1
2503		chk_add_tx_nr 4 4
2504		chk_add_nr 4 0
2505	fi
2506
2507	# add_addr timeout IPv6
2508	if reset_with_add_addr_timeout "signal address, ADD_ADDR6 timeout" 6; then
2509		pm_nl_set_limits $ns1 0 1
2510		pm_nl_set_limits $ns2 1 1
2511		pm_nl_add_endpoint $ns1 dead:beef:2::1 flags signal
2512		speed=slow \
2513			run_tests $ns1 $ns2 dead:beef:1::1
2514		chk_join_nr 1 1 1
2515		chk_add_nr 4 0
2516	fi
2517
2518	# signal addresses timeout
2519	if reset_with_add_addr_timeout "signal addresses, ADD_ADDR timeout"; then
2520		pm_nl_set_limits $ns1 2 2
2521		pm_nl_add_endpoint $ns1 10.0.2.1 flags signal
2522		pm_nl_add_endpoint $ns1 10.0.3.1 flags signal
2523		pm_nl_set_limits $ns2 2 2
2524		speed=10 \
2525			run_tests $ns1 $ns2 10.0.1.1
2526		chk_join_nr 2 2 2
2527		chk_add_nr 8 0
2528	fi
2529
2530	# signal invalid addresses timeout
2531	if reset_with_add_addr_timeout "invalid address, ADD_ADDR timeout"; then
2532		pm_nl_set_limits $ns1 2 2
2533		pm_nl_add_endpoint $ns1 10.0.12.1 flags signal
2534		pm_nl_add_endpoint $ns1 10.0.3.1 flags signal
2535		pm_nl_set_limits $ns2 2 2
2536		speed=10 \
2537			run_tests $ns1 $ns2 10.0.1.1
2538		join_syn_tx=2 \
2539			chk_join_nr 1 1 1
2540		chk_add_nr 8 0
2541	fi
2542}
2543
2544remove_tests()
2545{
2546	# single subflow, remove
2547	if reset "remove single subflow"; then
2548		pm_nl_set_limits $ns1 0 1
2549		pm_nl_set_limits $ns2 0 1
2550		pm_nl_add_endpoint $ns2 10.0.3.2 flags subflow
2551		addr_nr_ns2=-1 speed=slow \
2552			run_tests $ns1 $ns2 10.0.1.1
2553		chk_join_nr 1 1 1
2554		chk_rm_tx_nr 1
2555		chk_rm_nr 1 1
2556		chk_rst_nr 0 0
2557	fi
2558
2559	# multiple subflows, remove
2560	if reset "remove multiple subflows"; then
2561		pm_nl_set_limits $ns1 0 2
2562		pm_nl_set_limits $ns2 0 2
2563		pm_nl_add_endpoint $ns2 10.0.2.2 flags subflow
2564		pm_nl_add_endpoint $ns2 10.0.3.2 flags subflow
2565		addr_nr_ns2=-2 speed=slow \
2566			run_tests $ns1 $ns2 10.0.1.1
2567		chk_join_nr 2 2 2
2568		chk_rm_nr 2 2
2569		chk_rst_nr 0 0
2570	fi
2571
2572	# single address, remove
2573	if reset "remove single address"; then
2574		pm_nl_set_limits $ns1 0 1
2575		pm_nl_add_endpoint $ns1 10.0.2.1 flags signal
2576		pm_nl_set_limits $ns2 1 1
2577		addr_nr_ns1=-1 speed=slow \
2578			run_tests $ns1 $ns2 10.0.1.1
2579		chk_join_nr 1 1 1
2580		chk_add_nr 1 1
2581		chk_rm_nr 1 1 invert
2582		chk_rst_nr 0 0
2583	fi
2584
2585	# subflow and signal, remove
2586	if reset "remove subflow and signal"; then
2587		pm_nl_set_limits $ns1 0 2
2588		pm_nl_add_endpoint $ns1 10.0.2.1 flags signal
2589		pm_nl_set_limits $ns2 1 2
2590		pm_nl_add_endpoint $ns2 10.0.3.2 flags subflow
2591		addr_nr_ns1=-1 addr_nr_ns2=-1 speed=slow \
2592			run_tests $ns1 $ns2 10.0.1.1
2593		chk_join_nr 2 2 2
2594		chk_add_nr 1 1
2595		chk_rm_nr 1 1
2596		chk_rst_nr 0 0
2597	fi
2598
2599	# subflows and signal, remove
2600	if reset "remove subflows and signal"; then
2601		pm_nl_set_limits $ns1 0 3
2602		pm_nl_add_endpoint $ns1 10.0.2.1 flags signal
2603		pm_nl_set_limits $ns2 1 3
2604		pm_nl_add_endpoint $ns2 10.0.3.2 flags subflow
2605		pm_nl_add_endpoint $ns2 10.0.4.2 flags subflow
2606		addr_nr_ns1=-1 addr_nr_ns2=-2 speed=10 \
2607			run_tests $ns1 $ns2 10.0.1.1
2608		chk_join_nr 3 3 3
2609		chk_add_nr 1 1
2610		chk_rm_nr 2 2
2611		chk_rst_nr 0 0
2612	fi
2613
2614	# addresses remove
2615	if reset "remove addresses"; then
2616		pm_nl_set_limits $ns1 3 3
2617		pm_nl_add_endpoint $ns1 10.0.2.1 flags signal id 250
2618		pm_nl_add_endpoint $ns1 10.0.3.1 flags signal
2619		pm_nl_add_endpoint $ns1 10.0.4.1 flags signal
2620		pm_nl_set_limits $ns2 3 3
2621		addr_nr_ns1=-3 speed=10 \
2622			run_tests $ns1 $ns2 10.0.1.1
2623		chk_join_nr 3 3 3
2624		chk_add_nr 3 3
2625		chk_rm_nr 3 3 invert
2626		chk_rst_nr 0 0
2627	fi
2628
2629	# invalid addresses remove
2630	if reset "remove invalid addresses"; then
2631		pm_nl_set_limits $ns1 3 3
2632		pm_nl_add_endpoint $ns1 10.0.12.1 flags signal
2633		# broadcast IP: no packet for this address will be received on ns1
2634		pm_nl_add_endpoint $ns1 224.0.0.1 flags signal
2635		pm_nl_add_endpoint $ns1 10.0.3.1 flags signal
2636		pm_nl_set_limits $ns2 2 2
2637		addr_nr_ns1=-3 speed=10 \
2638			run_tests $ns1 $ns2 10.0.1.1
2639		join_syn_tx=2 join_connect_err=1 \
2640			chk_join_nr 1 1 1
2641		chk_add_nr 3 3
2642		chk_rm_nr 3 1 invert
2643		chk_rst_nr 0 0
2644	fi
2645
2646	# subflows and signal, flush
2647	if reset "flush subflows and signal"; then
2648		pm_nl_set_limits $ns1 0 3
2649		pm_nl_add_endpoint $ns1 10.0.2.1 flags signal
2650		pm_nl_set_limits $ns2 1 3
2651		pm_nl_add_endpoint $ns2 10.0.3.2 flags subflow
2652		pm_nl_add_endpoint $ns2 10.0.4.2 flags subflow
2653		addr_nr_ns1=-8 addr_nr_ns2=-8 speed=slow \
2654			run_tests $ns1 $ns2 10.0.1.1
2655		chk_join_nr 3 3 3
2656		chk_add_nr 1 1
2657		chk_rm_nr 1 3 invert simult
2658		chk_rst_nr 0 0
2659	fi
2660
2661	# subflows flush
2662	if reset "flush subflows"; then
2663		pm_nl_set_limits $ns1 3 3
2664		pm_nl_set_limits $ns2 3 3
2665		pm_nl_add_endpoint $ns2 10.0.2.2 flags subflow id 150
2666		pm_nl_add_endpoint $ns2 10.0.3.2 flags subflow
2667		pm_nl_add_endpoint $ns2 10.0.4.2 flags subflow
2668		addr_nr_ns1=-8 addr_nr_ns2=-8 speed=slow \
2669			run_tests $ns1 $ns2 10.0.1.1
2670		chk_join_nr 3 3 3
2671
2672		if mptcp_lib_kversion_ge 5.18; then
2673			chk_rm_tx_nr 0
2674			chk_rm_nr 0 3 simult
2675		else
2676			chk_rm_nr 3 3
2677		fi
2678		chk_rst_nr 0 0
2679	fi
2680
2681	# addresses flush
2682	if reset "flush addresses"; then
2683		pm_nl_set_limits $ns1 3 3
2684		pm_nl_add_endpoint $ns1 10.0.2.1 flags signal id 250
2685		pm_nl_add_endpoint $ns1 10.0.3.1 flags signal
2686		pm_nl_add_endpoint $ns1 10.0.4.1 flags signal
2687		pm_nl_set_limits $ns2 3 3
2688		addr_nr_ns1=-8 addr_nr_ns2=-8 speed=slow \
2689			run_tests $ns1 $ns2 10.0.1.1
2690		chk_join_nr 3 3 3
2691		chk_add_nr 3 3
2692		chk_rm_nr 3 3 invert simult
2693		chk_rst_nr 0 0
2694	fi
2695
2696	# invalid addresses flush
2697	if reset "flush invalid addresses"; then
2698		pm_nl_set_limits $ns1 3 3
2699		pm_nl_add_endpoint $ns1 10.0.12.1 flags signal
2700		pm_nl_add_endpoint $ns1 10.0.3.1 flags signal
2701		pm_nl_add_endpoint $ns1 10.0.14.1 flags signal
2702		pm_nl_set_limits $ns2 3 3
2703		addr_nr_ns1=-8 speed=slow \
2704			run_tests $ns1 $ns2 10.0.1.1
2705		join_syn_tx=3 \
2706			chk_join_nr 1 1 1
2707		chk_add_nr 3 3
2708		chk_rm_nr 3 1 invert
2709		chk_rst_nr 0 0
2710	fi
2711
2712	# remove id 0 subflow
2713	if reset "remove id 0 subflow"; then
2714		pm_nl_set_limits $ns1 0 1
2715		pm_nl_set_limits $ns2 0 1
2716		pm_nl_add_endpoint $ns2 10.0.3.2 flags subflow
2717		addr_nr_ns2=-9 speed=slow \
2718			run_tests $ns1 $ns2 10.0.1.1
2719		chk_join_nr 1 1 1
2720		chk_rm_nr 1 1
2721		chk_rst_nr 0 0
2722	fi
2723
2724	# remove id 0 address
2725	if reset "remove id 0 address"; then
2726		pm_nl_set_limits $ns1 0 1
2727		pm_nl_add_endpoint $ns1 10.0.2.1 flags signal
2728		pm_nl_set_limits $ns2 1 1
2729		addr_nr_ns1=-9 speed=slow \
2730			run_tests $ns1 $ns2 10.0.1.1
2731		chk_join_nr 1 1 1
2732		chk_add_nr 1 1
2733		chk_rm_nr 1 1 invert
2734		chk_rst_nr 0 0 invert
2735	fi
2736}
2737
2738add_tests()
2739{
2740	# add single subflow
2741	if reset "add single subflow"; then
2742		pm_nl_set_limits $ns1 0 1
2743		pm_nl_set_limits $ns2 0 1
2744		addr_nr_ns2=1 speed=slow cestab_ns2=1 \
2745			run_tests $ns1 $ns2 10.0.1.1
2746		chk_join_nr 1 1 1
2747		chk_cestab_nr $ns2 0
2748	fi
2749
2750	# add signal address
2751	if reset "add signal address"; then
2752		pm_nl_set_limits $ns1 0 1
2753		pm_nl_set_limits $ns2 1 1
2754		addr_nr_ns1=1 speed=slow cestab_ns1=1 \
2755			run_tests $ns1 $ns2 10.0.1.1
2756		chk_join_nr 1 1 1
2757		chk_add_nr 1 1
2758		chk_cestab_nr $ns1 0
2759	fi
2760
2761	# add multiple subflows
2762	if reset "add multiple subflows"; then
2763		pm_nl_set_limits $ns1 0 2
2764		pm_nl_set_limits $ns2 0 2
2765		addr_nr_ns2=2 speed=slow cestab_ns2=1 \
2766			run_tests $ns1 $ns2 10.0.1.1
2767		chk_join_nr 2 2 2
2768		chk_cestab_nr $ns2 0
2769	fi
2770
2771	# add multiple subflows IPv6
2772	if reset "add multiple subflows IPv6"; then
2773		pm_nl_set_limits $ns1 0 2
2774		pm_nl_set_limits $ns2 0 2
2775		addr_nr_ns2=2 speed=slow cestab_ns2=1 \
2776			run_tests $ns1 $ns2 dead:beef:1::1
2777		chk_join_nr 2 2 2
2778		chk_cestab_nr $ns2 0
2779	fi
2780
2781	# add multiple addresses IPv6
2782	if reset "add multiple addresses IPv6"; then
2783		pm_nl_set_limits $ns1 0 2
2784		pm_nl_set_limits $ns2 2 2
2785		addr_nr_ns1=2 speed=slow cestab_ns1=1 \
2786			run_tests $ns1 $ns2 dead:beef:1::1
2787		chk_join_nr 2 2 2
2788		chk_add_nr 2 2
2789		chk_cestab_nr $ns1 0
2790	fi
2791}
2792
2793ipv6_tests()
2794{
2795	# subflow IPv6
2796	if reset "single subflow IPv6"; then
2797		pm_nl_set_limits $ns1 0 1
2798		pm_nl_set_limits $ns2 0 1
2799		pm_nl_add_endpoint $ns2 dead:beef:3::2 dev ns2eth3 flags subflow
2800		speed=slow \
2801			run_tests $ns1 $ns2 dead:beef:1::1
2802		chk_join_nr 1 1 1
2803	fi
2804
2805	# add_address, unused IPv6
2806	if reset "unused signal address IPv6"; then
2807		pm_nl_add_endpoint $ns1 dead:beef:2::1 flags signal
2808		speed=slow \
2809			run_tests $ns1 $ns2 dead:beef:1::1
2810		chk_join_nr 0 0 0
2811		chk_add_nr 1 1
2812	fi
2813
2814	# signal address IPv6
2815	if reset "single address IPv6"; then
2816		pm_nl_set_limits $ns1 0 1
2817		pm_nl_add_endpoint $ns1 dead:beef:2::1 flags signal
2818		pm_nl_set_limits $ns2 1 1
2819		speed=slow \
2820			run_tests $ns1 $ns2 dead:beef:1::1
2821		chk_join_nr 1 1 1
2822		chk_add_nr 1 1
2823	fi
2824
2825	# single address IPv6, remove
2826	if reset "remove single address IPv6"; then
2827		pm_nl_set_limits $ns1 0 1
2828		pm_nl_add_endpoint $ns1 dead:beef:2::1 flags signal
2829		pm_nl_set_limits $ns2 1 1
2830		addr_nr_ns1=-1 speed=slow \
2831			run_tests $ns1 $ns2 dead:beef:1::1
2832		chk_join_nr 1 1 1
2833		chk_add_nr 1 1
2834		chk_rm_nr 1 1 invert
2835	fi
2836
2837	# subflow and signal IPv6, remove
2838	if reset "remove subflow and signal IPv6"; then
2839		pm_nl_set_limits $ns1 0 2
2840		pm_nl_add_endpoint $ns1 dead:beef:2::1 flags signal
2841		pm_nl_set_limits $ns2 1 2
2842		pm_nl_add_endpoint $ns2 dead:beef:3::2 dev ns2eth3 flags subflow
2843		addr_nr_ns1=-1 addr_nr_ns2=-1 speed=slow \
2844			run_tests $ns1 $ns2 dead:beef:1::1
2845		chk_join_nr 2 2 2
2846		chk_add_nr 1 1
2847		chk_rm_nr 1 1
2848	fi
2849}
2850
2851v4mapped_tests()
2852{
2853	# subflow IPv4-mapped to IPv4-mapped
2854	if reset "single subflow IPv4-mapped"; then
2855		pm_nl_set_limits $ns1 0 1
2856		pm_nl_set_limits $ns2 0 1
2857		pm_nl_add_endpoint $ns2 "::ffff:10.0.3.2" flags subflow
2858		run_tests $ns1 $ns2 "::ffff:10.0.1.1"
2859		chk_join_nr 1 1 1
2860	fi
2861
2862	# signal address IPv4-mapped with IPv4-mapped sk
2863	if reset "signal address IPv4-mapped"; then
2864		pm_nl_set_limits $ns1 0 1
2865		pm_nl_set_limits $ns2 1 1
2866		pm_nl_add_endpoint $ns1 "::ffff:10.0.2.1" flags signal
2867		run_tests $ns1 $ns2 "::ffff:10.0.1.1"
2868		chk_join_nr 1 1 1
2869		chk_add_nr 1 1
2870	fi
2871
2872	# subflow v4-map-v6
2873	if reset "single subflow v4-map-v6"; then
2874		pm_nl_set_limits $ns1 0 1
2875		pm_nl_set_limits $ns2 0 1
2876		pm_nl_add_endpoint $ns2 10.0.3.2 flags subflow
2877		run_tests $ns1 $ns2 "::ffff:10.0.1.1"
2878		chk_join_nr 1 1 1
2879	fi
2880
2881	# signal address v4-map-v6
2882	if reset "signal address v4-map-v6"; then
2883		pm_nl_set_limits $ns1 0 1
2884		pm_nl_set_limits $ns2 1 1
2885		pm_nl_add_endpoint $ns1 10.0.2.1 flags signal
2886		run_tests $ns1 $ns2 "::ffff:10.0.1.1"
2887		chk_join_nr 1 1 1
2888		chk_add_nr 1 1
2889	fi
2890
2891	# subflow v6-map-v4
2892	if reset "single subflow v6-map-v4"; then
2893		pm_nl_set_limits $ns1 0 1
2894		pm_nl_set_limits $ns2 0 1
2895		pm_nl_add_endpoint $ns2 "::ffff:10.0.3.2" flags subflow
2896		run_tests $ns1 $ns2 10.0.1.1
2897		chk_join_nr 1 1 1
2898	fi
2899
2900	# signal address v6-map-v4
2901	if reset "signal address v6-map-v4"; then
2902		pm_nl_set_limits $ns1 0 1
2903		pm_nl_set_limits $ns2 1 1
2904		pm_nl_add_endpoint $ns1 "::ffff:10.0.2.1" flags signal
2905		run_tests $ns1 $ns2 10.0.1.1
2906		chk_join_nr 1 1 1
2907		chk_add_nr 1 1
2908	fi
2909
2910	# no subflow IPv6 to v4 address
2911	if reset "no JOIN with diff families v4-v6"; then
2912		pm_nl_set_limits $ns1 0 1
2913		pm_nl_set_limits $ns2 0 1
2914		pm_nl_add_endpoint $ns2 dead:beef:2::2 flags subflow
2915		run_tests $ns1 $ns2 10.0.1.1
2916		chk_join_nr 0 0 0
2917	fi
2918
2919	# no subflow IPv6 to v4 address even if v6 has a valid v4 at the end
2920	if reset "no JOIN with diff families v4-v6-2"; then
2921		pm_nl_set_limits $ns1 0 1
2922		pm_nl_set_limits $ns2 0 1
2923		pm_nl_add_endpoint $ns2 dead:beef:2::10.0.3.2 flags subflow
2924		run_tests $ns1 $ns2 10.0.1.1
2925		chk_join_nr 0 0 0
2926	fi
2927
2928	# no subflow IPv4 to v6 address, no need to slow down too then
2929	if reset "no JOIN with diff families v6-v4"; then
2930		pm_nl_set_limits $ns1 0 1
2931		pm_nl_set_limits $ns2 0 1
2932		pm_nl_add_endpoint $ns2 10.0.3.2 flags subflow
2933		run_tests $ns1 $ns2 dead:beef:1::1
2934		chk_join_nr 0 0 0
2935	fi
2936}
2937
2938mixed_tests()
2939{
2940	if reset "IPv4 sockets do not use IPv6 addresses" &&
2941	   continue_if mptcp_lib_kversion_ge 6.3; then
2942		pm_nl_set_limits $ns1 0 1
2943		pm_nl_set_limits $ns2 1 1
2944		pm_nl_add_endpoint $ns1 dead:beef:2::1 flags signal
2945		speed=slow \
2946			run_tests $ns1 $ns2 10.0.1.1
2947		chk_join_nr 0 0 0
2948	fi
2949
2950	# Need an IPv6 mptcp socket to allow subflows of both families
2951	if reset "simult IPv4 and IPv6 subflows" &&
2952	   continue_if mptcp_lib_kversion_ge 6.3; then
2953		pm_nl_set_limits $ns1 0 1
2954		pm_nl_set_limits $ns2 1 1
2955		pm_nl_add_endpoint $ns1 10.0.1.1 flags signal
2956		speed=slow \
2957			run_tests $ns1 $ns2 dead:beef:2::1
2958		chk_join_nr 1 1 1
2959	fi
2960
2961	# cross families subflows will not be created even in fullmesh mode
2962	if reset "simult IPv4 and IPv6 subflows, fullmesh 1x1" &&
2963	   continue_if mptcp_lib_kversion_ge 6.3; then
2964		pm_nl_set_limits $ns1 0 4
2965		pm_nl_set_limits $ns2 1 4
2966		pm_nl_add_endpoint $ns2 dead:beef:2::2 flags subflow,fullmesh
2967		pm_nl_add_endpoint $ns1 10.0.1.1 flags signal
2968		speed=slow \
2969			run_tests $ns1 $ns2 dead:beef:2::1
2970		if mptcp_lib_kallsyms_has "mptcp_pm_get_endp_fullmesh_max$"; then
2971			chk_join_nr 0 0 0
2972		else
2973			chk_join_nr 1 1 1
2974		fi
2975	fi
2976
2977	# fullmesh still tries to create all the possibly subflows with
2978	# matching family
2979	if reset "simult IPv4 and IPv6 subflows, fullmesh 2x2" &&
2980	   continue_if mptcp_lib_kversion_ge 6.3; then
2981		pm_nl_set_limits $ns1 0 4
2982		pm_nl_set_limits $ns2 2 4
2983		pm_nl_add_endpoint $ns1 10.0.2.1 flags signal
2984		pm_nl_add_endpoint $ns1 dead:beef:2::1 flags signal
2985		fullmesh=1 speed=slow \
2986			run_tests $ns1 $ns2 dead:beef:1::1
2987		chk_join_nr 4 4 4
2988	fi
2989}
2990
2991backup_tests()
2992{
2993	# single subflow, backup
2994	if reset "single subflow, backup" &&
2995	   continue_if mptcp_lib_kallsyms_has "subflow_rebuild_header$"; then
2996		pm_nl_set_limits $ns1 0 1
2997		pm_nl_set_limits $ns2 0 1
2998		pm_nl_add_endpoint $ns2 10.0.3.2 flags subflow,backup
2999		sflags=nobackup speed=slow \
3000			run_tests $ns1 $ns2 10.0.1.1
3001		chk_join_nr 1 1 1
3002		chk_prio_nr 0 1 1 0
3003	fi
3004
3005	# single address, backup
3006	if reset "single address, backup" &&
3007	   continue_if mptcp_lib_kallsyms_has "subflow_rebuild_header$"; then
3008		pm_nl_set_limits $ns1 0 1
3009		pm_nl_add_endpoint $ns1 10.0.2.1 flags signal,backup
3010		pm_nl_set_limits $ns2 1 1
3011		sflags=nobackup speed=slow \
3012			run_tests $ns1 $ns2 10.0.1.1
3013		chk_join_nr 1 1 1
3014		chk_add_nr 1 1
3015		chk_prio_nr 1 0 0 1
3016	fi
3017
3018	# single address, switch to backup
3019	if reset "single address, switch to backup" &&
3020	   continue_if mptcp_lib_kallsyms_has "subflow_rebuild_header$"; then
3021		pm_nl_set_limits $ns1 0 1
3022		pm_nl_add_endpoint $ns1 10.0.2.1 flags signal
3023		pm_nl_set_limits $ns2 1 1
3024		sflags=backup speed=slow \
3025			run_tests $ns1 $ns2 10.0.1.1
3026		chk_join_nr 1 1 1
3027		chk_add_nr 1 1
3028		chk_prio_nr 1 1 0 0
3029	fi
3030
3031	# single address with port, backup
3032	if reset "single address with port, backup" &&
3033	   continue_if mptcp_lib_kallsyms_has "subflow_rebuild_header$"; then
3034		pm_nl_set_limits $ns1 0 1
3035		pm_nl_add_endpoint $ns1 10.0.2.1 flags signal,backup port 10100
3036		pm_nl_set_limits $ns2 1 1
3037		sflags=nobackup speed=slow \
3038			run_tests $ns1 $ns2 10.0.1.1
3039		chk_join_nr 1 1 1
3040		chk_add_nr 1 1
3041		chk_prio_nr 1 0 0 1
3042	fi
3043
3044	if reset "mpc backup" &&
3045	   continue_if mptcp_lib_kallsyms_doesnt_have "T mptcp_subflow_send_ack$"; then
3046		pm_nl_add_endpoint $ns2 10.0.1.2 flags subflow,backup
3047		speed=slow \
3048			run_tests $ns1 $ns2 10.0.1.1
3049		chk_join_nr 0 0 0
3050		chk_prio_nr 0 1 0 0
3051	fi
3052
3053	if reset "mpc backup both sides" &&
3054	   continue_if mptcp_lib_kallsyms_doesnt_have "T mptcp_subflow_send_ack$"; then
3055		pm_nl_set_limits $ns1 0 2
3056		pm_nl_set_limits $ns2 1 2
3057		pm_nl_add_endpoint $ns1 10.0.1.1 flags signal,backup
3058		pm_nl_add_endpoint $ns2 10.0.1.2 flags subflow,backup
3059
3060		# 10.0.2.2 (non-backup) -> 10.0.1.1 (backup)
3061		pm_nl_add_endpoint $ns2 10.0.2.2 flags subflow
3062		# 10.0.1.2 (backup) -> 10.0.2.1 (non-backup)
3063		pm_nl_add_endpoint $ns1 10.0.2.1 flags signal
3064		ip -net "$ns2" route add 10.0.2.1 via 10.0.1.1 dev ns2eth1 # force this path
3065
3066		speed=slow \
3067			run_tests $ns1 $ns2 10.0.1.1
3068		chk_join_nr 2 2 2
3069		chk_prio_nr 1 1 1 1
3070	fi
3071
3072	if reset "mpc switch to backup" &&
3073	   continue_if mptcp_lib_kallsyms_doesnt_have "T mptcp_subflow_send_ack$"; then
3074		pm_nl_add_endpoint $ns2 10.0.1.2 flags subflow
3075		sflags=backup speed=slow \
3076			run_tests $ns1 $ns2 10.0.1.1
3077		chk_join_nr 0 0 0
3078		chk_prio_nr 0 1 0 0
3079	fi
3080
3081	if reset "mpc switch to backup both sides" &&
3082	   continue_if mptcp_lib_kallsyms_doesnt_have "T mptcp_subflow_send_ack$"; then
3083		pm_nl_add_endpoint $ns1 10.0.1.1 flags subflow
3084		pm_nl_add_endpoint $ns2 10.0.1.2 flags subflow
3085		sflags=backup speed=slow \
3086			run_tests $ns1 $ns2 10.0.1.1
3087		chk_join_nr 0 0 0
3088		chk_prio_nr 1 1 0 0
3089	fi
3090}
3091
3092verify_listener_events()
3093{
3094	local e_type=$2
3095	local e_saddr=$4
3096	local e_sport=$5
3097	local name
3098
3099	if [ $e_type = $MPTCP_LIB_EVENT_LISTENER_CREATED ]; then
3100		name="LISTENER_CREATED"
3101	elif [ $e_type = $MPTCP_LIB_EVENT_LISTENER_CLOSED ]; then
3102		name="LISTENER_CLOSED "
3103	else
3104		name="$e_type"
3105	fi
3106
3107	print_check "$name $e_saddr:$e_sport"
3108
3109	if ! mptcp_lib_kallsyms_has "mptcp_event_pm_listener$"; then
3110		print_skip "event not supported"
3111		return
3112	fi
3113
3114	if mptcp_lib_verify_listener_events "${@}"; then
3115		print_ok
3116		return 0
3117	fi
3118	fail_test
3119}
3120
3121chk_mpc_endp_attempt()
3122{
3123	local retl=$1
3124	local attempts=$2
3125
3126	print_check "Connect"
3127
3128	if [ ${retl} = 124 ]; then
3129		fail_test "timeout on connect"
3130	elif [ ${retl} = 0 ]; then
3131		fail_test "unexpected successful connect"
3132	else
3133		print_ok
3134
3135		print_check "Attempts"
3136		count=$(mptcp_lib_get_counter ${ns1} "MPTcpExtMPCapableEndpAttempt")
3137		if [ -z "$count" ]; then
3138			print_skip
3139		elif [ "$count" != "$attempts" ]; then
3140			fail_test "got ${count} MPC attempt[s] on port-based endpoint, expected ${attempts}"
3141		else
3142			print_ok
3143		fi
3144	fi
3145}
3146
3147add_addr_ports_tests()
3148{
3149	# signal address with port
3150	if reset "signal address with port"; then
3151		pm_nl_set_limits $ns1 0 1
3152		pm_nl_set_limits $ns2 1 1
3153		pm_nl_add_endpoint $ns1 10.0.2.1 flags signal port 10100
3154		run_tests $ns1 $ns2 10.0.1.1
3155		chk_join_nr 1 1 1
3156		chk_add_nr 1 1 1
3157	fi
3158
3159	# subflow and signal with port
3160	if reset "subflow and signal with port"; then
3161		pm_nl_add_endpoint $ns1 10.0.2.1 flags signal port 10100
3162		pm_nl_set_limits $ns1 0 2
3163		pm_nl_set_limits $ns2 1 2
3164		pm_nl_add_endpoint $ns2 10.0.3.2 flags subflow
3165		run_tests $ns1 $ns2 10.0.1.1
3166		chk_join_nr 2 2 2
3167		chk_add_nr 1 1 1
3168	fi
3169
3170	# single address with port, remove
3171	# pm listener events
3172	if reset_with_events "remove single address with port"; then
3173		pm_nl_set_limits $ns1 0 1
3174		pm_nl_add_endpoint $ns1 10.0.2.1 flags signal port 10100
3175		pm_nl_set_limits $ns2 1 1
3176		addr_nr_ns1=-1 speed=slow \
3177			run_tests $ns1 $ns2 10.0.1.1
3178		chk_join_nr 1 1 1
3179		chk_add_nr 1 1 1
3180		chk_rm_nr 1 1 invert
3181
3182		verify_listener_events $evts_ns1 $MPTCP_LIB_EVENT_LISTENER_CREATED \
3183				       $MPTCP_LIB_AF_INET 10.0.2.1 10100
3184		verify_listener_events $evts_ns1 $MPTCP_LIB_EVENT_LISTENER_CLOSED \
3185				       $MPTCP_LIB_AF_INET 10.0.2.1 10100
3186		kill_events_pids
3187	fi
3188
3189	# subflow and signal with port, remove
3190	if reset "remove subflow and signal with port"; then
3191		pm_nl_set_limits $ns1 0 2
3192		pm_nl_add_endpoint $ns1 10.0.2.1 flags signal port 10100
3193		pm_nl_set_limits $ns2 1 2
3194		pm_nl_add_endpoint $ns2 10.0.3.2 flags subflow
3195		addr_nr_ns1=-1 addr_nr_ns2=-1 speed=slow \
3196			run_tests $ns1 $ns2 10.0.1.1
3197		chk_join_nr 2 2 2
3198		chk_add_nr 1 1 1
3199		chk_rm_nr 1 1
3200	fi
3201
3202	# subflows and signal with port, flush
3203	if reset "flush subflows and signal with port"; then
3204		pm_nl_set_limits $ns1 0 3
3205		pm_nl_add_endpoint $ns1 10.0.2.1 flags signal port 10100
3206		pm_nl_set_limits $ns2 1 3
3207		pm_nl_add_endpoint $ns2 10.0.3.2 flags subflow
3208		pm_nl_add_endpoint $ns2 10.0.4.2 flags subflow
3209		addr_nr_ns1=-8 addr_nr_ns2=-2 speed=slow \
3210			run_tests $ns1 $ns2 10.0.1.1
3211		chk_join_nr 3 3 3
3212		chk_add_nr 1 1
3213		chk_rm_nr 1 3 invert simult
3214	fi
3215
3216	# multiple addresses with port
3217	if reset "multiple addresses with port"; then
3218		pm_nl_set_limits $ns1 2 2
3219		pm_nl_add_endpoint $ns1 10.0.2.1 flags signal port 10100
3220		pm_nl_add_endpoint $ns1 10.0.3.1 flags signal port 10100
3221		pm_nl_set_limits $ns2 2 2
3222		run_tests $ns1 $ns2 10.0.1.1
3223		chk_join_nr 2 2 2
3224		chk_add_nr 2 2 2
3225	fi
3226
3227	# multiple addresses with ports
3228	if reset "multiple addresses with ports"; then
3229		pm_nl_set_limits $ns1 2 2
3230		pm_nl_add_endpoint $ns1 10.0.2.1 flags signal port 10100
3231		pm_nl_add_endpoint $ns1 10.0.3.1 flags signal port 10101
3232		pm_nl_set_limits $ns2 2 2
3233		run_tests $ns1 $ns2 10.0.1.1
3234		chk_join_nr 2 2 2
3235		chk_add_nr 2 2 2
3236	fi
3237
3238	if reset "port-based signal endpoint must not accept mpc"; then
3239		local port retl count
3240		port=$(get_port)
3241
3242		cond_start_capture ${ns1}
3243		pm_nl_add_endpoint ${ns1} 10.0.2.1 flags signal port ${port}
3244		mptcp_lib_wait_local_port_listen ${ns1} ${port}
3245
3246		timeout 1 ip netns exec ${ns2} \
3247			./mptcp_connect -t ${timeout_poll} -p $port -s MPTCP 10.0.2.1 >/dev/null 2>&1
3248		retl=$?
3249		cond_stop_capture
3250
3251		chk_mpc_endp_attempt ${retl} 1
3252	fi
3253}
3254
3255bind_tests()
3256{
3257	# bind to one address should not allow extra subflows to other addresses
3258	if reset "bind main address v4, no join v4"; then
3259		pm_nl_set_limits $ns1 0 2
3260		pm_nl_set_limits $ns2 2 2
3261		pm_nl_add_endpoint $ns1 10.0.2.1 flags signal
3262		bind_addr="10.0.1.1" \
3263			run_tests $ns1 $ns2 10.0.1.1
3264		join_syn_tx=1 \
3265			chk_join_nr 0 0 0
3266		chk_add_nr 1 1
3267	fi
3268
3269	# bind to one address should not allow extra subflows to other addresses
3270	if reset "bind main address v6, no join v6"; then
3271		pm_nl_set_limits $ns1 0 2
3272		pm_nl_set_limits $ns2 2 2
3273		pm_nl_add_endpoint $ns1 dead:beef:2::1 flags signal
3274		bind_addr="dead:beef:1::1" \
3275			run_tests $ns1 $ns2 dead:beef:1::1
3276		join_syn_tx=1 \
3277			chk_join_nr 0 0 0
3278		chk_add_nr 1 1
3279	fi
3280
3281	# multiple binds to allow extra subflows to other addresses
3282	if reset "multiple bind to allow joins v4"; then
3283		local extra_bind
3284
3285		pm_nl_set_limits $ns1 0 2
3286		pm_nl_set_limits $ns2 2 2
3287		pm_nl_add_endpoint $ns1 10.0.2.1 flags signal
3288
3289		# Launching another app listening on a different address
3290		# Note: it could be a totally different app, e.g. nc, socat, ...
3291		ip netns exec ${ns1} ./mptcp_connect -l -t -1 -p "$(get_port)" \
3292			-s MPTCP 10.0.2.1 &
3293		extra_bind=$!
3294
3295		bind_addr="10.0.1.1" \
3296			run_tests $ns1 $ns2 10.0.1.1
3297		chk_join_nr 1 1 1
3298		chk_add_nr 1 1
3299
3300		kill ${extra_bind}
3301	fi
3302
3303	# multiple binds to allow extra subflows to other addresses
3304	if reset "multiple bind to allow joins v6"; then
3305		local extra_bind
3306
3307		pm_nl_set_limits $ns1 0 2
3308		pm_nl_set_limits $ns2 2 2
3309		pm_nl_add_endpoint $ns1 dead:beef:2::1 flags signal
3310
3311		# Launching another app listening on a different address
3312		# Note: it could be a totally different app, e.g. nc, socat, ...
3313		ip netns exec ${ns1} ./mptcp_connect -l -t -1 -p "$(get_port)" \
3314			-s MPTCP dead:beef:2::1 &
3315		extra_bind=$!
3316
3317		bind_addr="dead:beef:1::1" \
3318			run_tests $ns1 $ns2 dead:beef:1::1
3319		chk_join_nr 1 1 1
3320		chk_add_nr 1 1
3321
3322		kill ${extra_bind}
3323	fi
3324
3325	# multiple binds to allow extra subflows to other addresses: v6 LL case
3326	if reset "multiple bind to allow joins v6 link-local routing"; then
3327		local extra_bind ns1ll1 ns1ll2
3328
3329		ns1ll1="$(get_ll_addr $ns1 ns1eth1)"
3330		ns1ll2="$(get_ll_addr $ns1 ns1eth2)"
3331
3332		pm_nl_set_limits $ns1 0 2
3333		pm_nl_set_limits $ns2 2 2
3334		pm_nl_add_endpoint $ns1 "${ns1ll2}" flags signal
3335
3336		wait_ll_ready $ns1 # to be able to bind
3337		wait_ll_ready $ns2 # also needed to bind on the client side
3338		ip netns exec ${ns1} ./mptcp_connect -l -t -1 -p "$(get_port)" \
3339			-s MPTCP "${ns1ll2}%ns1eth2" &
3340		extra_bind=$!
3341
3342		bind_addr="${ns1ll1}%ns1eth1" \
3343			run_tests $ns1 $ns2 "${ns1ll1}%ns2eth1"
3344		# it is not possible to connect to the announced LL addr without
3345		# specifying the outgoing interface.
3346		join_connect_err=1 \
3347			chk_join_nr 0 0 0
3348		chk_add_nr 1 1
3349
3350		kill ${extra_bind}
3351	fi
3352
3353	# multiple binds to allow extra subflows to v6 LL addresses: laminar
3354	if reset "multiple bind to allow joins v6 link-local laminar" &&
3355	   continue_if mptcp_lib_kallsyms_has "mptcp_pm_get_endp_laminar_max$"; then
3356		local extra_bind ns1ll1 ns1ll2 ns2ll2
3357
3358		ns1ll1="$(get_ll_addr $ns1 ns1eth1)"
3359		ns1ll2="$(get_ll_addr $ns1 ns1eth2)"
3360		ns2ll2="$(get_ll_addr $ns2 ns2eth2)"
3361
3362		pm_nl_set_limits $ns1 0 2
3363		pm_nl_set_limits $ns2 2 2
3364		pm_nl_add_endpoint $ns1 "${ns1ll2}" flags signal
3365		pm_nl_add_endpoint $ns2 "${ns2ll2}" flags laminar dev ns2eth2
3366
3367		wait_ll_ready $ns1 # to be able to bind
3368		wait_ll_ready $ns2 # also needed to bind on the client side
3369		ip netns exec ${ns1} ./mptcp_connect -l -t -1 -p "$(get_port)" \
3370			-s MPTCP "${ns1ll2}%ns1eth2" &
3371		extra_bind=$!
3372
3373		bind_addr="${ns1ll1}%ns1eth1" \
3374			run_tests $ns1 $ns2 "${ns1ll1}%ns2eth1"
3375		chk_join_nr 1 1 1
3376		chk_add_nr 1 1
3377
3378		kill ${extra_bind}
3379	fi
3380}
3381
3382syncookies_tests()
3383{
3384	# single subflow, syncookies
3385	if reset_with_cookies "single subflow with syn cookies"; then
3386		pm_nl_set_limits $ns1 0 1
3387		pm_nl_set_limits $ns2 0 1
3388		pm_nl_add_endpoint $ns2 10.0.3.2 flags subflow
3389		run_tests $ns1 $ns2 10.0.1.1
3390		chk_join_nr 1 1 1
3391	fi
3392
3393	# multiple subflows with syn cookies
3394	if reset_with_cookies "multiple subflows with syn cookies"; then
3395		pm_nl_set_limits $ns1 0 2
3396		pm_nl_set_limits $ns2 0 2
3397		pm_nl_add_endpoint $ns2 10.0.3.2 flags subflow
3398		pm_nl_add_endpoint $ns2 10.0.2.2 flags subflow
3399		run_tests $ns1 $ns2 10.0.1.1
3400		chk_join_nr 2 2 2
3401	fi
3402
3403	# multiple subflows limited by server
3404	if reset_with_cookies "subflows limited by server w cookies"; then
3405		pm_nl_set_limits $ns1 0 1
3406		pm_nl_set_limits $ns2 0 2
3407		pm_nl_add_endpoint $ns2 10.0.3.2 flags subflow
3408		pm_nl_add_endpoint $ns2 10.0.2.2 flags subflow
3409		run_tests $ns1 $ns2 10.0.1.1
3410		join_syn_rej=1 \
3411			chk_join_nr 2 1 1
3412	fi
3413
3414	# test signal address with cookies
3415	if reset_with_cookies "signal address with syn cookies"; then
3416		pm_nl_set_limits $ns1 0 1
3417		pm_nl_set_limits $ns2 1 1
3418		pm_nl_add_endpoint $ns1 10.0.2.1 flags signal
3419		run_tests $ns1 $ns2 10.0.1.1
3420		chk_join_nr 1 1 1
3421		chk_add_nr 1 1
3422	fi
3423
3424	# test cookie with subflow and signal
3425	if reset_with_cookies "subflow and signal w cookies"; then
3426		pm_nl_add_endpoint $ns1 10.0.2.1 flags signal
3427		pm_nl_set_limits $ns1 0 2
3428		pm_nl_set_limits $ns2 1 2
3429		pm_nl_add_endpoint $ns2 10.0.3.2 flags subflow
3430		run_tests $ns1 $ns2 10.0.1.1
3431		chk_join_nr 2 2 2
3432		chk_add_nr 1 1
3433	fi
3434
3435	# accept and use add_addr with additional subflows
3436	if reset_with_cookies "subflows and signal w. cookies"; then
3437		pm_nl_set_limits $ns1 0 3
3438		pm_nl_add_endpoint $ns1 10.0.2.1 flags signal
3439		pm_nl_set_limits $ns2 1 3
3440		pm_nl_add_endpoint $ns2 10.0.3.2 flags subflow
3441		pm_nl_add_endpoint $ns2 10.0.4.2 flags subflow
3442		run_tests $ns1 $ns2 10.0.1.1
3443		chk_join_nr 3 3 3
3444		chk_add_nr 1 1
3445	fi
3446}
3447
3448checksum_tests()
3449{
3450	local checksum_enable
3451	for checksum_enable in "0 0" "1 1" "0 1" "1 0"; do
3452		# checksum test 0 0, 1 1, 0 1, 1 0
3453		if reset_with_checksum ${checksum_enable}; then
3454			pm_nl_set_limits $ns1 0 1
3455			pm_nl_set_limits $ns2 0 1
3456			run_tests $ns1 $ns2 10.0.1.1
3457			chk_join_nr 0 0 0
3458		fi
3459	done
3460}
3461
3462deny_join_id0_tests()
3463{
3464	# subflow allow join id0 ns1
3465	if reset_with_allow_join_id0 "single subflow allow join id0 ns1" 1 0; then
3466		pm_nl_set_limits $ns1 1 1
3467		pm_nl_set_limits $ns2 1 1
3468		pm_nl_add_endpoint $ns2 10.0.3.2 flags subflow
3469		run_tests $ns1 $ns2 10.0.1.1
3470		chk_join_nr 1 1 1
3471	fi
3472
3473	# subflow allow join id0 ns2
3474	if reset_with_allow_join_id0 "single subflow allow join id0 ns2" 0 1; then
3475		pm_nl_set_limits $ns1 1 1
3476		pm_nl_set_limits $ns2 1 1
3477		pm_nl_add_endpoint $ns2 10.0.3.2 flags subflow
3478		run_tests $ns1 $ns2 10.0.1.1
3479		chk_join_nr 0 0 0
3480	fi
3481
3482	# signal address allow join id0 ns1
3483	# ADD_ADDRs are not affected by allow_join_id0 value.
3484	if reset_with_allow_join_id0 "signal address allow join id0 ns1" 1 0; then
3485		pm_nl_set_limits $ns1 1 1
3486		pm_nl_set_limits $ns2 1 1
3487		pm_nl_add_endpoint $ns1 10.0.2.1 flags signal
3488		run_tests $ns1 $ns2 10.0.1.1
3489		chk_join_nr 1 1 1
3490		chk_add_nr 1 1
3491	fi
3492
3493	# signal address allow join id0 ns2
3494	# ADD_ADDRs are not affected by allow_join_id0 value.
3495	if reset_with_allow_join_id0 "signal address allow join id0 ns2" 0 1; then
3496		pm_nl_set_limits $ns1 1 1
3497		pm_nl_set_limits $ns2 1 1
3498		pm_nl_add_endpoint $ns1 10.0.2.1 flags signal
3499		run_tests $ns1 $ns2 10.0.1.1
3500		chk_join_nr 1 1 1
3501		chk_add_nr 1 1
3502	fi
3503
3504	# subflow and address allow join id0 ns1
3505	if reset_with_allow_join_id0 "subflow and address allow join id0 1" 1 0; then
3506		pm_nl_set_limits $ns1 2 2
3507		pm_nl_set_limits $ns2 2 2
3508		pm_nl_add_endpoint $ns1 10.0.2.1 flags signal
3509		pm_nl_add_endpoint $ns2 10.0.3.2 flags subflow
3510		run_tests $ns1 $ns2 10.0.1.1
3511		chk_join_nr 2 2 2
3512	fi
3513
3514	# subflow and address allow join id0 ns2
3515	if reset_with_allow_join_id0 "subflow and address allow join id0 2" 0 1; then
3516		pm_nl_set_limits $ns1 2 2
3517		pm_nl_set_limits $ns2 2 2
3518		pm_nl_add_endpoint $ns1 10.0.2.1 flags signal
3519		pm_nl_add_endpoint $ns2 10.0.3.2 flags subflow
3520		run_tests $ns1 $ns2 10.0.1.1
3521		chk_join_nr 1 1 1
3522	fi
3523
3524	# default limits, server deny join id 0 + signal
3525	if reset_with_allow_join_id0 "default limits, server deny join id 0" 0 1; then
3526		pm_nl_set_limits $ns1 0 2
3527		pm_nl_set_limits $ns2 0 2
3528		pm_nl_add_endpoint $ns1 10.0.2.1 flags signal
3529		pm_nl_add_endpoint $ns2 10.0.3.2 flags subflow
3530		pm_nl_add_endpoint $ns2 10.0.4.2 flags subflow
3531		run_tests $ns1 $ns2 10.0.1.1
3532		chk_join_nr 2 2 2
3533	fi
3534}
3535
3536fullmesh_tests()
3537{
3538	# fullmesh 1
3539	# 2 fullmesh addrs in ns2, added before the connection,
3540	# 1 non-fullmesh addr in ns1, added during the connection.
3541	if reset "fullmesh test 2x1"; then
3542		pm_nl_set_limits $ns1 0 4
3543		pm_nl_set_limits $ns2 1 4
3544		pm_nl_add_endpoint $ns2 10.0.2.2 flags subflow,fullmesh
3545		pm_nl_add_endpoint $ns2 10.0.3.2 flags subflow,fullmesh
3546		addr_nr_ns1=1 speed=slow \
3547			run_tests $ns1 $ns2 10.0.1.1
3548		chk_join_nr 4 4 4
3549		chk_add_nr 1 1
3550	fi
3551
3552	# fullmesh 2
3553	# 1 non-fullmesh addr in ns1, added before the connection,
3554	# 1 fullmesh addr in ns2, added during the connection.
3555	if reset "fullmesh test 1x1"; then
3556		pm_nl_set_limits $ns1 1 3
3557		pm_nl_set_limits $ns2 1 3
3558		pm_nl_add_endpoint $ns1 10.0.2.1 flags signal
3559		if mptcp_lib_kallsyms_has "mptcp_pm_subflow_check_next$"; then
3560			pm_nl_add_endpoint $ns2 10.0.1.2 flags subflow,fullmesh
3561		fi
3562		fullmesh=1 speed=slow \
3563			run_tests $ns1 $ns2 10.0.1.1
3564		chk_join_nr 3 3 3
3565		chk_add_nr 1 1
3566	fi
3567
3568	# fullmesh 3
3569	# 1 non-fullmesh addr in ns1, added before the connection,
3570	# 2 fullmesh addrs in ns2, added during the connection.
3571	if reset "fullmesh test 1x2"; then
3572		pm_nl_set_limits $ns1 2 5
3573		pm_nl_set_limits $ns2 1 5
3574		pm_nl_add_endpoint $ns1 10.0.2.1 flags signal
3575		fullmesh=2 speed=slow \
3576			run_tests $ns1 $ns2 10.0.1.1
3577		chk_join_nr 5 5 5
3578		chk_add_nr 1 1
3579	fi
3580
3581	# fullmesh 4
3582	# 1 non-fullmesh addr in ns1, added before the connection,
3583	# 2 fullmesh addrs in ns2, added during the connection,
3584	# limit max_subflows to 4.
3585	if reset "fullmesh test 1x2, limited"; then
3586		pm_nl_set_limits $ns1 2 4
3587		pm_nl_set_limits $ns2 1 4
3588		pm_nl_add_endpoint $ns1 10.0.2.1 flags signal
3589		fullmesh=2 speed=slow \
3590			run_tests $ns1 $ns2 10.0.1.1
3591		chk_join_nr 4 4 4
3592		chk_add_nr 1 1
3593	fi
3594
3595	# set fullmesh flag
3596	if reset "set fullmesh flag test" &&
3597	   continue_if mptcp_lib_kversion_ge 5.18; then
3598		pm_nl_set_limits $ns1 4 4
3599		pm_nl_add_endpoint $ns1 10.0.2.1 flags subflow
3600		pm_nl_set_limits $ns2 4 4
3601		addr_nr_ns2=1 sflags=fullmesh speed=slow \
3602			run_tests $ns1 $ns2 10.0.1.1
3603		chk_join_nr 2 2 2
3604		chk_rm_nr 0 1
3605	fi
3606
3607	# set nofullmesh flag
3608	if reset "set nofullmesh flag test" &&
3609	   continue_if mptcp_lib_kversion_ge 5.18; then
3610		pm_nl_set_limits $ns1 4 4
3611		pm_nl_add_endpoint $ns1 10.0.2.1 flags subflow,fullmesh
3612		pm_nl_set_limits $ns2 4 4
3613		fullmesh=1 sflags=nofullmesh speed=slow \
3614			run_tests $ns1 $ns2 10.0.1.1
3615		chk_join_nr 2 2 2
3616		chk_rm_nr 0 1
3617	fi
3618
3619	# set backup,fullmesh flags
3620	if reset "set backup,fullmesh flags test" &&
3621	   continue_if mptcp_lib_kversion_ge 5.18; then
3622		pm_nl_set_limits $ns1 4 4
3623		pm_nl_add_endpoint $ns1 10.0.2.1 flags subflow
3624		pm_nl_set_limits $ns2 4 4
3625		addr_nr_ns2=1 sflags=backup,fullmesh speed=slow \
3626			run_tests $ns1 $ns2 10.0.1.1
3627		chk_join_nr 2 2 2
3628		chk_prio_nr 0 1 1 0
3629		chk_rm_nr 0 1
3630	fi
3631
3632	# set nobackup,nofullmesh flags
3633	if reset "set nobackup,nofullmesh flags test" &&
3634	   continue_if mptcp_lib_kversion_ge 5.18; then
3635		pm_nl_set_limits $ns1 4 4
3636		pm_nl_set_limits $ns2 4 4
3637		pm_nl_add_endpoint $ns2 10.0.2.2 flags subflow,backup,fullmesh
3638		sflags=nobackup,nofullmesh speed=slow \
3639			run_tests $ns1 $ns2 10.0.1.1
3640		chk_join_nr 2 2 2
3641		chk_prio_nr 0 1 1 0
3642		chk_rm_nr 0 1
3643	fi
3644}
3645
3646fastclose_tests()
3647{
3648	if reset_check_counter "fastclose test" "MPTcpExtMPFastcloseTx"; then
3649		MPTCP_LIB_SUBTEST_FLAKY=1
3650		test_linkfail=1024 fastclose=client \
3651			run_tests $ns1 $ns2 10.0.1.1
3652		chk_join_nr 0 0 0
3653		chk_fclose_nr 1 1
3654		chk_rst_nr 1 1 invert
3655	fi
3656
3657	if reset_check_counter "fastclose server test" "MPTcpExtMPFastcloseRx"; then
3658		MPTCP_LIB_SUBTEST_FLAKY=1
3659		test_linkfail=1024 fastclose=server \
3660			run_tests $ns1 $ns2 10.0.1.1
3661		join_rst_nr=1 \
3662			chk_join_nr 0 0 0
3663		chk_fclose_nr 1 1 invert
3664		chk_rst_nr 1 1
3665	fi
3666}
3667
3668pedit_action_pkts()
3669{
3670	tc -n $ns2 -j -s action show action pedit index 100 | \
3671		mptcp_lib_get_info_value \"packets\" packets
3672}
3673
3674fail_tests()
3675{
3676	# single subflow
3677	if reset_with_fail "Infinite map" 1; then
3678		MPTCP_LIB_SUBTEST_FLAKY=1
3679		test_linkfail=128 \
3680			run_tests $ns1 $ns2 10.0.1.1
3681		join_csum_ns1=+1 join_csum_ns2=+0 \
3682			join_fail_nr=1 join_rst_nr=0 join_infi_nr=1 \
3683			join_corrupted_pkts="$(pedit_action_pkts)" \
3684			fb_ns1="fb_dss=1" fb_ns2="fb_infinite_map_tx=1" \
3685			chk_join_nr 0 0 0
3686		chk_fail_nr 1 -1 invert
3687	fi
3688
3689	# multiple subflows
3690	if reset_with_fail "MP_FAIL MP_RST" 2; then
3691		MPTCP_LIB_SUBTEST_FLAKY=1
3692		tc -n $ns2 qdisc add dev ns2eth1 root netem rate 1mbit delay 5ms
3693		pm_nl_set_limits $ns1 0 1
3694		pm_nl_set_limits $ns2 0 1
3695		pm_nl_add_endpoint $ns2 10.0.2.2 dev ns2eth2 flags subflow
3696		test_linkfail=1024 \
3697			run_tests $ns1 $ns2 10.0.1.1
3698		join_csum_ns1=1 join_csum_ns2=0 \
3699			join_fail_nr=1 join_rst_nr=1 join_infi_nr=0 \
3700			join_corrupted_pkts="$(pedit_action_pkts)" \
3701			chk_join_nr 1 1 1
3702	fi
3703}
3704
3705# $1: ns ; $2: addr ; $3: id
3706userspace_pm_add_addr()
3707{
3708	local evts=$evts_ns1
3709	local tk
3710
3711	[ "$1" == "$ns2" ] && evts=$evts_ns2
3712	tk=$(mptcp_lib_evts_get_info token "$evts")
3713
3714	ip netns exec $1 ./pm_nl_ctl ann $2 token $tk id $3
3715	sleep 1
3716}
3717
3718# $1: ns ; $2: id
3719userspace_pm_rm_addr()
3720{
3721	local evts=$evts_ns1
3722	local tk
3723	local cnt
3724
3725	[ "$1" == "$ns2" ] && evts=$evts_ns2
3726	tk=$(mptcp_lib_evts_get_info token "$evts")
3727
3728	cnt=$(rm_addr_count ${1})
3729	ip netns exec $1 ./pm_nl_ctl rem token $tk id $2
3730	wait_rm_addr $1 "${cnt}"
3731}
3732
3733# $1: ns ; $2: addr ; $3: id
3734userspace_pm_add_sf()
3735{
3736	local evts=$evts_ns1
3737	local tk da dp
3738
3739	[ "$1" == "$ns2" ] && evts=$evts_ns2
3740	tk=$(mptcp_lib_evts_get_info token "$evts")
3741	da=$(mptcp_lib_evts_get_info daddr4 "$evts")
3742	dp=$(mptcp_lib_evts_get_info dport "$evts")
3743
3744	ip netns exec $1 ./pm_nl_ctl csf lip $2 lid $3 \
3745				rip $da rport $dp token $tk
3746	sleep 1
3747}
3748
3749# $1: ns ; $2: addr $3: event type
3750userspace_pm_rm_sf()
3751{
3752	local evts=$evts_ns1
3753	local t=${3:-1}
3754	local ip
3755	local tk da dp sp
3756	local cnt
3757
3758	[ "$1" == "$ns2" ] && evts=$evts_ns2
3759	[ -n "$(mptcp_lib_evts_get_info "saddr4" "$evts" $t)" ] && ip=4
3760	[ -n "$(mptcp_lib_evts_get_info "saddr6" "$evts" $t)" ] && ip=6
3761	tk=$(mptcp_lib_evts_get_info token "$evts")
3762	da=$(mptcp_lib_evts_get_info "daddr$ip" "$evts" $t $2)
3763	dp=$(mptcp_lib_evts_get_info dport "$evts" $t $2)
3764	sp=$(mptcp_lib_evts_get_info sport "$evts" $t $2)
3765
3766	cnt=$(rm_sf_count ${1})
3767	ip netns exec $1 ./pm_nl_ctl dsf lip $2 lport $sp \
3768				rip $da rport $dp token $tk
3769	wait_rm_sf $1 "${cnt}"
3770}
3771
3772check_output()
3773{
3774	local cmd="$1"
3775	local expected="$2"
3776	local msg="$3"
3777	local rc=0
3778
3779	mptcp_lib_check_output "${err}" "${cmd}" "${expected}" || rc=${?}
3780	if [ ${rc} -eq 2 ]; then
3781		fail_test "fail to check output # error ${rc}"
3782	elif [ ${rc} -eq 0 ]; then
3783		print_ok
3784	elif [ ${rc} -eq 1 ]; then
3785		fail_test "fail to check output # different output"
3786	fi
3787}
3788
3789# $1: ns
3790userspace_pm_dump()
3791{
3792	local evts=$evts_ns1
3793	local tk
3794
3795	[ "$1" == "$ns2" ] && evts=$evts_ns2
3796	tk=$(mptcp_lib_evts_get_info token "$evts")
3797
3798	ip netns exec $1 ./pm_nl_ctl dump token $tk
3799}
3800
3801# $1: ns ; $2: id
3802userspace_pm_get_addr()
3803{
3804	local evts=$evts_ns1
3805	local tk
3806
3807	[ "$1" == "$ns2" ] && evts=$evts_ns2
3808	tk=$(mptcp_lib_evts_get_info token "$evts")
3809
3810	ip netns exec $1 ./pm_nl_ctl get $2 token $tk
3811}
3812
3813userspace_pm_chk_dump_addr()
3814{
3815	local ns="${1}"
3816	local exp="${2}"
3817	local check="${3}"
3818
3819	print_check "dump addrs ${check}"
3820
3821	if mptcp_lib_kallsyms_has "mptcp_userspace_pm_dump_addr$"; then
3822		check_output "userspace_pm_dump ${ns}" "${exp}"
3823	else
3824		print_skip
3825	fi
3826}
3827
3828userspace_pm_chk_get_addr()
3829{
3830	local ns="${1}"
3831	local id="${2}"
3832	local exp="${3}"
3833
3834	print_check "get id ${id} addr"
3835
3836	if mptcp_lib_kallsyms_has "mptcp_userspace_pm_get_addr$"; then
3837		check_output "userspace_pm_get_addr ${ns} ${id}" "${exp}"
3838	else
3839		print_skip
3840	fi
3841}
3842
3843# $1: ns ; $2: event type ; $3: count
3844chk_evt_nr()
3845{
3846	local ns=${1}
3847	local evt_name="${2}"
3848	local exp="${3}"
3849
3850	local evts="${evts_ns1}"
3851	local evt="${!evt_name}"
3852	local count
3853
3854	evt_name="${evt_name:16}" # without MPTCP_LIB_EVENT_
3855	[ "${ns}" == "ns2" ] && evts="${evts_ns2}"
3856
3857	print_check "event ${ns} ${evt_name} (${exp})"
3858
3859	if [[ "${evt_name}" = "LISTENER_"* ]] &&
3860	   ! mptcp_lib_kallsyms_has "mptcp_event_pm_listener$"; then
3861		print_skip "event not supported"
3862		return
3863	fi
3864
3865	count=$(grep -cw "type:${evt}" "${evts}")
3866	if [ "${count}" != "${exp}" ]; then
3867		fail_test "got ${count} events, expected ${exp}"
3868	else
3869		print_ok
3870	fi
3871}
3872
3873userspace_tests()
3874{
3875	# userspace pm type prevents add_addr
3876	if reset "userspace pm type prevents add_addr" &&
3877	   continue_if mptcp_lib_has_file '/proc/sys/net/mptcp/pm_type'; then
3878		set_userspace_pm $ns1
3879		pm_nl_set_limits $ns1 0 2
3880		pm_nl_set_limits $ns2 0 2
3881		pm_nl_add_endpoint $ns1 10.0.2.1 flags signal
3882		run_tests $ns1 $ns2 10.0.1.1
3883		chk_join_nr 0 0 0
3884		chk_add_nr 0 0
3885	fi
3886
3887	# userspace pm type does not echo add_addr without daemon
3888	if reset "userspace pm no echo w/o daemon" &&
3889	   continue_if mptcp_lib_has_file '/proc/sys/net/mptcp/pm_type'; then
3890		set_userspace_pm $ns2
3891		pm_nl_set_limits $ns1 0 2
3892		pm_nl_set_limits $ns2 0 2
3893		pm_nl_add_endpoint $ns1 10.0.2.1 flags signal
3894		run_tests $ns1 $ns2 10.0.1.1
3895		chk_join_nr 0 0 0
3896		chk_add_nr 1 0
3897	fi
3898
3899	# userspace pm type rejects join
3900	if reset "userspace pm type rejects join" &&
3901	   continue_if mptcp_lib_has_file '/proc/sys/net/mptcp/pm_type'; then
3902		set_userspace_pm $ns1
3903		pm_nl_set_limits $ns1 1 1
3904		pm_nl_set_limits $ns2 1 1
3905		pm_nl_add_endpoint $ns2 10.0.3.2 flags subflow
3906		run_tests $ns1 $ns2 10.0.1.1
3907		join_syn_rej=1 \
3908			chk_join_nr 1 1 0
3909	fi
3910
3911	# userspace pm type does not send join
3912	if reset "userspace pm type does not send join" &&
3913	   continue_if mptcp_lib_has_file '/proc/sys/net/mptcp/pm_type'; then
3914		set_userspace_pm $ns2
3915		pm_nl_set_limits $ns1 1 1
3916		pm_nl_set_limits $ns2 1 1
3917		pm_nl_add_endpoint $ns2 10.0.3.2 flags subflow
3918		run_tests $ns1 $ns2 10.0.1.1
3919		chk_join_nr 0 0 0
3920	fi
3921
3922	# userspace pm type prevents mp_prio
3923	if reset "userspace pm type prevents mp_prio" &&
3924	   continue_if mptcp_lib_has_file '/proc/sys/net/mptcp/pm_type'; then
3925		set_userspace_pm $ns1
3926		pm_nl_set_limits $ns1 1 1
3927		pm_nl_set_limits $ns2 1 1
3928		pm_nl_add_endpoint $ns2 10.0.3.2 flags subflow
3929		sflags=backup speed=slow \
3930			run_tests $ns1 $ns2 10.0.1.1
3931		join_syn_rej=1 \
3932			chk_join_nr 1 1 0
3933		chk_prio_nr 0 0 0 0
3934	fi
3935
3936	# userspace pm type prevents rm_addr
3937	if reset "userspace pm type prevents rm_addr" &&
3938	   continue_if mptcp_lib_has_file '/proc/sys/net/mptcp/pm_type'; then
3939		set_userspace_pm $ns1
3940		set_userspace_pm $ns2
3941		pm_nl_set_limits $ns1 0 1
3942		pm_nl_set_limits $ns2 0 1
3943		pm_nl_add_endpoint $ns2 10.0.3.2 flags subflow
3944		addr_nr_ns2=-1 speed=slow \
3945			run_tests $ns1 $ns2 10.0.1.1
3946		chk_join_nr 0 0 0
3947		chk_rm_nr 0 0
3948	fi
3949
3950	# userspace pm add & remove address
3951	if reset_with_events "userspace pm add & remove address" &&
3952	   continue_if mptcp_lib_has_file '/proc/sys/net/mptcp/pm_type'; then
3953		set_userspace_pm $ns1
3954		pm_nl_set_limits $ns2 2 2
3955		{ speed=5 \
3956			run_tests $ns1 $ns2 10.0.1.1 & } 2>/dev/null
3957		local tests_pid=$!
3958		wait_mpj $ns1
3959		userspace_pm_add_addr $ns1 10.0.2.1 10
3960		userspace_pm_add_addr $ns1 10.0.3.1 20
3961		chk_join_nr 2 2 2
3962		chk_add_nr 2 2
3963		chk_mptcp_info subflows 2 subflows 2
3964		chk_subflows_total 3 3
3965		chk_mptcp_info add_addr_signal 2 add_addr_accepted 2
3966		userspace_pm_chk_dump_addr "${ns1}" \
3967			$'id 10 flags signal 10.0.2.1\nid 20 flags signal 10.0.3.1' \
3968			"signal"
3969		userspace_pm_chk_get_addr "${ns1}" "10" "id 10 flags signal 10.0.2.1"
3970		userspace_pm_chk_get_addr "${ns1}" "20" "id 20 flags signal 10.0.3.1"
3971		userspace_pm_rm_sf $ns1 "::ffff:10.0.2.1" $MPTCP_LIB_EVENT_SUB_ESTABLISHED
3972		userspace_pm_chk_dump_addr "${ns1}" \
3973			"id 20 flags signal 10.0.3.1" "after rm_sf 10"
3974		userspace_pm_rm_addr $ns1 20
3975		userspace_pm_chk_dump_addr "${ns1}" "" "after rm_addr 20"
3976		chk_rm_nr 1 1 invert
3977		chk_mptcp_info subflows 0 subflows 0
3978		chk_subflows_total 1 1
3979		kill_events_pids
3980		mptcp_lib_kill_wait $tests_pid
3981	fi
3982
3983	# userspace pm create destroy subflow
3984	if reset_with_events "userspace pm create destroy subflow" &&
3985	   continue_if mptcp_lib_has_file '/proc/sys/net/mptcp/pm_type'; then
3986		set_userspace_pm $ns2
3987		pm_nl_set_limits $ns1 0 1
3988		{ speed=5 \
3989			run_tests $ns1 $ns2 10.0.1.1 & } 2>/dev/null
3990		local tests_pid=$!
3991		wait_mpj $ns2
3992		userspace_pm_add_sf $ns2 10.0.3.2 20
3993		chk_join_nr 1 1 1
3994		chk_mptcp_info subflows 1 subflows 1
3995		chk_subflows_total 2 2
3996		userspace_pm_chk_dump_addr "${ns2}" \
3997			"id 20 flags subflow 10.0.3.2" \
3998			"subflow"
3999		userspace_pm_chk_get_addr "${ns2}" "20" "id 20 flags subflow 10.0.3.2"
4000		userspace_pm_rm_sf $ns2 10.0.3.2 $MPTCP_LIB_EVENT_SUB_ESTABLISHED
4001		userspace_pm_chk_dump_addr "${ns2}" \
4002			"" \
4003			"after rm_sf 20"
4004		chk_rm_nr 0 1
4005		chk_mptcp_info subflows 0 subflows 0
4006		chk_subflows_total 1 1
4007		kill_events_pids
4008		mptcp_lib_kill_wait $tests_pid
4009	fi
4010
4011	# userspace pm create id 0 subflow
4012	if reset_with_events "userspace pm create id 0 subflow" &&
4013	   continue_if mptcp_lib_has_file '/proc/sys/net/mptcp/pm_type'; then
4014		set_userspace_pm $ns2
4015		pm_nl_set_limits $ns1 0 1
4016		{ speed=5 \
4017			run_tests $ns1 $ns2 10.0.1.1 & } 2>/dev/null
4018		local tests_pid=$!
4019		wait_mpj $ns2
4020		chk_mptcp_info subflows 0 subflows 0
4021		chk_subflows_total 1 1
4022		userspace_pm_add_sf $ns2 10.0.3.2 0
4023		userspace_pm_chk_dump_addr "${ns2}" \
4024			"id 0 flags subflow 10.0.3.2" "id 0 subflow"
4025		chk_join_nr 1 1 1
4026		chk_mptcp_info subflows 1 subflows 1
4027		chk_subflows_total 2 2
4028		kill_events_pids
4029		mptcp_lib_kill_wait $tests_pid
4030	fi
4031
4032	# userspace pm remove initial subflow
4033	if reset_with_events "userspace pm remove initial subflow" &&
4034	   continue_if mptcp_lib_has_file '/proc/sys/net/mptcp/pm_type'; then
4035		set_userspace_pm $ns2
4036		pm_nl_set_limits $ns1 0 1
4037		{ speed=5 \
4038			run_tests $ns1 $ns2 10.0.1.1 & } 2>/dev/null
4039		local tests_pid=$!
4040		wait_mpj $ns2
4041		userspace_pm_add_sf $ns2 10.0.3.2 20
4042		chk_join_nr 1 1 1
4043		chk_mptcp_info subflows 1 subflows 1
4044		chk_subflows_total 2 2
4045		userspace_pm_rm_sf $ns2 10.0.1.2
4046		# we don't look at the counter linked to the RM_ADDR but
4047		# to the one linked to the subflows that have been removed
4048		chk_rm_nr 0 1
4049		chk_rst_nr 0 0 invert
4050		chk_mptcp_info subflows 1 subflows 1
4051		chk_subflows_total 1 1
4052		kill_events_pids
4053		mptcp_lib_kill_wait $tests_pid
4054	fi
4055
4056	# userspace pm send RM_ADDR for ID 0
4057	if reset_with_events "userspace pm send RM_ADDR for ID 0" &&
4058	   continue_if mptcp_lib_has_file '/proc/sys/net/mptcp/pm_type'; then
4059		set_userspace_pm $ns1
4060		pm_nl_set_limits $ns2 1 1
4061		{ speed=5 \
4062			run_tests $ns1 $ns2 10.0.1.1 & } 2>/dev/null
4063		local tests_pid=$!
4064		wait_mpj $ns1
4065		userspace_pm_add_addr $ns1 10.0.2.1 10
4066		chk_join_nr 1 1 1
4067		chk_add_nr 1 1
4068		chk_mptcp_info subflows 1 subflows 1
4069		chk_subflows_total 2 2
4070		chk_mptcp_info add_addr_signal 1 add_addr_accepted 1
4071		userspace_pm_rm_addr $ns1 0
4072		# we don't look at the counter linked to the subflows that
4073		# have been removed but to the one linked to the RM_ADDR
4074		chk_rm_nr 1 0 invert
4075		chk_rst_nr 0 0 invert
4076		chk_mptcp_info subflows 1 subflows 1
4077		chk_subflows_total 1 1
4078		kill_events_pids
4079		mptcp_lib_kill_wait $tests_pid
4080	fi
4081}
4082
4083endpoint_tests()
4084{
4085	# subflow_rebuild_header is needed to support the implicit flag
4086	# userspace pm type prevents add_addr
4087	if reset "implicit EP" &&
4088	   continue_if mptcp_lib_kallsyms_has "subflow_rebuild_header$"; then
4089		pm_nl_set_limits $ns1 2 2
4090		pm_nl_set_limits $ns2 2 2
4091		pm_nl_add_endpoint $ns1 10.0.2.1 flags signal
4092		{ speed=slow \
4093			run_tests $ns1 $ns2 10.0.1.1 & } 2>/dev/null
4094		local tests_pid=$!
4095
4096		wait_mpj $ns1
4097		pm_nl_check_endpoint "creation" \
4098			$ns2 10.0.2.2 id 1 flags implicit
4099		chk_mptcp_info subflows 1 subflows 1
4100		chk_mptcp_info add_addr_signal 1 add_addr_accepted 1
4101
4102		pm_nl_add_endpoint $ns2 10.0.2.2 id 33 2>/dev/null
4103		pm_nl_check_endpoint "ID change is prevented" \
4104			$ns2 10.0.2.2 id 1 flags implicit
4105
4106		pm_nl_add_endpoint $ns2 10.0.2.2 flags signal
4107		pm_nl_check_endpoint "modif is allowed" \
4108			$ns2 10.0.2.2 id 1 flags signal
4109		mptcp_lib_kill_wait $tests_pid
4110	fi
4111
4112	if reset_with_tcp_filter "delete and re-add" ns2 10.0.3.2 REJECT OUTPUT &&
4113	   continue_if mptcp_lib_kallsyms_has "subflow_rebuild_header$"; then
4114		start_events
4115		pm_nl_set_limits $ns1 0 3
4116		pm_nl_set_limits $ns2 0 3
4117		pm_nl_add_endpoint $ns2 10.0.1.2 id 1 dev ns2eth1 flags subflow
4118		pm_nl_add_endpoint $ns2 10.0.2.2 id 2 dev ns2eth2 flags subflow
4119		{ test_linkfail=4 speed=5 \
4120			run_tests $ns1 $ns2 10.0.1.1 & } 2>/dev/null
4121		local tests_pid=$!
4122
4123		wait_mpj $ns2
4124		pm_nl_check_endpoint "creation" \
4125			$ns2 10.0.2.2 id 2 flags subflow dev ns2eth2
4126		chk_subflow_nr "before delete id 2" 2
4127		chk_mptcp_info subflows 1 subflows 1
4128
4129		pm_nl_del_endpoint $ns2 2 10.0.2.2
4130		sleep 0.5
4131		chk_subflow_nr "after delete id 2" 1
4132		chk_mptcp_info subflows 0 subflows 0
4133
4134		pm_nl_add_endpoint $ns2 10.0.2.2 id 2 dev ns2eth2 flags subflow
4135		wait_mpj $ns2
4136		chk_subflow_nr "after re-add id 2" 2
4137		chk_mptcp_info subflows 1 subflows 1
4138
4139		pm_nl_add_endpoint $ns2 10.0.3.2 id 3 flags subflow
4140		wait_attempt_fail $ns2
4141		chk_subflow_nr "after new reject" 2
4142		chk_mptcp_info subflows 1 subflows 1
4143
4144		ip netns exec "${ns2}" ${iptables} -D OUTPUT -s "10.0.3.2" -p tcp -j REJECT
4145		pm_nl_del_endpoint $ns2 3 10.0.3.2
4146		pm_nl_add_endpoint $ns2 10.0.3.2 id 3 flags subflow
4147		wait_mpj $ns2
4148		chk_subflow_nr "after no reject" 3
4149		chk_mptcp_info subflows 2 subflows 2
4150
4151		local i
4152		for i in $(seq 3); do
4153			pm_nl_del_endpoint $ns2 1 10.0.1.2
4154			sleep 0.5
4155			chk_subflow_nr "after delete id 0 ($i)" 2
4156			chk_mptcp_info subflows 2 subflows 2 # only decr for additional sf
4157
4158			pm_nl_add_endpoint $ns2 10.0.1.2 id 1 dev ns2eth1 flags subflow
4159			wait_mpj $ns2
4160			chk_subflow_nr "after re-add id 0 ($i)" 3
4161			chk_mptcp_info subflows 3 subflows 3
4162		done
4163
4164		mptcp_lib_kill_wait $tests_pid
4165
4166		kill_events_pids
4167		chk_evt_nr ns1 MPTCP_LIB_EVENT_LISTENER_CREATED 1
4168		chk_evt_nr ns1 MPTCP_LIB_EVENT_CREATED 1
4169		chk_evt_nr ns1 MPTCP_LIB_EVENT_ESTABLISHED 1
4170		chk_evt_nr ns1 MPTCP_LIB_EVENT_ANNOUNCED 0
4171		chk_evt_nr ns1 MPTCP_LIB_EVENT_REMOVED 4
4172		chk_evt_nr ns1 MPTCP_LIB_EVENT_SUB_ESTABLISHED 6
4173		chk_evt_nr ns1 MPTCP_LIB_EVENT_SUB_CLOSED 4
4174
4175		chk_evt_nr ns2 MPTCP_LIB_EVENT_CREATED 1
4176		chk_evt_nr ns2 MPTCP_LIB_EVENT_ESTABLISHED 1
4177		chk_evt_nr ns2 MPTCP_LIB_EVENT_ANNOUNCED 0
4178		chk_evt_nr ns2 MPTCP_LIB_EVENT_REMOVED 0
4179		chk_evt_nr ns2 MPTCP_LIB_EVENT_SUB_ESTABLISHED 6
4180		chk_evt_nr ns2 MPTCP_LIB_EVENT_SUB_CLOSED 5 # one has been closed before estab
4181
4182		join_syn_tx=7 \
4183			chk_join_nr 6 6 6
4184		chk_rm_nr 4 4
4185	fi
4186
4187	# remove and re-add
4188	if reset_with_events "delete re-add signal" &&
4189	   continue_if mptcp_lib_kallsyms_has "subflow_rebuild_header$"; then
4190		ip netns exec $ns1 sysctl -q net.mptcp.add_addr_timeout=0
4191		pm_nl_set_limits $ns1 0 3
4192		pm_nl_set_limits $ns2 3 3
4193		pm_nl_add_endpoint $ns1 10.0.2.1 id 1 flags signal
4194		# broadcast IP: no packet for this address will be received on ns1
4195		pm_nl_add_endpoint $ns1 224.0.0.1 id 2 flags signal
4196		pm_nl_add_endpoint $ns1 10.0.1.1 id 42 flags signal
4197		{ test_linkfail=4 speed=5 \
4198			run_tests $ns1 $ns2 10.0.1.1 & } 2>/dev/null
4199		local tests_pid=$!
4200
4201		wait_mpj $ns2
4202		pm_nl_check_endpoint "creation" \
4203			$ns1 10.0.2.1 id 1 flags signal
4204		chk_subflow_nr "before delete" 2
4205		chk_mptcp_info subflows 1 subflows 1
4206
4207		pm_nl_del_endpoint $ns1 1 10.0.2.1
4208		pm_nl_del_endpoint $ns1 2 224.0.0.1
4209		sleep 0.5
4210		chk_subflow_nr "after delete" 1
4211		chk_mptcp_info subflows 0 subflows 0
4212
4213		pm_nl_add_endpoint $ns1 10.0.2.1 id 1 flags signal
4214		pm_nl_add_endpoint $ns1 10.0.3.1 id 2 flags signal
4215		wait_mpj $ns2
4216		chk_subflow_nr "after re-add" 3
4217		chk_mptcp_info subflows 2 subflows 2
4218
4219		pm_nl_del_endpoint $ns1 42 10.0.1.1
4220		sleep 0.5
4221		chk_subflow_nr "after delete ID 0" 2
4222		chk_mptcp_info subflows 2 subflows 2
4223
4224		pm_nl_add_endpoint $ns1 10.0.1.1 id 99 flags signal
4225		wait_mpj $ns2
4226		chk_subflow_nr "after re-add ID 0" 3
4227		chk_mptcp_info subflows 3 subflows 3
4228
4229		pm_nl_del_endpoint $ns1 99 10.0.1.1
4230		sleep 0.5
4231		chk_subflow_nr "after re-delete ID 0" 2
4232		chk_mptcp_info subflows 2 subflows 2
4233
4234		pm_nl_add_endpoint $ns1 10.0.1.1 id 88 flags signal
4235		wait_mpj $ns2
4236		chk_subflow_nr "after re-re-add ID 0" 3
4237		chk_mptcp_info subflows 3 subflows 3
4238		mptcp_lib_kill_wait $tests_pid
4239
4240		kill_events_pids
4241		chk_evt_nr ns1 MPTCP_LIB_EVENT_LISTENER_CREATED 1
4242		chk_evt_nr ns1 MPTCP_LIB_EVENT_CREATED 1
4243		chk_evt_nr ns1 MPTCP_LIB_EVENT_ESTABLISHED 1
4244		chk_evt_nr ns1 MPTCP_LIB_EVENT_ANNOUNCED 0
4245		chk_evt_nr ns1 MPTCP_LIB_EVENT_REMOVED 0
4246		chk_evt_nr ns1 MPTCP_LIB_EVENT_SUB_ESTABLISHED 5
4247		chk_evt_nr ns1 MPTCP_LIB_EVENT_SUB_CLOSED 3
4248
4249		chk_evt_nr ns2 MPTCP_LIB_EVENT_CREATED 1
4250		chk_evt_nr ns2 MPTCP_LIB_EVENT_ESTABLISHED 1
4251		chk_evt_nr ns2 MPTCP_LIB_EVENT_ANNOUNCED 6
4252		chk_evt_nr ns2 MPTCP_LIB_EVENT_REMOVED 4
4253		chk_evt_nr ns2 MPTCP_LIB_EVENT_SUB_ESTABLISHED 5
4254		chk_evt_nr ns2 MPTCP_LIB_EVENT_SUB_CLOSED 3
4255
4256		join_connect_err=1 \
4257			chk_join_nr 5 5 5
4258		chk_add_nr 6 6
4259		chk_rm_nr 4 3 invert
4260	fi
4261
4262	# flush and re-add
4263	if reset_with_tcp_filter "flush re-add" ns2 10.0.3.2 REJECT OUTPUT &&
4264	   continue_if mptcp_lib_kallsyms_has "subflow_rebuild_header$"; then
4265		pm_nl_set_limits $ns1 0 2
4266		pm_nl_set_limits $ns2 1 2
4267		# broadcast IP: no packet for this address will be received on ns1
4268		pm_nl_add_endpoint $ns1 224.0.0.1 id 2 flags signal
4269		pm_nl_add_endpoint $ns2 10.0.3.2 id 3 flags subflow
4270		{ test_linkfail=4 speed=20 \
4271			run_tests $ns1 $ns2 10.0.1.1 & } 2>/dev/null
4272		local tests_pid=$!
4273
4274		wait_attempt_fail $ns2
4275		chk_subflow_nr "before flush" 1
4276		chk_mptcp_info subflows 0 subflows 0
4277
4278		pm_nl_flush_endpoint $ns2
4279		pm_nl_flush_endpoint $ns1
4280		wait_rm_addr $ns2 0
4281		ip netns exec "${ns2}" ${iptables} -D OUTPUT -s "10.0.3.2" -p tcp -j REJECT
4282		pm_nl_add_endpoint $ns2 10.0.3.2 id 3 flags subflow
4283		wait_mpj $ns2
4284		pm_nl_add_endpoint $ns1 10.0.3.1 id 2 flags signal
4285		wait_mpj $ns2
4286		mptcp_lib_kill_wait $tests_pid
4287
4288		join_syn_tx=3 join_connect_err=1 \
4289			chk_join_nr 2 2 2
4290		chk_add_nr 2 2
4291		chk_rm_nr 1 0 invert
4292	fi
4293}
4294
4295# [$1: error message]
4296usage()
4297{
4298	if [ -n "${1}" ]; then
4299		echo "${1}"
4300		ret=${KSFT_FAIL}
4301	fi
4302
4303	echo "mptcp_join usage:"
4304
4305	local key
4306	for key in "${!all_tests[@]}"; do
4307		echo "  -${key} ${all_tests[${key}]}"
4308	done
4309
4310	echo "  -c capture pcap files"
4311	echo "  -C enable data checksum"
4312	echo "  -i use ip mptcp"
4313	echo "  -h help"
4314
4315	echo "[test ids|names]"
4316
4317	exit ${ret}
4318}
4319
4320
4321# Use a "simple" array to force an specific order we cannot have with an associative one
4322all_tests_sorted=(
4323	f@subflows_tests
4324	e@subflows_error_tests
4325	s@signal_address_tests
4326	L@laminar_endp_tests
4327	l@link_failure_tests
4328	t@add_addr_timeout_tests
4329	r@remove_tests
4330	a@add_tests
4331	6@ipv6_tests
4332	4@v4mapped_tests
4333	M@mixed_tests
4334	b@backup_tests
4335	p@add_addr_ports_tests
4336	B@bind_tests
4337	k@syncookies_tests
4338	S@checksum_tests
4339	d@deny_join_id0_tests
4340	m@fullmesh_tests
4341	z@fastclose_tests
4342	F@fail_tests
4343	u@userspace_tests
4344	I@endpoint_tests
4345)
4346
4347all_tests_args=""
4348all_tests_names=()
4349for subtests in "${all_tests_sorted[@]}"; do
4350	key="${subtests%@*}"
4351	value="${subtests#*@}"
4352
4353	all_tests_args+="${key}"
4354	all_tests_names+=("${value}")
4355	all_tests[${key}]="${value}"
4356done
4357
4358tests=()
4359while getopts "${all_tests_args}cCih" opt; do
4360	case $opt in
4361		["${all_tests_args}"])
4362			tests+=("${all_tests[${opt}]}")
4363			;;
4364		c)
4365			capture=true
4366			;;
4367		C)
4368			checksum=true
4369			;;
4370		i)
4371			mptcp_lib_set_ip_mptcp
4372			;;
4373		h)
4374			usage
4375			;;
4376		*)
4377			usage "Unknown option: -${opt}"
4378			;;
4379	esac
4380done
4381
4382shift $((OPTIND - 1))
4383
4384for arg in "${@}"; do
4385	if [[ "${arg}" =~ ^[0-9]+$ ]]; then
4386		only_tests_ids+=("${arg}")
4387	else
4388		only_tests_names+=("${arg}")
4389	fi
4390done
4391
4392if [ ${#tests[@]} -eq 0 ]; then
4393	tests=("${all_tests_names[@]}")
4394fi
4395
4396mptcp_lib_subtests_last_ts_reset
4397for subtests in "${tests[@]}"; do
4398	"${subtests}"
4399done
4400append_prev_results
4401
4402if [ ${ret} -ne 0 ]; then
4403	echo
4404	echo "${#failed_tests[@]} failure(s) has(ve) been detected:"
4405	for i in $(get_failed_tests_ids); do
4406		echo -e "\t- ${i}: ${failed_tests[${i}]}"
4407	done
4408	echo
4409fi
4410
4411mptcp_lib_result_print_all_tap
4412
4413exit $ret
4414