599 lines
		
	
	
		
			13 KiB
		
	
	
	
		
			Bash
		
	
	
	
	
	
			
		
		
	
	
			599 lines
		
	
	
		
			13 KiB
		
	
	
	
		
			Bash
		
	
	
	
	
	
| #!/bin/bash
 | |
| # SPDX-License-Identifier: GPL-2.0
 | |
| 
 | |
| # Kselftest framework requirement - SKIP code is 4.
 | |
| ksft_skip=4
 | |
| 
 | |
| ##############################################################################
 | |
| # Defines
 | |
| 
 | |
| if [[ ! -v DEVLINK_DEV ]]; then
 | |
| 	DEVLINK_DEV=$(devlink port show "${NETIFS[p1]:-$NETIF_NO_CABLE}" -j \
 | |
| 			     | jq -r '.port | keys[]' | cut -d/ -f-2)
 | |
| 	if [ -z "$DEVLINK_DEV" ]; then
 | |
| 		echo "SKIP: ${NETIFS[p1]} has no devlink device registered for it"
 | |
| 		exit $ksft_skip
 | |
| 	fi
 | |
| 	if [[ "$(echo $DEVLINK_DEV | grep -c pci)" -eq 0 ]]; then
 | |
| 		echo "SKIP: devlink device's bus is not PCI"
 | |
| 		exit $ksft_skip
 | |
| 	fi
 | |
| 
 | |
| 	DEVLINK_VIDDID=$(lspci -s $(echo $DEVLINK_DEV | cut -d"/" -f2) \
 | |
| 			 -n | cut -d" " -f3)
 | |
| elif [[ ! -z "$DEVLINK_DEV" ]]; then
 | |
| 	devlink dev show $DEVLINK_DEV &> /dev/null
 | |
| 	if [ $? -ne 0 ]; then
 | |
| 		echo "SKIP: devlink device \"$DEVLINK_DEV\" not found"
 | |
| 		exit $ksft_skip
 | |
| 	fi
 | |
| fi
 | |
| 
 | |
| ##############################################################################
 | |
| # Sanity checks
 | |
| 
 | |
| devlink help 2>&1 | grep resource &> /dev/null
 | |
| if [ $? -ne 0 ]; then
 | |
| 	echo "SKIP: iproute2 too old, missing devlink resource support"
 | |
| 	exit $ksft_skip
 | |
| fi
 | |
| 
 | |
| devlink help 2>&1 | grep trap &> /dev/null
 | |
| if [ $? -ne 0 ]; then
 | |
| 	echo "SKIP: iproute2 too old, missing devlink trap support"
 | |
| 	exit $ksft_skip
 | |
| fi
 | |
| 
 | |
| devlink dev help 2>&1 | grep info &> /dev/null
 | |
| if [ $? -ne 0 ]; then
 | |
| 	echo "SKIP: iproute2 too old, missing devlink dev info support"
 | |
| 	exit $ksft_skip
 | |
| fi
 | |
| 
 | |
| ##############################################################################
 | |
| # Devlink helpers
 | |
| 
 | |
| devlink_resource_names_to_path()
 | |
| {
 | |
| 	local resource
 | |
| 	local path=""
 | |
| 
 | |
| 	for resource in "${@}"; do
 | |
| 		if [ "$path" == "" ]; then
 | |
| 			path="$resource"
 | |
| 		else
 | |
| 			path="${path}/$resource"
 | |
| 		fi
 | |
| 	done
 | |
| 
 | |
| 	echo "$path"
 | |
| }
 | |
| 
 | |
| devlink_resource_get()
 | |
