1function hmm() {
2cat <<EOF
3
4Run "m help" for help with the build system itself.
5
6Invoke ". build/envsetup.sh" from your shell to add the following functions to your environment:
7- lunch:      lunch <product_name>-<build_variant>
8              Selects <product_name> as the product to build, and <build_variant> as the variant to
9              build, and stores those selections in the environment to be read by subsequent
10              invocations of 'm' etc.
11- tapas:      tapas [<App1> <App2> ...] [arm|x86|mips|arm64|x86_64|mips64] [eng|userdebug|user]
12- croot:      Changes directory to the top of the tree, or a subdirectory thereof.
13- m:          Makes from the top of the tree.
14- mm:         Builds all of the modules in the current directory, but not their dependencies.
15- mmm:        Builds all of the modules in the supplied directories, but not their dependencies.
16              To limit the modules being built use the syntax: mmm dir/:target1,target2.
17- mma:        Builds all of the modules in the current directory, and their dependencies.
18- mmma:       Builds all of the modules in the supplied directories, and their dependencies.
19- provision:  Flash device with all required partitions. Options will be passed on to fastboot.
20- cgrep:      Greps on all local C/C++ files.
21- ggrep:      Greps on all local Gradle files.
22- jgrep:      Greps on all local Java files.
23- resgrep:    Greps on all local res/*.xml files.
24- mangrep:    Greps on all local AndroidManifest.xml files.
25- mgrep:      Greps on all local Makefiles files.
26- sepgrep:    Greps on all local sepolicy files.
27- sgrep:      Greps on all local source files.
28- godir:      Go to the directory containing a file.
29- allmod:     List all modules.
30- gomod:      Go to the directory containing a module.
31- pathmod:    Get the directory containing a module.
32- refreshmod: Refresh list of modules for allmod/gomod.
33
34Environment options:
35- SANITIZE_HOST: Set to 'true' to use ASAN for all host modules. Note that
36                 ASAN_OPTIONS=detect_leaks=0 will be set by default until the
37                 build is leak-check clean.
38- ANDROID_QUIET_BUILD: set to 'true' to display only the essential messages.
39
40Look at the source to view more functions. The complete list is:
41EOF
42    local T=$(gettop)
43    local A=""
44    local i
45    for i in `cat $T/build/envsetup.sh | sed -n "/^[[:blank:]]*function /s/function \([a-z_]*\).*/\1/p" | sort | uniq`; do
46      A="$A $i"
47    done
48    echo $A
49}
50
51# Get all the build variables needed by this script in a single call to the build system.
52function build_build_var_cache()
53{
54    local T=$(gettop)
55    # Grep out the variable names from the script.
56    cached_vars=(`cat $T/build/envsetup.sh | tr '()' '  ' | awk '{for(i=1;i<=NF;i++) if($i~/get_build_var/) print $(i+1)}' | sort -u | tr '\n' ' '`)
57    cached_abs_vars=(`cat $T/build/envsetup.sh | tr '()' '  ' | awk '{for(i=1;i<=NF;i++) if($i~/get_abs_build_var/) print $(i+1)}' | sort -u | tr '\n' ' '`)
58    # Call the build system to dump the "<val>=<value>" pairs as a shell script.
59    build_dicts_script=`\builtin cd $T; build/soong/soong_ui.bash --dumpvars-mode \
60                        --vars="${cached_vars[*]}" \
61                        --abs-vars="${cached_abs_vars[*]}" \
62                        --var-prefix=var_cache_ \
63                        --abs-var-prefix=abs_var_cache_`
64    local ret=$?
65    if [ $ret -ne 0 ]
66    then
67        unset build_dicts_script
68        return $ret
69    fi
70    # Execute the script to store the "<val>=<value>" pairs as shell variables.
71    eval "$build_dicts_script"
72    ret=$?
73    unset build_dicts_script
74    if [ $ret -ne 0 ]
75    then
76        return $ret
77    fi
78    BUILD_VAR_CACHE_READY="true"
79}
80
81# Delete the build var cache, so that we can still call into the build system
82# to get build variables not listed in this script.
83function destroy_build_var_cache()
84{
85    unset BUILD_VAR_CACHE_READY
86    local v
87    for v in $cached_vars; do
88      unset var_cache_$v
89    done
90    unset cached_vars
91    for v in $cached_abs_vars; do
92      unset abs_var_cache_$v
93    done
94    unset cached_abs_vars
95}
96
97# Get the value of a build variable as an absolute path.
98function get_abs_build_var()
99{
100    if [ "$BUILD_VAR_CACHE_READY" = "true" ]
101    then
102        eval "echo \"\${abs_var_cache_$1}\""
103    return
104    fi
105
106    local T=$(gettop)
107    if [ ! "$T" ]; then
108        echo "Couldn't locate the top of the tree.  Try setting TOP." >&2
109        return
110    fi
111    (\cd $T; build/soong/soong_ui.bash --dumpvar-mode --abs $1)
112}
113
114# Get the exact value of a build variable.
115function get_build_var()
116{
117    if [ "$BUILD_VAR_CACHE_READY" = "true" ]
118    then
119        eval "echo \"\${var_cache_$1}\""
120    return
121    fi
122
123    local T=$(gettop)
124    if [ ! "$T" ]; then
125        echo "Couldn't locate the top of the tree.  Try setting TOP." >&2
126        return
127    fi
128    (\cd $T; build/soong/soong_ui.bash --dumpvar-mode $1)
129}
130
131# check to see if the supplied product is one we can build
132function check_product()
133{
134    local T=$(gettop)
135    if [ ! "$T" ]; then
136        echo "Couldn't locate the top of the tree.  Try setting TOP." >&2
137        return
138    fi
139        TARGET_PRODUCT=$1 \
140        TARGET_BUILD_VARIANT= \
141        TARGET_BUILD_TYPE= \
142        TARGET_BUILD_APPS= \
143        get_build_var TARGET_DEVICE > /dev/null
144    # hide successful answers, but allow the errors to show
145}
146
147VARIANT_CHOICES=(user userdebug eng)
148
149# check to see if the supplied variant is valid
150function check_variant()
151{
152    local v
153    for v in ${VARIANT_CHOICES[@]}
154    do
155        if [ "$v" = "$1" ]
156        then
157            return 0
158        fi
159    done
160    return 1
161}
162
163function setpaths()
164{
165    local T=$(gettop)
166    if [ ! "$T" ]; then
167        echo "Couldn't locate the top of the tree.  Try setting TOP."
168        return
169    fi
170
171    ##################################################################
172    #                                                                #
173    #              Read me before you modify this code               #
174    #                                                                #
175    #   This function sets ANDROID_BUILD_PATHS to what it is adding  #
176    #   to PATH, and the next time it is run, it removes that from   #
177    #   PATH.  This is required so lunch can be run more than once   #
178    #   and still have working paths.                                #
179    #                                                                #
180    ##################################################################
181
182    # Note: on windows/cygwin, ANDROID_BUILD_PATHS will contain spaces
183    # due to "C:\Program Files" being in the path.
184
185    # out with the old
186    if [ -n "$ANDROID_BUILD_PATHS" ] ; then
187        export PATH=${PATH/$ANDROID_BUILD_PATHS/}
188    fi
189    if [ -n "$ANDROID_PRE_BUILD_PATHS" ] ; then
190        export PATH=${PATH/$ANDROID_PRE_BUILD_PATHS/}
191        # strip leading ':', if any
192        export PATH=${PATH/:%/}
193    fi
194
195    # and in with the new
196    local prebuiltdir=$(getprebuilt)
197    local gccprebuiltdir=$(get_abs_build_var ANDROID_GCC_PREBUILTS)
198
199    # defined in core/config.mk
200    local targetgccversion=$(get_build_var TARGET_GCC_VERSION)
201    local targetgccversion2=$(get_build_var 2ND_TARGET_GCC_VERSION)
202    export TARGET_GCC_VERSION=$targetgccversion
203
204    # The gcc toolchain does not exists for windows/cygwin. In this case, do not reference it.
205    export ANDROID_TOOLCHAIN=
206    export ANDROID_TOOLCHAIN_2ND_ARCH=
207    local ARCH=$(get_build_var TARGET_ARCH)
208    local toolchaindir toolchaindir2=
209    case $ARCH in
210        x86) toolchaindir=x86/x86_64-linux-android-$targetgccversion/bin
211            ;;
212        x86_64) toolchaindir=x86/x86_64-linux-android-$targetgccversion/bin
213            ;;
214        arm) toolchaindir=arm/arm-linux-androideabi-$targetgccversion/bin
215            ;;
216        arm64) toolchaindir=aarch64/aarch64-linux-android-$targetgccversion/bin;
217               toolchaindir2=arm/arm-linux-androideabi-$targetgccversion2/bin
218            ;;
219        mips|mips64) toolchaindir=mips/mips64el-linux-android-$targetgccversion/bin
220            ;;
221        *)
222            echo "Can't find toolchain for unknown architecture: $ARCH"
223            toolchaindir=xxxxxxxxx
224            ;;
225    esac
226    if [ -d "$gccprebuiltdir/$toolchaindir" ]; then
227        export ANDROID_TOOLCHAIN=$gccprebuiltdir/$toolchaindir
228    fi
229
230    if [ "$toolchaindir2" -a -d "$gccprebuiltdir/$toolchaindir2" ]; then
231        export ANDROID_TOOLCHAIN_2ND_ARCH=$gccprebuiltdir/$toolchaindir2
232    fi
233
234    export ANDROID_DEV_SCRIPTS=$T/development/scripts:$T/prebuilts/devtools/tools:$T/external/selinux/prebuilts/bin
235
236    # add kernel specific binaries
237    case $(uname -s) in
238        Linux)
239            export ANDROID_DEV_SCRIPTS=$ANDROID_DEV_SCRIPTS:$T/prebuilts/misc/linux-x86/dtc:$T/prebuilts/misc/linux-x86/libufdt
240            ;;
241        *)
242            ;;
243    esac
244
245    ANDROID_BUILD_PATHS=$(get_build_var ANDROID_BUILD_PATHS):$ANDROID_TOOLCHAIN
246    if [ -n "$ANDROID_TOOLCHAIN_2ND_ARCH" ]; then
247        ANDROID_BUILD_PATHS=$ANDROID_BUILD_PATHS:$ANDROID_TOOLCHAIN_2ND_ARCH
248    fi
249    ANDROID_BUILD_PATHS=$ANDROID_BUILD_PATHS:$ANDROID_DEV_SCRIPTS:
250    export ANDROID_BUILD_PATHS
251
252    # If prebuilts/android-emulator/<system>/ exists, prepend it to our PATH
253    # to ensure that the corresponding 'emulator' binaries are used.
254    case $(uname -s) in
255        Darwin)
256            ANDROID_EMULATOR_PREBUILTS=$T/prebuilts/android-emulator/darwin-x86_64
257            ;;
258        Linux)
259            ANDROID_EMULATOR_PREBUILTS=$T/prebuilts/android-emulator/linux-x86_64
260            ;;
261        *)
262            ANDROID_EMULATOR_PREBUILTS=
263            ;;
264    esac
265    if [ -n "$ANDROID_EMULATOR_PREBUILTS" -a -d "$ANDROID_EMULATOR_PREBUILTS" ]; then
266        ANDROID_BUILD_PATHS=$ANDROID_BUILD_PATHS$ANDROID_EMULATOR_PREBUILTS:
267        export ANDROID_EMULATOR_PREBUILTS
268    fi
269
270    # Append asuite prebuilts path to ANDROID_BUILD_PATHS.
271    local os_arch=$(get_build_var HOST_PREBUILT_TAG)
272    local ACLOUD_PATH="$T/prebuilts/asuite/acloud/$os_arch:"
273    local AIDEGEN_PATH="$T/prebuilts/asuite/aidegen/$os_arch:"
274    local ATEST_PATH="$T/prebuilts/asuite/atest/$os_arch:"
275    export ANDROID_BUILD_PATHS=$ANDROID_BUILD_PATHS$ACLOUD_PATH$AIDEGEN_PATH$ATEST_PATH
276
277    export PATH=$ANDROID_BUILD_PATHS$PATH
278
279    # out with the duplicate old
280    if [ -n $ANDROID_PYTHONPATH ]; then
281        export PYTHONPATH=${PYTHONPATH//$ANDROID_PYTHONPATH/}
282    fi
283    # and in with the new
284    export ANDROID_PYTHONPATH=$T/development/python-packages:
285    export PYTHONPATH=$ANDROID_PYTHONPATH$PYTHONPATH
286
287    export ANDROID_JAVA_HOME=$(get_abs_build_var ANDROID_JAVA_HOME)
288    export JAVA_HOME=$ANDROID_JAVA_HOME
289    export ANDROID_JAVA_TOOLCHAIN=$(get_abs_build_var ANDROID_JAVA_TOOLCHAIN)
290    export ANDROID_PRE_BUILD_PATHS=$ANDROID_JAVA_TOOLCHAIN:
291    export PATH=$ANDROID_PRE_BUILD_PATHS$PATH
292
293    unset ANDROID_PRODUCT_OUT
294    export ANDROID_PRODUCT_OUT=$(get_abs_build_var PRODUCT_OUT)
295    export OUT=$ANDROID_PRODUCT_OUT
296
297    unset ANDROID_HOST_OUT
298    export ANDROID_HOST_OUT=$(get_abs_build_var HOST_OUT)
299
300    unset ANDROID_HOST_OUT_TESTCASES
301    export ANDROID_HOST_OUT_TESTCASES=$(get_abs_build_var HOST_OUT_TESTCASES)
302
303    unset ANDROID_TARGET_OUT_TESTCASES
304    export ANDROID_TARGET_OUT_TESTCASES=$(get_abs_build_var TARGET_OUT_TESTCASES)
305
306    # needed for building linux on MacOS
307    # TODO: fix the path
308    #export HOST_EXTRACFLAGS="-I "$T/system/kernel_headers/host_include
309}
310
311function printconfig()
312{
313    local T=$(gettop)
314    if [ ! "$T" ]; then
315        echo "Couldn't locate the top of the tree.  Try setting TOP." >&2
316        return
317    fi
318    get_build_var report_config
319}
320
321function set_stuff_for_environment()
322{
323    setpaths
324    set_sequence_number
325
326    export ANDROID_BUILD_TOP=$(gettop)
327    # With this environment variable new GCC can apply colors to warnings/errors
328    export GCC_COLORS='error=01;31:warning=01;35:note=01;36:caret=01;32:locus=01:quote=01'
329    export ASAN_OPTIONS=detect_leaks=0
330}
331
332function set_sequence_number()
333{
334    export BUILD_ENV_SEQUENCE_NUMBER=13
335}
336
337# Takes a command name, and check if it's in ENVSETUP_NO_COMPLETION or not.
338function should_add_completion() {
339    local cmd="$(basename $1| sed 's/_completion//' |sed 's/\.\(.*\)*sh$//')"
340    case :"$ENVSETUP_NO_COMPLETION": in
341        *:"$cmd":*)
342            return 1
343            ;;
344    esac
345    return 0
346}
347
348function addcompletions()
349{
350    local T dir f
351
352    # Keep us from trying to run in something that's neither bash nor zsh.
353    if [ -z "$BASH_VERSION" -a -z "$ZSH_VERSION" ]; then
354        return
355    fi
356
357    # Keep us from trying to run in bash that's too old.
358    if [ -n "$BASH_VERSION" -a ${BASH_VERSINFO[0]} -lt 3 ]; then
359        return
360    fi
361
362    local completion_files=(
363      system/core/adb/adb.bash
364      system/core/fastboot/fastboot.bash
365      tools/asuite/asuite.sh
366    )
367    # Completion can be disabled selectively to allow users to use non-standard completion.
368    # e.g.
369    # ENVSETUP_NO_COMPLETION=adb # -> disable adb completion
370    # ENVSETUP_NO_COMPLETION=adb:bit # -> disable adb and bit completion
371    for f in ${completion_files[*]}; do
372        if [ -f "$f" ] && should_add_completion "$f"; then
373            . $f
374        fi
375    done
376
377    if should_add_completion bit ; then
378        complete -C "bit --tab" bit
379    fi
380    if [ -z "$ZSH_VERSION" ]; then
381        # Doesn't work in zsh.
382        complete -o nospace -F _croot croot
383    fi
384    complete -F _lunch lunch
385
386    complete -F _complete_android_module_names gomod
387    complete -F _complete_android_module_names m
388}
389
390function choosetype()
391{
392    echo "Build type choices are:"
393    echo "     1. release"
394    echo "     2. debug"
395    echo
396
397    local DEFAULT_NUM DEFAULT_VALUE
398    DEFAULT_NUM=1
399    DEFAULT_VALUE=release
400
401    export TARGET_BUILD_TYPE=
402    local ANSWER
403    while [ -z $TARGET_BUILD_TYPE ]
404    do
405        echo -n "Which would you like? ["$DEFAULT_NUM"] "
406        if [ -z "$1" ] ; then
407            read ANSWER
408        else
409            echo $1
410            ANSWER=$1
411        fi
412        case $ANSWER in
413        "")
414            export TARGET_BUILD_TYPE=$DEFAULT_VALUE
415            ;;
416        1)
417            export TARGET_BUILD_TYPE=release
418            ;;
419        release)
420            export TARGET_BUILD_TYPE=release
421            ;;
422        2)
423            export TARGET_BUILD_TYPE=debug
424            ;;
425        debug)
426            export TARGET_BUILD_TYPE=debug
427            ;;
428        *)
429            echo
430            echo "I didn't understand your response.  Please try again."
431            echo
432            ;;
433        esac
434        if [ -n "$1" ] ; then
435            break
436        fi
437    done
438
439    build_build_var_cache
440    set_stuff_for_environment
441    destroy_build_var_cache
442}
443
444#
445# This function isn't really right:  It chooses a TARGET_PRODUCT
446# based on the list of boards.  Usually, that gets you something
447# that kinda works with a generic product, but really, you should
448# pick a product by name.
449#
450function chooseproduct()
451{
452    local default_value
453    if [ "x$TARGET_PRODUCT" != x ] ; then
454        default_value=$TARGET_PRODUCT
455    else
456        default_value=aosp_arm
457    fi
458
459    export TARGET_BUILD_APPS=
460    export TARGET_PRODUCT=
461    local ANSWER
462    while [ -z "$TARGET_PRODUCT" ]
463    do
464        echo -n "Which product would you like? [$default_value] "
465        if [ -z "$1" ] ; then
466            read ANSWER
467        else
468            echo $1
469            ANSWER=$1
470        fi
471
472        if [ -z "$ANSWER" ] ; then
473            export TARGET_PRODUCT=$default_value
474        else
475            if check_product $ANSWER
476            then
477                export TARGET_PRODUCT=$ANSWER
478            else
479                echo "** Not a valid product: $ANSWER"
480            fi
481        fi
482        if [ -n "$1" ] ; then
483            break
484        fi
485    done
486
487    build_build_var_cache
488    set_stuff_for_environment
489    destroy_build_var_cache
490}
491
492function choosevariant()
493{
494    echo "Variant choices are:"
495    local index=1
496    local v
497    for v in ${VARIANT_CHOICES[@]}
498    do
499        # The product name is the name of the directory containing
500        # the makefile we found, above.
501        echo "     $index. $v"
502        index=$(($index+1))
503    done
504
505    local default_value=eng
506    local ANSWER
507
508    export TARGET_BUILD_VARIANT=
509    while [ -z "$TARGET_BUILD_VARIANT" ]
510    do
511        echo -n "Which would you like? [$default_value] "
512        if [ -z "$1" ] ; then
513            read ANSWER
514        else
515            echo $1
516            ANSWER=$1
517        fi
518
519        if [ -z "$ANSWER" ] ; then
520            export TARGET_BUILD_VARIANT=$default_value
521        elif (echo -n $ANSWER | grep -q -e "^[0-9][0-9]*$") ; then
522            if [ "$ANSWER" -le "${#VARIANT_CHOICES[@]}" ] ; then
523                export TARGET_BUILD_VARIANT=${VARIANT_CHOICES[$(($ANSWER-1))]}
524            fi
525        else
526            if check_variant $ANSWER
527            then
528                export TARGET_BUILD_VARIANT=$ANSWER
529            else
530                echo "** Not a valid variant: $ANSWER"
531            fi
532        fi
533        if [ -n "$1" ] ; then
534            break
535        fi
536    done
537}
538
539function choosecombo()
540{
541    choosetype $1
542
543    echo
544    echo
545    chooseproduct $2
546
547    echo
548    echo
549    choosevariant $3
550
551    echo
552    build_build_var_cache
553    set_stuff_for_environment
554    printconfig
555    destroy_build_var_cache
556}
557
558function add_lunch_combo()
559{
560    if [ -n "$ZSH_VERSION" ]; then
561        echo -n "${funcfiletrace[1]}: "
562    else
563        echo -n "${BASH_SOURCE[1]}:${BASH_LINENO[0]}: "
564    fi
565    echo "add_lunch_combo is obsolete. Use COMMON_LUNCH_CHOICES in your AndroidProducts.mk instead."
566}
567
568function print_lunch_menu()
569{
570    local uname=$(uname)
571    echo
572    echo "You're building on" $uname
573    echo
574    echo "Lunch menu... pick a combo:"
575
576    local i=1
577    local choice
578    for choice in $(TARGET_BUILD_APPS= get_build_var COMMON_LUNCH_CHOICES)
579    do
580        echo "     $i. $choice"
581        i=$(($i+1))
582    done
583
584    echo
585}
586
587function lunch()
588{
589    local answer
590
591    if [ "$1" ] ; then
592        answer=$1
593    else
594        print_lunch_menu
595        echo -n "Which would you like? [aosp_arm-eng] "
596        read answer
597    fi
598
599    local selection=
600
601    if [ -z "$answer" ]
602    then
603        selection=aosp_arm-eng
604    elif (echo -n $answer | grep -q -e "^[0-9][0-9]*$")
605    then
606        local choices=($(TARGET_BUILD_APPS= get_build_var COMMON_LUNCH_CHOICES))
607        if [ $answer -le ${#choices[@]} ]
608        then
609            # array in zsh starts from 1 instead of 0.
610            if [ -n "$ZSH_VERSION" ]
611            then
612                selection=${choices[$(($answer))]}
613            else
614                selection=${choices[$(($answer-1))]}
615            fi
616        fi
617    else
618        selection=$answer
619    fi
620
621    export TARGET_BUILD_APPS=
622
623    local product variant_and_version variant version
624
625    product=${selection%%-*} # Trim everything after first dash
626    variant_and_version=${selection#*-} # Trim everything up to first dash
627    if [ "$variant_and_version" != "$selection" ]; then
628        variant=${variant_and_version%%-*}
629        if [ "$variant" != "$variant_and_version" ]; then
630            version=${variant_and_version#*-}
631        fi
632    fi
633
634    if [ -z "$product" ]
635    then
636        echo
637        echo "Invalid lunch combo: $selection"
638        return 1
639    fi
640
641    TARGET_PRODUCT=$product \
642    TARGET_BUILD_VARIANT=$variant \
643    TARGET_PLATFORM_VERSION=$version \
644    build_build_var_cache
645    if [ $? -ne 0 ]
646    then
647        return 1
648    fi
649
650    export TARGET_PRODUCT=$(get_build_var TARGET_PRODUCT)
651    export TARGET_BUILD_VARIANT=$(get_build_var TARGET_BUILD_VARIANT)
652    if [ -n "$version" ]; then
653      export TARGET_PLATFORM_VERSION=$(get_build_var TARGET_PLATFORM_VERSION)
654    else
655      unset TARGET_PLATFORM_VERSION
656    fi
657    export TARGET_BUILD_TYPE=release
658
659    echo
660
661    set_stuff_for_environment
662    printconfig
663    destroy_build_var_cache
664}
665
666unset COMMON_LUNCH_CHOICES_CACHE
667# Tab completion for lunch.
668function _lunch()
669{
670    local cur prev opts
671    COMPREPLY=()
672    cur="${COMP_WORDS[COMP_CWORD]}"
673    prev="${COMP_WORDS[COMP_CWORD-1]}"
674
675    if [ -z "$COMMON_LUNCH_CHOICES_CACHE" ]; then
676        COMMON_LUNCH_CHOICES_CACHE=$(TARGET_BUILD_APPS= get_build_var COMMON_LUNCH_CHOICES)
677    fi
678
679    COMPREPLY=( $(compgen -W "${COMMON_LUNCH_CHOICES_CACHE}" -- ${cur}) )
680    return 0
681}
682
683# Configures the build to build unbundled apps.
684# Run tapas with one or more app names (from LOCAL_PACKAGE_NAME)
685function tapas()
686{
687    local showHelp="$(echo $* | xargs -n 1 echo | \grep -E '^(help)$' | xargs)"
688    local arch="$(echo $* | xargs -n 1 echo | \grep -E '^(arm|x86|mips|arm64|x86_64|mips64)$' | xargs)"
689    local variant="$(echo $* | xargs -n 1 echo | \grep -E '^(user|userdebug|eng)$' | xargs)"
690    local density="$(echo $* | xargs -n 1 echo | \grep -E '^(ldpi|mdpi|tvdpi|hdpi|xhdpi|xxhdpi|xxxhdpi|alldpi)$' | xargs)"
691    local apps="$(echo $* | xargs -n 1 echo | \grep -E -v '^(user|userdebug|eng|arm|x86|mips|arm64|x86_64|mips64|ldpi|mdpi|tvdpi|hdpi|xhdpi|xxhdpi|xxxhdpi|alldpi)$' | xargs)"
692
693    if [ "$showHelp" != "" ]; then
694      $(gettop)/build/make/tapasHelp.sh
695      return
696    fi
697
698    if [ $(echo $arch | wc -w) -gt 1 ]; then
699        echo "tapas: Error: Multiple build archs supplied: $arch"
700        return
701    fi
702    if [ $(echo $variant | wc -w) -gt 1 ]; then
703        echo "tapas: Error: Multiple build variants supplied: $variant"
704        return
705    fi
706    if [ $(echo $density | wc -w) -gt 1 ]; then
707        echo "tapas: Error: Multiple densities supplied: $density"
708        return
709    fi
710
711    local product=aosp_arm
712    case $arch in
713      x86)    product=aosp_x86;;
714      mips)   product=aosp_mips;;
715      arm64)  product=aosp_arm64;;
716      x86_64) product=aosp_x86_64;;
717      mips64)  product=aosp_mips64;;
718    esac
719    if [ -z "$variant" ]; then
720        variant=eng
721    fi
722    if [ -z "$apps" ]; then
723        apps=all
724    fi
725    if [ -z "$density" ]; then
726        density=alldpi
727    fi
728
729    export TARGET_PRODUCT=$product
730    export TARGET_BUILD_VARIANT=$variant
731    export TARGET_BUILD_DENSITY=$density
732    export TARGET_BUILD_TYPE=release
733    export TARGET_BUILD_APPS=$apps
734
735    build_build_var_cache
736    set_stuff_for_environment
737    printconfig
738    destroy_build_var_cache
739}
740
741function gettop
742{
743    local TOPFILE=build/make/core/envsetup.mk
744    if [ -n "$TOP" -a -f "$TOP/$TOPFILE" ] ; then
745        # The following circumlocution ensures we remove symlinks from TOP.
746        (cd $TOP; PWD= /bin/pwd)
747    else
748        if [ -f $TOPFILE ] ; then
749            # The following circumlocution (repeated below as well) ensures
750            # that we record the true directory name and not one that is
751            # faked up with symlink names.
752            PWD= /bin/pwd
753        else
754            local HERE=$PWD
755            local T=
756            while [ \( ! \( -f $TOPFILE \) \) -a \( $PWD != "/" \) ]; do
757                \cd ..
758                T=`PWD= /bin/pwd -P`
759            done
760            \cd $HERE
761            if [ -f "$T/$TOPFILE" ]; then
762                echo $T
763            fi
764        fi
765    fi
766}
767
768function m()
769{
770    local T=$(gettop)
771    if [ "$T" ]; then
772        _wrap_build $T/build/soong/soong_ui.bash --make-mode $@
773    else
774        echo "Couldn't locate the top of the tree.  Try setting TOP."
775        return 1
776    fi
777}
778
779function findmakefile()
780{
781    local TOPFILE=build/make/core/envsetup.mk
782    local HERE=$PWD
783    if [ "$1" ]; then
784        \cd $1
785    fi;
786    local T=
787    while [ \( ! \( -f $TOPFILE \) \) -a \( $PWD != "/" \) ]; do
788        T=`PWD= /bin/pwd`
789        if [ -f "$T/Android.mk" -o -f "$T/Android.bp" ]; then
790            echo $T/Android.mk
791            \cd $HERE
792            return
793        fi
794        \cd ..
795    done
796    \cd $HERE
797    return 1
798}
799
800function mm()
801{
802    local T=$(gettop)
803    # If we're sitting in the root of the build tree, just do a
804    # normal build.
805    if [ -f build/soong/soong_ui.bash ]; then
806        _wrap_build $T/build/soong/soong_ui.bash --make-mode $@
807    else
808        # Find the closest Android.mk file.
809        local M=$(findmakefile)
810        local MODULES=
811        local GET_INSTALL_PATH=
812        local ARGS=
813        # Remove the path to top as the makefilepath needs to be relative
814        local M=`echo $M|sed 's:'$T'/::'`
815        if [ ! "$T" ]; then
816            echo "Couldn't locate the top of the tree.  Try setting TOP."
817            return 1
818        elif [ ! "$M" ]; then
819            echo "Couldn't locate a makefile from the current directory."
820            return 1
821        else
822            local ARG
823            for ARG in $@; do
824                case $ARG in
825                  GET-INSTALL-PATH) GET_INSTALL_PATH=$ARG;;
826                esac
827            done
828            if [ -n "$GET_INSTALL_PATH" ]; then
829              MODULES=
830              ARGS=GET-INSTALL-PATH-IN-$(dirname ${M})
831              ARGS=${ARGS//\//-}
832            else
833              MODULES=MODULES-IN-$(dirname ${M})
834              # Convert "/" to "-".
835              MODULES=${MODULES//\//-}
836              ARGS=$@
837            fi
838            if [ "1" = "${WITH_TIDY_ONLY}" -o "true" = "${WITH_TIDY_ONLY}" ]; then
839              MODULES=tidy_only
840            fi
841            ONE_SHOT_MAKEFILE=$M _wrap_build $T/build/soong/soong_ui.bash --make-mode $MODULES $ARGS
842        fi
843    fi
844}
845
846function mmm()
847{
848    local T=$(gettop)
849    if [ "$T" ]; then
850        local MAKEFILE=
851        local MODULES=
852        local MODULES_IN_PATHS=
853        local ARGS=
854        local DIR TO_CHOP
855        local DIR_MODULES
856        local GET_INSTALL_PATH=
857        local GET_INSTALL_PATHS=
858        local DASH_ARGS=$(echo "$@" | awk -v RS=" " -v ORS=" " '/^-.*$/')
859        local DIRS=$(echo "$@" | awk -v RS=" " -v ORS=" " '/^[^-].*$/')
860        for DIR in $DIRS ; do
861            DIR_MODULES=`echo $DIR | sed -n -e 's/.*:\(.*$\)/\1/p' | sed 's/,/ /'`
862            DIR=`echo $DIR | sed -e 's/:.*//' -e 's:/$::'`
863            # Remove the leading ./ and trailing / if any exists.
864            DIR=${DIR#./}
865            DIR=${DIR%/}
866            local M
867            if [ "$DIR_MODULES" = "" ]; then
868                M=$(findmakefile $DIR)
869            else
870                # Only check the target directory if a module is specified.
871                if [ -f $DIR/Android.mk -o -f $DIR/Android.bp ]; then
872                    local HERE=$PWD
873                    cd $DIR
874                    M=`PWD= /bin/pwd`
875                    M=$M/Android.mk
876                    cd $HERE
877                fi
878            fi
879            if [ "$M" ]; then
880                # Remove the path to top as the makefilepath needs to be relative
881                local M=`echo $M|sed 's:'$T'/::'`
882                if [ "$DIR_MODULES" = "" ]; then
883                    MODULES_IN_PATHS="$MODULES_IN_PATHS MODULES-IN-$(dirname ${M})"
884                    GET_INSTALL_PATHS="$GET_INSTALL_PATHS GET-INSTALL-PATH-IN-$(dirname ${M})"
885                else
886                    MODULES="$MODULES $DIR_MODULES"
887                fi
888                MAKEFILE="$MAKEFILE $M"
889            else
890                case $DIR in
891                  showcommands | snod | dist | *=*) ARGS="$ARGS $DIR";;
892                  GET-INSTALL-PATH) GET_INSTALL_PATH=$DIR;;
893                  *) if [ -d $DIR ]; then
894                         echo "No Android.mk in $DIR.";
895                     else
896                         echo "Couldn't locate the directory $DIR";
897                     fi
898                     return 1;;
899                esac
900            fi
901        done
902        if [ -n "$GET_INSTALL_PATH" ]; then
903          ARGS=${GET_INSTALL_PATHS//\//-}
904          MODULES=
905          MODULES_IN_PATHS=
906        fi
907        if [ "1" = "${WITH_TIDY_ONLY}" -o "true" = "${WITH_TIDY_ONLY}" ]; then
908          MODULES=tidy_only
909          MODULES_IN_PATHS=
910        fi
911        # Convert "/" to "-".
912        MODULES_IN_PATHS=${MODULES_IN_PATHS//\//-}
913        ONE_SHOT_MAKEFILE="$MAKEFILE" _wrap_build $T/build/soong/soong_ui.bash --make-mode $DASH_ARGS $MODULES $MODULES_IN_PATHS $ARGS
914    else
915        echo "Couldn't locate the top of the tree.  Try setting TOP."
916        return 1
917    fi
918}
919
920function mma()
921{
922  local T=$(gettop)
923  if [ -f build/soong/soong_ui.bash ]; then
924    _wrap_build $T/build/soong/soong_ui.bash --make-mode $@
925  else
926    if [ ! "$T" ]; then
927      echo "Couldn't locate the top of the tree.  Try setting TOP."
928      return 1
929    fi
930    local M=$(findmakefile || echo $(realpath $PWD)/Android.mk)
931    # Remove the path to top as the makefilepath needs to be relative
932    local M=`echo $M|sed 's:'$T'/::'`
933    local MODULES_IN_PATHS=MODULES-IN-$(dirname ${M})
934    # Convert "/" to "-".
935    MODULES_IN_PATHS=${MODULES_IN_PATHS//\//-}
936    _wrap_build $T/build/soong/soong_ui.bash --make-mode $@ $MODULES_IN_PATHS
937  fi
938}
939
940function mmma()
941{
942  local T=$(gettop)
943  if [ "$T" ]; then
944    local DASH_ARGS=$(echo "$@" | awk -v RS=" " -v ORS=" " '/^-.*$/')
945    local DIRS=$(echo "$@" | awk -v RS=" " -v ORS=" " '/^[^-].*$/')
946    local MY_PWD=`PWD= /bin/pwd`
947    if [ "$MY_PWD" = "$T" ]; then
948      MY_PWD=
949    else
950      MY_PWD=`echo $MY_PWD|sed 's:'$T'/::'`
951    fi
952    local DIR=
953    local MODULES_IN_PATHS=
954    local ARGS=
955    for DIR in $DIRS ; do
956      if [ -d $DIR ]; then
957        # Remove the leading ./ and trailing / if any exists.
958        DIR=${DIR#./}
959        DIR=${DIR%/}
960        if [ "$MY_PWD" != "" ]; then
961          DIR=$MY_PWD/$DIR
962        fi
963        MODULES_IN_PATHS="$MODULES_IN_PATHS MODULES-IN-$DIR"
964      else
965        case $DIR in
966          showcommands | snod | dist | *=*) ARGS="$ARGS $DIR";;
967          *) echo "Couldn't find directory $DIR"; return 1;;
968        esac
969      fi
970    done
971    # Convert "/" to "-".
972    MODULES_IN_PATHS=${MODULES_IN_PATHS//\//-}
973    _wrap_build $T/build/soong/soong_ui.bash --make-mode $DASH_ARGS $ARGS $MODULES_IN_PATHS
974  else
975    echo "Couldn't locate the top of the tree.  Try setting TOP."
976    return 1
977  fi
978}
979
980function croot()
981{
982    local T=$(gettop)
983    if [ "$T" ]; then
984        if [ "$1" ]; then
985            \cd $(gettop)/$1
986        else
987            \cd $(gettop)
988        fi
989    else
990        echo "Couldn't locate the top of the tree.  Try setting TOP."
991    fi
992}
993
994function _croot()
995{
996    local T=$(gettop)
997    if [ "$T" ]; then
998        local cur="${COMP_WORDS[COMP_CWORD]}"
999        k=0
1000        for c in $(compgen -d ${T}/${cur}); do
1001            COMPREPLY[k++]=${c#${T}/}/
1002        done
1003    fi
1004}
1005
1006function cproj()
1007{
1008    local TOPFILE=build/make/core/envsetup.mk
1009    local HERE=$PWD
1010    local T=
1011    while [ \( ! \( -f $TOPFILE \) \) -a \( $PWD != "/" \) ]; do
1012        T=$PWD
1013        if [ -f "$T/Android.mk" ]; then
1014            \cd $T
1015            return
1016        fi
1017        \cd ..
1018    done
1019    \cd $HERE
1020    echo "can't find Android.mk"
1021}
1022
1023# simplified version of ps; output in the form
1024# <pid> <procname>
1025function qpid() {
1026    local prepend=''
1027    local append=''
1028    if [ "$1" = "--exact" ]; then
1029        prepend=' '
1030        append='$'
1031        shift
1032    elif [ "$1" = "--help" -o "$1" = "-h" ]; then
1033        echo "usage: qpid [[--exact] <process name|pid>"
1034        return 255
1035    fi
1036
1037    local EXE="$1"
1038    if [ "$EXE" ] ; then
1039        qpid | \grep "$prepend$EXE$append"
1040    else
1041        adb shell ps \
1042            | tr -d '\r' \
1043            | sed -e 1d -e 's/^[^ ]* *\([0-9]*\).* \([^ ]*\)$/\1 \2/'
1044    fi
1045}
1046
1047# coredump_setup - enable core dumps globally for any process
1048#                  that has the core-file-size limit set correctly
1049#
1050# NOTE: You must call also coredump_enable for a specific process
1051#       if its core-file-size limit is not set already.
1052# NOTE: Core dumps are written to ramdisk; they will not survive a reboot!
1053
1054function coredump_setup()
1055{
1056    echo "Getting root...";
1057    adb root;
1058    adb wait-for-device;
1059
1060    echo "Remounting root partition read-write...";
1061    adb shell mount -w -o remount -t rootfs rootfs;
1062    sleep 1;
1063    adb wait-for-device;
1064    adb shell mkdir -p /cores;
1065    adb shell mount -t tmpfs tmpfs /cores;
1066    adb shell chmod 0777 /cores;
1067
1068    echo "Granting SELinux permission to dump in /cores...";
1069    adb shell restorecon -R /cores;
1070
1071    echo "Set core pattern.";
1072    adb shell 'echo /cores/core.%p > /proc/sys/kernel/core_pattern';
1073
1074    echo "Done."
1075}
1076
1077# coredump_enable - enable core dumps for the specified process
1078# $1 = PID of process (e.g., $(pid mediaserver))
1079#
1080# NOTE: coredump_setup must have been called as well for a core
1081#       dump to actually be generated.
1082
1083function coredump_enable()
1084{
1085    local PID=$1;
1086    if [ -z "$PID" ]; then
1087        printf "Expecting a PID!\n";
1088        return;
1089    fi;
1090    echo "Setting core limit for $PID to infinite...";
1091    adb shell /system/bin/ulimit -p $PID -c unlimited
1092}
1093
1094# core - send SIGV and pull the core for process
1095# $1 = PID of process (e.g., $(pid mediaserver))
1096#
1097# NOTE: coredump_setup must be called once per boot for core dumps to be
1098#       enabled globally.
1099
1100function core()
1101{
1102    local PID=$1;
1103
1104    if [ -z "$PID" ]; then
1105        printf "Expecting a PID!\n";
1106        return;
1107    fi;
1108
1109    local CORENAME=core.$PID;
1110    local COREPATH=/cores/$CORENAME;
1111    local SIG=SEGV;
1112
1113    coredump_enable $1;
1114
1115    local done=0;
1116    while [ $(adb shell "[ -d /proc/$PID ] && echo -n yes") ]; do
1117        printf "\tSending SIG%s to %d...\n" $SIG $PID;
1118        adb shell kill -$SIG $PID;
1119        sleep 1;
1120    done;
1121
1122    adb shell "while [ ! -f $COREPATH ] ; do echo waiting for $COREPATH to be generated; sleep 1; done"
1123    echo "Done: core is under $COREPATH on device.";
1124}
1125
1126# systemstack - dump the current stack trace of all threads in the system process
1127# to the usual ANR traces file
1128function systemstack()
1129{
1130    stacks system_server
1131}
1132
1133# Read the ELF header from /proc/$PID/exe to determine if the process is
1134# 64-bit.
1135function is64bit()
1136{
1137    local PID="$1"
1138    if [ "$PID" ] ; then
1139        if [[ "$(adb shell cat /proc/$PID/exe | xxd -l 1 -s 4 -p)" -eq "02" ]] ; then
1140            echo "64"
1141        else
1142            echo ""
1143        fi
1144    else
1145        echo ""
1146    fi
1147}
1148
1149case `uname -s` in
1150    Darwin)
1151        function sgrep()
1152        {
1153            find -E . -name .repo -prune -o -name .git -prune -o  -type f -iregex '.*\.(c|h|cc|cpp|hpp|S|java|xml|sh|mk|aidl|vts)' \
1154                -exec grep --color -n "$@" {} +
1155        }
1156
1157        ;;
1158    *)
1159        function sgrep()
1160        {
1161            find . -name .repo -prune -o -name .git -prune -o  -type f -iregex '.*\.\(c\|h\|cc\|cpp\|hpp\|S\|java\|xml\|sh\|mk\|aidl\|vts\)' \
1162                -exec grep --color -n "$@" {} +
1163        }
1164        ;;
1165esac
1166
1167function gettargetarch
1168{
1169    get_build_var TARGET_ARCH
1170}
1171
1172function ggrep()
1173{
1174    find . -name .repo -prune -o -name .git -prune -o -name out -prune -o -type f -name "*\.gradle" \
1175        -exec grep --color -n "$@" {} +
1176}
1177
1178function jgrep()
1179{
1180    find . -name .repo -prune -o -name .git -prune -o -name out -prune -o -type f -name "*\.java" \
1181        -exec grep --color -n "$@" {} +
1182}
1183
1184function cgrep()
1185{
1186    find . -name .repo -prune -o -name .git -prune -o -name out -prune -o -type f \( -name '*.c' -o -name '*.cc' -o -name '*.cpp' -o -name '*.h' -o -name '*.hpp' \) \
1187        -exec grep --color -n "$@" {} +
1188}
1189
1190function resgrep()
1191{
1192    local dir
1193    for dir in `find . -name .repo -prune -o -name .git -prune -o -name out -prune -o -name res -type d`; do
1194        find $dir -type f -name '*\.xml' -exec grep --color -n "$@" {} +
1195    done
1196}
1197
1198function mangrep()
1199{
1200    find . -name .repo -prune -o -name .git -prune -o -path ./out -prune -o -type f -name 'AndroidManifest.xml' \
1201        -exec grep --color -n "$@" {} +
1202}
1203
1204function sepgrep()
1205{
1206    find . -name .repo -prune -o -name .git -prune -o -path ./out -prune -o -name sepolicy -type d \
1207        -exec grep --color -n -r --exclude-dir=\.git "$@" {} +
1208}
1209
1210function rcgrep()
1211{
1212    find . -name .repo -prune -o -name .git -prune -o -name out -prune -o -type f -name "*\.rc*" \
1213        -exec grep --color -n "$@" {} +
1214}
1215
1216case `uname -s` in
1217    Darwin)
1218        function mgrep()
1219        {
1220            find -E . -name .repo -prune -o -name .git -prune -o -path ./out -prune -o \( -iregex '.*/(Makefile|Makefile\..*|.*\.make|.*\.mak|.*\.mk|.*\.bp)' -o -regex '(.*/)?soong/[^/]*.go' \) -type f \
1221                -exec grep --color -n "$@" {} +
1222        }
1223
1224        function treegrep()
1225        {
1226            find -E . -name .repo -prune -o -name .git -prune -o -type f -iregex '.*\.(c|h|cpp|hpp|S|java|xml)' \
1227                -exec grep --color -n -i "$@" {} +
1228        }
1229
1230        ;;
1231    *)
1232        function mgrep()
1233        {
1234            find . -name .repo -prune -o -name .git -prune -o -path ./out -prune -o \( -regextype posix-egrep -iregex '(.*\/Makefile|.*\/Makefile\..*|.*\.make|.*\.mak|.*\.mk|.*\.bp)' -o -regextype posix-extended -regex '(.*/)?soong/[^/]*.go' \) -type f \
1235                -exec grep --color -n "$@" {} +
1236        }
1237
1238        function treegrep()
1239        {
1240            find . -name .repo -prune -o -name .git -prune -o -regextype posix-egrep -iregex '.*\.(c|h|cpp|hpp|S|java|xml)' -type f \
1241                -exec grep --color -n -i "$@" {} +
1242        }
1243
1244        ;;
1245esac
1246
1247function getprebuilt
1248{
1249    get_abs_build_var ANDROID_PREBUILTS
1250}
1251
1252function tracedmdump()
1253{
1254    local T=$(gettop)
1255    if [ ! "$T" ]; then
1256        echo "Couldn't locate the top of the tree.  Try setting TOP."
1257        return
1258    fi
1259    local prebuiltdir=$(getprebuilt)
1260    local arch=$(gettargetarch)
1261    local KERNEL=$T/prebuilts/qemu-kernel/$arch/vmlinux-qemu
1262
1263    local TRACE=$1
1264    if [ ! "$TRACE" ] ; then
1265        echo "usage:  tracedmdump  tracename"
1266        return
1267    fi
1268
1269    if [ ! -r "$KERNEL" ] ; then
1270        echo "Error: cannot find kernel: '$KERNEL'"
1271        return
1272    fi
1273
1274    local BASETRACE=$(basename $TRACE)
1275    if [ "$BASETRACE" = "$TRACE" ] ; then
1276        TRACE=$ANDROID_PRODUCT_OUT/traces/$TRACE
1277    fi
1278
1279    echo "post-processing traces..."
1280    rm -f $TRACE/qtrace.dexlist
1281    post_trace $TRACE
1282    if [ $? -ne 0 ]; then
1283        echo "***"
1284        echo "*** Error: malformed trace.  Did you remember to exit the emulator?"
1285        echo "***"
1286        return
1287    fi
1288    echo "generating dexlist output..."
1289    /bin/ls $ANDROID_PRODUCT_OUT/system/framework/*.jar $ANDROID_PRODUCT_OUT/system/app/*.apk $ANDROID_PRODUCT_OUT/data/app/*.apk 2>/dev/null | xargs dexlist > $TRACE/qtrace.dexlist
1290    echo "generating dmtrace data..."
1291    q2dm -r $ANDROID_PRODUCT_OUT/symbols $TRACE $KERNEL $TRACE/dmtrace || return
1292    echo "generating html file..."
1293    dmtracedump -h $TRACE/dmtrace >| $TRACE/dmtrace.html || return
1294    echo "done, see $TRACE/dmtrace.html for details"
1295    echo "or run:"
1296    echo "    traceview $TRACE/dmtrace"
1297}
1298
1299# communicate with a running device or emulator, set up necessary state,
1300# and run the hat command.
1301function runhat()
1302{
1303    # process standard adb options
1304    local adbTarget=""
1305    if [ "$1" = "-d" -o "$1" = "-e" ]; then
1306        adbTarget=$1
1307        shift 1
1308    elif [ "$1" = "-s" ]; then
1309        adbTarget="$1 $2"
1310        shift 2
1311    fi
1312    local adbOptions=${adbTarget}
1313    #echo adbOptions = ${adbOptions}
1314
1315    # runhat options
1316    local targetPid=$1
1317
1318    if [ "$targetPid" = "" ]; then
1319        echo "Usage: runhat [ -d | -e | -s serial ] target-pid"
1320        return
1321    fi
1322
1323    # confirm hat is available
1324    if [ -z $(which hat) ]; then
1325        echo "hat is not available in this configuration."
1326        return
1327    fi
1328
1329    # issue "am" command to cause the hprof dump
1330    local devFile=/data/local/tmp/hprof-$targetPid
1331    echo "Poking $targetPid and waiting for data..."
1332    echo "Storing data at $devFile"
1333    adb ${adbOptions} shell am dumpheap $targetPid $devFile
1334    echo "Press enter when logcat shows \"hprof: heap dump completed\""
1335    echo -n "> "
1336    read
1337
1338    local localFile=/tmp/$$-hprof
1339
1340    echo "Retrieving file $devFile..."
1341    adb ${adbOptions} pull $devFile $localFile
1342
1343    adb ${adbOptions} shell rm $devFile
1344
1345    echo "Running hat on $localFile"
1346    echo "View the output by pointing your browser at http://localhost:7000/"
1347    echo ""
1348    hat -JXmx512m $localFile
1349}
1350
1351function getbugreports()
1352{
1353    local reports=(`adb shell ls /sdcard/bugreports | tr -d '\r'`)
1354
1355    if [ ! "$reports" ]; then
1356        echo "Could not locate any bugreports."
1357        return
1358    fi
1359
1360    local report
1361    for report in ${reports[@]}
1362    do
1363        echo "/sdcard/bugreports/${report}"
1364        adb pull /sdcard/bugreports/${report} ${report}
1365        gunzip ${report}
1366    done
1367}
1368
1369function getsdcardpath()
1370{
1371    adb ${adbOptions} shell echo -n \$\{EXTERNAL_STORAGE\}
1372}
1373
1374function getscreenshotpath()
1375{
1376    echo "$(getsdcardpath)/Pictures/Screenshots"
1377}
1378
1379function getlastscreenshot()
1380{
1381    local screenshot_path=$(getscreenshotpath)
1382    local screenshot=`adb ${adbOptions} ls ${screenshot_path} | grep Screenshot_[0-9-]*.*\.png | sort -rk 3 | cut -d " " -f 4 | head -n 1`
1383    if [ "$screenshot" = "" ]; then
1384        echo "No screenshots found."
1385        return
1386    fi
1387    echo "${screenshot}"
1388    adb ${adbOptions} pull ${screenshot_path}/${screenshot}
1389}
1390
1391function startviewserver()
1392{
1393    local port=4939
1394    if [ $# -gt 0 ]; then
1395            port=$1
1396    fi
1397    adb shell service call window 1 i32 $port
1398}
1399
1400function stopviewserver()
1401{
1402    adb shell service call window 2
1403}
1404
1405function isviewserverstarted()
1406{
1407    adb shell service call window 3
1408}
1409
1410function key_home()
1411{
1412    adb shell input keyevent 3
1413}
1414
1415function key_back()
1416{
1417    adb shell input keyevent 4
1418}
1419
1420function key_menu()
1421{
1422    adb shell input keyevent 82
1423}
1424
1425function smoketest()
1426{
1427    if [ ! "$ANDROID_PRODUCT_OUT" ]; then
1428        echo "Couldn't locate output files.  Try running 'lunch' first." >&2
1429        return
1430    fi
1431    local T=$(gettop)
1432    if [ ! "$T" ]; then
1433        echo "Couldn't locate the top of the tree.  Try setting TOP." >&2
1434        return
1435    fi
1436
1437    (\cd "$T" && mmm tests/SmokeTest) &&
1438      adb uninstall com.android.smoketest > /dev/null &&
1439      adb uninstall com.android.smoketest.tests > /dev/null &&
1440      adb install $ANDROID_PRODUCT_OUT/data/app/SmokeTestApp.apk &&
1441      adb install $ANDROID_PRODUCT_OUT/data/app/SmokeTest.apk &&
1442      adb shell am instrument -w com.android.smoketest.tests/android.test.InstrumentationTestRunner
1443}
1444
1445# simple shortcut to the runtest command
1446function runtest()
1447{
1448    local T=$(gettop)
1449    if [ ! "$T" ]; then
1450        echo "Couldn't locate the top of the tree.  Try setting TOP." >&2
1451        return
1452    fi
1453    ("$T"/development/testrunner/runtest.py $@)
1454}
1455
1456function godir () {
1457    if [[ -z "$1" ]]; then
1458        echo "Usage: godir <regex>"
1459        return
1460    fi
1461    local T=$(gettop)
1462    local FILELIST
1463    if [ ! "$OUT_DIR" = "" ]; then
1464        mkdir -p $OUT_DIR
1465        FILELIST=$OUT_DIR/filelist
1466    else
1467        FILELIST=$T/filelist
1468    fi
1469    if [[ ! -f $FILELIST ]]; then
1470        echo -n "Creating index..."
1471        (\cd $T; find . -wholename ./out -prune -o -wholename ./.repo -prune -o -type f > $FILELIST)
1472        echo " Done"
1473        echo ""
1474    fi
1475    local lines
1476    lines=($(\grep "$1" $FILELIST | sed -e 's/\/[^/]*$//' | sort | uniq))
1477    if [[ ${#lines[@]} = 0 ]]; then
1478        echo "Not found"
1479        return
1480    fi
1481    local pathname
1482    local choice
1483    if [[ ${#lines[@]} > 1 ]]; then
1484        while [[ -z "$pathname" ]]; do
1485            local index=1
1486            local line
1487            for line in ${lines[@]}; do
1488                printf "%6s %s\n" "[$index]" $line
1489                index=$(($index + 1))
1490            done
1491            echo
1492            echo -n "Select one: "
1493            unset choice
1494            read choice
1495            if [[ $choice -gt ${#lines[@]} || $choice -lt 1 ]]; then
1496                echo "Invalid choice"
1497                continue
1498            fi
1499            pathname=${lines[$(($choice-1))]}
1500        done
1501    else
1502        pathname=${lines[0]}
1503    fi
1504    \cd $T/$pathname
1505}
1506
1507# Update module-info.json in out.
1508function refreshmod() {
1509    if [ ! "$ANDROID_PRODUCT_OUT" ]; then
1510        echo "No ANDROID_PRODUCT_OUT. Try running 'lunch' first." >&2
1511        return 1
1512    fi
1513
1514    echo "Refreshing modules (building module-info.json). Log at $ANDROID_PRODUCT_OUT/module-info.json.build.log." >&2
1515
1516    # for the output of the next command
1517    mkdir -p $ANDROID_PRODUCT_OUT || return 1
1518
1519    # Note, can't use absolute path because of the way make works.
1520    m out/target/product/$(get_build_var TARGET_DEVICE)/module-info.json \
1521        > $ANDROID_PRODUCT_OUT/module-info.json.build.log 2>&1
1522}
1523
1524# List all modules for the current device, as cached in module-info.json. If any build change is
1525# made and it should be reflected in the output, you should run 'refreshmod' first.
1526function allmod() {
1527    if [ ! "$ANDROID_PRODUCT_OUT" ]; then
1528        echo "No ANDROID_PRODUCT_OUT. Try running 'lunch' first." >&2
1529        return 1
1530    fi
1531
1532    if [ ! -f "$ANDROID_PRODUCT_OUT/module-info.json" ]; then
1533        echo "Could not find module-info.json. It will only be built once, and it can be updated with 'refreshmod'" >&2
1534        refreshmod || return 1
1535    fi
1536
1537    python -c "import json; print '\n'.join(sorted(json.load(open('$ANDROID_PRODUCT_OUT/module-info.json')).keys()))"
1538}
1539
1540# Get the path of a specific module in the android tree, as cached in module-info.json. If any build change
1541# is made, and it should be reflected in the output, you should run 'refreshmod' first.
1542function pathmod() {
1543    if [ ! "$ANDROID_PRODUCT_OUT" ]; then
1544        echo "No ANDROID_PRODUCT_OUT. Try running 'lunch' first." >&2
1545        return 1
1546    fi
1547
1548    if [[ $# -ne 1 ]]; then
1549        echo "usage: pathmod <module>" >&2
1550        return 1
1551    fi
1552
1553    if [ ! -f "$ANDROID_PRODUCT_OUT/module-info.json" ]; then
1554        echo "Could not find module-info.json. It will only be built once, and it can be updated with 'refreshmod'" >&2
1555        refreshmod || return 1
1556    fi
1557
1558    local relpath=$(python -c "import json, os
1559module = '$1'
1560module_info = json.load(open('$ANDROID_PRODUCT_OUT/module-info.json'))
1561if module not in module_info:
1562    exit(1)
1563print module_info[module]['path'][0]" 2>/dev/null)
1564
1565    if [ -z "$relpath" ]; then
1566        echo "Could not find module '$1' (try 'refreshmod' if there have been build changes?)." >&2
1567        return 1
1568    else
1569        echo "$ANDROID_BUILD_TOP/$relpath"
1570    fi
1571}
1572
1573# Go to a specific module in the android tree, as cached in module-info.json. If any build change
1574# is made, and it should be reflected in the output, you should run 'refreshmod' first.
1575function gomod() {
1576    if [[ $# -ne 1 ]]; then
1577        echo "usage: gomod <module>" >&2
1578        return 1
1579    fi
1580
1581    local path="$(pathmod $@)"
1582    if [ -z "$path" ]; then
1583        return 1
1584    fi
1585    cd $path
1586}
1587
1588function _complete_android_module_names() {
1589    local word=${COMP_WORDS[COMP_CWORD]}
1590    COMPREPLY=( $(allmod | grep -E "^$word") )
1591}
1592
1593# Print colored exit condition
1594function pez {
1595    "$@"
1596    local retval=$?
1597    if [ $retval -ne 0 ]
1598    then
1599        echo $'\E'"[0;31mFAILURE\e[00m"
1600    else
1601        echo $'\E'"[0;32mSUCCESS\e[00m"
1602    fi
1603    return $retval
1604}
1605
1606function get_make_command()
1607{
1608    # If we're in the top of an Android tree, use soong_ui.bash instead of make
1609    if [ -f build/soong/soong_ui.bash ]; then
1610        # Always use the real make if -C is passed in
1611        for arg in "$@"; do
1612            if [[ $arg == -C* ]]; then
1613                echo command make
1614                return
1615            fi
1616        done
1617        echo build/soong/soong_ui.bash --make-mode
1618    else
1619        echo command make
1620    fi
1621}
1622
1623function _wrap_build()
1624{
1625    if [[ "${ANDROID_QUIET_BUILD:-}" == true ]]; then
1626      "$@"
1627      return $?
1628    fi
1629    local start_time=$(date +"%s")
1630    "$@"
1631    local ret=$?
1632    local end_time=$(date +"%s")
1633    local tdiff=$(($end_time-$start_time))
1634    local hours=$(($tdiff / 3600 ))
1635    local mins=$((($tdiff % 3600) / 60))
1636    local secs=$(($tdiff % 60))
1637    local ncolors=$(tput colors 2>/dev/null)
1638    if [ -n "$ncolors" ] && [ $ncolors -ge 8 ]; then
1639        color_failed=$'\E'"[0;31m"
1640        color_success=$'\E'"[0;32m"
1641        color_reset=$'\E'"[00m"
1642    else
1643        color_failed=""
1644        color_success=""
1645        color_reset=""
1646    fi
1647    echo
1648    if [ $ret -eq 0 ] ; then
1649        echo -n "${color_success}#### build completed successfully "
1650    else
1651        echo -n "${color_failed}#### failed to build some targets "
1652    fi
1653    if [ $hours -gt 0 ] ; then
1654        printf "(%02g:%02g:%02g (hh:mm:ss))" $hours $mins $secs
1655    elif [ $mins -gt 0 ] ; then
1656        printf "(%02g:%02g (mm:ss))" $mins $secs
1657    elif [ $secs -gt 0 ] ; then
1658        printf "(%s seconds)" $secs
1659    fi
1660    echo " ####${color_reset}"
1661    echo
1662    return $ret
1663}
1664
1665function make()
1666{
1667    _wrap_build $(get_make_command "$@") "$@"
1668}
1669
1670function provision()
1671{
1672    if [ ! "$ANDROID_PRODUCT_OUT" ]; then
1673        echo "Couldn't locate output files.  Try running 'lunch' first." >&2
1674        return 1
1675    fi
1676    if [ ! -e "$ANDROID_PRODUCT_OUT/provision-device" ]; then
1677        echo "There is no provisioning script for the device." >&2
1678        return 1
1679    fi
1680
1681    # Check if user really wants to do this.
1682    if [ "$1" = "--no-confirmation" ]; then
1683        shift 1
1684    else
1685        echo "This action will reflash your device."
1686        echo ""
1687        echo "ALL DATA ON THE DEVICE WILL BE IRREVOCABLY ERASED."
1688        echo ""
1689        echo -n "Are you sure you want to do this (yes/no)? "
1690        read
1691        if [[ "${REPLY}" != "yes" ]] ; then
1692            echo "Not taking any action. Exiting." >&2
1693            return 1
1694        fi
1695    fi
1696    "$ANDROID_PRODUCT_OUT/provision-device" "$@"
1697}
1698
1699# Zsh needs bashcompinit called to support bash-style completion.
1700function enable_zsh_completion() {
1701    # Don't override user's options if bash-style completion is already enabled.
1702    if ! declare -f complete >/dev/null; then
1703        autoload -U compinit && compinit
1704        autoload -U bashcompinit && bashcompinit
1705    fi
1706}
1707
1708function validate_current_shell() {
1709    local current_sh="$(ps -o command -p $$)"
1710    case "$current_sh" in
1711        *bash*)
1712            function check_type() { type -t "$1"; }
1713            ;;
1714        *zsh*)
1715            function check_type() { type "$1"; }
1716            enable_zsh_completion ;;
1717        *)
1718            echo -e "WARNING: Only bash and zsh are supported.\nUse of other shell would lead to erroneous results."
1719            ;;
1720    esac
1721}
1722
1723# Execute the contents of any vendorsetup.sh files we can find.
1724# Unless we find an allowed-vendorsetup_sh-files file, in which case we'll only
1725# load those.
1726#
1727# This allows loading only approved vendorsetup.sh files
1728function source_vendorsetup() {
1729    allowed=
1730    for f in $(find -L device vendor product -maxdepth 4 -name 'allowed-vendorsetup_sh-files' 2>/dev/null | sort); do
1731        if [ -n "$allowed" ]; then
1732            echo "More than one 'allowed_vendorsetup_sh-files' file found, not including any vendorsetup.sh files:"
1733            echo "  $allowed"
1734            echo "  $f"
1735            return
1736        fi
1737        allowed="$f"
1738    done
1739
1740    allowed_files=
1741    [ -n "$allowed" ] && allowed_files=$(cat "$allowed")
1742    for dir in device vendor product; do
1743        for f in $(test -d $dir && \
1744            find -L $dir -maxdepth 4 -name 'vendorsetup.sh' 2>/dev/null | sort); do
1745
1746            if [[ -z "$allowed" || "$allowed_files" =~ $f ]]; then
1747                echo "including $f"; . "$f"
1748            else
1749                echo "ignoring $f, not in $allowed"
1750            fi
1751        done
1752    done
1753}
1754
1755validate_current_shell
1756source_vendorsetup
1757addcompletions
1758