Linux Audio

Check our new training course

Loading...
v6.13.7
   1# bpftool(8) bash completion                               -*- shell-script -*-
   2#
   3# SPDX-License-Identifier: (GPL-2.0-only OR BSD-2-Clause)
   4# Copyright (C) 2017-2018 Netronome Systems, Inc.
   5#
   6# Author: Quentin Monnet <quentin.monnet@netronome.com>
   7
   8# Takes a list of words in argument; each one of them is added to COMPREPLY if
   9# it is not already present on the command line. Returns no value.
  10_bpftool_once_attr()
  11{
  12    local w idx found
  13    for w in $*; do
  14        found=0
  15        for (( idx=3; idx < ${#words[@]}-1; idx++ )); do
  16            if [[ $w == ${words[idx]} ]]; then
  17                found=1
  18                break
  19            fi
  20        done
  21        [[ $found -eq 0 ]] && \
  22            COMPREPLY+=( $( compgen -W "$w" -- "$cur" ) )
  23    done
  24}
  25
  26# Takes a list of words as argument; if any of those words is present on the
  27# command line, return 0. Otherwise, return 1.
  28_bpftool_search_list()
  29{
  30    local w idx
  31    for w in $*; do
  32        for (( idx=3; idx < ${#words[@]}-1; idx++ )); do
  33            [[ $w == ${words[idx]} ]] && return 0
  34        done
  35    done
  36    return 1
  37}
  38
  39# Takes a list of words in argument; adds them all to COMPREPLY if none of them
  40# is already present on the command line. Returns no value.
  41_bpftool_one_of_list()
  42{
  43    _bpftool_search_list $* && return 1
  44    COMPREPLY+=( $( compgen -W "$*" -- "$cur" ) )
  45}
  46
  47_bpftool_get_map_ids()
  48{
  49    COMPREPLY+=( $( compgen -W "$( bpftool -jp map  2>&1 | \
  50        command sed -n 's/.*"id": \(.*\),$/\1/p' )" -- "$cur" ) )
  51}
  52
  53# Takes map type and adds matching map ids to the list of suggestions.
  54_bpftool_get_map_ids_for_type()
  55{
  56    local type="$1"
  57    COMPREPLY+=( $( compgen -W "$( bpftool -jp map  2>&1 | \
  58        command grep -C2 "$type" | \
  59        command sed -n 's/.*"id": \(.*\),$/\1/p' )" -- "$cur" ) )
  60}
  61
  62_bpftool_get_map_names()
  63{
  64    COMPREPLY+=( $( compgen -W "$( bpftool -jp map  2>&1 | \
  65        command sed -n 's/.*"name": \(.*\),$/\1/p' )" -- "$cur" ) )
  66}
  67
  68# Takes map type and adds matching map names to the list of suggestions.
  69_bpftool_get_map_names_for_type()
  70{
  71    local type="$1"
  72    COMPREPLY+=( $( compgen -W "$( bpftool -jp map  2>&1 | \
  73        command grep -C2 "$type" | \
  74        command sed -n 's/.*"name": \(.*\),$/\1/p' )" -- "$cur" ) )
  75}
  76
  77_bpftool_get_prog_ids()
  78{
  79    COMPREPLY+=( $( compgen -W "$( bpftool -jp prog 2>&1 | \
  80        command sed -n 's/.*"id": \(.*\),$/\1/p' )" -- "$cur" ) )
  81}
  82
  83_bpftool_get_prog_tags()
  84{
  85    COMPREPLY+=( $( compgen -W "$( bpftool -jp prog 2>&1 | \
  86        command sed -n 's/.*"tag": "\(.*\)",$/\1/p' )" -- "$cur" ) )
  87}
  88
  89_bpftool_get_prog_names()
  90{
  91    COMPREPLY+=( $( compgen -W "$( bpftool -jp prog 2>&1 | \
  92        command sed -n 's/.*"name": "\(.*\)",$/\1/p' )" -- "$cur" ) )
  93}
  94
  95_bpftool_get_btf_ids()
  96{
  97    COMPREPLY+=( $( compgen -W "$( bpftool -jp btf 2>&1 | \
  98        command sed -n 's/.*"id": \(.*\),$/\1/p' )" -- "$cur" ) )
  99}
 100
 101_bpftool_get_link_ids()
 102{
 103    COMPREPLY+=( $( compgen -W "$( bpftool -jp link 2>&1 | \
 104        command sed -n 's/.*"id": \(.*\),$/\1/p' )" -- "$cur" ) )
 105}
 106
 107_bpftool_get_obj_map_names()
 108{
 109    local obj maps
 110
 111    obj=$1
 112
 113    maps=$(objdump -j .maps -t $obj 2>/dev/null | \
 114        command awk '/g     . .maps/ {print $NF}')
 115
 116    COMPREPLY+=( $( compgen -W "$maps" -- "$cur" ) )
 117}
 118
 119_bpftool_get_obj_map_idxs()
 120{
 121    local obj nmaps
 122
 123    obj=$1
 124
 125    nmaps=$(objdump -j maps -t $obj 2>/dev/null | grep -c 'g     . maps')
 126
 127    COMPREPLY+=( $( compgen -W "$(seq 0 $((nmaps - 1)))" -- "$cur" ) )
 128}
 129
 130_sysfs_get_netdevs()
 131{
 132    COMPREPLY+=( $( compgen -W "$( ls /sys/class/net 2>/dev/null )" -- \
 133        "$cur" ) )
 134}
 135
 136# Retrieve type of the map that we are operating on.
 137_bpftool_map_guess_map_type()
 138{
 139    local keyword idx ref=""
 140    for (( idx=3; idx < ${#words[@]}-1; idx++ )); do
 141        case "${words[$((idx-2))]}" in
 142            lookup|update)
 143                keyword=${words[$((idx-1))]}
 144                ref=${words[$((idx))]}
 145                ;;
 146            push)
 147                printf "stack"
 148                return 0
 149                ;;
 150            enqueue)
 151                printf "queue"
 152                return 0
 153                ;;
 154        esac
 155    done
 156    [[ -z $ref ]] && return 0
 157
 158    local type
 159    type=$(bpftool -jp map show $keyword $ref | \
 160        command sed -n 's/.*"type": "\(.*\)",$/\1/p')
 161    [[ -n $type ]] && printf $type
 162}
 163
 164_bpftool_map_update_get_id()
 165{
 166    local command="$1"
 167
 168    # Is it the map to update, or a map to insert into the map to update?
 169    # Search for "value" keyword.
 170    local idx value
 171    for (( idx=7; idx < ${#words[@]}-1; idx++ )); do
 172        if [[ ${words[idx]} == "value" ]]; then
 173            value=1
 174            break
 175        fi
 176    done
 177    if [[ $value -eq 0 ]]; then
 178        case "$command" in
 179            push)
 180                _bpftool_get_map_ids_for_type stack
 181                ;;
 182            enqueue)
 183                _bpftool_get_map_ids_for_type queue
 184                ;;
 185            *)
 186                _bpftool_get_map_ids
 187                ;;
 188        esac
 189        return 0
 190    fi
 191
 192    # Id to complete is for a value. It can be either prog id or map id. This
 193    # depends on the type of the map to update.
 194    local type=$(_bpftool_map_guess_map_type)
 195    case $type in
 196        array_of_maps|hash_of_maps)
 197            _bpftool_get_map_ids
 198            return 0
 199            ;;
 200        prog_array)
 201            _bpftool_get_prog_ids
 202            return 0
 203            ;;
 204        *)
 205            return 0
 206            ;;
 207    esac
 208}
 209
 210_bpftool_map_update_get_name()
 211{
 212    local command="$1"
 213
 214    # Is it the map to update, or a map to insert into the map to update?
 215    # Search for "value" keyword.
 216    local idx value
 217    for (( idx=7; idx < ${#words[@]}-1; idx++ )); do
 218        if [[ ${words[idx]} == "value" ]]; then
 219            value=1
 220            break
 221        fi
 222    done
 223    if [[ $value -eq 0 ]]; then
 224        case "$command" in
 225            push)
 226                _bpftool_get_map_names_for_type stack
 227                ;;
 228            enqueue)
 229                _bpftool_get_map_names_for_type queue
 230                ;;
 231            *)
 232                _bpftool_get_map_names
 233                ;;
 234        esac
 235        return 0
 236    fi
 237
 238    # Name to complete is for a value. It can be either prog name or map name. This
 239    # depends on the type of the map to update.
 240    local type=$(_bpftool_map_guess_map_type)
 241    case $type in
 242        array_of_maps|hash_of_maps)
 243            _bpftool_get_map_names
 244            return 0
 245            ;;
 246        prog_array)
 247            _bpftool_get_prog_names
 248            return 0
 249            ;;
 250        *)
 251            return 0
 252            ;;
 253    esac
 254}
 255
 256_bpftool()
 257{
 258    local cur prev words cword comp_args
 259    local json=0
 260    _init_completion -- "$@" || return
 261
 262    # Deal with options
 263    if [[ ${words[cword]} == -* ]]; then
 264        local c='--version --json --pretty --bpffs --mapcompat --debug \
 265            --use-loader --base-btf'
 266        COMPREPLY=( $( compgen -W "$c" -- "$cur" ) )
 267        return 0
 268    fi
 269    if _bpftool_search_list -j --json -p --pretty; then
 270        json=1
 271    fi
 272
 273    # Deal with simplest keywords
 274    case $prev in
 275        help|hex)
 276            return 0
 277            ;;
 278        tag)
 279            _bpftool_get_prog_tags
 280            return 0
 281            ;;
 282        dev|offload_dev|xdpmeta_dev)
 283            _sysfs_get_netdevs
 284            return 0
 285            ;;
 286        file|pinned|-B|--base-btf)
 287            _filedir
 288            return 0
 289            ;;
 290        batch)
 291            COMPREPLY=( $( compgen -W 'file' -- "$cur" ) )
 292            return 0
 293            ;;
 294    esac
 295
 296    # Remove all options so completions don't have to deal with them.
 297    local i pprev
 298    for (( i=1; i < ${#words[@]}; )); do
 299        if [[ ${words[i]::1} == - ]] &&
 300            [[ ${words[i]} != "-B" ]] && [[ ${words[i]} != "--base-btf" ]]; then
 301            words=( "${words[@]:0:i}" "${words[@]:i+1}" )
 302            [[ $i -le $cword ]] && cword=$(( cword - 1 ))
 303        else
 304            i=$(( ++i ))
 305        fi
 306    done
 307    cur=${words[cword]}
 308    prev=${words[cword - 1]}
 309    pprev=${words[cword - 2]}
 310
 311    local object=${words[1]}
 312
 313    if [[ -z $object || $cword -eq 1 ]]; then
 314        case $cur in
 315            *)
 316                COMPREPLY=( $( compgen -W "$( bpftool help 2>&1 | \
 317                    command sed \
 318                    -e '/OBJECT := /!d' \
 319                    -e 's/.*{//' \
 320                    -e 's/}.*//' \
 321                    -e 's/|//g' )" -- "$cur" ) )
 322                COMPREPLY+=( $( compgen -W 'batch help' -- "$cur" ) )
 323                return 0
 324                ;;
 325        esac
 326    fi
 327
 328    local command=${words[2]}
 329    [[ $command == help ]] && return 0
 330
 331    local MAP_TYPE='id pinned name'
 332    local PROG_TYPE='id pinned tag name'
 333
 334    # Completion depends on object and command in use
 335    case $object in
 336        prog)
 337            # Complete id and name, only for subcommands that use prog (but no
 338            # map) ids/names.
 339            case $command in
 340                show|list|dump|pin)
 341                    case $prev in
 342                        id)
 343                            _bpftool_get_prog_ids
 344                            return 0
 345                            ;;
 346                        name)
 347                            _bpftool_get_prog_names
 348                            return 0
 349                            ;;
 350                    esac
 351                    ;;
 352            esac
 353
 
 
 354            local METRIC_TYPE='cycles instructions l1d_loads llc_misses \
 355                itlb_misses dtlb_misses'
 356            case $command in
 357                show|list)
 358                    [[ $prev != "$command" ]] && return 0
 359                    COMPREPLY=( $( compgen -W "$PROG_TYPE" -- "$cur" ) )
 360                    return 0
 361                    ;;
 362                dump)
 363                    case $prev in
 364                        $command)
 365                            COMPREPLY+=( $( compgen -W "xlated jited" -- \
 366                                "$cur" ) )
 367                            return 0
 368                            ;;
 369                        xlated|jited)
 370                            COMPREPLY=( $( compgen -W "$PROG_TYPE" -- \
 371                                "$cur" ) )
 372                            return 0
 373                            ;;
 374                        *)
 375                            # "file" is not compatible with other keywords here
 376                            if _bpftool_search_list 'file'; then
 377                                return 0
 378                            fi
 379                            if ! _bpftool_search_list 'linum opcodes visual'; then
 380                                _bpftool_once_attr 'file'
 381                            fi
 382                            _bpftool_once_attr 'linum opcodes'
 383                            if _bpftool_search_list 'xlated' && [[ "$json" == 0 ]]; then
 384                                _bpftool_once_attr 'visual'
 385                            fi
 386                            return 0
 387                            ;;
 388                    esac
 389                    ;;
 390                pin)
 391                    if [[ $prev == "$command" ]]; then
 392                        COMPREPLY=( $( compgen -W "$PROG_TYPE" -- "$cur" ) )
 393                    else
 394                        _filedir
 395                    fi
 396                    return 0
 397                    ;;
 398                attach|detach)
 399                    case $cword in
 400                        3)
 401                            COMPREPLY=( $( compgen -W "$PROG_TYPE" -- "$cur" ) )
 402                            return 0
 403                            ;;
 404                        4)
 405                            case $prev in
 406                                id)
 407                                    _bpftool_get_prog_ids
 408                                    ;;
 409                                name)
 410                                    _bpftool_get_prog_names
 411                                    ;;
 412                                pinned)
 413                                    _filedir
 414                                    ;;
 415                            esac
 416                            return 0
 417                            ;;
 418                        5)
 419                            local BPFTOOL_PROG_ATTACH_TYPES='sk_msg_verdict \
 420                                sk_skb_verdict sk_skb_stream_verdict sk_skb_stream_parser \
 421                                flow_dissector'
 422                            COMPREPLY=( $( compgen -W "$BPFTOOL_PROG_ATTACH_TYPES" -- "$cur" ) )
 423                            return 0
 424                            ;;
 425                        6)
 426                            COMPREPLY=( $( compgen -W "$MAP_TYPE" -- "$cur" ) )
 427                            return 0
 428                            ;;
 429                        7)
 430                            case $prev in
 431                                id)
 432                                    _bpftool_get_map_ids
 433                                    ;;
 434                                name)
 435                                    _bpftool_get_map_names
 436                                    ;;
 437                                pinned)
 438                                    _filedir
 439                                    ;;
 440                            esac
 441                            return 0
 442                            ;;
 443                    esac
 444                    ;;
 445                load|loadall)
 446                    local obj
 447
 448                    # Propose "load/loadall" to complete "bpftool prog load",
 449                    # or bash tries to complete "load" as a filename below.
 450                    if [[ ${#words[@]} -eq 3 ]]; then
 451                        COMPREPLY=( $( compgen -W "load loadall" -- "$cur" ) )
 452                        return 0
 453                    fi
 454
 455                    if [[ ${#words[@]} -lt 6 ]]; then
 456                        _filedir
 457                        return 0
 458                    fi
 459
 460                    obj=${words[3]}
 461
 462                    if [[ ${words[-4]} == "map" ]]; then
 463                        COMPREPLY=( $( compgen -W "$MAP_TYPE" -- "$cur" ) )
 464                        return 0
 465                    fi
 466                    if [[ ${words[-3]} == "map" ]]; then
 467                        if [[ ${words[-2]} == "idx" ]]; then
 468                            _bpftool_get_obj_map_idxs $obj
 469                        elif [[ ${words[-2]} == "name" ]]; then
 470                            _bpftool_get_obj_map_names $obj
 471                        fi
 472                        return 0
 473                    fi
 474                    if [[ ${words[-2]} == "map" ]]; then
 475                        COMPREPLY=( $( compgen -W "idx name" -- "$cur" ) )
 476                        return 0
 477                    fi
 478
 479                    case $prev in
 480                        type)
 481                            local BPFTOOL_PROG_LOAD_TYPES='socket kprobe \
 482                                kretprobe classifier flow_dissector \
 483                                action tracepoint raw_tracepoint \
 484                                xdp perf_event cgroup/skb cgroup/sock \
 485                                cgroup/dev lwt_in lwt_out lwt_xmit \
 486                                lwt_seg6local sockops sk_skb sk_msg lirc_mode2 \
 487                                cgroup/bind4 cgroup/bind6 \
 488                                cgroup/connect4 cgroup/connect6 cgroup/connect_unix \
 489                                cgroup/getpeername4 cgroup/getpeername6 cgroup/getpeername_unix \
 490                                cgroup/getsockname4 cgroup/getsockname6 cgroup/getsockname_unix \
 491                                cgroup/sendmsg4 cgroup/sendmsg6 cgroup/sendmsg_unix \
 492                                cgroup/recvmsg4 cgroup/recvmsg6 cgroup/recvmsg_unix \
 493                                cgroup/post_bind4 cgroup/post_bind6 \
 494                                cgroup/sysctl cgroup/getsockopt \
 495                                cgroup/setsockopt cgroup/sock_release struct_ops \
 496                                fentry fexit freplace sk_lookup'
 497                            COMPREPLY=( $( compgen -W "$BPFTOOL_PROG_LOAD_TYPES" -- "$cur" ) )
 498                            return 0
 499                            ;;
 500                        id)
 501                            _bpftool_get_map_ids
 502                            return 0
 503                            ;;
 504                        name)
 505                            _bpftool_get_map_names
 506                            return 0
 507                            ;;
 508                        pinned|pinmaps)
 509                            _filedir
 510                            return 0
 511                            ;;
 512                        *)
 513                            COMPREPLY=( $( compgen -W "map" -- "$cur" ) )
 514                            _bpftool_once_attr 'type pinmaps autoattach'
 515                            _bpftool_one_of_list 'offload_dev xdpmeta_dev'
 516                            return 0
 517                            ;;
 518                    esac
 519                    ;;
 520                tracelog)
 521                    return 0
 522                    ;;
 523                profile)
 524                    case $cword in
 525                        3)
 526                            COMPREPLY=( $( compgen -W "$PROG_TYPE" -- "$cur" ) )
 527                            return 0
 528                            ;;
 529                        4)
 530                            case $prev in
 531                                id)
 532                                    _bpftool_get_prog_ids
 533                                    ;;
 534                                name)
 535                                    _bpftool_get_prog_names
 536                                    ;;
 537                                pinned)
 538                                    _filedir
 539                                    ;;
 540                            esac
 541                            return 0
 542                            ;;
 543                        5)
 544                            COMPREPLY=( $( compgen -W "$METRIC_TYPE duration" -- "$cur" ) )
 545                            return 0
 546                            ;;
 
 
 
 
 
 
 
 
 
 
 
 
 547                        *)
 548                            [[ $prev == duration ]] && return 0
 549                            _bpftool_once_attr "$METRIC_TYPE"
 550                            return 0
 551                            ;;
 552                    esac
 553                    ;;
 554                run)
 555                    if [[ ${#words[@]} -eq 4 ]]; then
 556                        COMPREPLY=( $( compgen -W "$PROG_TYPE" -- "$cur" ) )
 557                        return 0
 558                    fi
 559                    case $prev in
 560                        id)
 561                            _bpftool_get_prog_ids
 562                            return 0
 563                            ;;
 564                        name)
 565                            _bpftool_get_prog_names
 566                            return 0
 567                            ;;
 568                        data_in|data_out|ctx_in|ctx_out)
 569                            _filedir
 570                            return 0
 571                            ;;
 572                        repeat|data_size_out|ctx_size_out)
 573                            return 0
 574                            ;;
 575                        *)
 576                            _bpftool_once_attr 'data_in data_out data_size_out \
 577                                ctx_in ctx_out ctx_size_out repeat'
 578                            return 0
 579                            ;;
 580                    esac
 581                    ;;
 582                *)
 583                    [[ $prev == $object ]] && \
 584                        COMPREPLY=( $( compgen -W 'dump help pin attach detach \
 585                            load loadall show list tracelog run profile' -- "$cur" ) )
 586                    ;;
 587            esac
 588            ;;
 589        struct_ops)
 590            local STRUCT_OPS_TYPE='id name'
 591            case $command in
 592                show|list|dump|unregister)
 593                    case $prev in
 594                        $command)
 595                            COMPREPLY=( $( compgen -W "$STRUCT_OPS_TYPE" -- "$cur" ) )
 596                            ;;
 597                        id)
 598                            _bpftool_get_map_ids_for_type struct_ops
 599                            ;;
 600                        name)
 601                            _bpftool_get_map_names_for_type struct_ops
 602                            ;;
 603                    esac
 604                    return 0
 605                    ;;
 606                register)
 607                    [[ $prev == $command ]] && _filedir
 608                    return 0
 609                    ;;
 610                *)
 611                    [[ $prev == $object ]] && \
 612                        COMPREPLY=( $( compgen -W 'register unregister show list dump help' \
 613                            -- "$cur" ) )
 614                    ;;
 615            esac
 616            ;;
 617        iter)
 618            case $command in
 619                pin)
 620                    case $prev in
 621                        $command)
 622                            _filedir
 623                            ;;
 624                        id)
 625                            _bpftool_get_map_ids
 626                            ;;
 627                        name)
 628                            _bpftool_get_map_names
 629                            ;;
 630                        pinned)
 631                            _filedir
 632                            ;;
 633                        map)
 634                            _bpftool_one_of_list $MAP_TYPE
 635                            ;;
 636                        *)
 637                            _bpftool_once_attr 'map'
 638                            ;;
 639                    esac
 640                    return 0
 641                    ;;
 642                *)
 643                    [[ $prev == $object ]] && \
 644                        COMPREPLY=( $( compgen -W 'pin help' \
 645                            -- "$cur" ) )
 646                    ;;
 647            esac
 648            ;;
 649        map)
 
 650            case $command in
 651                show|list|dump|peek|pop|dequeue|freeze)
 652                    case $prev in
 653                        $command)
 654                            COMPREPLY=( $( compgen -W "$MAP_TYPE" -- "$cur" ) )
 655                            return 0
 656                            ;;
 657                        id)
 658                            case "$command" in
 659                                peek)
 660                                    _bpftool_get_map_ids_for_type stack
 661                                    _bpftool_get_map_ids_for_type queue
 662                                    ;;
 663                                pop)
 664                                    _bpftool_get_map_ids_for_type stack
 665                                    ;;
 666                                dequeue)
 667                                    _bpftool_get_map_ids_for_type queue
 668                                    ;;
 669                                *)
 670                                    _bpftool_get_map_ids
 671                                    ;;
 672                            esac
 673                            return 0
 674                            ;;
 675                        name)
 676                            case "$command" in
 677                                peek)
 678                                    _bpftool_get_map_names_for_type stack
 679                                    _bpftool_get_map_names_for_type queue
 680                                    ;;
 681                                pop)
 682                                    _bpftool_get_map_names_for_type stack
 683                                    ;;
 684                                dequeue)
 685                                    _bpftool_get_map_names_for_type queue
 686                                    ;;
 687                                *)
 688                                    _bpftool_get_map_names
 689                                    ;;
 690                            esac
 691                            return 0
 692                            ;;
 693                        *)
 694                            return 0
 695                            ;;
 696                    esac
 697                    ;;
 698                create)
 699                    case $prev in
 700                        $command)
 701                            _filedir
 702                            return 0
 703                            ;;
 704                        type)
 705                            local BPFTOOL_MAP_CREATE_TYPES="$(bpftool feature list_builtins map_types 2>/dev/null | \
 706                                grep -v '^unspec$')"
 707                            COMPREPLY=( $( compgen -W "$BPFTOOL_MAP_CREATE_TYPES" -- "$cur" ) )
 708                            return 0
 709                            ;;
 710                        key|value|flags|entries)
 711                            return 0
 712                            ;;
 713                        inner_map)
 714                            COMPREPLY=( $( compgen -W "$MAP_TYPE" -- "$cur" ) )
 715                            return 0
 716                            ;;
 717                        id)
 718                            _bpftool_get_map_ids
 719                            ;;
 720                        name)
 721                            case $pprev in
 722                                inner_map)
 723                                    _bpftool_get_map_names
 724                                    ;;
 725                                *)
 726                                    return 0
 727                                    ;;
 728                            esac
 729                            ;;
 730                        *)
 731                            _bpftool_once_attr 'type key value entries name flags offload_dev'
 732                            if _bpftool_search_list 'array_of_maps' 'hash_of_maps'; then
 733                                _bpftool_once_attr 'inner_map'
 734                            fi
 735                            return 0
 736                            ;;
 737                    esac
 738                    ;;
 739                lookup|getnext|delete)
 740                    case $prev in
 741                        $command)
 742                            COMPREPLY=( $( compgen -W "$MAP_TYPE" -- "$cur" ) )
 743                            return 0
 744                            ;;
 745                        id)
 746                            _bpftool_get_map_ids
 747                            return 0
 748                            ;;
 749                        name)
 750                            _bpftool_get_map_names
 751                            return 0
 752                            ;;
 753                        key)
 754                            COMPREPLY+=( $( compgen -W 'hex' -- "$cur" ) )
 755                            ;;
 756                        *)
 757                            case $(_bpftool_map_guess_map_type) in
 758                                queue|stack)
 759                                    return 0
 760                                    ;;
 761                            esac
 762
 763                            _bpftool_once_attr 'key'
 764                            return 0
 765                            ;;
 766                    esac
 767                    ;;
 768                update|push|enqueue)
 769                    case $prev in
 770                        $command)
 771                            COMPREPLY=( $( compgen -W "$MAP_TYPE" -- "$cur" ) )
 772                            return 0
 773                            ;;
 774                        id)
 775                            _bpftool_map_update_get_id $command
 776                            return 0
 777                            ;;
 778                        name)
 779                            _bpftool_map_update_get_name $command
 780                            return 0
 781                            ;;
 782                        key)
 783                            COMPREPLY+=( $( compgen -W 'hex' -- "$cur" ) )
 784                            ;;
 785                        value)
 786                            # We can have bytes, or references to a prog or a
 787                            # map, depending on the type of the map to update.
 788                            case "$(_bpftool_map_guess_map_type)" in
 789                                array_of_maps|hash_of_maps)
 
 790                                    COMPREPLY+=( $( compgen -W "$MAP_TYPE" \
 791                                        -- "$cur" ) )
 792                                    return 0
 793                                    ;;
 794                                prog_array)
 
 795                                    COMPREPLY+=( $( compgen -W "$PROG_TYPE" \
 796                                        -- "$cur" ) )
 797                                    return 0
 798                                    ;;
 799                                *)
 800                                    COMPREPLY+=( $( compgen -W 'hex' \
 801                                        -- "$cur" ) )
 802                                    return 0
 803                                    ;;
 804                            esac
 805                            return 0
 806                            ;;
 807                        *)
 808                            case $(_bpftool_map_guess_map_type) in
 809                                queue|stack)
 810                                    _bpftool_once_attr 'value'
 811                                    return 0;
 812                                    ;;
 813                            esac
 814
 815                            _bpftool_once_attr 'key'
 816                            local UPDATE_FLAGS='any exist noexist' idx
 817                            for (( idx=3; idx < ${#words[@]}-1; idx++ )); do
 818                                if [[ ${words[idx]} == 'value' ]]; then
 819                                    # 'value' is present, but is not the last
 820                                    # word i.e. we can now have UPDATE_FLAGS.
 821                                    _bpftool_one_of_list "$UPDATE_FLAGS"
 822                                    return 0
 823                                fi
 824                            done
 825                            for (( idx=3; idx < ${#words[@]}-1; idx++ )); do
 826                                if [[ ${words[idx]} == 'key' ]]; then
 827                                    # 'key' is present, but is not the last
 828                                    # word i.e. we can now have 'value'.
 829                                    _bpftool_once_attr 'value'
 830                                    return 0
 831                                fi
 832                            done
 833
 834                            return 0
 835                            ;;
 836                    esac
 837                    ;;
 838                pin)
 839                    case $prev in
 840                        $command)
 841                            COMPREPLY=( $( compgen -W "$MAP_TYPE" -- "$cur" ) )
 842                            ;;
 843                        id)
 844                            _bpftool_get_map_ids
 845                            ;;
 846                        name)
 847                            _bpftool_get_map_names
 848                            ;;
 849                    esac
 850                    return 0
 851                    ;;
 852                event_pipe)
 853                    case $prev in
 854                        $command)
 855                            COMPREPLY=( $( compgen -W "$MAP_TYPE" -- "$cur" ) )
 856                            return 0
 857                            ;;
 858                        id)
 859                            _bpftool_get_map_ids_for_type perf_event_array
 860                            return 0
 861                            ;;
 862                        name)
 863                            _bpftool_get_map_names_for_type perf_event_array
 864                            return 0
 865                            ;;
 866                        cpu)
 867                            return 0
 868                            ;;
 869                        index)
 870                            return 0
 871                            ;;
 872                        *)
 873                            _bpftool_once_attr 'cpu index'
 874                            return 0
 875                            ;;
 876                    esac
 877                    ;;
 878                *)
 879                    [[ $prev == $object ]] && \
 880                        COMPREPLY=( $( compgen -W 'delete dump getnext help \
 881                            lookup pin event_pipe show list update create \
 882                            peek push enqueue pop dequeue freeze' -- \
 883                            "$cur" ) )
 884                    ;;
 885            esac
 886            ;;
 887        btf)
 
 888            local MAP_TYPE='id pinned name'
 889            case $command in
 890                dump)
 891                    case $prev in
 892                        $command)
 893                            COMPREPLY+=( $( compgen -W "id map prog file" -- \
 894                                "$cur" ) )
 895                            return 0
 896                            ;;
 897                        prog)
 898                            COMPREPLY=( $( compgen -W "$PROG_TYPE" -- "$cur" ) )
 899                            return 0
 900                            ;;
 901                        map)
 902                            COMPREPLY=( $( compgen -W "$MAP_TYPE" -- "$cur" ) )
 903                            return 0
 904                            ;;
 905                        id)
 906                            case $pprev in
 907                                prog)
 908                                    _bpftool_get_prog_ids
 909                                    ;;
 910                                map)
 911                                    _bpftool_get_map_ids
 912                                    ;;
 913                                $command)
 914                                    _bpftool_get_btf_ids
 915                                    ;;
 916                            esac
 917                            return 0
 918                            ;;
 919                        name)
 920                            case $pprev in
 921                                prog)
 922                                    _bpftool_get_prog_names
 923                                    ;;
 924                                map)
 925                                    _bpftool_get_map_names
 926                                    ;;
 927                            esac
 928                            return 0
 929                            ;;
 930                        format)
 931                            COMPREPLY=( $( compgen -W "c raw" -- "$cur" ) )
 932                            ;;
 933                        c)
 934                            COMPREPLY=( $( compgen -W "unsorted" -- "$cur" ) )
 935                            ;;
 936                        *)
 937                            # emit extra options
 938                            case ${words[3]} in
 939                                id|file)
 940                                    _bpftool_once_attr 'format'
 941                                    ;;
 942                                map|prog)
 943                                    if [[ ${words[3]} == "map" ]] && [[ $cword == 6 ]]; then
 944                                        COMPREPLY+=( $( compgen -W "key value kv all" -- "$cur" ) )
 945                                    fi
 946                                    _bpftool_once_attr 'format'
 947                                    ;;
 948                                *)
 949                                    ;;
 950                            esac
 951                            return 0
 952                            ;;
 953                    esac
 954                    ;;
 955                show|list)
 956                    case $prev in
 957                        $command)
 958                            COMPREPLY+=( $( compgen -W "id" -- "$cur" ) )
 959                            ;;
 960                        id)
 961                            _bpftool_get_btf_ids
 962                            ;;
 963                    esac
 964                    return 0
 965                    ;;
 966                *)
 967                    [[ $prev == $object ]] && \
 968                        COMPREPLY=( $( compgen -W 'dump help show list' \
 969                            -- "$cur" ) )
 970                    ;;
 971            esac
 972            ;;
 973        gen)
 974            case $command in
 975                object)
 976                    _filedir
 977                    return 0
 978                    ;;
 979                skeleton)
 980                    case $prev in
 981                        $command)
 982                            _filedir
 983                            return 0
 984                            ;;
 985                        *)
 986                            _bpftool_once_attr 'name'
 987                            return 0
 988                            ;;
 989                    esac
 990                    ;;
 991                subskeleton)
 992                    case $prev in
 993                        $command)
 994                            _filedir
 995                            return 0
 996                            ;;
 997                        *)
 998                            _bpftool_once_attr 'name'
 999                            return 0