| {
 | |
| 	local name=$1
 | |
| 	local resource_name=.[][\"$DEVLINK_DEV\"]
 | |
| 
 | |
| 	resource_name="$resource_name | .[] | select (.name == \"$name\")"
 | |
| 
 | |
| 	shift
 | |
| 	for resource in "${@}"; do
 | |
| 		resource_name="${resource_name} | .[\"resources\"][] | \
 | |
| 			       select (.name == \"$resource\")"
 | |
| 	done
 | |
| 
 | |
| 	devlink -j resource show "$DEVLINK_DEV" | jq "$resource_name"
 | |
| }
 | |
| 
 | |
| devlink_resource_size_get()
 | |
| {
 | |
| 	local size=$(devlink_resource_get "$@" | jq '.["size_new"]')
 | |
| 
 | |
| 	if [ "$size" == "null" ]; then
 | |
| 		devlink_resource_get "$@" | jq '.["size"]'
 | |
| 	else
 | |
| 		echo "$size"
 | |
| 	fi
 | |
| }
 | |
| 
 | |
| devlink_resource_size_set()
 | |
| {
 | |
| 	local new_size=$1
 | |
| 	local path
 | |
| 
 | |
| 	shift
 | |
| 	path=$(devlink_resource_names_to_path "$@")
 | |
| 	devlink resource set "$DEVLINK_DEV" path "$path" size "$new_size"
 | |
| 	check_err $? "Failed setting path $path to size $size"
 | |
| }
 | |
| 
 | |
| devlink_resource_occ_get()
 | |
| {
 | |
| 	devlink_resource_get "$@" | jq '.["occ"]'
 | |
| }
 | |
| 
 | |
| devlink_reload()
 | |
| {
 | |
| 	local still_pending
 | |
| 
 | |
| 	devlink dev reload "$DEVLINK_DEV" &> /dev/null
 | |
| 	check_err $? "Failed reload"
 | |
| 
 | |
| 	still_pending=$(devlink resource show "$DEVLINK_DEV" | \
 | |
| 			grep -c "size_new")
 | |
| 	check_err $still_pending "Failed reload - There are still unset sizes"
 | |
| 
 | |
| 	udevadm settle
 | |
| }
 | |
| 
 | |
| declare -A DEVLINK_ORIG
 | |
| 
 | |
| # Changing pool type from static to dynamic causes reinterpretation of threshold
 | |
| # values. They therefore need to be saved before pool type is changed, then the
 | |
| # pool type can be changed, and then the new values need to be set up. Therefore
 | |
| # instead of saving the current state implicitly in the _set call, provide
 | |
| # functions for all three primitives: save, set, and restore.
 | |
| 
 | |
| devlink_port_pool_threshold()
 | |
| {
 | |
| 	local port=$1; shift
 | |
| 	local pool=$1; shift
 | |
| 
 | |
| 	devlink sb port pool show $port pool $pool -j \
 | |
| 		| jq '.port_pool."'"$port"'"[].threshold'
 | |
| }
 | |
| 
 | |
| devlink_port_pool_th_save()
 | |
| {
 | |
| 	local port=$1; shift
 | |
| 	local pool=$1; shift
 | |
| 	local key="port_pool($port,$pool).threshold"
 | |
| 
 | |
| 	DEVLINK_ORIG[$key]=$(devlink_port_pool_threshold $port $pool)
 | |
| }
 | |
| 
 | |
| devlink_port_pool_th_set()
 | |
| {
 | |
| 	local port=$1; shift
 | |
| 	local pool=$1; shift
 | |
| 	local th=$1; shift
 | |
| 
 | |
| 	devlink sb port pool set $port pool $pool th $th
 | |
| }
 | |
| 
 | |
| devlink_port_pool_th_restore()
 | |
| {
 | |
| 	local port=$1; shift
 | |
| 	local pool=$1; shift
 | |
| 	local key="port_pool($port,$pool).threshold"
 | |
| 	local -a orig=(${DEVLINK_ORIG[$key]})
 | |
| 
 | |
| 	if [[ -z $orig ]]; then
 | |
| 		echo "WARNING: Mismatched devlink_port_pool_th_restore"
 | |
| 	else
 | |
| 		devlink sb port pool set $port pool $pool th $orig
 | |
| 	fi
 | |
| }
 | |
| 
 | |
| devlink_pool_size_thtype()
 | |
| {
 | |
| 	local pool=$1; shift
 | |
| 
 | |
| 	devlink sb pool show "$DEVLINK_DEV" pool $pool -j \
 | |
| 	    | jq -r '.pool[][] | (.size, .thtype)'
 | |
| }
 | |
| 
 | |
| devlink_pool_size_thtype_save()
 | |
| {
 | |
| 	local pool=$1; shift
 | |
| 	local key="pool($pool).size_thtype"
 | |
| 
 | |
| 	DEVLINK_ORIG[$key]=$(devlink_pool_size_thtype $pool)
 | |
| }
 | |
| 
 | |
| devlink_pool_size_thtype_set()
 | |
| {
 | |
| 	local pool=$1; shift
 | |
| 	local thtype=$1; shift
 | |
| 	local size=$1; shift
 | |
| 
 | |
| 	devlink sb pool set "$DEVLINK_DEV" pool $pool size $size thtype $thtype
 | |
| }
 | |
| 
 | |
| devlink_pool_size_thtype_restore()
 | |
| {
 | |
| 	local pool=$1; shift
 | |
| 	local key="pool($pool).size_thtype"
 | |
| 	local -a orig=(${DEVLINK_ORIG[$key]})
 | |
| 
 | |
| 	if [[ -z ${orig[0]} ]]; then
 | |
| 		echo "WARNING: Mismatched devlink_pool_size_thtype_restore"
 | |
| 	else
 | |
| 		devlink sb pool set "$DEVLINK_DEV" pool $pool \
 | |
| 			size ${orig[0]} thtype ${orig[1]}
 | |
| 	fi
 | |
| }
 | |
| 
 | |
| devlink_tc_bind_pool_th()
 | |
| {
 | |
| 	local port=$1; shift
 | |
| 	local tc=$1; shift
 | |
| 	local dir=$1; shift
 | |
| 
 | |
| 	devlink sb tc bind show $port tc $tc type $dir -j \
 | |
| 	    | jq -r '.tc_bind[][] | (.pool, .threshold)'
 | |
| }
 | |
| 
 | |
| devlink_tc_bind_pool_th_save()
 | |
| {
 | |
| 	local port=$1; shift
 | |
| 	local tc=$1; shift
 | |
| 	local dir=$1; shift
 | |
| 	local key="tc_bind($port,$dir,$tc).pool_th"
 | |
| 
 | |
| 	DEVLINK_ORIG[$key]=$(devlink_tc_bind_pool_th $port $tc $dir)
 | |
| }
 | |
| 
 | |
| devlink_tc_bind_pool_th_set()
 | |
| {
 | |
| 	local port=$1; shift
 | |
| 	local tc=$1; shift
 | |
| 	local dir=$1; shift
 | |
| 	local pool=$1; shift
 | |
| 	local th=$1; shift
 | |
| 
 | |
| 	devlink sb tc bind set $port tc $tc type $dir pool $pool th $th
 | |
| }
 | |
| 
 | |
| devlink_tc_bind_pool_th_restore()
 | |
| {
 | |
| 	local port=$1; shift
 | |
| 	local tc=$1; shift
 | |
| 	local dir=$1; shift
 | |
| 	local key="tc_bind($port,$dir,$tc).pool_th"
 | |
| 	local -a orig=(${DEVLINK_ORIG[$key]})
 | |
| 
 | |
| 	if [[ -z ${orig[0]} ]]; then
 | |
| 		echo "WARNING: Mismatched devlink_tc_bind_pool_th_restore"
 | |
| 	else
 | |
| 		devlink sb tc bind set $port tc $tc type $dir \
 | |
| 			pool ${orig[0]} th ${orig[1]}
 | |
| 	fi
 | |
| }
 | |
| 
 | |
| devlink_traps_num_get()
 | |
| {
 | |
| 	devlink -j trap | jq '.[]["'$DEVLINK_DEV'"] | length'
 | |
| }
 | |
| 
 | |
| devlink_traps_get()
 | |
| {
 | |
| 	devlink -j trap | jq -r '.[]["'$DEVLINK_DEV'"][].name'
 | |
| }
 | |
| 
 | |
| devlink_trap_type_get()
 | |
| {
 | |
| 	local trap_name=$1; shift
 | |
| 
 | |
| 	devlink -j trap show $DEVLINK_DEV trap $trap_name \
 | |
| 		| jq -r '.[][][].type'
 | |
| }
 | |
| 
 | |
| devlink_trap_action_set()
 | |
| {
 | |
| 	local trap_name=$1; shift
 | |
| 	local action=$1; shift
 | |
| 
 | |
| 	# Pipe output to /dev/null to avoid expected warnings.
 | |
| 	devlink trap set $DEVLINK_DEV trap $trap_name \
 | |
| 		action $action &> /dev/null
 | |
| }
 | |
| 
 | |
| devlink_trap_action_get()
 | |
| {
 | |
| 	local trap_name=$1; shift
 | |
| 
 | |
| 	devlink -j trap show $DEVLINK_DEV trap $trap_name \
 | |
| 		| jq -r '.[][][].action'
 | |
| }
 | |
| 
 | |
| devlink_trap_group_get()
 | |
| {
 | |
| 	devlink -j trap show $DEVLINK_DEV trap $trap_name \
 | |
| 		| jq -r '.[][][].group'
 | |
| }
 | |
| 
 | |
| devlink_trap_metadata_test()
 | |
| {
 | |
| 	local trap_name=$1; shift
 | |
| 	local metadata=$1; shift
 | |
| 
 | |
| 	devlink -jv trap show $DEVLINK_DEV trap $trap_name \
 | |
| 		| jq -e '.[][][].metadata | contains(["'$metadata'"])' \
 | |
| 		&> /dev/null
 | |
| }
 | |
| 
 | |
| devlink_trap_rx_packets_get()
 | |
| {
 | |
| 	local trap_name=$1; shift
 | |
| 
 | |
| 	devlink -js trap show $DEVLINK_DEV trap $trap_name \
 | |
| 		| jq '.[][][]["stats"]["rx"]["packets"]'
 | |
| }
 | |
| 
 | |
| devlink_trap_rx_bytes_get()
 | |
| {
 | |
| 	local trap_name=$1; shift
 | |
| 
 | |
| 	devlink -js trap show $DEVLINK_DEV trap $trap_name \
 | |
| 		| jq '.[][][]["stats"]["rx"]["bytes"]'
 | |
| }
 | |
| 
 | |
| devlink_trap_drop_packets_get()
 | |
| {
 | |
| 	local trap_name=$1; shift
 | |
| 
 | |
| 	devlink -js trap show $DEVLINK_DEV trap $trap_name \
 | |
| 		| jq '.[][][]["stats"]["rx"]["dropped"]'
 | |
| }
 | |
| 
 | |
| devlink_trap_stats_idle_test()
 | |
| {
 | |
| 	local trap_name=$1; shift
 | |
| 	local t0_packets t0_bytes
 | |
| 	local t1_packets t1_bytes
 | |
| 
 | |
| 	t0_packets=$(devlink_trap_rx_packets_get $trap_name)
 | |
| 	t0_bytes=$(devlink_trap_rx_bytes_get $trap_name)
 | |
| 
 | |
| 	sleep 1
 | |
| 
 | |
| 	t1_packets=$(devlink_trap_rx_packets_get $trap_name)
 | |
| 	t1_bytes=$(devlink_trap_rx_bytes_get $trap_name)
 | |
| 
 | |
| 	if [[ $t0_packets -eq $t1_packets && $t0_bytes -eq $t1_bytes ]]; then
 | |
| 		return 0
 | |
| 	else
 | |
| 		return 1
 | |
| 	fi
 | |
| }
 | |
| 
 | |
| devlink_trap_drop_stats_idle_test()
 | |
| {
 | |
| 	local trap_name=$1; shift
 | |
| 	local t0_packets t0_bytes
 | |
| 
 | |
| 	t0_packets=$(devlink_trap_drop_packets_get $trap_name)
 | |
| 
 | |
| 	sleep 1
 | |
| 
 | |
| 	t1_packets=$(devlink_trap_drop_packets_get $trap_name)
 | |
| 
 | |
| 	if [[ $t0_packets -eq $t1_packets ]]; then
 | |
| 		return 0
 | |
| 	else
 | |
| 		return 1
 | |
| 	fi
 | |
| }
 | |
| 
 | |
| devlink_traps_enable_all()
 | |
| {
 | |
| 	local trap_name
 | |
| 
 | |
| 	for trap_name in $(devlink_traps_get); do
 | |
| 		devlink_trap_action_set $trap_name "trap"
 | |
| 	done
 | |
| }
 | |
| 
 | |
| devlink_traps_disable_all()
 | |
| {
 | |
| 	for trap_name in $(devlink_traps_get); do
 | |
| 		devlink_trap_action_set $trap_name "drop"
 | |
| 	done
 | |
| }
 | |
| 
 | |
| devlink_trap_groups_get()
 | |
| {
 | |
| 	devlink -j trap group | jq -r '.[]["'$DEVLINK_DEV'"][].name'
 | |
| }
 | |
| 
 | |
| devlink_trap_group_action_set()
 | |
| {
 | |
| 	local group_name=$1; shift
 | |
| 	local action=$1; shift
 | |
| 
 | |
| 	# Pipe output to /dev/null to avoid expected warnings.
 | |
| 	devlink trap group set $DEVLINK_DEV group $group_name action $action \
 | |
| 		&> /dev/null
 | |
| }
 | |
| 
 | |
| devlink_trap_group_rx_packets_get()
 | |
| {
 | |
| 	local group_name=$1; shift
 | |
| 
 | |
| 	devlink -js trap group show $DEVLINK_DEV group $group_name \
 | |
| 		| jq '.[][][]["stats"]["rx"]["packets"]'
 | |
| }
 | |
| 
 | |
| devlink_trap_group_rx_bytes_get()
 | |
| {
 | |
| 	local group_name=$1; shift
 | |
| 
 | |
| 	devlink -js trap group show $DEVLINK_DEV group $group_name \
 | |
| 		| jq '.[][][]["stats"]["rx"]["bytes"]'
 | |
| }
 | |
| 
 | |
| devlink_trap_group_stats_idle_test()
 | |
| {
 | |
| 	local group_name=$1; shift
 | |
| 	local t0_packets t0_bytes
 | |
| 	local t1_packets t1_bytes
 | |
| 
 | |
| 	t0_packets=$(devlink_trap_group_rx_packets_get $group_name)
 | |
| 	t0_bytes=$(devlink_trap_group_rx_bytes_get $group_name)
 | |
| 
 | |
| 	sleep 1
 | |
| 
 | |
| 	t1_packets=$(devlink_trap_group_rx_packets_get $group_name)
 | |
| 	t1_bytes=$(devlink_trap_group_rx_bytes_get $group_name)
 | |
| 
 | |
| 	if [[ $t0_packets -eq $t1_packets && $t0_bytes -eq $t1_bytes ]]; then
 | |
| 		return 0
 | |
| 	else
 | |
| 		return 1
 | |
| 	fi
 | |
| }
 | |
| 
 | |
| devlink_trap_exception_test()
 | |
| {
 | |
| 	local trap_name=$1; shift
 | |
| 	local group_name
 | |
| 
 | |
| 	group_name=$(devlink_trap_group_get $trap_name)
 | |
| 
 | |
| 	devlink_trap_stats_idle_test $trap_name
 | |
| 	check_fail $? "Trap stats idle when packets should have been trapped"
 | |
| 
 | |
| 	devlink_trap_group_stats_idle_test $group_name
 | |
| 	check_fail $? "Trap group idle when packets should have been trapped"
 | |
| }
 | |
| 
 | |
| devlink_trap_drop_test()
 | |
| {
 | |
| 	local trap_name=$1; shift
 | |
| 	local dev=$1; shift
 | |
| 	local handle=$1; shift
 | |
| 	local group_name
 | |
| 
 | |
| 	group_name=$(devlink_trap_group_get $trap_name)
 | |
| 
 | |
| 	# This is the common part of all the tests. It checks that stats are
 | |
| 	# initially idle, then non-idle after changing the trap action and
 | |
| 	# finally idle again. It also makes sure the packets are dropped and
 | |
| 	# never forwarded.
 | |
| 	devlink_trap_stats_idle_test $trap_name
 | |
| 	check_err $? "Trap stats not idle with initial drop action"
 | |
| 	devlink_trap_group_stats_idle_test $group_name
 | |
| 	check_err $? "Trap group stats not idle with initial drop action"
 | |
| 
 | |
| 	devlink_trap_action_set $trap_name "trap"
 | |
| 	devlink_trap_stats_idle_test $trap_name
 | |
| 	check_fail $? "Trap stats idle after setting action to trap"
 | |
| 	devlink_trap_group_stats_idle_test $group_name
 | |
| 	check_fail $? "Trap group stats idle after setting action to trap"
 | |
| 
 | |
| 	devlink_trap_action_set $trap_name "drop"
 | |
| 
 | |
| 	devlink_trap_stats_idle_test $trap_name
 | |
| 	check_err $? "Trap stats not idle after setting action to drop"
 | |
| 	devlink_trap_group_stats_idle_test $group_name
 | |
| 	check_err $? "Trap group stats not idle after setting action to drop"
 | |
| 
 | |
| 	tc_check_packets "dev $dev egress" $handle 0
 | |
| 	check_err $? "Packets were not dropped"
 | |
| }
 | |
| 
 | |
| devlink_trap_drop_cleanup()
 | |
| {
 | |
| 	local mz_pid=$1; shift
 | |
| 	local dev=$1; shift
 | |
| 	local proto=$1; shift
 | |
| 	local pref=$1; shift
 | |
| 	local handle=$1; shift
 | |
| 
 | |
| 	kill $mz_pid && wait $mz_pid &> /dev/null
 | |
| 	tc filter del dev $dev egress protocol $proto pref $pref handle $handle flower
 | |
| }
 | |
| 
 | |
| devlink_trap_stats_check()
 | |
| {
 | |
| 	local trap_name=$1; shift
 | |
| 	local send_one="$@"
 | |
| 	local t0_packets
 | |
| 	local t1_packets
 | |
| 
 | |
| 	t0_packets=$(devlink_trap_rx_packets_get $trap_name)
 | |
| 
 | |
| 	$send_one && sleep 1
 | |
| 
 | |
| 	t1_packets=$(devlink_trap_rx_packets_get $trap_name)
 | |
| 
 | |
| 	[[ $t1_packets -ne $t0_packets ]]
 | |
| }
 | |
| 
 | |
| devlink_trap_stats_test()
 | |
| {
 | |
| 	local test_name=$1; shift
 | |
| 
 | |
| 	RET=0
 | |
| 
 | |
| 	devlink_trap_stats_check "$@"
 | |
| 	check_err $? "Trap stats did not increase"
 | |
| 
 | |
| 	log_test "$test_name"
 | |
| }
 | |
| 
 | |
| devlink_trap_policers_num_get()
 | |
| {
 | |
| 	devlink -j -p trap policer show | jq '.[]["'$DEVLINK_DEV'"] | length'
 | |
| }
 | |
| 
 | |
| devlink_trap_policer_rate_get()
 | |
| {
 | |
| 	local policer_id=$1; shift
 | |
| 
 | |
| 	devlink -j -p trap policer show $DEVLINK_DEV policer $policer_id \
 | |
| 		| jq '.[][][]["rate"]'
 | |
| }
 | |
| 
 | |
| devlink_trap_policer_burst_get()
 | |
| {
 | |
| 	local policer_id=$1; shift
 | |
| 
 | |
| 	devlink -j -p trap policer show $DEVLINK_DEV policer $policer_id \
 | |
| 		| jq '.[][][]["burst"]'
 | |
| }
 | |
| 
 | |
| devlink_trap_policer_rx_dropped_get()
 | |
| {
 | |
| 	local policer_id=$1; shift
 | |
| 
 | |
| 	devlink -j -p -s trap policer show $DEVLINK_DEV policer $policer_id \
 | |
| 		| jq '.[][][]["stats"]["rx"]["dropped"]'
 | |
| }
 | |
| 
 | |
| devlink_trap_group_policer_get()
 | |
| {
 | |
| 	local group_name=$1; shift
 | |
| 
 | |
| 	devlink -j -p trap group show $DEVLINK_DEV group $group_name \
 | |
| 		| jq '.[][][]["policer"]'
 | |
| }
 | |
| 
 | |
| devlink_port_by_netdev()
 | |
| {
 | |
| 	local if_name=$1
 | |
| 
 | |
| 	devlink -j port show $if_name | jq -e '.[] | keys' | jq -r '.[]'
 | |
| }
 | |
| 
 | |
| devlink_cpu_port_get()
 | |
| {
 | |
| 	local cpu_dl_port_num=$(devlink port list | grep "$DEVLINK_DEV" |
 | |
| 				grep cpu | cut -d/ -f3 | cut -d: -f1 |
 | |
| 				sed -n '1p')
 | |
| 
 | |
| 	echo "$DEVLINK_DEV/$cpu_dl_port_num"
 | |
| }
 | |
| 
 | |
| devlink_cell_size_get()
 | |
| {
 | |
| 	devlink sb pool show "$DEVLINK_DEV" pool 0 -j \
 | |
| 	    | jq '.pool[][].cell_size'
 | |
| }
 | |
| 
 | |
| devlink_pool_size_get()
 | |
| {
 | |
| 	devlink sb show "$DEVLINK_DEV" -j | jq '.[][][]["size"]'
 | |
| }
 |