Linux kernel mirror (for testing)
git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
kernel
os
linux
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
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
269 # Deal with simplest keywords
270 case $prev in
271 help|hex|opcodes|visual|linum)
272 return 0
273 ;;
274 tag)
275 _bpftool_get_prog_tags
276 return 0
277 ;;
278 dev)
279 _sysfs_get_netdevs
280 return 0
281 ;;
282 file|pinned|-B|--base-btf)
283 _filedir
284 return 0
285 ;;
286 batch)
287 COMPREPLY=( $( compgen -W 'file' -- "$cur" ) )
288 return 0
289 ;;
290 esac
291
292 # Remove all options so completions don't have to deal with them.
293 local i
294 for (( i=1; i < ${#words[@]}; )); do
295 if [[ ${words[i]::1} == - ]] &&
296 [[ ${words[i]} != "-B" ]] && [[ ${words[i]} != "--base-btf" ]]; then
297 words=( "${words[@]:0:i}" "${words[@]:i+1}" )
298 [[ $i -le $cword ]] && cword=$(( cword - 1 ))
299 else
300 i=$(( ++i ))
301 fi
302 done
303 cur=${words[cword]}
304 prev=${words[cword - 1]}
305 pprev=${words[cword - 2]}
306
307 local object=${words[1]} command=${words[2]}
308
309 if [[ -z $object || $cword -eq 1 ]]; then
310 case $cur in
311 *)
312 COMPREPLY=( $( compgen -W "$( bpftool help 2>&1 | \
313 command sed \
314 -e '/OBJECT := /!d' \
315 -e 's/.*{//' \
316 -e 's/}.*//' \
317 -e 's/|//g' )" -- "$cur" ) )
318 COMPREPLY+=( $( compgen -W 'batch help' -- "$cur" ) )
319 return 0
320 ;;
321 esac
322 fi
323
324 [[ $command == help ]] && return 0
325
326 # Completion depends on object and command in use
327 case $object in
328 prog)
329 # Complete id and name, only for subcommands that use prog (but no
330 # map) ids/names.
331 case $command in
332 show|list|dump|pin)
333 case $prev in
334 id)
335 _bpftool_get_prog_ids
336 return 0
337 ;;
338 name)
339 _bpftool_get_prog_names
340 return 0
341 ;;
342 esac
343 ;;
344 esac
345
346 local PROG_TYPE='id pinned tag name'
347 local MAP_TYPE='id pinned name'
348 local METRIC_TYPE='cycles instructions l1d_loads llc_misses \
349 itlb_misses dtlb_misses'
350 case $command in
351 show|list)
352 [[ $prev != "$command" ]] && return 0
353 COMPREPLY=( $( compgen -W "$PROG_TYPE" -- "$cur" ) )
354 return 0
355 ;;
356 dump)
357 case $prev in
358 $command)
359 COMPREPLY+=( $( compgen -W "xlated jited" -- \
360 "$cur" ) )
361 return 0
362 ;;
363 xlated|jited)
364 COMPREPLY=( $( compgen -W "$PROG_TYPE" -- \
365 "$cur" ) )
366 return 0
367 ;;
368 *)
369 _bpftool_once_attr 'file'
370 if _bpftool_search_list 'xlated'; then
371 COMPREPLY+=( $( compgen -W 'opcodes visual linum' -- \
372 "$cur" ) )
373 else
374 COMPREPLY+=( $( compgen -W 'opcodes linum' -- \
375 "$cur" ) )
376 fi
377 return 0
378 ;;
379 esac
380 ;;
381 pin)
382 if [[ $prev == "$command" ]]; then
383 COMPREPLY=( $( compgen -W "$PROG_TYPE" -- "$cur" ) )
384 else
385 _filedir
386 fi
387 return 0
388 ;;
389 attach|detach)
390 case $cword in
391 3)
392 COMPREPLY=( $( compgen -W "$PROG_TYPE" -- "$cur" ) )
393 return 0
394 ;;
395 4)
396 case $prev in
397 id)
398 _bpftool_get_prog_ids
399 ;;
400 name)
401 _bpftool_get_prog_names
402 ;;
403 pinned)
404 _filedir
405 ;;
406 esac
407 return 0
408 ;;
409 5)
410 local BPFTOOL_PROG_ATTACH_TYPES='msg_verdict \
411 skb_verdict stream_verdict stream_parser \
412 flow_dissector'
413 COMPREPLY=( $( compgen -W "$BPFTOOL_PROG_ATTACH_TYPES" -- "$cur" ) )
414 return 0
415 ;;
416 6)
417 COMPREPLY=( $( compgen -W "$MAP_TYPE" -- "$cur" ) )
418 return 0
419 ;;
420 7)
421 case $prev in
422 id)
423 _bpftool_get_map_ids
424 ;;
425 name)
426 _bpftool_get_map_names
427 ;;
428 pinned)
429 _filedir
430 ;;
431 esac
432 return 0
433 ;;
434 esac
435 ;;
436 load|loadall)
437 local obj
438
439 # Propose "load/loadall" to complete "bpftool prog load",
440 # or bash tries to complete "load" as a filename below.
441 if [[ ${#words[@]} -eq 3 ]]; then
442 COMPREPLY=( $( compgen -W "load loadall" -- "$cur" ) )
443 return 0
444 fi
445
446 if [[ ${#words[@]} -lt 6 ]]; then
447 _filedir
448 return 0
449 fi
450
451 obj=${words[3]}
452
453 if [[ ${words[-4]} == "map" ]]; then
454 COMPREPLY=( $( compgen -W "id pinned" -- "$cur" ) )
455 return 0
456 fi
457 if [[ ${words[-3]} == "map" ]]; then
458 if [[ ${words[-2]} == "idx" ]]; then
459 _bpftool_get_obj_map_idxs $obj
460 elif [[ ${words[-2]} == "name" ]]; then
461 _bpftool_get_obj_map_names $obj
462 fi
463 return 0
464 fi
465 if [[ ${words[-2]} == "map" ]]; then
466 COMPREPLY=( $( compgen -W "idx name" -- "$cur" ) )
467 return 0
468 fi
469
470 case $prev in
471 type)
472 local BPFTOOL_PROG_LOAD_TYPES='socket kprobe \
473 kretprobe classifier flow_dissector \
474 action tracepoint raw_tracepoint \
475 xdp perf_event cgroup/skb cgroup/sock \
476 cgroup/dev lwt_in lwt_out lwt_xmit \
477 lwt_seg6local sockops sk_skb sk_msg \
478 lirc_mode2 cgroup/bind4 cgroup/bind6 \
479 cgroup/connect4 cgroup/connect6 \
480 cgroup/getpeername4 cgroup/getpeername6 \
481 cgroup/getsockname4 cgroup/getsockname6 \
482 cgroup/sendmsg4 cgroup/sendmsg6 \
483 cgroup/recvmsg4 cgroup/recvmsg6 \
484 cgroup/post_bind4 cgroup/post_bind6 \
485 cgroup/sysctl cgroup/getsockopt \
486 cgroup/setsockopt cgroup/sock_release struct_ops \
487 fentry fexit freplace sk_lookup'
488 COMPREPLY=( $( compgen -W "$BPFTOOL_PROG_LOAD_TYPES" -- "$cur" ) )
489 return 0
490 ;;
491 id)
492 _bpftool_get_map_ids
493 return 0
494 ;;
495 name)
496 _bpftool_get_map_names
497 return 0
498 ;;
499 pinned|pinmaps)
500 _filedir
501 return 0
502 ;;
503 *)
504 COMPREPLY=( $( compgen -W "map" -- "$cur" ) )
505 _bpftool_once_attr 'type'
506 _bpftool_once_attr 'dev'
507 _bpftool_once_attr 'pinmaps'
508 return 0
509 ;;
510 esac
511 ;;
512 tracelog)
513 return 0
514 ;;
515 profile)
516 case $cword in
517 3)
518 COMPREPLY=( $( compgen -W "$PROG_TYPE" -- "$cur" ) )
519 return 0
520 ;;
521 4)
522 case $prev in
523 id)
524 _bpftool_get_prog_ids
525 ;;
526 name)
527 _bpftool_get_prog_names
528 ;;
529 pinned)
530 _filedir
531 ;;
532 esac
533 return 0
534 ;;
535 5)
536 COMPREPLY=( $( compgen -W "$METRIC_TYPE duration" -- "$cur" ) )
537 return 0
538 ;;
539 6)
540 case $prev in
541 duration)
542 return 0
543 ;;
544 *)
545 COMPREPLY=( $( compgen -W "$METRIC_TYPE" -- "$cur" ) )
546 return 0
547 ;;
548 esac
549 return 0
550 ;;
551 *)
552 COMPREPLY=( $( compgen -W "$METRIC_TYPE" -- "$cur" ) )
553 return 0
554 ;;
555 esac
556 ;;
557 run)
558 if [[ ${#words[@]} -eq 4 ]]; then
559 COMPREPLY=( $( compgen -W "$PROG_TYPE" -- "$cur" ) )
560 return 0
561 fi
562 case $prev in
563 id)
564 _bpftool_get_prog_ids
565 return 0
566 ;;
567 name)
568 _bpftool_get_prog_names
569 return 0
570 ;;
571 data_in|data_out|ctx_in|ctx_out)
572 _filedir
573 return 0
574 ;;
575 repeat|data_size_out|ctx_size_out)
576 return 0
577 ;;
578 *)
579 _bpftool_once_attr 'data_in data_out data_size_out \
580 ctx_in ctx_out ctx_size_out repeat'
581 return 0
582 ;;
583 esac
584 ;;
585 *)
586 [[ $prev == $object ]] && \
587 COMPREPLY=( $( compgen -W 'dump help pin attach detach \
588 load loadall show list tracelog run profile' -- "$cur" ) )
589 ;;
590 esac
591 ;;
592 struct_ops)
593 local STRUCT_OPS_TYPE='id name'
594 case $command in
595 show|list|dump|unregister)
596 case $prev in
597 $command)
598 COMPREPLY=( $( compgen -W "$STRUCT_OPS_TYPE" -- "$cur" ) )
599 ;;
600 id)
601 _bpftool_get_map_ids_for_type struct_ops
602 ;;
603 name)
604 _bpftool_get_map_names_for_type struct_ops
605 ;;
606 esac
607 return 0
608 ;;
609 register)
610 _filedir
611 return 0
612 ;;
613 *)
614 [[ $prev == $object ]] && \
615 COMPREPLY=( $( compgen -W 'register unregister show list dump help' \
616 -- "$cur" ) )
617 ;;
618 esac
619 ;;
620 iter)
621 case $command in
622 pin)
623 case $prev in
624 $command)
625 _filedir
626 ;;
627 id)
628 _bpftool_get_map_ids
629 ;;
630 name)
631 _bpftool_get_map_names
632 ;;
633 pinned)
634 _filedir
635 ;;
636 *)
637 _bpftool_one_of_list $MAP_TYPE
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 local MAP_TYPE='id pinned name'
651 case $command in
652 show|list|dump|peek|pop|dequeue|freeze)
653 case $prev in
654 $command)
655 COMPREPLY=( $( compgen -W "$MAP_TYPE" -- "$cur" ) )
656 return 0
657 ;;
658 id)
659 case "$command" in
660 peek)
661 _bpftool_get_map_ids_for_type stack
662 _bpftool_get_map_ids_for_type queue
663 ;;
664 pop)
665 _bpftool_get_map_ids_for_type stack
666 ;;
667 dequeue)
668 _bpftool_get_map_ids_for_type queue
669 ;;
670 *)
671 _bpftool_get_map_ids
672 ;;
673 esac
674 return 0
675 ;;
676 name)
677 case "$command" in
678 peek)
679 _bpftool_get_map_names_for_type stack
680 _bpftool_get_map_names_for_type queue
681 ;;
682 pop)
683 _bpftool_get_map_names_for_type stack
684 ;;
685 dequeue)
686 _bpftool_get_map_names_for_type queue
687 ;;
688 *)
689 _bpftool_get_map_names
690 ;;
691 esac
692 return 0
693 ;;
694 *)
695 return 0
696 ;;
697 esac
698 ;;
699 create)
700 case $prev in
701 $command)
702 _filedir
703 return 0
704 ;;
705 type)
706 local BPFTOOL_MAP_CREATE_TYPES='hash array \
707 prog_array perf_event_array percpu_hash \
708 percpu_array stack_trace cgroup_array lru_hash \
709 lru_percpu_hash lpm_trie array_of_maps \
710 hash_of_maps devmap devmap_hash sockmap cpumap \
711 xskmap sockhash cgroup_storage reuseport_sockarray \
712 percpu_cgroup_storage queue stack sk_storage \
713 struct_ops inode_storage task_storage ringbuf'
714 COMPREPLY=( $( compgen -W "$BPFTOOL_MAP_CREATE_TYPES" -- "$cur" ) )
715 return 0
716 ;;
717 key|value|flags|entries)
718 return 0
719 ;;
720 inner_map)
721 COMPREPLY=( $( compgen -W "$MAP_TYPE" -- "$cur" ) )
722 return 0
723 ;;
724 id)
725 _bpftool_get_map_ids
726 ;;
727 name)
728 case $pprev in
729 inner_map)
730 _bpftool_get_map_names
731 ;;
732 *)
733 return 0
734 ;;
735 esac
736 ;;
737 *)
738 _bpftool_once_attr 'type'
739 _bpftool_once_attr 'key'
740 _bpftool_once_attr 'value'
741 _bpftool_once_attr 'entries'
742 _bpftool_once_attr 'name'
743 _bpftool_once_attr 'flags'
744 if _bpftool_search_list 'array_of_maps' 'hash_of_maps'; then
745 _bpftool_once_attr 'inner_map'
746 fi
747 _bpftool_once_attr 'dev'
748 return 0
749 ;;
750 esac
751 ;;
752 lookup|getnext|delete)
753 case $prev in
754 $command)
755 COMPREPLY=( $( compgen -W "$MAP_TYPE" -- "$cur" ) )
756 return 0
757 ;;
758 id)
759 _bpftool_get_map_ids
760 return 0
761 ;;
762 name)
763 _bpftool_get_map_names
764 return 0
765 ;;
766 key)
767 COMPREPLY+=( $( compgen -W 'hex' -- "$cur" ) )
768 ;;
769 *)
770 case $(_bpftool_map_guess_map_type) in
771 queue|stack)
772 return 0
773 ;;
774 esac
775
776 _bpftool_once_attr 'key'
777 return 0
778 ;;
779 esac
780 ;;
781 update|push|enqueue)
782 case $prev in
783 $command)
784 COMPREPLY=( $( compgen -W "$MAP_TYPE" -- "$cur" ) )
785 return 0
786 ;;
787 id)
788 _bpftool_map_update_get_id $command
789 return 0
790 ;;
791 name)
792 _bpftool_map_update_get_name $command
793 return 0
794 ;;
795 key)
796 COMPREPLY+=( $( compgen -W 'hex' -- "$cur" ) )
797 ;;
798 value)
799 # We can have bytes, or references to a prog or a
800 # map, depending on the type of the map to update.
801 case "$(_bpftool_map_guess_map_type)" in
802 array_of_maps|hash_of_maps)
803 local MAP_TYPE='id pinned name'
804 COMPREPLY+=( $( compgen -W "$MAP_TYPE" \
805 -- "$cur" ) )
806 return 0
807 ;;
808 prog_array)
809 local PROG_TYPE='id pinned tag name'
810 COMPREPLY+=( $( compgen -W "$PROG_TYPE" \
811 -- "$cur" ) )
812 return 0
813 ;;
814 *)
815 COMPREPLY+=( $( compgen -W 'hex' \
816 -- "$cur" ) )
817 return 0
818 ;;
819 esac
820 return 0
821 ;;
822 *)
823 case $(_bpftool_map_guess_map_type) in
824 queue|stack)
825 _bpftool_once_attr 'value'
826 return 0;
827 ;;
828 esac
829
830 _bpftool_once_attr 'key'
831 local UPDATE_FLAGS='any exist noexist'
832 for (( idx=3; idx < ${#words[@]}-1; idx++ )); do
833 if [[ ${words[idx]} == 'value' ]]; then
834 # 'value' is present, but is not the last
835 # word i.e. we can now have UPDATE_FLAGS.
836 _bpftool_one_of_list "$UPDATE_FLAGS"
837 return 0
838 fi
839 done
840 for (( idx=3; idx < ${#words[@]}-1; idx++ )); do
841 if [[ ${words[idx]} == 'key' ]]; then
842 # 'key' is present, but is not the last
843 # word i.e. we can now have 'value'.
844 _bpftool_once_attr 'value'
845 return 0
846 fi
847 done
848
849 return 0
850 ;;
851 esac
852 ;;
853 pin)
854 case $prev in
855 $command)
856 COMPREPLY=( $( compgen -W "$MAP_TYPE" -- "$cur" ) )
857 ;;
858 id)
859 _bpftool_get_map_ids
860 ;;
861 name)
862 _bpftool_get_map_names
863 ;;
864 esac
865 return 0
866 ;;
867 event_pipe)
868 case $prev in
869 $command)
870 COMPREPLY=( $( compgen -W "$MAP_TYPE" -- "$cur" ) )
871 return 0
872 ;;
873 id)
874 _bpftool_get_map_ids_for_type perf_event_array
875 return 0
876 ;;
877 name)
878 _bpftool_get_map_names_for_type perf_event_array
879 return 0
880 ;;
881 cpu)
882 return 0
883 ;;
884 index)
885 return 0
886 ;;
887 *)
888 _bpftool_once_attr 'cpu'
889 _bpftool_once_attr 'index'
890 return 0
891 ;;
892 esac
893 ;;
894 *)
895 [[ $prev == $object ]] && \
896 COMPREPLY=( $( compgen -W 'delete dump getnext help \
897 lookup pin event_pipe show list update create \
898 peek push enqueue pop dequeue freeze' -- \
899 "$cur" ) )
900 ;;
901 esac
902 ;;
903 btf)
904 local PROG_TYPE='id pinned tag name'
905 local MAP_TYPE='id pinned name'
906 case $command in
907 dump)
908 case $prev in
909 $command)
910 COMPREPLY+=( $( compgen -W "id map prog file" -- \
911 "$cur" ) )
912 return 0
913 ;;
914 prog)
915 COMPREPLY=( $( compgen -W "$PROG_TYPE" -- "$cur" ) )
916 return 0
917 ;;
918 map)
919 COMPREPLY=( $( compgen -W "$MAP_TYPE" -- "$cur" ) )
920 return 0
921 ;;
922 id)
923 case $pprev in
924 prog)
925 _bpftool_get_prog_ids
926 ;;
927 map)
928 _bpftool_get_map_ids
929 ;;
930 $command)
931 _bpftool_get_btf_ids
932 ;;
933 esac
934 return 0
935 ;;
936 name)
937 case $pprev in
938 prog)
939 _bpftool_get_prog_names
940 ;;
941 map)
942 _bpftool_get_map_names
943 ;;
944 esac
945 return 0
946 ;;
947 format)
948 COMPREPLY=( $( compgen -W "c raw" -- "$cur" ) )
949 ;;
950 *)
951 # emit extra options
952 case ${words[3]} in
953 id|file)
954 _bpftool_once_attr 'format'
955 ;;
956 map|prog)
957 if [[ ${words[3]} == "map" ]] && [[ $cword == 6 ]]; then
958 COMPREPLY+=( $( compgen -W "key value kv all" -- "$cur" ) )
959 fi
960 _bpftool_once_attr 'format'
961 ;;
962 *)
963 ;;
964 esac
965 return 0
966 ;;
967 esac
968 ;;
969 show|list)
970 case $prev in
971 $command)
972 COMPREPLY+=( $( compgen -W "id" -- "$cur" ) )
973 ;;
974 id)
975 _bpftool_get_btf_ids
976 ;;
977 esac
978 return 0
979 ;;
980 *)
981 [[ $prev == $object ]] && \
982 COMPREPLY=( $( compgen -W 'dump help show list' \
983 -- "$cur" ) )
984 ;;
985 esac
986 ;;
987 gen)
988 case $command in
989 object)
990 _filedir
991 return 0
992 ;;
993 skeleton)
994 case $prev in
995 $command)
996 _filedir
997 return 0
998 ;;
999 *)
1000 _bpftool_once_attr 'name'
1001 return 0
1002 ;;
1003 esac
1004 ;;
1005 *)
1006 [[ $prev == $object ]] && \
1007 COMPREPLY=( $( compgen -W 'object skeleton help' -- "$cur" ) )
1008 ;;
1009 esac
1010 ;;
1011 cgroup)
1012 case $command in
1013 show|list|tree)
1014 case $cword in
1015 3)
1016 _filedir
1017 ;;
1018 4)
1019 COMPREPLY=( $( compgen -W 'effective' -- "$cur" ) )
1020 ;;
1021 esac
1022 return 0
1023 ;;
1024 attach|detach)
1025 local BPFTOOL_CGROUP_ATTACH_TYPES='ingress egress \
1026 sock_create sock_ops device \
1027 bind4 bind6 post_bind4 post_bind6 connect4 connect6 \
1028 getpeername4 getpeername6 getsockname4 getsockname6 \
1029 sendmsg4 sendmsg6 recvmsg4 recvmsg6 sysctl getsockopt \
1030 setsockopt sock_release'
1031 local ATTACH_FLAGS='multi override'
1032 local PROG_TYPE='id pinned tag name'
1033 # Check for $prev = $command first
1034 if [ $prev = $command ]; then
1035 _filedir
1036 return 0
1037 # Then check for attach type. This is done outside of the
1038 # "case $prev in" to avoid writing the whole list of attach
1039 # types again as pattern to match (where we cannot reuse
1040 # our variable).
1041 elif [[ $BPFTOOL_CGROUP_ATTACH_TYPES =~ $prev ]]; then
1042 COMPREPLY=( $( compgen -W "$PROG_TYPE" -- \
1043 "$cur" ) )
1044 return 0
1045 fi
1046 # case/esac for the other cases
1047 case $prev in
1048 id)
1049 _bpftool_get_prog_ids
1050 return 0
1051 ;;
1052 *)
1053 if ! _bpftool_search_list "$BPFTOOL_CGROUP_ATTACH_TYPES"; then
1054 COMPREPLY=( $( compgen -W \
1055 "$BPFTOOL_CGROUP_ATTACH_TYPES" -- "$cur" ) )
1056 elif [[ "$command" == "attach" ]]; then
1057 # We have an attach type on the command line,
1058 # but it is not the previous word, or
1059 # "id|pinned|tag|name" (we already checked for
1060 # that). This should only leave the case when
1061 # we need attach flags for "attach" commamnd.
1062 _bpftool_one_of_list "$ATTACH_FLAGS"
1063 fi
1064 return 0
1065 ;;
1066 esac
1067 ;;
1068 *)
1069 [[ $prev == $object ]] && \
1070 COMPREPLY=( $( compgen -W 'help attach detach \
1071 show list tree' -- "$cur" ) )
1072 ;;
1073 esac
1074 ;;
1075 perf)
1076 case $command in
1077 *)
1078 [[ $prev == $object ]] && \
1079 COMPREPLY=( $( compgen -W 'help \
1080 show list' -- "$cur" ) )
1081 ;;
1082 esac
1083 ;;
1084 net)
1085 local PROG_TYPE='id pinned tag name'
1086 local ATTACH_TYPES='xdp xdpgeneric xdpdrv xdpoffload'
1087 case $command in
1088 show|list)
1089 [[ $prev != "$command" ]] && return 0
1090 COMPREPLY=( $( compgen -W 'dev' -- "$cur" ) )
1091 return 0
1092 ;;
1093 attach)
1094 case $cword in
1095 3)
1096 COMPREPLY=( $( compgen -W "$ATTACH_TYPES" -- "$cur" ) )
1097 return 0
1098 ;;
1099 4)
1100 COMPREPLY=( $( compgen -W "$PROG_TYPE" -- "$cur" ) )
1101 return 0
1102 ;;
1103 5)
1104 case $prev in
1105 id)
1106 _bpftool_get_prog_ids
1107 ;;
1108 name)
1109 _bpftool_get_prog_names
1110 ;;
1111 pinned)
1112 _filedir
1113 ;;
1114 esac
1115 return 0
1116 ;;
1117 6)
1118 COMPREPLY=( $( compgen -W 'dev' -- "$cur" ) )
1119 return 0
1120 ;;
1121 8)
1122 _bpftool_once_attr 'overwrite'
1123 return 0
1124 ;;
1125 esac
1126 ;;
1127 detach)
1128 case $cword in
1129 3)
1130 COMPREPLY=( $( compgen -W "$ATTACH_TYPES" -- "$cur" ) )
1131 return 0
1132 ;;
1133 4)
1134 COMPREPLY=( $( compgen -W 'dev' -- "$cur" ) )
1135 return 0
1136 ;;
1137 esac
1138 ;;
1139 *)
1140 [[ $prev == $object ]] && \
1141 COMPREPLY=( $( compgen -W 'help \
1142 show list attach detach' -- "$cur" ) )
1143 ;;
1144 esac
1145 ;;
1146 feature)
1147 case $command in
1148 probe)
1149 [[ $prev == "prefix" ]] && return 0
1150 if _bpftool_search_list 'macros'; then
1151 _bpftool_once_attr 'prefix'
1152 else
1153 COMPREPLY+=( $( compgen -W 'macros' -- "$cur" ) )
1154 fi
1155 _bpftool_one_of_list 'kernel dev'
1156 _bpftool_once_attr 'full unprivileged'
1157 return 0
1158 ;;
1159 *)
1160 [[ $prev == $object ]] && \
1161 COMPREPLY=( $( compgen -W 'help probe' -- "$cur" ) )
1162 ;;
1163 esac
1164 ;;
1165 link)
1166 case $command in
1167 show|list|pin|detach)
1168 case $prev in
1169 id)
1170 _bpftool_get_link_ids
1171 return 0
1172 ;;
1173 esac
1174 ;;
1175 esac
1176
1177 local LINK_TYPE='id pinned'
1178 case $command in
1179 show|list)
1180 [[ $prev != "$command" ]] && return 0
1181 COMPREPLY=( $( compgen -W "$LINK_TYPE" -- "$cur" ) )
1182 return 0
1183 ;;
1184 pin|detach)
1185 if [[ $prev == "$command" ]]; then
1186 COMPREPLY=( $( compgen -W "$LINK_TYPE" -- "$cur" ) )
1187 else
1188 _filedir
1189 fi
1190 return 0
1191 ;;
1192 *)
1193 [[ $prev == $object ]] && \
1194 COMPREPLY=( $( compgen -W 'help pin show list' -- "$cur" ) )
1195 ;;
1196 esac
1197 ;;
1198 esac
1199} &&
1200complete -F _bpftool bpftool
1201
1202# ex: ts=4 sw=4 et filetype=sh