1000                            ;;
1001                    esac
1002                    ;;
1003                min_core_btf)
1004                    _filedir
1005                    return 0
1006                    ;;
1007                *)
1008                    [[ $prev == $object ]] && \
1009                        COMPREPLY=( $( compgen -W 'object skeleton subskeleton help min_core_btf' -- "$cur" ) )
1010                    ;;
1011            esac
1012            ;;
1013        cgroup)
1014            case $command in
1015                show|list|tree)
1016                    case $cword in
1017                        3)
1018                            _filedir
1019                            ;;
1020                        4)
1021                            COMPREPLY=( $( compgen -W 'effective' -- "$cur" ) )
1022                            ;;
1023                    esac
1024                    return 0
1025                    ;;
1026                attach|detach)
1027                    local BPFTOOL_CGROUP_ATTACH_TYPES="$(bpftool feature list_builtins attach_types 2>/dev/null | \
1028                        grep '^cgroup_')"
1029                    local ATTACH_FLAGS='multi override'
 
1030                    # Check for $prev = $command first
1031                    if [ $prev = $command ]; then
1032                        _filedir
1033                        return 0
1034                    # Then check for attach type. This is done outside of the
1035                    # "case $prev in" to avoid writing the whole list of attach
1036                    # types again as pattern to match (where we cannot reuse
1037                    # our variable).
1038                    elif [[ $BPFTOOL_CGROUP_ATTACH_TYPES =~ $prev ]]; then
1039                            COMPREPLY=( $( compgen -W "$PROG_TYPE" -- \
1040                                "$cur" ) )
1041                            return 0
1042                    fi
1043                    # case/esac for the other cases
1044                    case $prev in
1045                        id)
1046                            _bpftool_get_prog_ids
1047                            return 0
1048                            ;;
1049                        *)
1050                            if ! _bpftool_search_list "$BPFTOOL_CGROUP_ATTACH_TYPES"; then
1051                                COMPREPLY=( $( compgen -W \
1052                                    "$BPFTOOL_CGROUP_ATTACH_TYPES" -- "$cur" ) )
1053                            elif [[ "$command" == "attach" ]]; then
1054                                # We have an attach type on the command line,
1055                                # but it is not the previous word, or
1056                                # "id|pinned|tag|name" (we already checked for
1057                                # that). This should only leave the case when
1058                                # we need attach flags for "attach" commamnd.
1059                                _bpftool_one_of_list "$ATTACH_FLAGS"
1060                            fi
1061                            return 0
1062                            ;;
1063                    esac
1064                    ;;
1065                *)
1066                    [[ $prev == $object ]] && \
1067                        COMPREPLY=( $( compgen -W 'help attach detach \
1068                            show list tree' -- "$cur" ) )
1069                    ;;
1070            esac
1071            ;;
1072        perf)
1073            case $command in
1074                *)
1075                    [[ $prev == $object ]] && \
1076                        COMPREPLY=( $( compgen -W 'help \
1077                            show list' -- "$cur" ) )
1078                    ;;
1079            esac
1080            ;;
1081        net)
1082            local ATTACH_TYPES='xdp xdpgeneric xdpdrv xdpoffload tcx_ingress tcx_egress'
 
