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