| #!/bin/bash | 
 | # | 
 | # Copyright (C) 2015 The Android Open Source Project | 
 | # | 
 | # Licensed under the Apache License, Version 2.0 (the "License"); | 
 | # you may not use this file except in compliance with the License. | 
 | # You may obtain a copy of the License at | 
 | # | 
 | #      http://www.apache.org/licenses/LICENSE-2.0 | 
 | # | 
 | # Unless required by applicable law or agreed to in writing, software | 
 | # distributed under the License is distributed on an "AS IS" BASIS, | 
 | # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | 
 | # See the License for the specific language governing permissions and | 
 | # limitations under the License. | 
 |  | 
 | if [ ! -d libcore ]; then | 
 |   echo "Script needs to be run at the root of the android tree" | 
 |   exit 1 | 
 | fi | 
 |  | 
 | source build/envsetup.sh >&/dev/null # for get_build_var, setpaths | 
 | setpaths # include platform prebuilt java, javac, etc in $PATH. | 
 |  | 
 | if [ -z "$ANDROID_HOST_OUT" ] ; then | 
 |   ANDROID_HOST_OUT=${OUT_DIR-$ANDROID_BUILD_TOP/out}/host/linux-x86 | 
 | fi | 
 |  | 
 | # "Root" (actually "system") directory on device (in the case of | 
 | # target testing). | 
 | android_root=${ART_TEST_ANDROID_ROOT:-/system} | 
 |  | 
 | java_lib_location="${ANDROID_HOST_OUT}/../common/obj/JAVA_LIBRARIES" | 
 | make_target_name="apache-harmony-jdwp-tests-hostdex" | 
 |  | 
 | function boot_classpath_arg { | 
 |   local dir="$1" | 
 |   local suffix="$2" | 
 |   shift 2 | 
 |   local separator="" | 
 |   for var | 
 |   do | 
 |     if [ "$var" = "conscrypt" ] && [ "$mode" = "target" ]; then | 
 |       printf -- "${separator}/apex/com.android.conscrypt/javalib/conscrypt.jar"; | 
 |     elif [ "$var" = "core-icu4j" ] && [ "$mode" = "target" ]; then | 
 |       printf -- "${separator}/apex/com.android.i18n/javalib/core-icu4j.jar"; | 
 |     else | 
 |       printf -- "${separator}${dir}/${var}${suffix}.jar"; | 
 |     fi | 
 |     separator=":" | 
 |   done | 
 | } | 
 |  | 
 | # Note: This must start with the CORE_IMG_JARS in Android.common_path.mk | 
 | # because that's what we use for compiling the boot.art image. | 
 | # It may contain additional modules from TEST_CORE_JARS. | 
 | BOOT_CLASSPATH_JARS="core-oj core-libart okhttp bouncycastle apache-xml core-icu4j conscrypt" | 
 |  | 
 | vm_args="" | 
 | art="$android_root/bin/art" | 
 | mode="target" | 
 | art_debugee="sh $android_root/bin/art" | 
 | args=$@ | 
 | chroot_option= | 
 | debuggee_args="-Xcompiler-option --debuggable" | 
 | device_dir="--device-dir=/data/local/tmp" | 
 | # We use the art script on target to ensure the runner and the debuggee share the same | 
 | # image. | 
 | vm_command="--vm-command=$art" | 
 | image_compiler_option="" | 
 | plugin="" | 
 | debug="no" | 
 | explicit_debug="no" | 
 | verbose="no" | 
 | image="-Ximage:/system/framework/art_boot_images/boot.art" | 
 | boot_classpath="$(boot_classpath_arg /apex/com.android.art/javalib "" $BOOT_CLASSPATH_JARS)" | 
 | boot_classpath_locations="" | 
 | with_jdwp_path="" | 
 | agent_wrapper="" | 
 | vm_args="" | 
 | # By default, we run the whole JDWP test suite. | 
 | has_specific_test="no" | 
 | test="org.apache.harmony.jpda.tests.share.AllTests" | 
 | # Use JIT compiling by default. | 
 | use_jit=true | 
 | instant_jit=false | 
 | variant_cmdline_parameter="--variant=X32" | 
 | dump_command="/bin/true" | 
 | called_from_libjdwp=${RUN_JDWP_TESTS_CALLED_FROM_LIBJDWP:-false} | 
 | run_internal_jdwp_test=false | 
 | # Let LUCI bots do what they want. | 
 | if test -v LUCI_CONTEXT; then | 
 |   run_internal_jdwp_test=true | 
 | fi | 
 | # Timeout of JDWP test in ms. | 
 | # | 
 | # Note: some tests expect a timeout to check that *no* reply/event is received for a specific case. | 
 | # A lower timeout can save up several minutes when running the whole test suite, especially for | 
 | # continuous testing. This value can be adjusted to fit the configuration of the host machine(s). | 
 | jdwp_test_timeout=10000 | 
 |  | 
 | skip_tests= | 
 | gdb_target= | 
 | has_gdb="no" | 
 |  | 
 | while true; do | 
 |   if [[ "$1" == "--mode=host" ]]; then | 
 |     mode="host" | 
 |     # Specify bash explicitly since the art script cannot, since it has to run on the device | 
 |     # with mksh. | 
 |     art="bash ${OUT_DIR-out}/host/linux-x86/bin/art" | 
 |     art_debugee="bash ${OUT_DIR-out}/host/linux-x86/bin/art" | 
 |     # We force generation of a new image to avoid build-time and run-time classpath differences. | 
 |     image="-Ximage:/system/non/existent/vogar.art" | 
 |     # Pass the host boot classpath. | 
 |     if [ "${ANDROID_HOST_OUT:0:${#ANDROID_BUILD_TOP}+1}" = "${ANDROID_BUILD_TOP}/" ]; then | 
 |       framework_location="${ANDROID_HOST_OUT:${#ANDROID_BUILD_TOP}+1}/framework" | 
 |     else | 
 |       echo "error: ANDROID_BUILD_TOP/ is not a prefix of ANDROID_HOST_OUT" | 
 |       echo "ANDROID_BUILD_TOP=${ANDROID_BUILD_TOP}" | 
 |       echo "ANDROID_HOST_OUT=${ANDROID_HOST_OUT}" | 
 |       exit | 
 |     fi | 
 |     boot_classpath="$(boot_classpath_arg ${ANDROID_HOST_OUT}/framework -hostdex $BOOT_CLASSPATH_JARS)" | 
 |     boot_classpath_locations="$(boot_classpath_arg ${framework_location} -hostdex $BOOT_CLASSPATH_JARS)" | 
 |     # We do not need a device directory on host. | 
 |     device_dir="" | 
 |     # Vogar knows which VM to use on host. | 
 |     vm_command="" | 
 |     shift | 
 |   elif [[ "$1" == "--mode=jvm" ]]; then | 
 |     mode="ri" | 
 |     make_target_name="apache-harmony-jdwp-tests" | 
 |     run_internal_jdwp_test=true | 
 |     art="$(which java)" | 
 |     art_debugee="$(which java)" | 
 |     # No need for extra args. | 
 |     debuggee_args="" | 
 |     # No image. On the RI. | 
 |     image="" | 
 |     boot_classpath="" | 
 |     boot_classpath_locations="" | 
 |     # We do not need a device directory on RI. | 
 |     device_dir="" | 
 |     # Vogar knows which VM to use on RI. | 
 |     vm_command="" | 
 |     # We don't care about jit with the RI | 
 |     use_jit=false | 
 |     shift | 
 |   elif [[ $1 == --skip-test ]]; then | 
 |     skip_tests="${skip_tests},${2}" | 
 |     # remove the --skip-test | 
 |     args=${args/$1} | 
 |     shift | 
 |     # remove the arg | 
 |     args=${args/$1} | 
 |     shift | 
 |   elif [[ $1 == --force-run-test ]]; then | 
 |     run_internal_jdwp_test=true | 
 |     # remove the --force-run-test | 
 |     args=${args/$1} | 
 |     shift | 
 |   elif [[ $1 == --test-timeout-ms ]]; then | 
 |     # Remove the --test-timeout-ms from the arguments. | 
 |     args=${args/$1} | 
 |     shift | 
 |     jdwp_test_timeout=$1 | 
 |     # Remove the argument | 
 |     args=${args/$1} | 
 |     shift | 
 |   elif [[ $1 == --agent-wrapper ]]; then | 
 |     # Remove the --agent-wrapper from the arguments. | 
 |     args=${args/$1} | 
 |     shift | 
 |     agent_wrapper=${agent_wrapper}${1}, | 
 |     # Remove the argument | 
 |     args=${args/$1} | 
 |     shift | 
 |   elif [[ $1 == -Ximage:* ]]; then | 
 |     image="$1" | 
 |     shift | 
 |   elif [[ "$1" == "--instant-jit" ]]; then | 
 |     instant_jit=true | 
 |     # Remove the --instant-jit from the arguments. | 
 |     args=${args/$1} | 
 |     shift | 
 |   elif [[ "$1" == "--no-jit" ]]; then | 
 |     use_jit=false | 
 |     # Remove the --no-jit from the arguments. | 
 |     args=${args/$1} | 
 |     shift | 
 |   elif [[ $1 == "--no-debug" ]]; then | 
 |     explicit_debug="yes" | 
 |     debug="no" | 
 |     # Remove the --no-debug from the arguments. | 
 |     args=${args/$1} | 
 |     shift | 
 |   elif [[ $1 == "--debug" ]]; then | 
 |     explicit_debug="yes" | 
 |     debug="yes" | 
 |     # Remove the --debug from the arguments. | 
 |     args=${args/$1} | 
 |     shift | 
 |   elif [[ $1 == "--verbose" ]]; then | 
 |     verbose="yes" | 
 |     # Remove the --verbose from the arguments. | 
 |     args=${args/$1} | 
 |     shift | 
 |   elif [[ $1 == "--gdbserver" ]]; then | 
 |     # Remove the --gdbserver from the arguments. | 
 |     args=${args/$1} | 
 |     has_gdb="yes" | 
 |     shift | 
 |     gdb_target=$1 | 
 |     # Remove the target from the arguments. | 
 |     args=${args/$1} | 
 |     shift | 
 |   elif [[ $1 == "--test" ]]; then | 
 |     # Remove the --test from the arguments. | 
 |     args=${args/$1} | 
 |     shift | 
 |     has_specific_test="yes" | 
 |     test=$1 | 
 |     # Remove the test from the arguments. | 
 |     args=${args/$1} | 
 |     shift | 
 |   elif [[ "$1" == "--jdwp-path" ]]; then | 
 |     # Remove the --jdwp-path from the arguments. | 
 |     args=${args/$1} | 
 |     shift | 
 |     with_jdwp_path=$1 | 
 |     # Remove the path from the arguments. | 
 |     args=${args/$1} | 
 |     shift | 
 |   elif [[ "$1" == "" ]]; then | 
 |     break | 
 |   elif [[ $1 == --variant=* ]]; then | 
 |     variant_cmdline_parameter=$1 | 
 |     shift | 
 |   elif [[ $1 == -Xplugin:* ]]; then | 
 |     plugin="$1" | 
 |     args=${args/$1} | 
 |     shift | 
 |   else | 
 |     shift | 
 |   fi | 
 | done | 
 |  | 
 | if [ ! -t 1 ] ; then | 
 |   # Suppress color codes if not attached to a terminal | 
 |   args="$args --no-color" | 
 | fi | 
 |  | 
 | if [[ $mode == "target" ]]; then | 
 |   # Honor environment variable ART_TEST_CHROOT. | 
 |   if [[ -n "$ART_TEST_CHROOT" ]]; then | 
 |     # Set Vogar's `--chroot` option. | 
 |     chroot_option="--chroot $ART_TEST_CHROOT" | 
 |     # Adjust settings for chroot environment. | 
 |     art="/system/bin/art" | 
 |     art_debugee="sh /system/bin/art" | 
 |     vm_command="--vm-command=$art" | 
 |     device_dir="--device-dir=/tmp" | 
 |   fi | 
 | fi | 
 |  | 
 | if [[ $called_from_libjdwp != "true" ]]; then | 
 |   if [[ $run_internal_jdwp_test = "false" ]]; then | 
 |     echo "Calling run_jdwp_tests.sh directly is probably not what you want. You probably want to" | 
 |     echo "run ./art/tools/run-libjdwp-tests.sh instead in order to test the JDWP implementation" | 
 |     echo "used by apps. If you really wish to run these tests using the deprecated internal JDWP" | 
 |     echo "implementation pass the '--force-run-test' flag." | 
 |     exit 1 | 
 |   fi | 
 | fi | 
 |  | 
 | if [[ $has_gdb = "yes" ]]; then | 
 |   if [[ $explicit_debug = "no" ]]; then | 
 |     debug="yes" | 
 |   fi | 
 | fi | 
 |  | 
 | if [[ $mode == "ri" ]]; then | 
 |   if [[ "x$with_jdwp_path" != "x" ]]; then | 
 |     vm_args="${vm_args} --vm-arg -Djpda.settings.debuggeeAgentArgument=-agentpath:${agent_wrapper}" | 
 |     vm_args="${vm_args} --vm-arg -Djpda.settings.debuggeeAgentName=$with_jdwp_path" | 
 |   fi | 
 |   if [[ "x$image" != "x" ]]; then | 
 |     echo "Cannot use -Ximage: with --mode=jvm" | 
 |     exit 1 | 
 |   elif [[ $has_gdb = "yes" ]]; then | 
 |     echo "Cannot use --gdbserver with --mode=jvm" | 
 |     exit 1 | 
 |   elif [[ $debug == "yes" ]]; then | 
 |     echo "Cannot use --debug with --mode=jvm" | 
 |     exit 1 | 
 |   fi | 
 | else | 
 |   if [[ "$mode" == "host" ]]; then | 
 |     dump_command="/bin/kill -3" | 
 |   else | 
 |     # Note that this dumping command won't work when `$android_root` | 
 |     # is different from `/system` (e.g. on ART Buildbot devices) when | 
 |     # the device is running Android N, as the debuggerd protocol | 
 |     # changed in an incompatible way in Android O (see b/32466479). | 
 |     dump_command="$android_root/xbin/su root $android_root/bin/debuggerd" | 
 |   fi | 
 |   if [[ $has_gdb = "yes" ]]; then | 
 |     if [[ $mode == "target" ]]; then | 
 |       echo "Cannot use --gdbserver with --mode=target" | 
 |       exit 1 | 
 |     else | 
 |       art_debugee="$art_debugee --gdbserver $gdb_target" | 
 |       # The tests absolutely require some timeout. We set a ~2 week timeout since we can kill the | 
 |       # test with gdb if it goes on too long. | 
 |       jdwp_test_timeout="1000000000" | 
 |     fi | 
 |   fi | 
 |   if [[ "x$with_jdwp_path" != "x" ]]; then | 
 |     vm_args="${vm_args} --vm-arg -Djpda.settings.debuggeeAgentArgument=-agentpath:${agent_wrapper}" | 
 |     vm_args="${vm_args} --vm-arg -Djpda.settings.debuggeeAgentName=${with_jdwp_path}" | 
 |   fi | 
 |   vm_args="$vm_args --vm-arg -Xcompiler-option --vm-arg --debuggable" | 
 |   # we don't want to be trying to connect to adbconnection which might not have | 
 |   # been built. | 
 |   vm_args="${vm_args} --vm-arg -XjdwpProvider:none" | 
 |   # Make sure the debuggee doesn't re-generate, nor clean up what the debugger has generated. | 
 |   art_debugee="$art_debugee --no-compile --no-clean" | 
 | fi | 
 |  | 
 | function jlib_name { | 
 |   local path=$1 | 
 |   local str="classes" | 
 |   local suffix="jar" | 
 |   if [[ $mode == "ri" ]]; then | 
 |     str="javalib" | 
 |   fi | 
 |   echo "$path/$str.$suffix" | 
 | } | 
 |  | 
 | # Jar containing all the tests. | 
 | test_jar=$(jlib_name "${java_lib_location}/${make_target_name}_intermediates") | 
 |  | 
 | if [[ ! -f $test_jar ]]; then | 
 |   echo "Before running, you must build jdwp tests and vogar:" \ | 
 |        "m ${make_target_name} vogar" | 
 |   exit 1 | 
 | fi | 
 |  | 
 | # For the host: | 
 | # | 
 | # If, on the other hand, there is a variant set, use it to modify the art_debugee parameter to | 
 | # force the fork to have the same bitness as the controller. This should be fine and not impact | 
 | # testing (cross-bitness), as the protocol is always 64-bit anyways (our implementation). | 
 | # | 
 | # Note: this isn't necessary for the device as the BOOTCLASSPATH environment variable is set there | 
 | #       and used as a fallback. | 
 | if [[ $mode == "host" ]]; then | 
 |   variant=${variant_cmdline_parameter:10} | 
 |   if [[ $variant == "x32" || $variant == "X32" ]]; then | 
 |     art_debugee="$art_debugee --32" | 
 |   elif [[ $variant == "x64" || $variant == "X64" ]]; then | 
 |     art_debugee="$art_debugee --64" | 
 |   else | 
 |     echo "Error, do not understand variant $variant_cmdline_parameter." | 
 |     exit 1 | 
 |   fi | 
 | fi | 
 |  | 
 | if [[ "$image" != "" ]]; then | 
 |   vm_args="$vm_args --vm-arg $image" | 
 |   debuggee_args="$debuggee_args $image" | 
 | fi | 
 | if [[ "$boot_classpath" != "" ]]; then | 
 |   vm_args="$vm_args --vm-arg -Xbootclasspath:${boot_classpath}" | 
 |   debuggee_args="$debuggee_args -Xbootclasspath:${boot_classpath}" | 
 | fi | 
 | if [[ "$boot_classpath_locations" != "" ]]; then | 
 |   vm_args="$vm_args --vm-arg -Xbootclasspath-locations:${boot_classpath_locations}" | 
 |   debuggee_args="$debuggee_args -Xbootclasspath-locations:${boot_classpath_locations}" | 
 | fi | 
 |  | 
 | if [[ "$plugin" != "" ]]; then | 
 |   vm_args="$vm_args --vm-arg $plugin" | 
 | fi | 
 |  | 
 | if [[ $mode != "ri" ]]; then | 
 |   # Because we're running debuggable, we discard any AOT code. | 
 |   # Therefore we run dex2oat with 'verify' to avoid spending time compiling. | 
 |   vm_args="$vm_args --vm-arg -Xcompiler-option --vm-arg --compiler-filter=verify" | 
 |   debuggee_args="$debuggee_args -Xcompiler-option --compiler-filter=verify" | 
 |  | 
 |   if $instant_jit; then | 
 |     debuggee_args="$debuggee_args -Xjitthreshold:0" | 
 |   fi | 
 |  | 
 |   vm_args="$vm_args --vm-arg -Xusejit:$use_jit" | 
 |   debuggee_args="$debuggee_args -Xusejit:$use_jit" | 
 | fi | 
 |  | 
 | if [[ $debug == "yes" ]]; then | 
 |   art="$art -d" | 
 |   art_debugee="$art_debugee -d" | 
 |   vm_args="$vm_args --vm-arg -XXlib:libartd.so --vm-arg -XX:SlowDebug=true" | 
 | fi | 
 | if [[ $verbose == "yes" ]]; then | 
 |   # Enable JDWP logs in the debuggee. | 
 |   art_debugee="$art_debugee -verbose:jdwp" | 
 | fi | 
 |  | 
 | if [[ $mode != "ri" ]]; then | 
 |   toolchain_args="--toolchain d8 --language CUR" | 
 |   if [[ "x$with_jdwp_path" == "x" ]]; then | 
 |     # Need to enable the internal jdwp implementation. | 
 |     art_debugee="${art_debugee} -XjdwpProvider:internal" | 
 |   else | 
 |     # need to disable the jdwpProvider since we give the agent explicitly on the | 
 |     # cmdline. | 
 |     art_debugee="${art_debugee} -XjdwpProvider:none" | 
 |   fi | 
 | else | 
 |   toolchain_args="--toolchain javac --language CUR" | 
 | fi | 
 |  | 
 | # Run the tests using vogar. | 
 | vogar $vm_command \ | 
 |       $vm_args \ | 
 |       --verbose \ | 
 |       $args \ | 
 |       $chroot_option \ | 
 |       $device_dir \ | 
 |       $image_compiler_option \ | 
 |       --timeout 800 \ | 
 |       --vm-arg -Djpda.settings.verbose=true \ | 
 |       --vm-arg -Djpda.settings.timeout=$jdwp_test_timeout \ | 
 |       --vm-arg -Djpda.settings.waitingTime=$jdwp_test_timeout \ | 
 |       --vm-arg -Djpda.settings.transportAddress=127.0.0.1:55107 \ | 
 |       --vm-arg -Djpda.settings.dumpProcess="$dump_command" \ | 
 |       --vm-arg -Djpda.settings.debuggeeJavaPath="$art_debugee $plugin $debuggee_args" \ | 
 |       --vm-arg -Djpda.settings.badTestCases="$skip_tests" \ | 
 |       --classpath "$test_jar" \ | 
 |       $toolchain_args \ | 
 |       $test | 
 |  | 
 | vogar_exit_status=$? | 
 |  | 
 | echo "Killing stalled dalvikvm processes..." | 
 | if [[ $mode == "host" ]]; then | 
 |   pkill -9 -f /bin/dalvikvm | 
 | else | 
 |   # Tests may run on older Android versions where pkill requires "-l SIGNAL" | 
 |   # rather than "-SIGNAL". | 
 |   adb shell pkill -l 9 -f /bin/dalvikvm | 
 | fi | 
 | echo "Done." | 
 |  | 
 | exit $vogar_exit_status |