1083            case $command in
1084                show|list)
1085                    [[ $prev != "$command" ]] && return 0
1086                    COMPREPLY=( $( compgen -W 'dev' -- "$cur" ) )
1087                    return 0
1088                    ;;
1089                attach)
1090                    case $cword in
1091                        3)
1092                            COMPREPLY=( $( compgen -W "$ATTACH_TYPES" -- "$cur" ) )
1093                            return 0
1094                            ;;
1095                        4)
1096                            COMPREPLY=( $( compgen -W "$PROG_TYPE" -- "$cur" ) )
1097                            return 0
1098                            ;;
1099                        5)
1100                            case $prev in
1101                                id)
1102                                    _bpftool_get_prog_ids
1103                                    ;;
1104                                name)
1105                                    _bpftool_get_prog_names
1106                                    ;;
1107                                pinned)
1108                                    _filedir
1109                                    ;;
1110                            esac
1111                            return 0
1112                            ;;
1113                        6)
1114                            COMPREPLY=( $( compgen -W 'dev' -- "$cur" ) )
1115                            return 0
1116                            ;;
1117                        8)
1118                            _bpftool_once_attr 'overwrite'
1119                            return 0
1120                            ;;
1121                    esac
1122                    ;;
1123                detach)
1124                    case $cword in
1125                        3)
1126                            COMPREPLY=( $( compgen -W "$ATTACH_TYPES" -- "$cur" ) )
1127                            return 0
1128                            ;;
1129                        4)
1130                            COMPREPLY=( $( compgen -W 'dev' -- "$cur" ) )
1131                            return 0
1132                            ;;
1133                    esac
1134                    ;;
1135                *)
1136                    [[ $prev == $object ]] && \
1137                        COMPREPLY=( $( compgen -W 'help \
1138                            show list attach detach' -- "$cur" ) )
1139                    ;;
1140            esac
1141            ;;
1142        feature)
1143            case $command in
1144                probe)
1145                    [[ $prev == "prefix" ]] && return 0
1146                    if _bpftool_search_list 'macros'; then
1147                        _bpftool_once_attr 'prefix'
1148                    else
1149                        COMPREPLY+=( $( compgen -W 'macros' -- "$cur" ) )
1150                    fi
1151                    _bpftool_one_of_list 'kernel dev'
1152                    _bpftool_once_attr 'full unprivileged'
1153                    return 0
1154                    ;;
1155                list_builtins)
1156                    [[ $prev != "$command" ]] && return 0
1157                    COMPREPLY=( $( compgen -W 'prog_types map_types \
1158                        attach_types link_types helpers' -- "$cur" ) )
1159                    ;;
1160                *)
1161                    [[ $prev == $object ]] && \
1162                        COMPREPLY=( $( compgen -W 'help list_builtins probe' -- "$cur" ) )
1163                    ;;
1164            esac
1165            ;;
1166        link)
1167            case $command in
1168                show|list|pin|detach)
1169                    case $prev in
1170                        id)
1171                            _bpftool_get_link_ids
1172                            return 0
1173                            ;;
1174                    esac
1175                    ;;
1176            esac
1177
1178            local LINK_TYPE='id pinned'
1179            case $command in
1180                show|list)
1181                    [[ $prev != "$command" ]] && return 0
1182                    COMPREPLY=( $( compgen -W "$LINK_TYPE" -- "$cur" ) )
1183                    return 0
1184                    ;;
1185                pin|detach)
1186                    if [[ $prev == "$command" ]]; then
1187                        COMPREPLY=( $( compgen -W "$LINK_TYPE" -- "$cur" ) )
1188                    elif [[ $pprev == "$command" ]]; then
1189                        _filedir
1190                    fi
1191                    return 0
1192                    ;;
1193                *)
1194                    [[ $prev == $object ]] && \
1195                        COMPREPLY=( $( compgen -W 'help pin detach show list' -- "$cur" ) )
1196                    ;;
1197            esac
1198            ;;
1199    esac
1200} &&
1201complete -F _bpftool bpftool
1202
1203# ex: ts=4 sw=4 et filetype=sh
v6.8
   1# bpftool(8) bash completion                               -*- shell-script -*-
   2#
   3# SPDX-License-Identifier: (GPL-2.0-only OR BSD-2-Clause)
   4# Copyright (C) 2017-2018 Netronome Systems, Inc.
   5#
   6# Author: Quentin Monnet <quentin.monnet@netronome.com>
   7
   8# Takes a list of words in argument; each one of them is added to COMPREPLY if
   9# it is not already present on the command line. Returns no value.
  10_bpftool_once_attr()
  11{
  12    local w idx found
  13    for w in $*; do
  14        found=0
  15        for (( idx=3; idx < ${#words[@]}-1; idx++ )); do
  16            if [[ $w == ${words[idx]} ]]; then
  17                found=1
  18                break
  19            fi
  20        done
  21        [[ $found -eq 0 ]] && \
  22            COMPREPLY+=( $( compgen -W "$w" -- "$cur" ) )
  23    done
  24}
  25
  26# Takes a list of words as argument; if any of those words is present on the
  27# command line, return 0. Otherwise, return 1.
  28_bpftool_search_list()
  29{
  30    local w idx
  31    for w in $*; do
  32        for (( idx=3; idx < ${#words[@]}-1; idx++ )); do
  33            [[ $w == ${words[idx]} ]] && return 0
  34        done
  35    done
  36    return 1
  37}
  38
  39# Takes a list of words in argument; adds them all to COMPREPLY if none of them
  40# is already present on the command line. Returns no value.
  41_bpftool_one_of_list()
  42{
  43    _bpftool_search_list $* && return 1
  44    COMPREPLY+=( $( compgen -W "$*" -- "$cur" ) )
  45}
  46
  47_bpftool_get_map_ids()
  48{
  49    COMPREPLY+=( $( compgen -W "$( bpftool -jp map  2>&1 | \
  50        command sed -n 's/.*"id": \(.*\),$/\1/p' )" -- "$cur" ) )
  51}
  52
  53# Takes map type and adds matching map ids to the list of suggestions.
  54_bpftool_get_map_ids_for_type()
  55{
  56    local type="$1"
  57    COMPREPLY+=( $( compgen -W "$( bpftool -jp map  2>&1 | \
  58        command grep -C2 "$type" | \
  59        command sed -n 's/.*"id": \(.*\),$/\1/p' )" -- "$cur" ) )
  60}
  61
  62_bpftool_get_map_names()
  63{
  64    COMPREPLY+=( $( compgen -W "$( bpftool -jp map  2>&1 | \
  65        command sed -n 's/.*"name": \(.*\),$/\1/p' )" -- "$cur" ) )
  66}
  67
  68# Takes map type and adds matching map names to the list of suggestions.
  69_bpftool_get_map_names_for_type()
  70{
  71    local type="$1"
  72    COMPREPLY+=( $( compgen -W "$( bpftool -jp map  2>&1 | \
  73        command grep -C2 "$type" | \
  74        command sed -n 's/.*"name": \(.*\),$/\1/p' )" -- "$cur" ) )
  75}
  76
  77_bpftool_get_prog_ids()
  78{
  79    COMPREPLY+=( $( compgen -W "$( bpftool -jp prog 2>&1 | \
  80        command sed -n 's/.*"id": \(.*\),$/\1/p' )" -- "$cur" ) )
  81}
  82
  83_bpftool_get_prog_tags()
  84{
  85    COMPREPLY+=( $( compgen -W "$( bpftool -jp prog 2>&1 | \
  86        command sed -n 's/.*"tag": "\(.*\)",$/\1/p' )" -- "$cur" ) )
  87}
  88
  89_bpftool_get_prog_names()
  90{
  91    COMPREPLY+=( $( compgen -W "$( bpftool -jp prog 2>&1 | \
  92        command sed -n 's/.*"name": "\(.*\)",$/\1/p' )" -- "$cur" ) )
  93}
  94
  95_bpftool_get_btf_ids()
  96{
  97    COMPREPLY+=( $( compgen -W "$( bpftool -jp btf 2>&1 | \
  98        command sed -n 's/.*"id": \(.*\),$/\1/p' )" -- "$cur" ) )
  99}
 100
 101_bpftool_get_link_ids()
 102{
 103    COMPREPLY+=( $( compgen -W "$( bpftool -jp link 2>&1 | \
 104        command sed -n 's/.*"id": \(.*\),$/\1/p' )" -- "$cur" ) )
 105}
 106
 107_bpftool_get_obj_map_names()
 108{
 109    local obj
 110
 111    obj=$1
 112
 113    maps=$(objdump -j maps -t $obj 2>/dev/null | \
 114        command awk '/g     . maps/ {print $NF}')
 115
 116    COMPREPLY+=( $( compgen -W "$maps" -- "$cur" ) )
 117}
 118
 119_bpftool_get_obj_map_idxs()
 120{
 121    local obj
 122
 123    obj=$1
 124
 125    nmaps=$(objdump -j maps -t $obj 2>/dev/null | grep -c 'g     . maps')
 126
 127    COMPREPLY+=( $( compgen -W "$(seq 0 $((nmaps - 1)))" -- "$cur" ) )
 128}
 129
 130_sysfs_get_netdevs()
 131{
 132    COMPREPLY+=( $( compgen -W "$( ls /sys/class/net 2>/dev/null )" -- \
 133        "$cur" ) )
 134}
 135
 136# Retrieve type of the map that we are operating on.
 137_bpftool_map_guess_map_type()
 138{
 139    local keyword ref
 140    for (( idx=3; idx < ${#words[@]}-1; idx++ )); do
 141        case "${words[$((idx-2))]}" in
 142            lookup|update)
 143                keyword=${words[$((idx-1))]}
 144                ref=${words[$((idx))]}
 145                ;;
 146            push)
 147                printf "stack"
 148                return 0
 149                ;;
 150            enqueue)
 151                printf "queue"
 152                return 0
 153                ;;
 154        esac
 155    done
 156    [[ -z $ref ]] && return 0
 157
 158    local type
 159    type=$(bpftool -jp map show $keyword $ref | \
 160        command sed -n 's/.*"type": "\(.*\)",$/\1/p')
 161    [[ -n $type ]] && printf $type
 162}
 163
 164_bpftool_map_update_get_id()
 165{
 166    local command="$1"
 167
 168    # Is it the map to update, or a map to insert into the map to update?
 169    # Search for "value" keyword.
 170    local idx value
 171    for (( idx=7; idx < ${#words[@]}-1; idx++ )); do
 172        if [[ ${words[idx]} == "value" ]]; then
 173            value=1
 174            break
 175        fi
 176    done
 177    if [[ $value -eq 0 ]]; then
 178        case "$command" in
 179            push)
 180                _bpftool_get_map_ids_for_type stack
 181                ;;
 182            enqueue)
 183                _bpftool_get_map_ids_for_type queue
 184                ;;
 185            *)
 186                _bpftool_get_map_ids
 187                ;;
 188        esac
 189        return 0
 190    fi
 191
 192    # Id to complete is for a value. It can be either prog id or map id. This
 193    # depends on the type of the map to update.
 194    local type=$(_bpftool_map_guess_map_type)
 195    case $type in
 196        array_of_maps|hash_of_maps)
 197            _bpftool_get_map_ids
 198            return 0
 199            ;;
 200        prog_array)
 201            _bpftool_get_prog_ids
 202            return 0
 203            ;;
 204        *)
 205            return 0
 206            ;;
 207    esac
 208}
 209
 210_bpftool_map_update_get_name()
 211{
 212    local command="$1"
 213
 214    # Is it the map to update, or a map to insert into the map to update?
 215    # Search for "value" keyword.
 216    local idx value
 217    for (( idx=7; idx < ${#words[@]}-1; idx++ )); do
 218        if [[ ${words[idx]} == "value" ]]; then
 219            value=1
 220            break
 221        fi
 222    done
 223    if [[ $value -eq 0 ]]; then
 224        case "$command" in
 225            push)
 226                _bpftool_get_map_names_for_type stack
 227                ;;
 228            enqueue)
 229                _bpftool_get_map_names_for_type queue
 230                ;;
 231            *)
 232                _bpftool_get_map_names
 233                ;;
 234        esac
 235        return 0
 236    fi
 237
 238    # Name to complete is for a value. It can be either prog name or map name. This
 239    # depends on the type of the map to update.
 240    local type=$(_bpftool_map_guess_map_type)
 241    case $type in
 242        array_of_maps|hash_of_maps)
 243            _bpftool_get_map_names
 244            return 0
 245            ;;
 246        prog_array)
 247            _bpftool_get_prog_names
 248            return 0
 249            ;;
 250        *)
 251            return 0
 252            ;;
 253    esac
 254}
 255
 256_bpftool()
 257{
 258    local cur prev words objword json=0
 259    _init_completion || return
 
 260
 261    # Deal with options
 262    if [[ ${words[cword]} == -* ]]; then
 263        local c='--version --json --pretty --bpffs --mapcompat --debug \
 264            --use-loader --base-btf'
 265        COMPREPLY=( $( compgen -W "$c" -- "$cur" ) )
 266        return 0
 267    fi
 268    if _bpftool_search_list -j --json -p --pretty; then
 269        json=1
 270    fi
 271
 272    # Deal with simplest keywords
 273    case $prev in
 274        help|hex)
 275            return 0
 276            ;;
 277        tag)
 278            _bpftool_get_prog_tags
 279            return 0
 280            ;;
 281        dev|offload_dev|xdpmeta_dev)
 282            _sysfs_get_netdevs
 283            return 0
 284            ;;
 285        file|pinned|-B|--base-btf)
 286            _filedir
 287            return 0
 288            ;;
 289        batch)
 290            COMPREPLY=( $( compgen -W 'file' -- "$cur" ) )
 291            return 0
 292            ;;
 293    esac
 294
 295    # Remove all options so completions don't have to deal with them.
 296    local i
 297    for (( i=1; i < ${#words[@]}; )); do
 298        if [[ ${words[i]::1} == - ]] &&
 299            [[ ${words[i]} != "-B" ]] && [[ ${words[i]} != "--base-btf" ]]; then
 300            words=( "${words[@]:0:i}" "${words[@]:i+1}" )
 301            [[ $i -le $cword ]] && cword=$(( cword - 1 ))
 302        else
 303            i=$(( ++i ))
 304        fi
 305    done
 306    cur=${words[cword]}
 307    prev=${words[cword - 1]}
 308    pprev=${words[cword - 2]}
 309
 310    local object=${words[1]} command=${words[2]}
 311
 312    if [[ -z $object || $cword -eq 1 ]]; then
 313        case $cur in
 314            *)
 315                COMPREPLY=( $( compgen -W "$( bpftool help 2>&1 | \
 316                    command sed \
 317                    -e '/OBJECT := /!d' \
 318                    -e 's/.*{//' \
 319                    -e 's/}.*//' \
 320                    -e 's/|//g' )" -- "$cur" ) )
 321                COMPREPLY+=( $( compgen -W 'batch help' -- "$cur" ) )
 322                return 0
 323                ;;
 324        esac
 325    fi
 326
 
 327    [[ $command == help ]] && return 0
 328
 
 
 
 329    # Completion depends on object and command in use
 330    case $object in
 331        prog)
 332            # Complete id and name, only for subcommands that use prog (but no
 333            # map) ids/names.
 334            case $command in
 335                show|list|dump|pin)
 336                    case $prev in
 337                        id)
 338                            _bpftool_get_prog_ids
 339                            return 0
 340                            ;;
 341                        name)
 342                            _bpftool_get_prog_names
 343                            return 0
 344                            ;;
 345                    esac
 346                    ;;
 347            esac
 348
 349            local PROG_TYPE='id pinned tag name'
 350            local MAP_TYPE='id pinned name'
 351            local METRIC_TYPE='cycles instructions l1d_loads llc_misses \
 352                itlb_misses dtlb_misses'
 353            case $command in
 354                show|list)
 355                    [[ $prev != "$command" ]] && return 0
 356                    COMPREPLY=( $( compgen -W "$PROG_TYPE" -- "$cur" ) )
 357                    return 0
 358                    ;;
 359                dump)
 360                    case $prev in
 361                        $command)
 362                            COMPREPLY+=( $( compgen -W "xlated jited" -- \
 363                                "$cur" ) )
 364                            return 0
 365                            ;;
 366                        xlated|jited)
 367                            COMPREPLY=( $( compgen -W "$PROG_TYPE" -- \
 368                                "$cur" ) )
 369                            return 0
 370                            ;;
 371                        *)
 372                            # "file" is not compatible with other keywords here
 373                            if _bpftool_search_list 'file'; then
 374                                return 0
 375                            fi
 376                            if ! _bpftool_search_list 'linum opcodes visual'; then
 377                                _bpftool_once_attr 'file'
 378                            fi
 379                            _bpftool_once_attr 'linum opcodes'
 380                            if _bpftool_search_list 'xlated' && [[ "$json" == 0 ]]; then
 381                                _bpftool_once_attr 'visual'
 382                            fi
 383                            return 0
 384                            ;;
 385                    esac
 386                    ;;
 387                pin)
 388                    if [[ $prev == "$command" ]]; then
 389                        COMPREPLY=( $( compgen -W "$PROG_TYPE" -- "$cur" ) )
 390                    else
 391                        _filedir
 392                    fi
 393                    return 0
 394                    ;;
 395                attach|detach)
 396                    case $cword in
 397                        3)
 398                            COMPREPLY=( $( compgen -W "$PROG_TYPE" -- "$cur" ) )
 399                            return 0
 400                            ;;
 401                        4)
 402                            case $prev in
 403                                id)
 404                                    _bpftool_get_prog_ids
 405                                    ;;
 406                                name)
 407                                    _bpftool_get_prog_names
 408                                    ;;
 409                                pinned)
 410                                    _filedir
 411                                    ;;
 412                            esac
 413                            return 0
 414                            ;;
 415                        5)
 416                            local BPFTOOL_PROG_ATTACH_TYPES='sk_msg_verdict \
 417                                sk_skb_verdict sk_skb_stream_verdict sk_skb_stream_parser \
 418                                flow_dissector'
 419                            COMPREPLY=( $( compgen -W "$BPFTOOL_PROG_ATTACH_TYPES" -- "$cur" ) )
 420                            return 0
 421                            ;;
 422                        6)
 423                            COMPREPLY=( $( compgen -W "$MAP_TYPE" -- "$cur" ) )
 424                            return 0
 425                            ;;
 426                        7)
 427                            case $prev in
 428                                id)
 429                                    _bpftool_get_map_ids
 430                                    ;;
 431                                name)
 432                                    _bpftool_get_map_names
 433                                    ;;
 434                                pinned)
 435                                    _filedir
 436                                    ;;
 437                            esac
 438                            return 0
 439                            ;;
 440                    esac
 441                    ;;
 442                load|loadall)
 443                    local obj
 444
 445                    # Propose "load/loadall" to complete "bpftool prog load",
 446                    # or bash tries to complete "load" as a filename below.
 447                    if [[ ${#words[@]} -eq 3 ]]; then
 448                        COMPREPLY=( $( compgen -W "load loadall" -- "$cur" ) )
 449                        return 0
 450                    fi
 451
 452                    if [[ ${#words[@]} -lt 6 ]]; then
 453                        _filedir
 454                        return 0
 455                    fi
 456
 457                    obj=${words[3]}
 458
 459                    if [[ ${words[-4]} == "map" ]]; then
 460                        COMPREPLY=( $( compgen -W "id pinned" -- "$cur" ) )
 461                        return 0
 462                    fi
 463                    if [[ ${words[-3]} == "map" ]]; then
 464                        if [[ ${words[-2]} == "idx" ]]; then
 465                            _bpftool_get_obj_map_idxs $obj
 466                        elif [[ ${words[-2]} == "name" ]]; then
 467                            _bpftool_get_obj_map_names $obj
 468                        fi
 469                        return 0
 470                    fi
 471                    if [[ ${words[-2]} == "map" ]]; then
 472                        COMPREPLY=( $( compgen -W "idx name" -- "$cur" ) )
 473                        return 0
 474                    fi
 475
 476                    case $prev in
 477                        type)
 478                            local BPFTOOL_PROG_LOAD_TYPES='socket kprobe \
 479                                kretprobe classifier flow_dissector \
 480                                action tracepoint raw_tracepoint \
 481                                xdp perf_event cgroup/skb cgroup/sock \
 482                                cgroup/dev lwt_in lwt_out lwt_xmit \
 483                                lwt_seg6local sockops sk_skb sk_msg lirc_mode2 \
 484                                cgroup/bind4 cgroup/bind6 \
 485                                cgroup/connect4 cgroup/connect6 cgroup/connect_unix \
 486                                cgroup/getpeername4 cgroup/getpeername6 cgroup/getpeername_unix \
 487                                cgroup/getsockname4 cgroup/getsockname6 cgroup/getsockname_unix \
 488                                cgroup/sendmsg4 cgroup/sendmsg6 cgroup/sendmsg_unix \
 489                                cgroup/recvmsg4 cgroup/recvmsg6 cgroup/recvmsg_unix \
 490                                cgroup/post_bind4 cgroup/post_bind6 \
 491                                cgroup/sysctl cgroup/getsockopt \
 492                                cgroup/setsockopt cgroup/sock_release struct_ops \
 493                                fentry fexit freplace sk_lookup'
 494                            COMPREPLY=( $( compgen -W "$BPFTOOL_PROG_LOAD_TYPES" -- "$cur" ) )
 495                            return 0
 496                            ;;
 497                        id)
 498                            _bpftool_get_map_ids
 499                            return 0
 500                            ;;
 501                        name)
 502                            _bpftool_get_map_names
 503                            return 0
 504                            ;;
 505                        pinned|pinmaps)
 506                            _filedir
 507                            return 0
 508                            ;;
 509                        *)
 510                            COMPREPLY=( $( compgen -W "map" -- "$cur" ) )
 511                            _bpftool_once_attr 'type pinmaps autoattach'
 512                            _bpftool_one_of_list 'offload_dev xdpmeta_dev'
 513                            return 0
 514                            ;;
 515                    esac
 516                    ;;
 517                tracelog)
 518                    return 0
 519                    ;;
 520                profile)
 521                    case $cword in
 522                        3)
 523                            COMPREPLY=( $( compgen -W "$PROG_TYPE" -- "$cur" ) )
 524                            return 0
 525                            ;;
 526                        4)
 527                            case $prev in
 528                                id)
 529                                    _bpftool_get_prog_ids
 530                                    ;;
 531                                name)
 532                                    _bpftool_get_prog_names
 533                                    ;;
 534                                pinned)
 535                                    _filedir
 536                                    ;;
 537                            esac
 538                            return 0
 539                            ;;
 540                        5)
 541                            COMPREPLY=( $( compgen -W "$METRIC_TYPE duration" -- "$cur" ) )
 542                            return 0
 543                            ;;
 544                        6)
 545                            case $prev in
 546                                duration)
 547                                    return 0
 548                                    ;;
 549                                *)
 550                                    COMPREPLY=( $( compgen -W "$METRIC_TYPE" -- "$cur" ) )
 551                                    return 0
 552                                    ;;
 553                            esac
 554                            return 0
 555                            ;;
 556                        *)
 557                            COMPREPLY=( $( compgen -W "$METRIC_TYPE" -- "$cur" ) )
 
 558                            return 0
 559                            ;;
 560                    esac
 561                    ;;
 562                run)
 563                    if [[ ${#words[@]} -eq 4 ]]; then
 564                        COMPREPLY=( $( compgen -W "$PROG_TYPE" -- "$cur" ) )
 565                        return 0
 566                    fi
 567                    case $prev in
 568                        id)
 569                            _bpftool_get_prog_ids
 570                            return 0
 571                            ;;
 572                        name)
 573                            _bpftool_get_prog_names
 574                            return 0
 575                            ;;
 576                        data_in|data_out|ctx_in|ctx_out)
 577                            _filedir
 578                            return 0
 579                            ;;
 580                        repeat|data_size_out|ctx_size_out)
 581                            return 0
 582                            ;;
 583                        *)
 584                            _bpftool_once_attr 'data_in data_out data_size_out \
 585                                ctx_in ctx_out ctx_size_out repeat'
 586                            return 0
 587                            ;;
 588                    esac
 589                    ;;
 590                *)
 591                    [[ $prev == $object ]] && \
 592                        COMPREPLY=( $( compgen -W 'dump help pin attach detach \
 593                            load loadall show list tracelog run profile' -- "$cur" ) )
 594                    ;;
 595            esac
 596            ;;
 597        struct_ops)
 598            local STRUCT_OPS_TYPE='id name'
 599            case $command in
 600                show|list|dump|unregister)
 601                    case $prev in
 602                        $command)
 603                            COMPREPLY=( $( compgen -W "$STRUCT_OPS_TYPE" -- "$cur" ) )
 604                            ;;
 605                        id)
 606                            _bpftool_get_map_ids_for_type struct_ops
 607                            ;;
 608                        name)
 609                            _bpftool_get_map_names_for_type struct_ops
 610                            ;;
 611                    esac
 612                    return 0
 613                    ;;
 614                register)
 615                    _filedir
 616                    return 0
 617                    ;;
 618                *)
 619                    [[ $prev == $object ]] && \
 620                        COMPREPLY=( $( compgen -W 'register unregister show list dump help' \
 621                            -- "$cur" ) )
 622                    ;;
 623            esac
 624            ;;
 625        iter)
 626            case $command in
 627                pin)
 628                    case $prev in
 629                        $command)
 630                            _filedir
 631                            ;;
 632                        id)
 633                            _bpftool_get_map_ids
 634                            ;;
 635                        name)
 636                            _bpftool_get_map_names
 637                            ;;
 638                        pinned)
 639                            _filedir
 640                            ;;
 
 
 
 641                        *)
 642                            _bpftool_one_of_list $MAP_TYPE
 643                            ;;
 644                    esac
 645                    return 0
 646                    ;;
 647                *)
 648                    [[ $prev == $object ]] && \
 649                        COMPREPLY=( $( compgen -W 'pin help' \
 650                            -- "$cur" ) )
 651                    ;;
 652            esac
 653            ;;
 654        map)
 655            local MAP_TYPE='id pinned name'
 656            case $command in
 657                show|list|dump|peek|pop|dequeue|freeze)
 658                    case $prev in
 659                        $command)
 660                            COMPREPLY=( $( compgen -W "$MAP_TYPE" -- "$cur" ) )
 661                            return 0
 662                            ;;
 663                        id)
 664                            case "$command" in
 665                                peek)
 666                                    _bpftool_get_map_ids_for_type stack
 667                                    _bpftool_get_map_ids_for_type queue
 668                                    ;;
 669                                pop)
 670                                    _bpftool_get_map_ids_for_type stack
 671                                    ;;
 672                                dequeue)
 673                                    _bpftool_get_map_ids_for_type queue
 674                                    ;;
 675                                *)
 676                                    _bpftool_get_map_ids
 677                                    ;;
 678                            esac
 679                            return 0
 680                            ;;
 681                        name)
 682                            case "$command" in
 683                                peek)
 684                                    _bpftool_get_map_names_for_type stack
 685                                    _bpftool_get_map_names_for_type queue
 686                                    ;;
 687                                pop)
 688                                    _bpftool_get_map_names_for_type stack
 689                                    ;;
 690                                dequeue)
 691                                    _bpftool_get_map_names_for_type queue
 692                                    ;;
 693                                *)
 694                                    _bpftool_get_map_names
 695                                    ;;
 696                            esac
 697                            return 0
 698                            ;;
 699                        *)
 700                            return 0
 701                            ;;
 702                    esac
 703                    ;;
 704                create)
 705                    case $prev in
 706                        $command)
 707                            _filedir
 708                            return 0
 709                            ;;
 710                        type)
 711                            local BPFTOOL_MAP_CREATE_TYPES="$(bpftool feature list_builtins map_types 2>/dev/null | \
 712                                grep -v '^unspec$')"
 713                            COMPREPLY=( $( compgen -W "$BPFTOOL_MAP_CREATE_TYPES" -- "$cur" ) )
 714                            return 0
 715                            ;;
 716                        key|value|flags|entries)
 717                            return 0
 718                            ;;
 719                        inner_map)
 720                            COMPREPLY=( $( compgen -W "$MAP_TYPE" -- "$cur" ) )
 721                            return 0
 722                            ;;
 723                        id)
 724                            _bpftool_get_map_ids
 725                            ;;
 726                        name)
 727                            case $pprev in
 728                                inner_map)
 729                                    _bpftool_get_map_names
 730                                    ;;
 731                                *)
 732                                    return 0
 733                                    ;;
 734                            esac
 735                            ;;
 736                        *)
 737                            _bpftool_once_attr 'type key value entries name flags offload_dev'
 738                            if _bpftool_search_list 'array_of_maps' 'hash_of_maps'; then
 739                                _bpftool_once_attr 'inner_map'
 740                            fi
 741                            return 0
 742                            ;;
 743                    esac
 744                    ;;
 745                lookup|getnext|delete)
 746                    case $prev in
 747                        $command)
 748                            COMPREPLY=( $( compgen -W "$MAP_TYPE" -- "$cur" ) )
 749                            return 0
 750                            ;;
 751                        id)
 752                            _bpftool_get_map_ids
 753                            return 0
 754                            ;;
 755                        name)
 756                            _bpftool_get_map_names
 757                            return 0
 758                            ;;
 759                        key)
 760                            COMPREPLY+=( $( compgen -W 'hex' -- "$cur" ) )
 761                            ;;
 762                        *)
 763                            case $(_bpftool_map_guess_map_type) in
 764                                queue|stack)
 765                                    return 0
 766                                    ;;
 767                            esac
 768
 769                            _bpftool_once_attr 'key'
 770                            return 0
 771                            ;;
 772                    esac
 773                    ;;
 774                update|push|enqueue)
 775                    case $prev in
 776                        $command)
 777                            COMPREPLY=( $( compgen -W "$MAP_TYPE" -- "$cur" ) )
 778                            return 0
 779                            ;;
 780                        id)
 781                            _bpftool_map_update_get_id $command
 782                            return 0
 783                            ;;
 784                        name)
 785                            _bpftool_map_update_get_name $command
 786                            return 0
 787                            ;;
 788                        key)
 789                            COMPREPLY+=( $( compgen -W 'hex' -- "$cur" ) )
 790                            ;;
 791                        value)
 792                            # We can have bytes, or references to a prog or a
 793                            # map, depending on the type of the map to update.
 794                            case "$(_bpftool_map_guess_map_type)" in
 795                                array_of_maps|hash_of_maps)
 796                                    local MAP_TYPE='id pinned name'
 797                                    COMPREPLY+=( $( compgen -W "$MAP_TYPE" \
 798                                        -- "$cur" ) )
 799                                    return 0
 800                                    ;;
 801                                prog_array)
 802                                    local PROG_TYPE='id pinned tag name'
 803                                    COMPREPLY+=( $( compgen -W "$PROG_TYPE" \
 804                                        -- "$cur" ) )
 805                                    return 0
 806                                    ;;
 807                                *)
 808                                    COMPREPLY+=( $( compgen -W 'hex' \
 809                                        -- "$cur" ) )
 810                                    return 0
 811                                    ;;
 812                            esac
 813                            return 0
 814                            ;;
 815                        *)
 816                            case $(_bpftool_map_guess_map_type) in
 817                                queue|stack)
 818                                    _bpftool_once_attr 'value'
 819                                    return 0;
 820                                    ;;
 821                            esac
 822
 823                            _bpftool_once_attr 'key'
 824                            local UPDATE_FLAGS='any exist noexist'
 825                            for (( idx=3; idx < ${#words[@]}-1; idx++ )); do
 826                                if [[ ${words[idx]} == 'value' ]]; then
 827                                    # 'value' is present, but is not the last
 828                                    # word i.e. we can now have UPDATE_FLAGS.
 829                                    _bpftool_one_of_list "$UPDATE_FLAGS"
 830                                    return 0
 831                                fi
 832                            done
 833                            for (( idx=3; idx < ${#words[@]}-1; idx++ )); do
 834                                if [[ ${words[idx]} == 'key' ]]; then
 835                                    # 'key' is present, but is not the last
 836                                    # word i.e. we can now have 'value'.
 837                                    _bpftool_once_attr 'value'
 838                                    return 0
 839                                fi
 840                            done
 841
 842                            return 0
 843                            ;;
 844                    esac
 845                    ;;
 846                pin)
 847                    case $prev in
 848                        $command)
 849                            COMPREPLY=( $( compgen -W "$MAP_TYPE" -- "$cur" ) )
 850                            ;;
 851                        id)
 852                            _bpftool_get_map_ids
 853                            ;;
 854                        name)
 855                            _bpftool_get_map_names
 856                            ;;
 857                    esac
 858                    return 0
 859                    ;;
 860                event_pipe)
 861                    case $prev in
 862                        $command)
 863                            COMPREPLY=( $( compgen -W "$MAP_TYPE" -- "$cur" ) )
 864                            return 0
 865                            ;;
 866                        id)
 867                            _bpftool_get_map_ids_for_type perf_event_array
 868                            return 0
 869                            ;;
 870                        name)
 871                            _bpftool_get_map_names_for_type perf_event_array
 872                            return 0
 873                            ;;
 874                        cpu)
 875                            return 0
 876                            ;;
 877                        index)
 878                            return 0
 879                            ;;
 880                        *)
 881                            _bpftool_once_attr 'cpu index'
 882                            return 0
 883                            ;;
 884                    esac
 885                    ;;
 886                *)
 887                    [[ $prev == $object ]] && \
 888                        COMPREPLY=( $( compgen -W 'delete dump getnext help \
 889                            lookup pin event_pipe show list update create \
 890                            peek push enqueue pop dequeue freeze' -- \
 891                            "$cur" ) )
 892                    ;;
 893            esac
 894            ;;
 895        btf)
 896            local PROG_TYPE='id pinned tag name'
 897            local MAP_TYPE='id pinned name'
 898            case $command in
 899                dump)
 900                    case $prev in
 901                        $command)
 902                            COMPREPLY+=( $( compgen -W "id map prog file" -- \
 903                                "$cur" ) )
 904                            return 0
 905                            ;;
 906                        prog)
 907                            COMPREPLY=( $( compgen -W "$PROG_TYPE" -- "$cur" ) )
 908                            return 0
 909                            ;;
 910                        map)
 911                            COMPREPLY=( $( compgen -W "$MAP_TYPE" -- "$cur" ) )
 912                            return 0
 913                            ;;
 914                        id)
 915                            case $pprev in
 916                                prog)
 917                                    _bpftool_get_prog_ids
 918                                    ;;
 919                                map)
 920                                    _bpftool_get_map_ids
 921                                    ;;
 922                                $command)
 923                                    _bpftool_get_btf_ids
 924                                    ;;
 925                            esac
 926                            return 0
 927                            ;;
 928                        name)
 929                            case $pprev in
 930                                prog)
 931                                    _bpftool_get_prog_names
 932                                    ;;
 933                                map)
 934                                    _bpftool_get_map_names
 935                                    ;;
 936                            esac
 937                            return 0
 938                            ;;
 939                        format)
 940                            COMPREPLY=( $( compgen -W "c raw" -- "$cur" ) )
 941                            ;;
 
 
 
 942                        *)
 943                            # emit extra options
 944                            case ${words[3]} in
 945                                id|file)
 946                                    _bpftool_once_attr 'format'
 947                                    ;;
 948                                map|prog)
 949                                    if [[ ${words[3]} == "map" ]] && [[ $cword == 6 ]]; then
 950                                        COMPREPLY+=( $( compgen -W "key value kv all" -- "$cur" ) )
 951                                    fi
 952                                    _bpftool_once_attr 'format'
 953                                    ;;
 954                                *)
 955                                    ;;
 956                            esac
 957                            return 0
 958                            ;;
 959                    esac
 960                    ;;
 961                show|list)
 962                    case $prev in
 963                        $command)
 964                            COMPREPLY+=( $( compgen -W "id" -- "$cur" ) )
 965                            ;;
 966                        id)
 967                            _bpftool_get_btf_ids
 968                            ;;
 969                    esac
 970                    return 0
 971                    ;;
 972                *)
 973                    [[ $prev == $object ]] && \
 974                        COMPREPLY=( $( compgen -W 'dump help show list' \
 975                            -- "$cur" ) )
 976                    ;;
 977            esac
 978            ;;
 979        gen)
 980            case $command in
 981                object)
 982                    _filedir
 983                    return 0
 984                    ;;
 985                skeleton)
 986                    case $prev in
 987                        $command)
 988                            _filedir
 989                            return 0
 990                            ;;
 991                        *)
 992                            _bpftool_once_attr 'name'
 993                            return 0
 994                            ;;
 995                    esac
 996                    ;;
 997                subskeleton)
 998                    case $prev in
 999                        $command)
