| # /* vim: set ai ts=4 ft=sh: */ | 
 | # | 
 | # Copyright 2011, The Android Open Source Project | 
 | # | 
 | # Licensed under the Apache License, Version 2.0 (the "License"); | 
 | # you may not use this file except in compliance with the License. | 
 | # You may obtain a copy of the License at | 
 | # | 
 | #     http://www.apache.org/licenses/LICENSE-2.0 | 
 | # | 
 | # Unless required by applicable law or agreed to in writing, software | 
 | # distributed under the License is distributed on an "AS IS" BASIS, | 
 | # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | 
 | # See the License for the specific language governing permissions and | 
 | # limitations under the License. | 
 | # | 
 |  | 
 | _adb() { | 
 |     if ! check_type "$1" >/dev/null; then | 
 |         return | 
 |     fi | 
 |  | 
 |     if check_type _init_completion >/dev/null; then | 
 |         _init_completion || return | 
 |     fi | 
 |  | 
 |     local where i cur serial | 
 |     COMPREPLY=() | 
 |  | 
 |     serial="${ANDROID_SERIAL:-none}" | 
 |     where=OPTIONS | 
 |     for ((i=1; i <= COMP_CWORD; i++)); do | 
 |         cur="${COMP_WORDS[i]}" | 
 |         case "${cur}" in | 
 |             -s) | 
 |                 where=OPT_SERIAL | 
 |                 ;; | 
 |             -p) | 
 |                 where=OPT_PATH | 
 |                 ;; | 
 |             -*) | 
 |                 where=OPTIONS | 
 |                 ;; | 
 |             *) | 
 |                 if [[ $where == OPT_SERIAL ]]; then | 
 |                     where=OPT_SERIAL_ARG | 
 |                     serial=${cur} | 
 |                 else | 
 |                     where=COMMAND | 
 |                     break | 
 |                 fi | 
 |                 ;; | 
 |         esac | 
 |     done | 
 |  | 
 |     if [[ $where == COMMAND && $i -ge $COMP_CWORD ]]; then | 
 |         where=OPTIONS | 
 |     fi | 
 |  | 
 |     OPTIONS="-d -e -s -p" | 
 |     COMMAND="devices connect disconnect push pull sync shell emu logcat lolcat forward jdwp install uninstall bugreport help version start-server kill-server get-state get-serialno status-window remount reboot reboot-bootloader root usb tcpip disable-verity" | 
 |  | 
 |     case $where in | 
 |         OPTIONS|OPT_SERIAL|OPT_PATH) | 
 |             COMPREPLY=( $(compgen -W "$OPTIONS $COMMAND" -- "$cur") ) | 
 |             ;; | 
 |         OPT_SERIAL_ARG) | 
 |             local devices=$(command adb devices 2> /dev/null | grep -v "List of devices" | awk '{ print $1 }') | 
 |             COMPREPLY=( $(compgen -W "${devices}" -- ${cur}) ) | 
 |             ;; | 
 |         COMMAND) | 
 |             if [[ $i -eq $COMP_CWORD ]]; then | 
 |                 COMPREPLY=( $(compgen -W "$COMMAND" -- "$cur") ) | 
 |             else | 
 |                 i=$((i+1)) | 
 |                 case "${cur}" in | 
 |                     install) | 
 |                         _adb_cmd_install "$serial" $i | 
 |                         ;; | 
 |                     sideload) | 
 |                         _adb_cmd_sideload "$serial" $i | 
 |                         ;; | 
 |                     pull) | 
 |                         _adb_cmd_pull "$serial" $i | 
 |                         ;; | 
 |                     push) | 
 |                         _adb_cmd_push "$serial" $i | 
 |                         ;; | 
 |                     reboot) | 
 |                         if [[ $COMP_CWORD == $i ]]; then | 
 |                             args="bootloader recovery" | 
 |                             COMPREPLY=( $(compgen -W "${args}" -- "${COMP_WORDS[i]}") ) | 
 |                         fi | 
 |                         ;; | 
 |                     shell) | 
 |                         _adb_cmd_shell "$serial" $i | 
 |                         ;; | 
 |                     uninstall) | 
 |                         _adb_cmd_uninstall "$serial" $i | 
 |                         ;; | 
 |                 esac | 
 |             fi | 
 |             ;; | 
 |     esac | 
 |  | 
 |     return 0 | 
 | } | 
 |  | 
 | _adb_cmd_install() { | 
 |     local serial i cur where | 
 |  | 
 |     serial=$1 | 
 |     i=$2 | 
 |  | 
 |     where=OPTIONS | 
 |     for ((; i <= COMP_CWORD; i++)); do | 
 |         cur="${COMP_WORDS[i]}" | 
 |         case "${cur}" in | 
 |             -*) | 
 |                 where=OPTIONS | 
 |                 ;; | 
 |             *) | 
 |                 where=FILE | 
 |                 break | 
 |                 ;; | 
 |         esac | 
 |     done | 
 |  | 
 |     cur="${COMP_WORDS[COMP_CWORD]}" | 
 |     if [[ $where == OPTIONS ]]; then | 
 |         COMPREPLY=( $(compgen -W "-d -l -r -s" -- "${cur}") ) | 
 |         return | 
 |     fi | 
 |  | 
 |     _adb_util_complete_local_file "${cur}" '!*.apk' | 
 | } | 
 |  | 
 | _adb_cmd_sideload() { | 
 |     local serial i cur | 
 |  | 
 |     serial=$1 | 
 |     i=$2 | 
 |  | 
 |     cur="${COMP_WORDS[COMP_CWORD]}" | 
 |  | 
 |     _adb_util_complete_local_file "${cur}" '!*.zip' | 
 | } | 
 |  | 
 | _adb_cmd_push() { | 
 |     local serial IFS=$'\n' i cur | 
 |  | 
 |     serial=$1 | 
 |     i=$2 | 
 |  | 
 |     cur="${COMP_WORDS[COMP_CWORD]}" | 
 |  | 
 |     if [[ $COMP_CWORD == $i ]]; then | 
 |         _adb_util_complete_local_file "${cur}" | 
 |     elif [[ $COMP_CWORD == $(($i+1)) ]]; then | 
 |         if [ "${cur}" == "" ]; then | 
 |             cur="/" | 
 |         fi | 
 |         _adb_util_list_files $serial "${cur}" | 
 |     fi | 
 | } | 
 |  | 
 | _adb_cmd_pull() { | 
 |     local serial IFS=$'\n' i cur | 
 |  | 
 |     serial=$1 | 
 |     i=$2 | 
 |  | 
 |     cur="${COMP_WORDS[COMP_CWORD]}" | 
 |  | 
 |     if [[ $COMP_CWORD == $i ]]; then | 
 |         if [ "${cur}" == "" ]; then | 
 |             cur="/" | 
 |         fi | 
 |         _adb_util_list_files $serial "${cur}" | 
 |     elif [[ $COMP_CWORD == $(($i+1)) ]]; then | 
 |         _adb_util_complete_local_file "${cur}" | 
 |     fi | 
 | } | 
 |  | 
 | _adb_cmd_shell() { | 
 |     local serial IFS=$'\n' i cur | 
 |     local -a args | 
 |  | 
 |     serial=$1 | 
 |     i=$2 | 
 |  | 
 |     cur="${COMP_WORDS[i]}" | 
 |     if [ "$serial" != "none" ]; then | 
 |         args=(-s $serial) | 
 |     fi | 
 |  | 
 |     if [[ $i -eq $COMP_CWORD && ${cur:0:1} != "/" ]]; then | 
 |         paths=$(command adb ${args[@]} shell echo '$'PATH 2> /dev/null | tr -d '\r' | tr : '\n') | 
 |         COMMAND=$(command adb ${args[@]} shell ls $paths '2>' /dev/null | tr -d '\r' | { | 
 |             while read -r tmp; do | 
 |                 command=${tmp##*/} | 
 |                 printf '%s\n' "$command" | 
 |             done | 
 |         }) | 
 |         COMPREPLY=( $(compgen -W "$COMMAND" -- "$cur") ) | 
 |         return 0 | 
 |     fi | 
 |  | 
 |     i=$((i+1)) | 
 |     case "$cur" in | 
 |         ls) | 
 |             _adb_shell_file_command $serial $i "--color -A -C -F -H -L -R -S -Z -a -c -d -f -h -i -k -l -m -n -p -q -r -s -t -u -x -1" | 
 |             ;; | 
 |         cat) | 
 |             _adb_shell_file_command $serial $i "-h -e -t -u -v" | 
 |             ;; | 
 |         dumpsys) | 
 |             _adb_cmd_shell_dumpsys "$serial" $i | 
 |             ;; | 
 |         am) | 
 |             _adb_cmd_shell_am "$serial" $i | 
 |             ;; | 
 |         pm) | 
 |             _adb_cmd_shell_pm "$serial" $i | 
 |             ;; | 
 |         /*) | 
 |             _adb_util_list_files $serial "$cur" | 
 |             ;; | 
 |         *) | 
 |             COMPREPLY=( ) | 
 |             ;; | 
 |     esac | 
 |  | 
 |     return 0 | 
 | } | 
 |  | 
 | _adb_cmd_shell_dumpsys() { | 
 |     local serial i cur | 
 |     local -a args | 
 |     local candidates | 
 |  | 
 |     unset IFS | 
 |  | 
 |     serial=$1 | 
 |     i=$2 | 
 |  | 
 |     if [ "$serial" != "none" ]; then | 
 |         args=(-s $serial) | 
 |     fi | 
 |  | 
 |     if (( $i == $COMP_CWORD )) ; then | 
 |         cur="${COMP_WORDS[COMP_CWORD]}" | 
 |         # First line is a header, so need "1d". | 
 |         candidates=$(command adb ${args[@]} shell dumpsys -l 2> /dev/null | sed -e '1d;s/^  *//' | tr -d '\r') | 
 |         candidates="-l $candidates" | 
 |         COMPREPLY=( $(compgen -W "$candidates" -- "$cur") ) | 
 |         return 0 | 
 |     fi | 
 |  | 
 |     COMPREPLY=( ) | 
 |     return 0 | 
 | } | 
 |  | 
 | _adb_cmd_shell_am() { | 
 |     local serial i cur | 
 |     local candidates | 
 |  | 
 |     unset IFS | 
 |  | 
 |     serial=$1 | 
 |     i=$2 | 
 |  | 
 |     if (( $i == $COMP_CWORD )) ; then | 
 |         cur="${COMP_WORDS[COMP_CWORD]}" | 
 |         candidates="broadcast clear-debug-app clear-watch-heap dumpheap force-stop get-config get-inactive hang idle-maintenance instrument kill kill-all monitor package-importance profile restart screen-compat send-trim-memory set-debug-app set-inactive set-watch-heap stack start startservice start-user stopservice stop-user suppress-resize-config-changes switch-user task to-app-uri to-intent-uri to-uri" | 
 |         COMPREPLY=( $(compgen -W "$candidates" -- "$cur") ) | 
 |         return 0 | 
 |     fi | 
 |  | 
 |     COMPREPLY=( ) | 
 |     return 0 | 
 | } | 
 |  | 
 |  | 
 | _adb_cmd_shell_pm() { | 
 |     local serial i cur | 
 |     local candidates | 
 |  | 
 |     unset IFS | 
 |  | 
 |     serial=$1 | 
 |     i=$2 | 
 |  | 
 |     if (( $i == $COMP_CWORD )) ; then | 
 |         cur="${COMP_WORDS[COMP_CWORD]}" | 
 |         candidates="-l -lf -p clear create-user default-state disable" | 
 |         candidates+=" disable-until-used disable-user dump enable" | 
 |         candidates+=" get-app-link get-install-location get-max-users" | 
 |         candidates+=" get-max-running-users grant hide install" | 
 |         candidates+=" install-abandon install-commit install-create" | 
 |         candidates+=" install-write list move-package" | 
 |         candidates+=" move-primary-storage path remove-user" | 
 |         candidates+=" reset-permissions revoke set-app-link" | 
 |         candidates+=" set-installer set-install-location" | 
 |         candidates+=" set-permission-enforced trim-caches unhide" | 
 |         candidates+=" uninstall" | 
 |         COMPREPLY=( $(compgen -W "$candidates" -- "$cur") ) | 
 |         return 0 | 
 |     fi | 
 |  | 
 |     if (( $i + 1 == $COMP_CWORD )) && [[ "${COMP_WORDS[COMP_CWORD -1]}" == "list" ]]  ; then | 
 |         cur="${COMP_WORDS[COMP_CWORD]}" | 
 |         candidates="packages permission-groups permissions instrumentation features libraries users" | 
 |         COMPREPLY=( $(compgen -W "$candidates" -- "$cur") ) | 
 |         return 0 | 
 |     fi | 
 |  | 
 |     COMPREPLY=( ) | 
 |     return 0 | 
 | } | 
 |  | 
 | _adb_cmd_uninstall() { | 
 |     local serial i where cur packages | 
 |  | 
 |     serial=$1 | 
 |     i=$2 | 
 |     if [ "$serial" != "none" ]; then | 
 |         args=(-s $serial) | 
 |     fi | 
 |  | 
 |     where=OPTIONS | 
 |     for ((; i <= COMP_CWORD; i++)); do | 
 |         cur="${COMP_WORDS[i]}" | 
 |         case "${cur}" in | 
 |             -*) | 
 |                 where=OPTIONS | 
 |                 ;; | 
 |             *) | 
 |                 where=FILE | 
 |                 break | 
 |                 ;; | 
 |         esac | 
 |     done | 
 |  | 
 |     cur="${COMP_WORDS[COMP_CWORD]}" | 
 |     if [[ $where == OPTIONS ]]; then | 
 |         COMPREPLY=( $(compgen -W "-k" -- "${cur}") ) | 
 |     fi | 
 |  | 
 |     packages="$( | 
 |         command adb ${args[@]} shell pm list packages '2>' /dev/null 2> /dev/null | tr -d '\r' | { | 
 |             while read -r tmp; do | 
 |                 local package=${tmp#package:} | 
 |                 echo -n "${package} " | 
 |             done | 
 |         } | 
 |     )" | 
 |  | 
 |     COMPREPLY=( ${COMPREPLY[@]:-} $(compgen -W "${packages}" -- "${cur}") ) | 
 | } | 
 |  | 
 | _adb_shell_file_command() { | 
 |     local serial i cur file options | 
 |     local -a args | 
 |  | 
 |     serial=$1 | 
 |     i=$2 | 
 |     if [ "$serial" != "none" ]; then | 
 |         args=(-s $serial) | 
 |     fi | 
 |     options=$3 | 
 |  | 
 |     where=OPTIONS | 
 |     for ((; i <= COMP_CWORD; i++)); do | 
 |         cur="${COMP_WORDS[i]}" | 
 |         case "${cur}" in | 
 |             -*) | 
 |                 where=OPTIONS | 
 |                 ;; | 
 |             *) | 
 |                 where=FILE | 
 |                 break | 
 |                 ;; | 
 |         esac | 
 |     done | 
 |  | 
 |     file="${COMP_WORDS[COMP_CWORD]}" | 
 |     if [[ ${file} == "" ]]; then | 
 |         file="/" | 
 |     fi | 
 |  | 
 |     case $where in | 
 |         OPTIONS) | 
 |             unset IFS | 
 |             COMPREPLY=( $(compgen -W "$options" -- "$cur") ) | 
 |             ;; | 
 |         FILE) | 
 |             _adb_util_list_files $serial "$file" | 
 |             ;; | 
 |     esac | 
 |  | 
 |     return 0 | 
 | } | 
 |  | 
 | _adb_util_list_files() { | 
 |     local serial dir IFS=$'\n' | 
 |     local -a toks | 
 |     local -a args | 
 |  | 
 |     serial="$1" | 
 |     file="$2" | 
 |  | 
 |     if [ "$serial" != "none" ]; then | 
 |         args=(-s $serial) | 
 |     fi | 
 |  | 
 |     if [[ $( command adb ${args[@]} shell ls -dF / '2>/dev/null' | tr -d '\r' ) == "d /" ]] ; then | 
 |         toks=( ${toks[@]-} $( | 
 |             command adb ${args[@]} shell ls -dF ${file}"*" '2>' /dev/null 2> /dev/null | tr -d '\r' | { | 
 |                 while read -r tmp; do | 
 |                     filetype=${tmp%% *} | 
 |                     filename=${tmp:${#filetype}+1} | 
 |                     if [[ ${filetype:${#filetype}-1:1} == d ]]; then | 
 |                         printf '%s/\n' "$filename" | 
 |                     else | 
 |                         printf '%s\n' "$filename" | 
 |                     fi | 
 |                 done | 
 |             } | 
 |         )) | 
 |     else | 
 |         toks=( ${toks[@]-} $( | 
 |             command adb ${args[@]} shell ls -dp ${file}"*" '2>/dev/null' 2> /dev/null | tr -d '\r' | 
 |         )) | 
 |     fi | 
 |  | 
 |     # Since we're probably doing file completion here, don't add a space after. | 
 |     if [[ $(check_type compopt) == "builtin" ]]; then | 
 |         compopt -o nospace | 
 |     fi | 
 |  | 
 |     COMPREPLY=( ${COMPREPLY[@]:-} "${toks[@]}" ) | 
 | } | 
 |  | 
 | _adb_util_complete_local_file() | 
 | { | 
 |     local file xspec i j IFS=$'\n' | 
 |     local -a dirs files | 
 |  | 
 |     file=$1 | 
 |     xspec=$2 | 
 |  | 
 |     # Since we're probably doing file completion here, don't add a space after. | 
 |     if [[ $(check_type compopt) == "builtin" ]]; then | 
 |         compopt -o plusdirs | 
 |         if [[ "${xspec}" == "" ]]; then | 
 |             COMPREPLY=( ${COMPREPLY[@]:-} $(compgen -f -- "${cur}") ) | 
 |         else | 
 |             compopt +o filenames | 
 |             COMPREPLY=( ${COMPREPLY[@]:-} $(compgen -f -X "${xspec}" -- "${cur}") ) | 
 |         fi | 
 |     else | 
 |         # Work-around for shells with no compopt | 
 |  | 
 |         dirs=( $(compgen -d -- "${cur}" ) ) | 
 |  | 
 |         if [[ "${xspec}" == "" ]]; then | 
 |             files=( ${COMPREPLY[@]:-} $(compgen -f -- "${cur}") ) | 
 |         else | 
 |             files=( ${COMPREPLY[@]:-} $(compgen -f -X "${xspec}" -- "${cur}") ) | 
 |         fi | 
 |  | 
 |         COMPREPLY=( $( | 
 |             for i in "${files[@]}"; do | 
 |                 local skip= | 
 |                 for j in "${dirs[@]}"; do | 
 |                     if [[ $i == $j ]]; then | 
 |                         skip=1 | 
 |                         break | 
 |                     fi | 
 |                 done | 
 |                 [[ -n $skip ]] || printf "%s\n" "$i" | 
 |             done | 
 |         )) | 
 |  | 
 |         COMPREPLY=( ${COMPREPLY[@]:-} $( | 
 |             for i in "${dirs[@]}"; do | 
 |                 printf "%s/\n" "$i" | 
 |             done | 
 |         )) | 
 |     fi | 
 | } | 
 |  | 
 |  | 
 | if [[ $(check_type compopt) == "builtin" ]]; then | 
 |     complete -F _adb adb | 
 | else | 
 |     complete -o nospace -F _adb adb | 
 | fi |