1#!/bin/bash 2# SPDX-License-Identifier: GPL-2.0 3 4############################################################################## 5# Defines 6 7if [[ ! -v DEVLINK_DEV ]]; then 8 DEVLINK_DEV=$(devlink port show "${NETIFS[p1]}" -j \ 9 | jq -r '.port | keys[]' | cut -d/ -f-2) 10 if [ -z "$DEVLINK_DEV" ]; then 11 echo "SKIP: ${NETIFS[p1]} has no devlink device registered for it" 12 exit 1 13 fi 14 if [[ "$(echo $DEVLINK_DEV | grep -c pci)" -eq 0 ]]; then 15 echo "SKIP: devlink device's bus is not PCI" 16 exit 1 17 fi 18 19 DEVLINK_VIDDID=$(lspci -s $(echo $DEVLINK_DEV | cut -d"/" -f2) \ 20 -n | cut -d" " -f3) 21fi 22 23############################################################################## 24# Sanity checks 25 26devlink help 2>&1 | grep resource &> /dev/null 27if [ $? -ne 0 ]; then 28 echo "SKIP: iproute2 too old, missing devlink resource support" 29 exit 1 30fi 31 32devlink help 2>&1 | grep trap &> /dev/null 33if [ $? -ne 0 ]; then 34 echo "SKIP: iproute2 too old, missing devlink trap support" 35 exit 1 36fi 37 38devlink dev help 2>&1 | grep info &> /dev/null 39if [ $? -ne 0 ]; then 40 echo "SKIP: iproute2 too old, missing devlink dev info support" 41 exit 1 42fi 43 44############################################################################## 45# Devlink helpers 46 47devlink_resource_names_to_path() 48{ 49 local resource 50 local path="" 51 52 for resource in "${@}"; do 53 if [ "$path" == "" ]; then 54 path="$resource" 55 else 56 path="${path}/$resource" 57 fi 58 done 59 60 echo "$path" 61} 62 63devlink_resource_get() 64{ 65 local name=$1 66 local resource_name=.[][\"$DEVLINK_DEV\"] 67 68 resource_name="$resource_name | .[] | select (.name == \"$name\")" 69 70 shift 71 for resource in "${@}"; do 72 resource_name="${resource_name} | .[\"resources\"][] | \ 73 select (.name == \"$resource\")" 74 done 75 76 devlink -j resource show "$DEVLINK_DEV" | jq "$resource_name" 77} 78 79devlink_resource_size_get() 80{ 81 local size=$(devlink_resource_get "$@" | jq '.["size_new"]') 82 83 if [ "$size" == "null" ]; then 84 devlink_resource_get "$@" | jq '.["size"]' 85 else 86 echo "$size" 87 fi 88} 89 90devlink_resource_size_set() 91{ 92 local new_size=$1 93 local path 94 95 shift 96 path=$(devlink_resource_names_to_path "$@") 97 devlink resource set "$DEVLINK_DEV" path "$path" size "$new_size" 98 check_err $? "Failed setting path $path to size $size" 99} 100 101devlink_resource_occ_get() 102{ 103 devlink_resource_get "$@" | jq '.["occ"]' 104} 105 106devlink_reload() 107{ 108 local still_pending 109 110 devlink dev reload "$DEVLINK_DEV" &> /dev/null 111 check_err $? "Failed reload" 112 113 still_pending=$(devlink resource show "$DEVLINK_DEV" | \ 114 grep -c "size_new") 115 check_err $still_pending "Failed reload - There are still unset sizes" 116} 117 118declare -A DEVLINK_ORIG 119 120devlink_port_pool_threshold() 121{ 122 local port=$1; shift 123 local pool=$1; shift 124 125 devlink sb port pool show $port pool $pool -j \ 126 | jq '.port_pool."'"$port"'"[].threshold' 127} 128 129devlink_port_pool_th_set() 130{ 131 local port=$1; shift 132 local pool=$1; shift 133 local th=$1; shift 134 local key="port_pool($port,$pool).threshold" 135 136 DEVLINK_ORIG[$key]=$(devlink_port_pool_threshold $port $pool) 137 devlink sb port pool set $port pool $pool th $th 138} 139 140devlink_port_pool_th_restore() 141{ 142 local port=$1; shift 143 local pool=$1; shift 144 local key="port_pool($port,$pool).threshold" 145 146 devlink sb port pool set $port pool $pool th ${DEVLINK_ORIG[$key]} 147} 148 149devlink_pool_size_thtype() 150{ 151 local pool=$1; shift 152 153 devlink sb pool show "$DEVLINK_DEV" pool $pool -j \ 154 | jq -r '.pool[][] | (.size, .thtype)' 155} 156 157devlink_pool_size_thtype_set() 158{ 159 local pool=$1; shift 160 local thtype=$1; shift 161 local size=$1; shift 162 local key="pool($pool).size_thtype" 163 164 DEVLINK_ORIG[$key]=$(devlink_pool_size_thtype $pool) 165 devlink sb pool set "$DEVLINK_DEV" pool $pool size $size thtype $thtype 166} 167 168devlink_pool_size_thtype_restore() 169{ 170 local pool=$1; shift 171 local key="pool($pool).size_thtype" 172 local -a orig=(${DEVLINK_ORIG[$key]}) 173 174 devlink sb pool set "$DEVLINK_DEV" pool $pool \ 175 size ${orig[0]} thtype ${orig[1]} 176} 177 178devlink_tc_bind_pool_th() 179{ 180 local port=$1; shift 181 local tc=$1; shift 182 local dir=$1; shift 183 184 devlink sb tc bind show $port tc $tc type $dir -j \ 185 | jq -r '.tc_bind[][] | (.pool, .threshold)' 186} 187 188devlink_tc_bind_pool_th_set() 189{ 190 local port=$1; shift 191 local tc=$1; shift 192 local dir=$1; shift 193 local pool=$1; shift 194 local th=$1; shift 195 local key="tc_bind($port,$dir,$tc).pool_th" 196 197 DEVLINK_ORIG[$key]=$(devlink_tc_bind_pool_th $port $tc $dir) 198 devlink sb tc bind set $port tc $tc type $dir pool $pool th $th 199} 200 201devlink_tc_bind_pool_th_restore() 202{ 203 local port=$1; shift 204 local tc=$1; shift 205 local dir=$1; shift 206 local key="tc_bind($port,$dir,$tc).pool_th" 207 local -a orig=(${DEVLINK_ORIG[$key]}) 208 209 devlink sb tc bind set $port tc $tc type $dir \ 210 pool ${orig[0]} th ${orig[1]} 211} 212 213devlink_traps_num_get() 214{ 215 devlink -j trap | jq '.[]["'$DEVLINK_DEV'"] | length' 216} 217 218devlink_traps_get() 219{ 220 devlink -j trap | jq -r '.[]["'$DEVLINK_DEV'"][].name' 221} 222 223devlink_trap_type_get() 224{ 225 local trap_name=$1; shift 226 227 devlink -j trap show $DEVLINK_DEV trap $trap_name \ 228 | jq -r '.[][][].type' 229} 230 231devlink_trap_action_set() 232{ 233 local trap_name=$1; shift 234 local action=$1; shift 235 236 # Pipe output to /dev/null to avoid expected warnings. 237 devlink trap set $DEVLINK_DEV trap $trap_name \ 238 action $action &> /dev/null 239} 240 241devlink_trap_action_get() 242{ 243 local trap_name=$1; shift 244 245 devlink -j trap show $DEVLINK_DEV trap $trap_name \ 246 | jq -r '.[][][].action' 247} 248 249devlink_trap_group_get() 250{ 251 devlink -j trap show $DEVLINK_DEV trap $trap_name \ 252 | jq -r '.[][][].group' 253} 254 255devlink_trap_metadata_test() 256{ 257 local trap_name=$1; shift 258 local metadata=$1; shift 259 260 devlink -jv trap show $DEVLINK_DEV trap $trap_name \ 261 | jq -e '.[][][].metadata | contains(["'$metadata'"])' \ 262 &> /dev/null 263} 264 265devlink_trap_rx_packets_get() 266{ 267 local trap_name=$1; shift 268 269 devlink -js trap show $DEVLINK_DEV trap $trap_name \ 270 | jq '.[][][]["stats"]["rx"]["packets"]' 271} 272 273devlink_trap_rx_bytes_get() 274{ 275 local trap_name=$1; shift 276 277 devlink -js trap show $DEVLINK_DEV trap $trap_name \ 278 | jq '.[][][]["stats"]["rx"]["bytes"]' 279} 280 281devlink_trap_stats_idle_test() 282{ 283 local trap_name=$1; shift 284 local t0_packets t0_bytes 285 local t1_packets t1_bytes 286 287 t0_packets=$(devlink_trap_rx_packets_get $trap_name) 288 t0_bytes=$(devlink_trap_rx_bytes_get $trap_name) 289 290 sleep 1 291 292 t1_packets=$(devlink_trap_rx_packets_get $trap_name) 293 t1_bytes=$(devlink_trap_rx_bytes_get $trap_name) 294 295 if [[ $t0_packets -eq $t1_packets && $t0_bytes -eq $t1_bytes ]]; then 296 return 0 297 else 298 return 1 299 fi 300} 301 302devlink_traps_enable_all() 303{ 304 local trap_name 305 306 for trap_name in $(devlink_traps_get); do 307 devlink_trap_action_set $trap_name "trap" 308 done 309} 310 311devlink_traps_disable_all() 312{ 313 for trap_name in $(devlink_traps_get); do 314 devlink_trap_action_set $trap_name "drop" 315 done 316} 317 318devlink_trap_groups_get() 319{ 320 devlink -j trap group | jq -r '.[]["'$DEVLINK_DEV'"][].name' 321} 322 323devlink_trap_group_action_set() 324{ 325 local group_name=$1; shift 326 local action=$1; shift 327 328 # Pipe output to /dev/null to avoid expected warnings. 329 devlink trap group set $DEVLINK_DEV group $group_name action $action \ 330 &> /dev/null 331} 332 333devlink_trap_group_rx_packets_get() 334{ 335 local group_name=$1; shift 336 337 devlink -js trap group show $DEVLINK_DEV group $group_name \ 338 | jq '.[][][]["stats"]["rx"]["packets"]' 339} 340 341devlink_trap_group_rx_bytes_get() 342{ 343 local group_name=$1; shift 344 345 devlink -js trap group show $DEVLINK_DEV group $group_name \ 346 | jq '.[][][]["stats"]["rx"]["bytes"]' 347} 348 349devlink_trap_group_stats_idle_test() 350{ 351 local group_name=$1; shift 352 local t0_packets t0_bytes 353 local t1_packets t1_bytes 354 355 t0_packets=$(devlink_trap_group_rx_packets_get $group_name) 356 t0_bytes=$(devlink_trap_group_rx_bytes_get $group_name) 357 358 sleep 1 359 360 t1_packets=$(devlink_trap_group_rx_packets_get $group_name) 361 t1_bytes=$(devlink_trap_group_rx_bytes_get $group_name) 362 363 if [[ $t0_packets -eq $t1_packets && $t0_bytes -eq $t1_bytes ]]; then 364 return 0 365 else 366 return 1 367 fi 368} 369 370devlink_trap_exception_test() 371{ 372 local trap_name=$1; shift 373 local group_name 374 375 group_name=$(devlink_trap_group_get $trap_name) 376 377 devlink_trap_stats_idle_test $trap_name 378 check_fail $? "Trap stats idle when packets should have been trapped" 379 380 devlink_trap_group_stats_idle_test $group_name 381 check_fail $? "Trap group idle when packets should have been trapped" 382} 383 384devlink_trap_drop_test() 385{ 386 local trap_name=$1; shift 387 local dev=$1; shift 388 local handle=$1; shift 389 local group_name 390 391 group_name=$(devlink_trap_group_get $trap_name) 392 393 # This is the common part of all the tests. It checks that stats are 394 # initially idle, then non-idle after changing the trap action and 395 # finally idle again. It also makes sure the packets are dropped and 396 # never forwarded. 397 devlink_trap_stats_idle_test $trap_name 398 check_err $? "Trap stats not idle with initial drop action" 399 devlink_trap_group_stats_idle_test $group_name 400 check_err $? "Trap group stats not idle with initial drop action" 401 402 devlink_trap_action_set $trap_name "trap" 403 devlink_trap_stats_idle_test $trap_name 404 check_fail $? "Trap stats idle after setting action to trap" 405 devlink_trap_group_stats_idle_test $group_name 406 check_fail $? "Trap group stats idle after setting action to trap" 407 408 devlink_trap_action_set $trap_name "drop" 409 410 devlink_trap_stats_idle_test $trap_name 411 check_err $? "Trap stats not idle after setting action to drop" 412 devlink_trap_group_stats_idle_test $group_name 413 check_err $? "Trap group stats not idle after setting action to drop" 414 415 tc_check_packets "dev $dev egress" $handle 0 416 check_err $? "Packets were not dropped" 417} 418 419devlink_trap_drop_cleanup() 420{ 421 local mz_pid=$1; shift 422 local dev=$1; shift 423 local proto=$1; shift 424 local pref=$1; shift 425 local handle=$1; shift 426 427 kill $mz_pid && wait $mz_pid &> /dev/null 428 tc filter del dev $dev egress protocol $proto pref $pref handle $handle flower 429} 430 431devlink_trap_stats_test() 432{ 433 local test_name=$1; shift 434 local trap_name=$1; shift 435 local send_one="$@" 436 local t0_packets 437 local t1_packets 438 439 RET=0 440 441 t0_packets=$(devlink_trap_rx_packets_get $trap_name) 442 443 $send_one && sleep 1 444 445 t1_packets=$(devlink_trap_rx_packets_get $trap_name) 446 447 if [[ $t1_packets -eq $t0_packets ]]; then 448 check_err 1 "Trap stats did not increase" 449 fi 450 451 log_test "$test_name" 452} 453 454devlink_trap_policers_num_get() 455{ 456 devlink -j -p trap policer show | jq '.[]["'$DEVLINK_DEV'"] | length' 457} 458 459devlink_trap_policer_rate_get() 460{ 461 local policer_id=$1; shift 462 463 devlink -j -p trap policer show $DEVLINK_DEV policer $policer_id \ 464 | jq '.[][][]["rate"]' 465} 466 467devlink_trap_policer_burst_get() 468{ 469 local policer_id=$1; shift 470 471 devlink -j -p trap policer show $DEVLINK_DEV policer $policer_id \ 472 | jq '.[][][]["burst"]' 473} 474 475devlink_trap_policer_rx_dropped_get() 476{ 477 local policer_id=$1; shift 478 479 devlink -j -p -s trap policer show $DEVLINK_DEV policer $policer_id \ 480 | jq '.[][][]["stats"]["rx"]["dropped"]' 481} 482 483devlink_trap_group_policer_get() 484{ 485 local group_name=$1; shift 486 487 devlink -j -p trap group show $DEVLINK_DEV group $group_name \ 488 | jq '.[][][]["policer"]' 489} 490 491devlink_trap_policer_ids_get() 492{ 493 devlink -j -p trap policer show \ 494 | jq '.[]["'$DEVLINK_DEV'"][]["policer"]' 495} 496 497devlink_port_by_netdev() 498{ 499 local if_name=$1 500 501 devlink -j port show $if_name | jq -e '.[] | keys' | jq -r '.[]' 502} 503 504devlink_cpu_port_get() 505{ 506 local cpu_dl_port_num=$(devlink port list | grep "$DEVLINK_DEV" | 507 grep cpu | cut -d/ -f3 | cut -d: -f1 | 508 sed -n '1p') 509 510 echo "$DEVLINK_DEV/$cpu_dl_port_num" 511} 512