1#!/bin/bash
2# TODO:
3# 1. Check for ANDROID_SERIAL/multiple devices
4
5if [ -z "$ANDROID_BUILD_TOP" ]; then
6  >&2 echo '$ANDROID_BUILD_TOP is not set. Source build/envsetup.sh.'
7  exit 1
8fi
9
10# We can use environment variables (like ANDROID_BUILD_TOP) from the user's
11# shell, but not functions (like gettop), so we need to source envsetup in here
12# as well.
13source $ANDROID_BUILD_TOP/build/envsetup.sh
14echo
15
16function adb_get_product_device() {
17  local candidate=`adb shell getprop ro.hardware | tr -d '\r\n'`
18  if [[ "$candidate" =~ ^(goldfish|ranchu)$ ]]; then
19    # Emulator builds use product.device for OUT folder
20    candidate=`adb shell getprop ro.product.device | tr -d '\r\n'`
21  fi
22  echo $candidate
23}
24
25# returns 0 when process is not traced
26function adb_get_traced_by() {
27  echo `adb shell cat /proc/$1/status | grep -e "^TracerPid:" | sed "s/^TracerPid:[[:blank:]]//" | tr -d '\r\n'`
28}
29
30function get_symbols_directory()
31{
32    echo $(get_abs_build_var TARGET_OUT_UNSTRIPPED)
33}
34
35function gdbwrapper()
36{
37    local GDB_CMD="$1"
38    shift 1
39    $GDB_CMD -x "$@"
40}
41
42function gdbclient() {
43  local PROCESS_NAME="n/a"
44  local PID=$1
45  local PORT=5039
46  if [ -z "$PID" ]; then
47    echo "Usage: gdbclient <pid|processname> [port number]"
48    return -1
49  fi
50  local DEVICE=$(adb_get_product_device)
51
52  if [ -z "$DEVICE" ]; then
53    echo "Error: Unable to get device name. Please check if device is connected and ANDROID_SERIAL is set."
54    return -2
55  fi
56
57  if [ -n "$2" ]; then
58    PORT=$2
59  fi
60
61  local ROOT=$(gettop)
62  if [ -z "$ROOT" ]; then
63    # This is for the situation with downloaded symbols (from the build server)
64    # we check if they are available.
65    ROOT=`realpath .`
66  fi
67
68  local SYS_OUT_ROOT=$(get_build_var OUT_DIR)
69  local OUT_ROOT="${SYS_OUT_ROOT:-${OUT_DIR:-$ROOT/out}}/target/product/$DEVICE"
70  local SYMBOLS_DIR="$OUT_ROOT/symbols"
71  local IS_TAPAS_USER="$(get_build_var TARGET_BUILD_APPS)"
72  local TAPAS_SYMBOLS_DIR=
73
74  if [ $IS_TAPAS_USER ]; then
75    TAPAS_SYMBOLS_DIR=$(get_symbols_directory)
76  fi
77
78  if [ ! -d $SYMBOLS_DIR ]; then
79    if [ $IS_TAPAS_USER ]; then
80      mkdir -p $SYMBOLS_DIR/system/bin
81    else
82      echo "Error: couldn't find symbols: $SYMBOLS_DIR does not exist or is not a directory."
83      return -3
84    fi
85  fi
86
87  # let's figure out which executable we are about to debug
88
89  # check if user specified a name -> resolve to pid
90  if [[ ! "$PID" =~ ^[0-9]+$ ]] ; then
91    PROCESS_NAME=$PID
92    PID=$(pid --exact $PROCESS_NAME)
93    if [ -z "$PID" ]; then
94      echo "Error: couldn't resolve pid by process name: $PROCESS_NAME"
95      return -4
96    else
97      echo "Resolved pid for $PROCESS_NAME is $PID"
98    fi
99  fi
100
101  local ID=`adb shell id -u`
102  if [ "$ID" != "0" ]; then
103    echo "Error: gdbclient only works if you've run 'adb root'"
104    return -4
105  fi
106
107  local EXE=`adb shell readlink /proc/$PID/exe | tr -d '\r\n'`
108  if [ -z "$EXE" ]; then
109    echo "Error: couldn't find executable for pid $PID --- is the process still alive?"
110    return -4
111  fi
112
113  local LOCAL_EXE_PATH=$SYMBOLS_DIR$EXE
114
115  if [ ! -f $LOCAL_EXE_PATH ]; then
116    if [ $IS_TAPAS_USER ]; then
117      adb pull $EXE $LOCAL_EXE_PATH
118    else
119      echo "Error: unable to find symbols for executable $EXE: file $LOCAL_EXE_PATH does not exist"
120      return -5
121    fi
122  fi
123
124  local USE64BIT=""
125
126  if [[ "$(file $LOCAL_EXE_PATH)" =~ 64-bit ]]; then
127    USE64BIT="64"
128  fi
129
130  # and now linker for tapas users...
131  if [ -n "$IS_TAPAS_USER" -a ! -f "$SYMBOLS_DIR/system/bin/linker$USE64BIT" ]; then
132    adb pull /system/bin/linker$USE64BIT $SYMBOLS_DIR/system/bin/linker$USE64BIT
133  fi
134
135  local GDB
136  case $(uname -s) in
137    Darwin)
138      GDB=$ANDROID_BUILD_TOP/prebuilts/gdb/darwin-x86/bin/gdb
139      ;;
140
141    Linux)
142      GDB=$ANDROID_BUILD_TOP/prebuilts/gdb/linux-x86/bin/gdb
143      ;;
144
145    *)
146      echo "Error: Unknown platform '$(uname -s)'"
147      return 1
148      ;;
149  esac
150
151  local CPU_ABI=`adb shell getprop ro.product.cpu.abilist | tr -d '\r\n'`
152
153  # TODO: check if tracing process is gdbserver and not some random strace...
154  if [ "$(adb_get_traced_by $PID)" -eq 0 ]; then
155    # start gdbserver
156    echo "Starting gdbserver..."
157    # TODO: check if adb is already listening $PORT
158    # to avoid unnecessary calls
159    echo ". adb forward for port=$PORT..."
160    adb forward tcp:$PORT tcp:$PORT
161    echo ". starting gdbserver to attach to pid=$PID..."
162    adb shell gdbserver$USE64BIT :$PORT --attach $PID &
163    echo ". give it couple of seconds to start..."
164    sleep 2
165    echo ". done"
166  else
167    echo "It looks like gdbserver is already attached to $PID (process is traced), trying to connect to it using local port=$PORT"
168    adb forward tcp:$PORT tcp:$PORT
169  fi
170
171  local OUT_SO_SYMBOLS=$SYMBOLS_DIR/system/lib$USE64BIT
172  local TAPAS_OUT_SO_SYMBOLS=$TAPAS_SYMBOLS_DIR/system/lib$USE64BIT
173  local OUT_VENDOR_SO_SYMBOLS=$SYMBOLS_DIR/vendor/lib$USE64BIT
174  local ART_CMD=""
175
176  local SOLIB_SYSROOT=$SYMBOLS_DIR
177  local SOLIB_SEARCHPATH=$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
178
179  if [ $IS_TAPAS_USER ]; then
180    SOLIB_SYSROOT=$TAPAS_SYMBOLS_DIR:$SOLIB_SYSROOT
181    SOLIB_SEARCHPATH=$TAPAS_OUT_SO_SYMBOLS:$SOLIB_SEARCHPATH
182  fi
183
184  echo >|"$OUT_ROOT/gdbclient.cmds" "set solib-absolute-prefix $SOLIB_SYSROOT"
185  echo >>"$OUT_ROOT/gdbclient.cmds" "set solib-search-path $SOLIB_SEARCHPATH"
186  local DALVIK_GDB_SCRIPT=$ROOT/development/scripts/gdb/dalvik.gdb
187  if [ -f $DALVIK_GDB_SCRIPT ]; then
188    echo >>"$OUT_ROOT/gdbclient.cmds" "source $DALVIK_GDB_SCRIPT"
189    ART_CMD="art-on"
190  else
191    echo "Warning: couldn't find $DALVIK_GDB_SCRIPT - ART debugging options will not be available"
192  fi
193  echo >>"$OUT_ROOT/gdbclient.cmds" "target remote :$PORT"
194  if [[ $EXE =~ (^|/)(app_process|dalvikvm)(|32|64)$ ]]; then
195    echo >> "$OUT_ROOT/gdbclient.cmds" $ART_CMD
196  fi
197
198  echo >>"$OUT_ROOT/gdbclient.cmds" ""
199
200  gdbwrapper $GDB "$OUT_ROOT/gdbclient.cmds" "$LOCAL_EXE_PATH"
201}
202
203gdbclient $*
204