#!/bin/bash # SPDX-License-Identifier: GPL-2.0 ############################################################################## # Defines : "${WAIT_TIMEOUT:=20}" BUSYWAIT_TIMEOUT=$((WAIT_TIMEOUT * 1000)) # ms # Kselftest framework constants. ksft_pass=0 ksft_fail=1 ksft_xfail=2 ksft_skip=4 # namespace list created by setup_ns NS_LIST=() ############################################################################## # Helpers __ksft_status_merge() { local a=$1; shift local b=$1; shift local -A weights local weight=0 local i for i in "$@"; do weights[$i]=$((weight++)) done if [[ ${weights[$a]} > ${weights[$b]} ]]; then echo "$a" return 0 else echo "$b" return 1 fi } ksft_status_merge() { local a=$1; shift local b=$1; shift __ksft_status_merge "$a" "$b" \ $ksft_pass $ksft_xfail $ksft_skip $ksft_fail } ksft_exit_status_merge() { local a=$1; shift local b=$1; shift __ksft_status_merge "$a" "$b" \ $ksft_xfail $ksft_pass $ksft_skip $ksft_fail } loopy_wait() { local sleep_cmd=$1; shift local timeout_ms=$1; shift local start_time="$(date -u +%s%3N)" while true do local out if out=$("$@"); then echo -n "$out" return 0 fi local current_time="$(date -u +%s%3N)" if ((current_time - start_time > timeout_ms)); then echo -n "$out" return 1 fi $sleep_cmd done } busywait() { local timeout_ms=$1; shift loopy_wait : "$timeout_ms" "$@" } # timeout in seconds slowwait() { local timeout_sec=$1; shift loopy_wait "sleep 0.1" "$((timeout_sec * 1000))" "$@" } until_counter_is() { local expr=$1; shift local current=$("$@") echo $((current)) ((current $expr)) } busywait_for_counter() { local timeout=$1; shift local delta=$1; shift local base=$("$@") busywait "$timeout" until_counter_is ">= $((base + delta))" "$@" } slowwait_for_counter() { local timeout=$1; shift local delta=$1; shift local base=$("$@") slowwait "$timeout" until_counter_is ">= $((base + delta))" "$@" } remove_ns_list() { local item=$1 local ns local ns_list=("${NS_LIST[@]}") NS_LIST=() for ns in "${ns_list[@]}"; do if [ "${ns}" != "${item}" ]; then NS_LIST+=("${ns}") fi done } cleanup_ns() { local ns="" local ret=0 for ns in "$@"; do [ -z "${ns}" ] && continue ip netns delete "${ns}" &> /dev/null || true if ! busywait $BUSYWAIT_TIMEOUT ip netns list \| grep -vq "^$ns$" &> /dev/null; then echo "Warn: Failed to remove namespace $ns" ret=1 else remove_ns_list "${ns}" fi done return $ret } cleanup_all_ns() { cleanup_ns "${NS_LIST[@]}" } # setup netns with given names as prefix. e.g # setup_ns local remote setup_ns() { local ns_name="" local ns_list=() for ns_name in "$@"; do # avoid conflicts with local var: internal error if [ "${ns_name}" = "ns_name" ]; then echo "Failed to setup namespace '${ns_name}': invalid name" cleanup_ns "${ns_list[@]}" exit $ksft_fail fi # Some test may setup/remove same netns multi times if [ -z "${!ns_name}" ]; then eval "${ns_name}=${ns_name,,}-$(mktemp -u XXXXXX)" else cleanup_ns "${!ns_name}" fi if ! ip netns add "${!ns_name}"; then echo "Failed to create namespace $ns_name" cleanup_ns "${ns_list[@]}" return $ksft_skip fi ip -n "${!ns_name}" link set lo up ns_list+=("${!ns_name}") done NS_LIST+=("${ns_list[@]}") } tc_rule_stats_get() { local dev=$1; shift local pref=$1; shift local dir=${1:-ingress}; shift local selector=${1:-.packets}; shift tc -j -s filter show dev $dev $dir pref $pref \ | jq ".[1].options.actions[].stats$selector" } tc_rule_handle_stats_get() { local id=$1; shift local handle=$1; shift local selector=${1:-.packets}; shift local netns=${1:-""}; shift tc $netns -j -s filter show $id \ | jq ".[] | select(.options.handle == $handle) | \ .options.actions[0].stats$selector" }