123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172 |
- #!/bin/bash
- #
- # Copyright (c) 2012 The Chromium Authors. All rights reserved.
- # Use of this source code is governed by a BSD-style license that can be
- # found in the LICENSE file.
- #
- # Attach gdb to a running android application. Similar to ndk-gdb.
- # Run with --annotate=3 if running under emacs (M-x gdb).
- #
- # By default it is used to debug content shell, if it is used to
- # debug other piceces, '-p' and '-l' options are needed.
- # For *unittests_apk (like base_unittests_apk), run with:
- # "gdb_apk -p org.chromium.native_test -l out/Release/lib.target -r"
- # Run a command through adb shell, strip the extra \r from the output
- # and return the correct status code to detect failures. This assumes
- # that the adb shell command prints a final \n to stdout.
- # args: command to run
- # Prints the command's stdout on stdout
- # Returns the command's status
- # Note: the command's stderr is lost
- adb_shell () {
- local TMPOUT="$(mktemp)"
- local LASTLINE RET
- local ADB=${ADB:-adb}
- # The weird sed rule is to strip the final \r on each output line
- # Since 'adb shell' never returns the command's proper exit/status code,
- # we force it to print it as '%%<status>' in the temporary output file,
- # which we will later strip from it.
- $ADB shell $@ ";" echo "%%\$?" 2>/dev/null | sed -e 's![[:cntrl:]]!!g' > $TMPOUT
- # Get last line in log, which contains the exit code from the command
- LASTLINE=$(sed -e '$!d' $TMPOUT)
- # Extract the status code from the end of the line, which must be '%%<code>'
- RET=$(echo "$LASTLINE" | awk '{ if (match($0, "%%[0-9]+$")) { print substr($0,RSTART+2); } }')
- # Remove the status code from the last line. Note that this may result in an empty line
- LASTLINE=$(echo "$LASTLINE" | awk '{ if (match($0, "%%[0-9]+$")) { print substr($0,1,RSTART-1); } }')
- # The output itself: all lines except the status code
- sed -e '$d' $TMPOUT && echo -n "$LASTLINE"
- # Remove temp file
- rm -f $TMPOUT
- # Exit with the appropriate status
- return $RET
- }
- adb=$(which adb)
- if [[ "$adb" = "" ]] ; then
- echo "Need adb in your path"
- exit 1
- fi
- usage() {
- echo "usage: ${0##*/} [-p package_name] [-l shared_lib_dir] [-g gdb] [-r]"
- echo "-p package_name the android APK package to be debugged"
- echo "-l shared_lib_dir directory containes native shared library"
- echo "-g gdb_args agruments for gdb, eg: -g '-n -write'"
- echo "-r the target device is rooted"
- }
- process_options() {
- local OPTNAME OPTIND OPTERR OPTARG
- while getopts ":p:l:g:r" OPTNAME; do
- case "$OPTNAME" in
- p)
- package_name="$OPTARG"
- ;;
- l)
- shared_lib_dir="$OPTARG"
- ;;
- g)
- gdb_args="$OPTARG"
- ;;
- r)
- rooted_phone=1
- ;;
- \:)
- echo "'-$OPTARG' needs an argument."
- usage
- exit 1
- ;;
- *)
- echo "invalid command line option: $OPTARG"
- usage
- exit 1
- ;;
- esac
- done
- if [ $# -ge ${OPTIND} ]; then
- eval echo "Unexpected command line argument: \${${OPTIND}}"
- usage
- exit 1
- fi
- }
- rooted_phone=0
- root=$(dirname $0)/../..
- package_name=org.chromium.content_shell
- shared_lib_dir=$root/out/${BUILDTYPE:-Debug}/lib.target
- gdb_args=''
- #process options
- process_options "$@"
- echo "Debug package $package_name"
- echo "Assume native shared library is under $shared_lib_dir"
- data_dir=/data/data/$package_name
- gdb_server_on_device=$data_dir/lib/gdbserver
- # Kill any running gdbserver
- pid=$(adb shell ps | awk '/gdbserver/ {print $2}')
- if [[ "$pid" != "" ]] ; then
- if [[ $rooted_phone -eq 1 ]] ; then
- adb shell kill $pid
- else
- adb shell run-as $package_name kill $pid
- fi
- fi
- pid=$(adb_shell ps | awk "/$package_name$/ {print \$2}")
- if [[ "$pid" = "" ]] ; then
- echo "No $package_name running?"
- echo "Try this: adb shell am start -a android.intent.action.VIEW " \
- "-n $package_name/.SomethingActivity (Something might be ContentShell)"
- exit 2
- fi
- no_gdb_server=$(adb shell ls $gdb_server_on_device | grep 'No such file')
- if [[ "$no_gdb_server" != "" ]] ; then
- echo "No gdb server on device at $gdb_server_on_device"
- echo "Please install a debug build."
- exit 3
- fi
- if [[ $rooted_phone -eq 1 ]] ; then
- adb shell $gdb_server_on_device :4321 --attach $pid &
- adb forward tcp:4321 tcp:4321
- else
- adb shell run-as $package_name lib/gdbserver +debug-socket --attach $pid &
- adb forward tcp:4321 localfilesystem:$data_dir/debug-socket
- fi
- sleep 2
- # Pull app_process and C libraries from device if needed
- app_process=${shared_lib_dir}/app_process
- if [[ ! -f ${app_process} ]] ; then
- adb pull /system/bin/app_process ${app_process}
- adb pull /system/lib/libc.so ${shared_lib_dir}
- fi
- # gdb commands
- cmdfile=$(mktemp /tmp/gdb_android_XXXXXXXX)
- cat >$cmdfile<<EOF
- # set solib-absolute-prefix null
- set solib-search-path ${shared_lib_dir}
- file ${app_process}
- target remote :4321
- EOF
- gdb=$(echo $ANDROID_TOOLCHAIN/../../linux-x86/bin/*gdb)
- if [[ ! -f ${gdb} ]] ; then
- echo "Wow no gdb in env var ANDROID_TOOLCHAIN which is $ANDROID_TOOLCHAIN"
- exit 4
- else
- echo Using $gdb
- fi
- # ${gdb} -x $cmdfile $* $app_process
- ${gdb} -x $cmdfile $gdb_args
- rm $cmdfile
|