301 lines
6.4 KiB
Bash
301 lines
6.4 KiB
Bash
|
#! /bin/bash
|
||
|
# SPDX-License-Identifier: GPL-2.0
|
||
|
|
||
|
readonly KSFT_PASS=0
|
||
|
readonly KSFT_FAIL=1
|
||
|
readonly KSFT_SKIP=4
|
||
|
|
||
|
# shellcheck disable=SC2155 # declare and assign separately
|
||
|
readonly KSFT_TEST="${MPTCP_LIB_KSFT_TEST:-$(basename "${0}" .sh)}"
|
||
|
|
||
|
MPTCP_LIB_SUBTESTS=()
|
||
|
|
||
|
# only if supported (or forced) and not disabled, see no-color.org
|
||
|
if { [ -t 1 ] || [ "${SELFTESTS_MPTCP_LIB_COLOR_FORCE:-}" = "1" ]; } &&
|
||
|
[ "${NO_COLOR:-}" != "1" ]; then
|
||
|
readonly MPTCP_LIB_COLOR_RED="\E[1;31m"
|
||
|
readonly MPTCP_LIB_COLOR_GREEN="\E[1;32m"
|
||
|
readonly MPTCP_LIB_COLOR_YELLOW="\E[1;33m"
|
||
|
readonly MPTCP_LIB_COLOR_BLUE="\E[1;34m"
|
||
|
readonly MPTCP_LIB_COLOR_RESET="\E[0m"
|
||
|
else
|
||
|
readonly MPTCP_LIB_COLOR_RED=
|
||
|
readonly MPTCP_LIB_COLOR_GREEN=
|
||
|
readonly MPTCP_LIB_COLOR_YELLOW=
|
||
|
readonly MPTCP_LIB_COLOR_BLUE=
|
||
|
readonly MPTCP_LIB_COLOR_RESET=
|
||
|
fi
|
||
|
|
||
|
# $1: color, $2: text
|
||
|
mptcp_lib_print_color() {
|
||
|
echo -e "${MPTCP_LIB_START_PRINT:-}${*}${MPTCP_LIB_COLOR_RESET}"
|
||
|
}
|
||
|
|
||
|
mptcp_lib_print_ok() {
|
||
|
mptcp_lib_print_color "${MPTCP_LIB_COLOR_GREEN}${*}"
|
||
|
}
|
||
|
|
||
|
mptcp_lib_print_warn() {
|
||
|
mptcp_lib_print_color "${MPTCP_LIB_COLOR_YELLOW}${*}"
|
||
|
}
|
||
|
|
||
|
mptcp_lib_print_info() {
|
||
|
mptcp_lib_print_color "${MPTCP_LIB_COLOR_BLUE}${*}"
|
||
|
}
|
||
|
|
||
|
mptcp_lib_print_err() {
|
||
|
mptcp_lib_print_color "${MPTCP_LIB_COLOR_RED}${*}"
|
||
|
}
|
||
|
|
||
|
# SELFTESTS_MPTCP_LIB_EXPECT_ALL_FEATURES env var can be set when validating all
|
||
|
# features using the last version of the kernel and the selftests to make sure
|
||
|
# a test is not being skipped by mistake.
|
||
|
mptcp_lib_expect_all_features() {
|
||
|
[ "${SELFTESTS_MPTCP_LIB_EXPECT_ALL_FEATURES:-}" = "1" ]
|
||
|
}
|
||
|
|
||
|
# $1: msg
|
||
|
mptcp_lib_fail_if_expected_feature() {
|
||
|
if mptcp_lib_expect_all_features; then
|
||
|
echo "ERROR: missing feature: ${*}"
|
||
|
exit ${KSFT_FAIL}
|
||
|
fi
|
||
|
|
||
|
return 1
|
||
|
}
|
||
|
|
||
|
# $1: file
|
||
|
mptcp_lib_has_file() {
|
||
|
local f="${1}"
|
||
|
|
||
|
if [ -f "${f}" ]; then
|
||
|
return 0
|
||
|
fi
|
||
|
|
||
|
mptcp_lib_fail_if_expected_feature "${f} file not found"
|
||
|
}
|
||
|
|
||
|
mptcp_lib_check_mptcp() {
|
||
|
if ! mptcp_lib_has_file "/proc/sys/net/mptcp/enabled"; then
|
||
|
echo "SKIP: MPTCP support is not available"
|
||
|
exit ${KSFT_SKIP}
|
||
|
fi
|
||
|
}
|
||
|
|
||
|
mptcp_lib_check_kallsyms() {
|
||
|
if ! mptcp_lib_has_file "/proc/kallsyms"; then
|
||
|
echo "SKIP: CONFIG_KALLSYMS is missing"
|
||
|
exit ${KSFT_SKIP}
|
||
|
fi
|
||
|
}
|
||
|
|
||
|
# Internal: use mptcp_lib_kallsyms_has() instead
|
||
|
__mptcp_lib_kallsyms_has() {
|
||
|
local sym="${1}"
|
||
|
|
||
|
mptcp_lib_check_kallsyms
|
||
|
|
||
|
grep -q " ${sym}" /proc/kallsyms
|
||
|
}
|
||
|
|
||
|
# $1: part of a symbol to look at, add '$' at the end for full name
|
||
|
mptcp_lib_kallsyms_has() {
|
||
|
local sym="${1}"
|
||
|
|
||
|
if __mptcp_lib_kallsyms_has "${sym}"; then
|
||
|
return 0
|
||
|
fi
|
||
|
|
||
|
mptcp_lib_fail_if_expected_feature "${sym} symbol not found"
|
||
|
}
|
||
|
|
||
|
# $1: part of a symbol to look at, add '$' at the end for full name
|
||
|
mptcp_lib_kallsyms_doesnt_have() {
|
||
|
local sym="${1}"
|
||
|
|
||
|
if ! __mptcp_lib_kallsyms_has "${sym}"; then
|
||
|
return 0
|
||
|
fi
|
||
|
|
||
|
mptcp_lib_fail_if_expected_feature "${sym} symbol has been found"
|
||
|
}
|
||
|
|
||
|
# !!!AVOID USING THIS!!!
|
||
|
# Features might not land in the expected version and features can be backported
|
||
|
#
|
||
|
# $1: kernel version, e.g. 6.3
|
||
|
mptcp_lib_kversion_ge() {
|
||
|
local exp_maj="${1%.*}"
|
||
|
local exp_min="${1#*.}"
|
||
|
local v maj min
|
||
|
|
||
|
# If the kernel has backported features, set this env var to 1:
|
||
|
if [ "${SELFTESTS_MPTCP_LIB_NO_KVERSION_CHECK:-}" = "1" ]; then
|
||
|
return 0
|
||
|
fi
|
||
|
|
||
|
v=$(uname -r | cut -d'.' -f1,2)
|
||
|
maj=${v%.*}
|
||
|
min=${v#*.}
|
||
|
|
||
|
if [ "${maj}" -gt "${exp_maj}" ] ||
|
||
|
{ [ "${maj}" -eq "${exp_maj}" ] && [ "${min}" -ge "${exp_min}" ]; }; then
|
||
|
return 0
|
||
|
fi
|
||
|
|
||
|
mptcp_lib_fail_if_expected_feature "kernel version ${1} lower than ${v}"
|
||
|
}
|
||
|
|
||
|
__mptcp_lib_result_add() {
|
||
|
local result="${1}"
|
||
|
shift
|
||
|
|
||
|
local id=$((${#MPTCP_LIB_SUBTESTS[@]} + 1))
|
||
|
|
||
|
MPTCP_LIB_SUBTESTS+=("${result} ${id} - ${KSFT_TEST}: ${*}")
|
||
|
}
|
||
|
|
||
|
# $1: test name
|
||
|
mptcp_lib_result_pass() {
|
||
|
__mptcp_lib_result_add "ok" "${1}"
|
||
|
}
|
||
|
|
||
|
# $1: test name
|
||
|
mptcp_lib_result_fail() {
|
||
|
__mptcp_lib_result_add "not ok" "${1}"
|
||
|
}
|
||
|
|
||
|
# $1: test name
|
||
|
mptcp_lib_result_skip() {
|
||
|
__mptcp_lib_result_add "ok" "${1} # SKIP"
|
||
|
}
|
||
|
|
||
|
# $1: result code ; $2: test name
|
||
|
mptcp_lib_result_code() {
|
||
|
local ret="${1}"
|
||
|
local name="${2}"
|
||
|
|
||
|
case "${ret}" in
|
||
|
"${KSFT_PASS}")
|
||
|
mptcp_lib_result_pass "${name}"
|
||
|
;;
|
||
|
"${KSFT_FAIL}")
|
||
|
mptcp_lib_result_fail "${name}"
|
||
|
;;
|
||
|
"${KSFT_SKIP}")
|
||
|
mptcp_lib_result_skip "${name}"
|
||
|
;;
|
||
|
*)
|
||
|
echo "ERROR: wrong result code: ${ret}"
|
||
|
exit ${KSFT_FAIL}
|
||
|
;;
|
||
|
esac
|
||
|
}
|
||
|
|
||
|
mptcp_lib_result_print_all_tap() {
|
||
|
local subtest
|
||
|
|
||
|
if [ ${#MPTCP_LIB_SUBTESTS[@]} -eq 0 ] ||
|
||
|
[ "${SELFTESTS_MPTCP_LIB_NO_TAP:-}" = "1" ]; then
|
||
|
return
|
||
|
fi
|
||
|
|
||
|
printf "\nTAP version 13\n"
|
||
|
printf "1..%d\n" "${#MPTCP_LIB_SUBTESTS[@]}"
|
||
|
|
||
|
for subtest in "${MPTCP_LIB_SUBTESTS[@]}"; do
|
||
|
printf "%s\n" "${subtest}"
|
||
|
done
|
||
|
}
|
||
|
|
||
|
# get the value of keyword $1 in the line marked by keyword $2
|
||
|
mptcp_lib_get_info_value() {
|
||
|
grep "${2}" | sed -n 's/.*\('"${1}"':\)\([0-9a-f:.]*\).*$/\2/p;q'
|
||
|
}
|
||
|
|
||
|
# $1: info name ; $2: evts_ns ; [$3: event type; [$4: addr]]
|
||
|
mptcp_lib_evts_get_info() {
|
||
|
grep "${4:-}" "${2}" | mptcp_lib_get_info_value "${1}" "^type:${3:-1},"
|
||
|
}
|
||
|
|
||
|
# $1: PID
|
||
|
mptcp_lib_kill_wait() {
|
||
|
[ "${1}" -eq 0 ] && return 0
|
||
|
|
||
|
kill -SIGUSR1 "${1}" > /dev/null 2>&1
|
||
|
kill "${1}" > /dev/null 2>&1
|
||
|
wait "${1}" 2>/dev/null
|
||
|
}
|
||
|
|
||
|
# $1: IP address
|
||
|
mptcp_lib_is_v6() {
|
||
|
[ -z "${1##*:*}" ]
|
||
|
}
|
||
|
|
||
|
# $1: ns, $2: MIB counter
|
||
|
mptcp_lib_get_counter() {
|
||
|
local ns="${1}"
|
||
|
local counter="${2}"
|
||
|
local count
|
||
|
|
||
|
count=$(ip netns exec "${ns}" nstat -asz "${counter}" |
|
||
|
awk 'NR==1 {next} {print $2}')
|
||
|
if [ -z "${count}" ]; then
|
||
|
mptcp_lib_fail_if_expected_feature "${counter} counter"
|
||
|
return 1
|
||
|
fi
|
||
|
|
||
|
echo "${count}"
|
||
|
}
|
||
|
|
||
|
mptcp_lib_make_file() {
|
||
|
local name="${1}"
|
||
|
local bs="${2}"
|
||
|
local size="${3}"
|
||
|
|
||
|
dd if=/dev/urandom of="${name}" bs="${bs}" count="${size}" 2> /dev/null
|
||
|
echo -e "\nMPTCP_TEST_FILE_END_MARKER" >> "${name}"
|
||
|
}
|
||
|
|
||
|
# $1: file
|
||
|
mptcp_lib_print_file_err() {
|
||
|
ls -l "${1}" 1>&2
|
||
|
echo "Trailing bytes are: "
|
||
|
tail -c 27 "${1}"
|
||
|
}
|
||
|
|
||
|
# $1: input file ; $2: output file ; $3: what kind of file
|
||
|
mptcp_lib_check_transfer() {
|
||
|
local in="${1}"
|
||
|
local out="${2}"
|
||
|
local what="${3}"
|
||
|
|
||
|
if ! cmp "$in" "$out" > /dev/null 2>&1; then
|
||
|
echo "[ FAIL ] $what does not match (in, out):"
|
||
|
mptcp_lib_print_file_err "$in"
|
||
|
mptcp_lib_print_file_err "$out"
|
||
|
|
||
|
return 1
|
||
|
fi
|
||
|
|
||
|
return 0
|
||
|
}
|
||
|
|
||
|
# $1: ns, $2: port
|
||
|
mptcp_lib_wait_local_port_listen() {
|
||
|
local listener_ns="${1}"
|
||
|
local port="${2}"
|
||
|
|
||
|
local port_hex
|
||
|
port_hex="$(printf "%04X" "${port}")"
|
||
|
|
||
|
local _
|
||
|
for _ in $(seq 10); do
|
||
|
ip netns exec "${listener_ns}" cat /proc/net/tcp* | \
|
||
|
awk "BEGIN {rc=1} {if (\$2 ~ /:${port_hex}\$/ && \$4 ~ /0A/) \
|
||
|
{rc=0; exit}} END {exit rc}" &&
|
||
|
break
|
||
|
sleep 0.1
|
||
|
done
|
||
|
}
|