1000                            _filedir
1001                            return 0
1002                            ;;
1003                        *)
1004                            _bpftool_once_attr 'name'
1005                            return 0
1006                            ;;
1007                    esac
1008                    ;;
1009                min_core_btf)
1010                    _filedir
1011                    return 0
1012                    ;;
1013                *)
1014                    [[ $prev == $object ]] && \
1015                        COMPREPLY=( $( compgen -W 'object skeleton subskeleton help min_core_btf' -- "$cur" ) )
1016                    ;;
1017            esac
1018            ;;
1019        cgroup)
1020            case $command in
1021                show|list|tree)
1022                    case $cword in
1023                        3)
1024                            _filedir
1025                            ;;
1026                        4)
1027                            COMPREPLY=( $( compgen -W 'effective' -- "$cur" ) )
1028                            ;;
1029                    esac
1030                    return 0
1031                    ;;
1032                attach|detach)
1033                    local BPFTOOL_CGROUP_ATTACH_TYPES="$(bpftool feature list_builtins attach_types 2>/dev/null | \
1034                        grep '^cgroup_')"
1035                    local ATTACH_FLAGS='multi override'
1036                    local PROG_TYPE='id pinned tag name'
1037                    # Check for $prev = $command first
1038                    if [ $prev = $command ]; then
1039                        _filedir
1040                        return 0
1041                    # Then check for attach type. This is done outside of the
1042                    # "case $prev in" to avoid writing the whole list of attach
1043                    # types again as pattern to match (where we cannot reuse
1044                    # our variable).
1045                    elif [[ $BPFTOOL_CGROUP_ATTACH_TYPES =~ $prev ]]; then
1046                            COMPREPLY=( $( compgen -W "$PROG_TYPE" -- \
1047                                "$cur" ) )
1048                            return 0
1049                    fi
1050                    # case/esac for the other cases
1051                    case $prev in
1052                        id)
1053                            _bpftool_get_prog_ids
1054                            return 0
1055                            ;;
1056                        *)
1057                            if ! _bpftool_search_list "$BPFTOOL_CGROUP_ATTACH_TYPES"; then
1058                                COMPREPLY=( $( compgen -W \
1059                                    "$BPFTOOL_CGROUP_ATTACH_TYPES" -- "$cur" ) )
1060                            elif [[ "$command" == "attach" ]]; then
1061                                # We have an attach type on the command line,
1062                                # but it is not the previous word, or
1063                                # "id|pinned|tag|name" (we already checked for
1064                                # that). This should only leave the case when
1065                                # we need attach flags for "attach" commamnd.
1066                                _bpftool_one_of_list "$ATTACH_FLAGS"
1067                            fi
1068                            return 0
1069                            ;;
1070                    esac
1071                    ;;
1072                *)
1073                    [[ $prev == $object ]] && \
1074                        COMPREPLY=( $( compgen -W 'help attach detach \
1075                            show list tree' -- "$cur" ) )
1076                    ;;
1077            esac
1078            ;;
1079        perf)
1080            case $command in
1081                *)
1082                    [[ $prev == $object ]] && \
1083                        COMPREPLY=( $( compgen -W 'help \
1084                            show list' -- "$cur" ) )
1085                    ;;
1086            esac
1087            ;;
1088        net)
1089            local PROG_TYPE='id pinned tag name'
1090            local ATTACH_TYPES='xdp xdpgeneric xdpdrv xdpoffload'
1091            case $command in
1092                show|list)
1093                    [[ $prev != "$command" ]] && return 0
1094                    COMPREPLY=( $( compgen -W 'dev' -- "$cur" ) )
1095                    return 0
1096                    ;;
1097                attach)
1098                    case $cword in
1099                        3)
1100                            COMPREPLY=( $( compgen -W "$ATTACH_TYPES" -- "$cur" ) )
1101                            return 0
1102                            ;;
1103                        4)
1104                            COMPREPLY=( $( compgen -W "$PROG_TYPE" -- "$cur" ) )
1105                            return 0
1106                            ;;
1107                        5)
1108                            case $prev in
1109                                id)
1110                                    _bpftool_get_prog_ids
1111                                    ;;
1112                                name)
1113                                    _bpftool_get_prog_names
1114                                    ;;
1115                                pinned)
1116                                    _filedir
1117                                    ;;
1118                            esac
1119                            return 0
1120                            ;;
1121                        6)
1122                            COMPREPLY=( $( compgen -W 'dev' -- "$cur" ) )
1123                            return 0
1124                            ;;
1125                        8)
1126                            _bpftool_once_attr 'overwrite'
1127                            return 0
1128                            ;;
1129                    esac
1130                    ;;
1131                detach)
1132                    case $cword in
1133                        3)
1134                            COMPREPLY=( $( compgen -W "$ATTACH_TYPES" -- "$cur" ) )
1135                            return 0
1136                            ;;
1137                        4)
1138                            COMPREPLY=( $( compgen -W 'dev' -- "$cur" ) )
1139                            return 0
1140                            ;;
1141                    esac
1142                    ;;
1143                *)
1144                    [[ $prev == $object ]] && \
1145                        COMPREPLY=( $( compgen -W 'help \
1146                            show list attach detach' -- "$cur" ) )
1147                    ;;
1148            esac
1149            ;;
1150        feature)
1151            case $command in
1152                probe)
1153                    [[ $prev == "prefix" ]] && return 0
1154                    if _bpftool_search_list 'macros'; then
1155                        _bpftool_once_attr 'prefix'
1156                    else
1157                        COMPREPLY+=( $( compgen -W 'macros' -- "$cur" ) )
1158                    fi
1159                    _bpftool_one_of_list 'kernel dev'
1160                    _bpftool_once_attr 'full unprivileged'
1161                    return 0
1162                    ;;
1163                list_builtins)
1164                    [[ $prev != "$command" ]] && return 0
1165                    COMPREPLY=( $( compgen -W 'prog_types map_types \
1166                        attach_types link_types helpers' -- "$cur" ) )
1167                    ;;
1168                *)
1169                    [[ $prev == $object ]] && \
1170                        COMPREPLY=( $( compgen -W 'help list_builtins probe' -- "$cur" ) )
1171                    ;;
1172            esac
1173            ;;
1174        link)
1175            case $command in
1176                show|list|pin|detach)
1177                    case $prev in
1178                        id)
1179                            _bpftool_get_link_ids
1180                            return 0
1181                            ;;
1182                    esac
1183                    ;;
1184            esac
1185
1186            local LINK_TYPE='id pinned'
1187            case $command in
1188                show|list)
1189                    [[ $prev != "$command" ]] && return 0
1190                    COMPREPLY=( $( compgen -W "$LINK_TYPE" -- "$cur" ) )
1191                    return 0
1192                    ;;
1193                pin|detach)
1194                    if [[ $prev == "$command" ]]; then
1195                        COMPREPLY=( $( compgen -W "$LINK_TYPE" -- "$cur" ) )
1196                    else
1197                        _filedir
1198                    fi
1199                    return 0
1200                    ;;
1201                *)
1202                    [[ $prev == $object ]] && \
1203                        COMPREPLY=( $( compgen -W 'help pin show list' -- "$cur" ) )
1204                    ;;
1205            esac
1206            ;;
1207    esac
1208} &&
1209complete -F _bpftool bpftool
1210
1211# ex: ts=4 sw=4 et filetype=sh