d8310830a18b1f7fb2d7de473d8f3a48ac6c9e13
[sfrench/cifs-2.6.git] / tools / perf / perf-completion.sh
1 # perf bash and zsh completion
2 # SPDX-License-Identifier: GPL-2.0
3
4 # Taken from git.git's completion script.
5 __my_reassemble_comp_words_by_ref()
6 {
7         local exclude i j first
8         # Which word separators to exclude?
9         exclude="${1//[^$COMP_WORDBREAKS]}"
10         cword_=$COMP_CWORD
11         if [ -z "$exclude" ]; then
12                 words_=("${COMP_WORDS[@]}")
13                 return
14         fi
15         # List of word completion separators has shrunk;
16         # re-assemble words to complete.
17         for ((i=0, j=0; i < ${#COMP_WORDS[@]}; i++, j++)); do
18                 # Append each nonempty word consisting of just
19                 # word separator characters to the current word.
20                 first=t
21                 while
22                         [ $i -gt 0 ] &&
23                         [ -n "${COMP_WORDS[$i]}" ] &&
24                         # word consists of excluded word separators
25                         [ "${COMP_WORDS[$i]//[^$exclude]}" = "${COMP_WORDS[$i]}" ]
26                 do
27                         # Attach to the previous token,
28                         # unless the previous token is the command name.
29                         if [ $j -ge 2 ] && [ -n "$first" ]; then
30                                 ((j--))
31                         fi
32                         first=
33                         words_[$j]=${words_[j]}${COMP_WORDS[i]}
34                         if [ $i = $COMP_CWORD ]; then
35                                 cword_=$j
36                         fi
37                         if (($i < ${#COMP_WORDS[@]} - 1)); then
38                                 ((i++))
39                         else
40                                 # Done.
41                                 return
42                         fi
43                 done
44                 words_[$j]=${words_[j]}${COMP_WORDS[i]}
45                 if [ $i = $COMP_CWORD ]; then
46                         cword_=$j
47                 fi
48         done
49 }
50
51 # Define preload_get_comp_words_by_ref="false", if the function
52 # __perf_get_comp_words_by_ref() is required instead.
53 preload_get_comp_words_by_ref="true"
54
55 if [ $preload_get_comp_words_by_ref = "true" ]; then
56         type _get_comp_words_by_ref &>/dev/null ||
57         preload_get_comp_words_by_ref="false"
58 fi
59 [ $preload_get_comp_words_by_ref = "true" ] ||
60 __perf_get_comp_words_by_ref()
61 {
62         local exclude cur_ words_ cword_
63         if [ "$1" = "-n" ]; then
64                 exclude=$2
65                 shift 2
66         fi
67         __my_reassemble_comp_words_by_ref "$exclude"
68         cur_=${words_[cword_]}
69         while [ $# -gt 0 ]; do
70                 case "$1" in
71                 cur)
72                         cur=$cur_
73                         ;;
74                 prev)
75                         prev=${words_[$cword_-1]}
76                         ;;
77                 words)
78                         words=("${words_[@]}")
79                         ;;
80                 cword)
81                         cword=$cword_
82                         ;;
83                 esac
84                 shift
85         done
86 }
87
88 # Define preload__ltrim_colon_completions="false", if the function
89 # __perf__ltrim_colon_completions() is required instead.
90 preload__ltrim_colon_completions="true"
91
92 if [ $preload__ltrim_colon_completions = "true" ]; then
93         type __ltrim_colon_completions &>/dev/null ||
94         preload__ltrim_colon_completions="false"
95 fi
96 [ $preload__ltrim_colon_completions = "true" ] ||
97 __perf__ltrim_colon_completions()
98 {
99         if [[ "$1" == *:* && "$COMP_WORDBREAKS" == *:* ]]; then
100                 # Remove colon-word prefix from COMPREPLY items
101                 local colon_word=${1%"${1##*:}"}
102                 local i=${#COMPREPLY[*]}
103                 while [[ $((--i)) -ge 0 ]]; do
104                         COMPREPLY[$i]=${COMPREPLY[$i]#"$colon_word"}
105                 done
106         fi
107 }
108
109 __perfcomp ()
110 {
111         COMPREPLY=( $( compgen -W "$1" -- "$2" ) )
112 }
113
114 __perfcomp_colon ()
115 {
116         __perfcomp "$1" "$2"
117         if [ $preload__ltrim_colon_completions = "true" ]; then
118                 __ltrim_colon_completions $cur
119         else
120                 __perf__ltrim_colon_completions $cur
121         fi
122 }
123
124 __perf_prev_skip_opts ()
125 {
126         local i cmd_ cmds_
127
128         let i=cword-1
129         cmds_=$($cmd $1 --list-cmds)
130         prev_skip_opts=()
131         while [ $i -ge 0 ]; do
132                 if [[ ${words[i]} == $1 ]]; then
133                         return
134                 fi
135                 for cmd_ in $cmds_; do
136                         if [[ ${words[i]} == $cmd_ ]]; then
137                                 prev_skip_opts=${words[i]}
138                                 return
139                         fi
140                 done
141                 ((i--))
142         done
143 }
144
145 __perf_main ()
146 {
147         local cmd
148
149         cmd=${words[0]}
150         COMPREPLY=()
151
152         # Skip options backward and find the last perf command
153         __perf_prev_skip_opts
154         # List perf subcommands or long options
155         if [ -z $prev_skip_opts ]; then
156                 if [[ $cur == --* ]]; then
157                         cmds=$($cmd --list-opts)
158                 else
159                         cmds=$($cmd --list-cmds)
160                 fi
161                 __perfcomp "$cmds" "$cur"
162         # List possible events for -e option
163         elif [[ $prev == @("-e"|"--event") &&
164                 $prev_skip_opts == @(record|stat|top) ]]; then
165
166                 local cur1=${COMP_WORDS[COMP_CWORD]}
167                 local raw_evts=$($cmd list --raw-dump)
168                 local arr s tmp result
169
170                 if [[ "$cur1" == */* && ${cur1#*/} =~ ^[A-Z] ]]; then
171                         OLD_IFS="$IFS"
172                         IFS=" "
173                         arr=($raw_evts)
174                         IFS="$OLD_IFS"
175
176                         for s in ${arr[@]}
177                         do
178                                 if [[ "$s" == *cpu/* ]]; then
179                                         tmp=${s#*cpu/}
180                                         result=$result" ""cpu/"${tmp^^}
181                                 else
182                                         result=$result" "$s
183                                 fi
184                         done
185
186                         evts=${result}+$(ls /sys/bus/event_source/devices/cpu/events)
187                 else
188                         evts=${raw_evts}+$(ls /sys/bus/event_source/devices/cpu/events)
189                 fi
190
191                 __perfcomp_colon "$evts" "$cur1"
192         else
193                 # List subcommands for perf commands
194                 if [[ $prev_skip_opts == @(kvm|kmem|mem|lock|sched|
195                         |data|help|script|test|timechart|trace) ]]; then
196                         subcmds=$($cmd $prev_skip_opts --list-cmds)
197                         __perfcomp_colon "$subcmds" "$cur"
198                 fi
199                 # List long option names
200                 if [[ $cur == --* ]];  then
201                         subcmd=$prev_skip_opts
202                         __perf_prev_skip_opts $subcmd
203                         subcmd=$subcmd" "$prev_skip_opts
204                         opts=$($cmd $subcmd --list-opts)
205                         __perfcomp "$opts" "$cur"
206                 fi
207         fi
208 }
209
210 if [[ -n ${ZSH_VERSION-} ]]; then
211         autoload -U +X compinit && compinit
212
213         __perfcomp ()
214         {
215                 emulate -L zsh
216
217                 local c IFS=$' \t\n'
218                 local -a array
219
220                 for c in ${=1}; do
221                         case $c in
222                         --*=*|*.) ;;
223                         *) c="$c " ;;
224                         esac
225                         array[${#array[@]}+1]="$c"
226                 done
227
228                 compset -P '*[=:]'
229                 compadd -Q -S '' -a -- array && _ret=0
230         }
231
232         __perfcomp_colon ()
233         {
234                 emulate -L zsh
235
236                 local cur_="${2-$cur}"
237                 local c IFS=$' \t\n'
238                 local -a array
239
240                 if [[ "$cur_" == *:* ]]; then
241                         local colon_word=${cur_%"${cur_##*:}"}
242                 fi
243
244                 for c in ${=1}; do
245                         case $c in
246                         --*=*|*.) ;;
247                         *) c="$c " ;;
248                         esac
249                         array[$#array+1]=${c#"$colon_word"}
250                 done
251
252                 compset -P '*[=:]'
253                 compadd -Q -S '' -a -- array && _ret=0
254         }
255
256         _perf ()
257         {
258                 local _ret=1 cur cword prev
259                 cur=${words[CURRENT]}
260                 prev=${words[CURRENT-1]}
261                 let cword=CURRENT-1
262                 emulate ksh -c __perf_main
263                 let _ret && _default && _ret=0
264                 return _ret
265         }
266
267         compdef _perf perf
268         return
269 fi
270
271 type perf &>/dev/null &&
272 _perf()
273 {
274         if [[ "$COMP_WORDBREAKS" != *,* ]]; then
275                 COMP_WORDBREAKS="${COMP_WORDBREAKS},"
276                 export COMP_WORDBREAKS
277         fi
278
279         local cur words cword prev
280         if [ $preload_get_comp_words_by_ref = "true" ]; then
281                 _get_comp_words_by_ref -n =:, cur words cword prev
282         else
283                 __perf_get_comp_words_by_ref -n =:, cur words cword prev
284         fi
285         __perf_main
286 } &&
287
288 complete -o bashdefault -o default -o nospace -F _perf perf 2>/dev/null \
289         || complete -o default -o nospace -F _perf perf