mirror of
git://git.sv.gnu.org/emacs.git
synced 2025-12-06 06:20:55 -08:00
* .gitignore: Add new files.
* INSTALL.android: Explain how to build Emacs for ancient
versions of Android.
* admin/merge-gnulib (GNULIB_MODULES): Add getdelim.
* build-aux/config.guess (timestamp, version):
* build-aux/config.sub (timestamp, version): Autoupdate.
* configure.ac (BUILD_DETAILS, ANDROID_MIN_SDK):
(ANDROID_STUBIFY): Allow specifying CFLAGS via ANDROID_CFLAGS.
Add new configure tests for Android API version when not
explicitly specified.
* doc/emacs/android.texi (Android): Add reference to ``Other
Input Devices''.
(Android File System): Remove restrictions on directory-files on
the assets directory.
* doc/emacs/emacs.texi (Top): Add Other Input Devices to menu.
* doc/emacs/input.texi (Other Input Devices): New node.
* doc/lispref/commands.texi (Touchscreen Events): Document
changes to touchscreen input events.
* doc/lispref/frames.texi (Pop-Up Menus): Likewise.
* etc/NEWS: Announce changes.
* java/Makefile.in: Use lib-src/asset-directory-tool to generate
an `directory-tree' file placed in /assets.
* java/debug.sh: Large adjustments to support Android 2.2 and
later.
* java/org/gnu/emacs/EmacsContextMenu.java (inflateMenuItems):
* java/org/gnu/emacs/EmacsCopyArea.java (perform):
* java/org/gnu/emacs/EmacsDialog.java (toAlertDialog):
* java/org/gnu/emacs/EmacsDrawLine.java (perform):
* java/org/gnu/emacs/EmacsDrawRectangle.java (perform):
* java/org/gnu/emacs/EmacsDrawable.java (EmacsDrawable):
* java/org/gnu/emacs/EmacsFillPolygon.java (perform):
* java/org/gnu/emacs/EmacsFillRectangle.java (perform):
* java/org/gnu/emacs/EmacsGC.java (EmacsGC):
* java/org/gnu/emacs/EmacsPixmap.java (EmacsPixmap):
(destroyHandle):
* java/org/gnu/emacs/EmacsSdk7FontDriver.java (draw): Avoid
redundant canvas saves and restores.
* java/org/gnu/emacs/EmacsService.java (run):
* java/org/gnu/emacs/EmacsView.java (EmacsView):
(handleDirtyBitmap):
* java/org/gnu/emacs/EmacsWindow.java (changeWindowBackground)
(EmacsWindow): Make compatible with Android 2.2 and later.
* lib-src/Makefile.in (DONT_INSTALL): Add asset-directory-tool
on Android.:(asset-directory-tool{EXEEXT}): New target.
* lib-src/asset-directory-tool.c (struct directory_tree, xmalloc)
(main_1, main_2, main): New file.
* lib, m4: Merge from gnulib. This will be reverted before
merging to master.
* lisp/button.el (button-map):
(push-button):
* lisp/frame.el (display-popup-menus-p): Improve touchscreen
support.
* lisp/subr.el (event-start):
(event-end): Handle touchscreen events.
* lisp/touch-screen.el (touch-screen-handle-timeout):
(touch-screen-handle-point-update):
(touch-screen-handle-point-up):
(touch-screen-track-tap):
(touch-screen-track-drag):
(touch-screen-drag-mode-line-1):
(touch-screen-drag-mode-line): New functions.
([mode-line touchscreen-begin]):
([bottom-divider touchscreen-begin]): Bind new events.
* lisp/wid-edit.el (widget-event-point):
(widget-keymap):
(widget-event-start):
(widget-button--check-and-call-button):
(widget-button-click): Improve touchscreen support.
* src/alloc.c (make_lisp_symbol): Avoid ICE on Android NDK GCC.
(mark_pinned_symbols): Likewise.
* src/android.c (struct android_emacs_window): New struct.
(window_class): New variable.
(android_run_select_thread): Add workaround for Android platform
bug.
(android_extract_long, android_scan_directory_tree): New
functions.
(android_file_access_p): Use those functions instead.
(android_init_emacs_window): New function.
(android_init_emacs_gc_class): Update signature of `markDirty'.
(android_change_gc, android_set_clip_rectangles): Tell the GC
whether or not clip rects were dirtied.
(android_swap_buffers): Do not look up method every time.
(struct android_dir): Adjust for new directory tree lookup.
(android_opendir, android_readdir, android_closedir): Likewise.
(android_four_corners_bilinear): Fix coding style.
(android_ftruncate): New function.
* src/android.h: Update prototypes. Replace ftruncate with
android_ftruncate when necessary.
* src/androidterm.c (handle_one_android_event): Pacify GCC. Fix
touch screen tool bar bug.
* src/emacs.c (using_utf8): Fix compilation error.
* src/fileio.c (Ffile_system_info): Return Qnil when fsusage.o
is not built.
* src/filelock.c (BOOT_TIME_FILE): Fix definition for Android.
* src/frame.c (Fx_parse_geometry): Fix uninitialized variable
uses.
* src/keyboard.c (lispy_function_keys): Fix `back'.
* src/menu.c (x_popup_menu_1): Handle touch screen events.
(Fx_popup_menu): Document changes.
* src/sfnt.c (main): Improve tests.
* src/sfntfont-android.c (sfntfont_android_put_glyphs): Fix
minor problem.
(init_sfntfont_android): Check for
HAVE_DECL_ANDROID_GET_DEVICE_API_LEVEL.
* src/sfntfont.c (struct sfnt_font_desc): New fields `adstyle'
and `languages'.
(sfnt_parse_style): Append tokens to adstyle.
(sfnt_parse_languages): New function.
(sfnt_enum_font_1): Parse supported languages and adstyle.
(sfntfont_list_1): Handle new fields.
(sfntfont_text_extents): Fix uninitialized variable use.
(syms_of_sfntfont, mark_sfntfont): Adjust accordingly.
330 lines
8.3 KiB
Bash
Executable file
330 lines
8.3 KiB
Bash
Executable file
#!/bin/bash
|
|
### Run Emacs under GDB or JDB on Android.
|
|
|
|
## Copyright (C) 2023 Free Software Foundation, Inc.
|
|
|
|
## This file is part of GNU Emacs.
|
|
|
|
## GNU Emacs is free software: you can redistribute it and/or modify
|
|
## it under the terms of the GNU General Public License as published by
|
|
## the Free Software Foundation, either version 3 of the License, or
|
|
## (at your option) any later version.
|
|
|
|
## GNU Emacs is distributed in the hope that it will be useful,
|
|
## but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
## MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
## GNU General Public License for more details.
|
|
|
|
## You should have received a copy of the GNU General Public License
|
|
## along with GNU Emacs. If not, see <https://www.gnu.org/licenses/>.
|
|
|
|
set -m
|
|
oldpwd=`pwd`
|
|
cd `dirname $0`
|
|
|
|
devices=`adb devices | grep device | awk -- '/device\y/ { print $1 }' -`
|
|
device=
|
|
progname=$0
|
|
package=org.gnu.emacs
|
|
activity=org.gnu.emacs.EmacsActivity
|
|
gdb_port=5039
|
|
jdb_port=64013
|
|
jdb=no
|
|
attach_existing=no
|
|
gdbserver=
|
|
|
|
while [ $# -gt 0 ]; do
|
|
case "$1" in
|
|
## This option specifies the serial number of a device to use.
|
|
"--device" )
|
|
device="$2"
|
|
if [ -z device ]; then
|
|
echo "You must specify an argument to --device"
|
|
exit 1
|
|
fi
|
|
shift
|
|
;;
|
|
"--help" )
|
|
echo "Usage: $progname [options] -- [gdb options]"
|
|
echo ""
|
|
echo " --device DEVICE run Emacs on the specified device"
|
|
echo " --port PORT run the GDB server on a specific port"
|
|
echo " --jdb-port PORT run the JDB server on a specific port"
|
|
echo " --jdb run JDB instead of GDB"
|
|
echo " --attach-existing attach to an existing process"
|
|
echo " --gdbserver BINARY upload and use the specified gdbserver binary"
|
|
echo " --help print this message"
|
|
echo ""
|
|
echo "Available devices:"
|
|
for device in $devices; do
|
|
echo " " $device
|
|
done
|
|
echo ""
|
|
exit 0
|
|
;;
|
|
"--jdb" )
|
|
jdb=yes
|
|
;;
|
|
"--gdbserver" )
|
|
shift
|
|
gdbserver=$1
|
|
;;
|
|
"--port" )
|
|
shift
|
|
gdb_port=$1
|
|
;;
|
|
"--jdb-port" )
|
|
shift
|
|
jdb_port=$1
|
|
;;
|
|
"--attach-existing" )
|
|
attach_existing=yes
|
|
;;
|
|
"--" )
|
|
shift
|
|
gdbargs=$@
|
|
break;
|
|
;;
|
|
* )
|
|
echo "$progname: Unrecognized argument $1"
|
|
exit 1
|
|
;;
|
|
esac
|
|
shift
|
|
done
|
|
|
|
if [ -z "$devices" ]; then
|
|
echo "No devices are available."
|
|
exit 1
|
|
fi
|
|
|
|
if [ -z $device ]; then
|
|
device=$devices
|
|
fi
|
|
|
|
if [ `wc -w <<< "$devices"` -gt 1 ] && [ -z device ]; then
|
|
echo "Multiple devices are available. Please pick one using"
|
|
echo "--device and try again."
|
|
fi
|
|
|
|
echo "Looking for $package on device $device"
|
|
|
|
# Find the application data directory
|
|
app_data_dir=`adb -s $device shell run-as $package sh -c 'pwd 2> /dev/null'`
|
|
|
|
if [ -z $app_data_dir ]; then
|
|
echo "The data directory for the package $package was not found."
|
|
echo "Is it installed?"
|
|
fi
|
|
|
|
echo "Found application data directory at" "$app_data_dir"
|
|
|
|
# Generate an awk script to extract PIDs from Android ps output. It
|
|
# is enough to run `ps' as the package user on newer versions of
|
|
# Android, but that doesn't work on Android 2.3.
|
|
cat << EOF > tmp.awk
|
|
BEGIN {
|
|
pid = 0;
|
|
pid_column = 2;
|
|
}
|
|
|
|
{
|
|
# Remove any trailing carriage return from the input line.
|
|
gsub ("\r", "", \$NF)
|
|
|
|
# If this is line 1, figure out which column contains the PID.
|
|
if (NR == 1)
|
|
{
|
|
for (n = 1; n <= NF; ++n)
|
|
{
|
|
if (\$n == "PID")
|
|
pid_column=n;
|
|
}
|
|
}
|
|
else if (\$NF == "$package")
|
|
print \$pid_column
|
|
}
|
|
EOF
|
|
|
|
# Make sure that file disappears once this script exits.
|
|
trap "rm -f $(pwd)/tmp.awk" 0
|
|
|
|
# First, run ps to fetch the list of process IDs.
|
|
package_pids=`adb -s $device shell ps`
|
|
|
|
# Next, extract the list of PIDs currently running.
|
|
package_pids=`awk -f tmp.awk <<< $package_pids`
|
|
|
|
if [ "$attach_existing" != "yes" ]; then
|
|
# Finally, kill each existing process.
|
|
for pid in $package_pids; do
|
|
echo "Killing existing process $pid..."
|
|
adb -s $device shell run-as $package kill -9 $pid &> /dev/null
|
|
done
|
|
|
|
# Now run the main activity. This must be done as the adb user and
|
|
# not as the package user.
|
|
echo "Starting activity $activity and attaching debugger"
|
|
|
|
# Exit if the activity could not be started.
|
|
adb -s $device shell am start -D -n "$package/$activity"
|
|
if [ ! $? ]; then
|
|
exit 1;
|
|
fi
|
|
|
|
# Sleep for a bit. Otherwise, the process may not have started
|
|
# yet.
|
|
sleep 1
|
|
|
|
# Now look for processes matching the package again.
|
|
package_pids=`adb -s $device shell ps`
|
|
|
|
# Next, remove lines matching "ps" itself.
|
|
package_pids=`awk -f tmp.awk <<< $package_pids`
|
|
fi
|
|
|
|
pid=$package_pids
|
|
num_pids=`wc -w <<< "$package_pids"`
|
|
|
|
if [ $num_pids -gt 1 ]; then
|
|
echo "More than one process was started:"
|
|
echo ""
|
|
adb -s $device shell run-as $package ps | awk -- "{
|
|
if (!match (\$0, /ps/) && match (\$0, /$package/))
|
|
print \$0
|
|
}"
|
|
echo ""
|
|
printf "Which one do you want to attach to? "
|
|
read pid
|
|
elif [ -z $package_pids ]; then
|
|
echo "No processes were found to attach to."
|
|
exit 1
|
|
fi
|
|
|
|
# If either --jdb was specified or debug.sh is not connecting to an
|
|
# existing process, then store a suitable JDB invocation in
|
|
# jdb_command. GDB will then run JDB to unblock the application from
|
|
# the wait dialog after startup.
|
|
|
|
if [ "$jdb" = "yes" ] || [ "$attach_existing" != yes ]; then
|
|
adb -s $device forward --remove-all
|
|
adb -s $device forward "tcp:$jdb_port" "jdwp:$pid"
|
|
|
|
if [ ! $? ]; then
|
|
echo "Failed to forward jdwp:$pid to $jdb_port!"
|
|
echo "Perhaps you need to specify a different port with --port?"
|
|
exit 1;
|
|
fi
|
|
|
|
jdb_command="jdb -connect \
|
|
com.sun.jdi.SocketAttach:hostname=localhost,port=$jdb_port"
|
|
|
|
if [ $jdb = "yes" ]; then
|
|
# Just start JDB and then exit
|
|
$jdb_command
|
|
exit 1
|
|
fi
|
|
fi
|
|
|
|
if [ -n "$jdb_command" ]; then
|
|
echo "Starting JDB to unblock application."
|
|
|
|
# Start JDB to unblock the application.
|
|
coproc JDB { $jdb_command; }
|
|
|
|
# Tell JDB to first suspend all threads.
|
|
echo "suspend" >&${JDB[1]}
|
|
|
|
# Tell JDB to print a magic string once the program is
|
|
# initialized.
|
|
echo "print \"__verify_jdb_has_started__\"" >&${JDB[1]}
|
|
|
|
# Now wait for JDB to give the string back.
|
|
line=
|
|
while :; do
|
|
read -u ${JDB[0]} line
|
|
if [ ! $? ]; then
|
|
echo "Failed to read JDB output"
|
|
exit 1
|
|
fi
|
|
|
|
case "$line" in
|
|
*__verify_jdb_has_started__*)
|
|
# Android only polls for a Java debugger every 200ms, so
|
|
# the debugger must be connected for at least that long.
|
|
echo "Pausing 1 second for the program to continue."
|
|
sleep 1
|
|
break
|
|
;;
|
|
esac
|
|
done
|
|
|
|
# Note that JDB does not exit until GDB is fully attached!
|
|
fi
|
|
|
|
# See if gdbserver has to be uploaded
|
|
if [ -z "$gdbserver" ]; then
|
|
gdbserver_bin=/system/bin/gdbserver
|
|
else
|
|
gdbserver_bin=/data/local/tmp/gdbserver
|
|
|
|
# Upload the specified gdbserver binary to the device.
|
|
adb -s $device push "$gdbserver" "$gdbserver_bin"
|
|
adb -s $device shell chmod +x "$gdbserver_bin"
|
|
fi
|
|
|
|
# Now start gdbserver on the device asynchronously.
|
|
|
|
echo "Attaching gdbserver to $pid on $device..."
|
|
exec 5<> /tmp/file-descriptor-stamp
|
|
rm -f /tmp/file-descriptor-stamp
|
|
|
|
if [ -z "$gdbserver" ]; then
|
|
adb -s $device shell run-as $package $gdbserver_bin --once \
|
|
"+debug.$package.socket" --attach $pid >&5 &
|
|
gdb_socket="localfilesystem:$app_data_dir/debug.$package.socket"
|
|
else
|
|
# Normally the program cannot access $gdbserver_bin when it is
|
|
# placed in /data/local/tmp.
|
|
adb -s $device shell $gdbserver_bin --once \
|
|
"+/data/local/tmp/debug.$package.socket" \
|
|
--attach $pid >&5 &
|
|
gdb_socket="localfilesystem:/data/local/tmp/debug.$package.socket"
|
|
fi
|
|
|
|
# Wait until gdbserver successfully runs.
|
|
line=
|
|
while read -u 5 line; do
|
|
case "$line" in
|
|
*Attached* )
|
|
break;
|
|
;;
|
|
*error* | *Error* | failed )
|
|
echo "GDB error:" $line
|
|
exit 1
|
|
;;
|
|
* )
|
|
;;
|
|
esac
|
|
done
|
|
|
|
# Now that GDB is attached, tell the Java debugger to resume execution
|
|
# and then exit.
|
|
|
|
if [ -n "$jdb_command" ]; then
|
|
echo "resume" >&${JDB[1]}
|
|
echo "exit" >&${JDB[1]}
|
|
fi
|
|
|
|
# Forward the gdb server port here.
|
|
adb -s $device forward "tcp:$gdb_port" $gdb_socket
|
|
if [ ! $? ]; then
|
|
echo "Failed to forward $app_data_dir/debug.$package.socket"
|
|
echo "to $gdb_port! Perhaps you need to specify a different port"
|
|
echo "with --port?"
|
|
exit 1;
|
|
fi
|
|
|
|
# Finally, start gdb with any extra arguments needed.
|
|
cd "$oldpwd"
|
|
gdb --eval-command "target remote localhost:$gdb_port" $gdbargs
|