#!/usr/bin/env python3 # # [VPYTHON:BEGIN] # python_version: "3.8" # [VPYTHON:END] # # Copyright (C) 2021 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. import sys, os, argparse, subprocess, shlex, re, concurrent.futures, multiprocessing def parse_args(): parser = argparse.ArgumentParser(description="Run libcore tests using the vogar testing tool.") parser.add_argument('--mode', choices=['device', 'host', 'jvm'], required=True, help='Specify where tests should be run.') parser.add_argument('--variant', choices=['X32', 'X64'], help='Which dalvikvm variant to execute with.') parser.add_argument('-j', '--jobs', type=int, help='Number of tests to run simultaneously.') parser.add_argument('--timeout', type=int, help='How long to run the test before aborting (seconds).') parser.add_argument('--debug', action='store_true', help='Use debug version of ART (device|host only).') parser.add_argument('--dry-run', action='store_true', help='Print vogar command-line, but do not run.') parser.add_argument('--no-getrandom', action='store_false', dest='getrandom', help='Ignore failures from getrandom() (for kernel < 3.17).') parser.add_argument('--no-jit', action='store_false', dest='jit', help='Disable JIT (device|host only).') parser.add_argument('--gcstress', action='store_true', help='Enable GC stress configuration (device|host only).') parser.add_argument('tests', nargs="*", help='Name(s) of the test(s) to run') parser.add_argument('--verbose', action='store_true', help='Print verbose output from vogar.') return parser.parse_args() ART_TEST_ANDROID_ROOT = os.environ.get("ART_TEST_ANDROID_ROOT", "/system") ART_TEST_CHROOT = os.environ.get("ART_TEST_CHROOT") ANDROID_PRODUCT_OUT = os.environ.get("ANDROID_PRODUCT_OUT") LIBCORE_TEST_NAMES = [ ### luni tests. ### # Naive critical path optimization: Run the longest tests first. "org.apache.harmony.tests.java.util", # 90min under gcstress "libcore.java.lang", # 90min under gcstress "jsr166", # 60min under gcstress "libcore.java.util", # 60min under gcstress "libcore.java.math", # 50min under gcstress "org.apache.harmony.crypto", # 30min under gcstress "org.apache.harmony.tests.java.io", # 30min under gcstress "org.apache.harmony.tests.java.text", # 30min under gcstress # Split highmemorytest to individual classes since it is too big. "libcore.highmemorytest.java.text.DateFormatTest", "libcore.highmemorytest.java.text.DecimalFormatTest", "libcore.highmemorytest.java.text.SimpleDateFormatTest", "libcore.highmemorytest.java.time.format.DateTimeFormatterTest", "libcore.highmemorytest.java.util.CalendarTest", "libcore.highmemorytest.java.util.CurrencyTest", "libcore.highmemorytest.libcore.icu.SimpleDateFormatDataTest", # All other luni tests in alphabetical order. "libcore.android.system", "libcore.build", "libcore.dalvik.system", "libcore.java.awt", "libcore.java.text", "libcore.javax.crypto", "libcore.javax.net", "libcore.javax.security", "libcore.javax.sql", "libcore.javax.xml", "libcore.libcore.icu", "libcore.libcore.internal", "libcore.libcore.io", "libcore.libcore.net", "libcore.libcore.reflect", "libcore.libcore.util", "libcore.sun.invoke", "libcore.sun.misc", "libcore.sun.net", "libcore.sun.security", "libcore.sun.util", "libcore.xml", "org.apache.harmony.annotation", "org.apache.harmony.luni.tests.internal.net.www.protocol.http.HttpURLConnection", "org.apache.harmony.luni.tests.internal.net.www.protocol.https.HttpsURLConnection", "org.apache.harmony.luni.tests.java.io", "org.apache.harmony.luni.tests.java.net", "org.apache.harmony.nio", "org.apache.harmony.regex", "org.apache.harmony.testframework", "org.apache.harmony.tests.java.lang", "org.apache.harmony.tests.java.math", "org.apache.harmony.tests.javax.security", "tests.java.lang.String", ### OpenJDK upstream tests (ojluni). ### # "test.java.awt", "test.java.awt", # test.java.io "test.java.io.ByteArrayInputStream", "test.java.io.ByteArrayOutputStream", "test.java.io.FileReader", "test.java.io.FileWriter", "test.java.io.InputStream", "test.java.io.OutputStream", "test.java.io.PrintStream", "test.java.io.PrintWriter", "test.java.io.Reader", "test.java.io.Writer", # test.java.lang "test.java.lang.Boolean", "test.java.lang.ClassLoader", "test.java.lang.Double", "test.java.lang.Float", "test.java.lang.Integer", "test.java.lang.Long", # Sharded test.java.lang.StrictMath "test.java.lang.StrictMath.CubeRootTests", # TODO: disable the test until b/248208762 is fixed. # "test.java.lang.StrictMath.ExactArithTests", "test.java.lang.StrictMath.Expm1Tests", "test.java.lang.StrictMath.ExpTests", "test.java.lang.StrictMath.HyperbolicTests", "test.java.lang.StrictMath.HypotTests#testAgainstTranslit_shard1", "test.java.lang.StrictMath.HypotTests#testAgainstTranslit_shard2", "test.java.lang.StrictMath.HypotTests#testAgainstTranslit_shard3", "test.java.lang.StrictMath.HypotTests#testAgainstTranslit_shard4", "test.java.lang.StrictMath.HypotTests#testHypot", "test.java.lang.StrictMath.Log1pTests", "test.java.lang.StrictMath.Log10Tests", "test.java.lang.StrictMath.MultiplicationTests", "test.java.lang.StrictMath.PowTests", "test.java.lang.String", "test.java.lang.Thread", # test.java.lang.invoke "test.java.lang.invoke", # test.java.lang.ref "test.java.lang.ref.SoftReference", "test.java.lang.ref.BasicTest", "test.java.lang.ref.EnqueueNullRefTest", "test.java.lang.ref.EnqueuePollRaceTest", "test.java.lang.ref.ReferenceCloneTest", "test.java.lang.ref.ReferenceEnqueuePendingTest", # test.java.math "test.java.math.BigDecimal", # Sharded test.java.math.BigInteger "test.java.math.BigInteger#testArithmetic", "test.java.math.BigInteger#testBitCount", "test.java.math.BigInteger#testBitLength", "test.java.math.BigInteger#testbitOps", "test.java.math.BigInteger#testBitwise", "test.java.math.BigInteger#testByteArrayConv", "test.java.math.BigInteger#testConstructor", "test.java.math.BigInteger#testDivideAndReminder", "test.java.math.BigInteger#testDivideLarge", "test.java.math.BigInteger#testModExp", "test.java.math.BigInteger#testMultiplyLarge", "test.java.math.BigInteger#testNextProbablePrime", "test.java.math.BigInteger#testPow", "test.java.math.BigInteger#testSerialize", "test.java.math.BigInteger#testShift", "test.java.math.BigInteger#testSquare", "test.java.math.BigInteger#testSquareLarge", "test.java.math.BigInteger#testSquareRootAndReminder", "test.java.math.BigInteger#testStringConv_generic", "test.java.math.RoundingMode", # test.java.net "test.java.net.DatagramSocket", "test.java.net.Socket", "test.java.net.SocketOptions", "test.java.net.URLDecoder", "test.java.net.URLEncoder", # test.java.nio "test.java.nio.channels.Channels", "test.java.nio.channels.SelectionKey", "test.java.nio.channels.Selector", "test.java.nio.file", # test.java.security "test.java.security.cert", # Sharded test.java.security.KeyAgreement "test.java.security.KeyAgreement.KeyAgreementTest", "test.java.security.KeyAgreement.KeySizeTest#testECDHKeySize", "test.java.security.KeyAgreement.KeySpecTest", "test.java.security.KeyAgreement.MultiThreadTest", "test.java.security.KeyAgreement.NegativeTest", "test.java.security.KeyStore", "test.java.security.Provider", # test.java.time "test.java.time", # test.java.util "test.java.util.Arrays", "test.java.util.Collection", "test.java.util.Collections", "test.java.util.Date", "test.java.util.EnumMap", "test.java.util.EnumSet", "test.java.util.GregorianCalendar", "test.java.util.LinkedHashMap", "test.java.util.LinkedHashSet", "test.java.util.List", "test.java.util.Map", "test.java.util.Optional", "test.java.util.TestFormatter", "test.java.util.TimeZone", # test.java.util.concurrent "test.java.util.concurrent", # test.java.util.function "test.java.util.function", # test.java.util.stream "test.java.util.stream", # test.java.util.zip "test.java.util.zip.ZipFile", # tck.java.time "tck.java.time", ] # "org.apache.harmony.security", # We don't have rights to revert changes in case of failures. # 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 = [ "/apex/com.android.art/javalib/core-oj.jar", "/apex/com.android.art/javalib/core-libart.jar", "/apex/com.android.art/javalib/okhttp.jar", "/apex/com.android.art/javalib/bouncycastle.jar", "/apex/com.android.art/javalib/apache-xml.jar", "/apex/com.android.i18n/javalib/core-icu4j.jar", "/apex/com.android.conscrypt/javalib/conscrypt.jar", ] CLASSPATH = ["core-tests", "core-ojtests", "jsr166-tests", "mockito-target"] SLOW_OJLUNI_TESTS = { "test.java.awt", "test.java.lang.String", "test.java.lang.invoke", "test.java.nio.channels.Selector", "test.java.time", "test.java.util.Arrays", "test.java.util.Map", "test.java.util.concurrent", "test.java.util.stream", "test.java.util.zip.ZipFile", "tck.java.time", } # Disabled to unblock art-buildbot # These tests fail with "java.io.IOException: Stream closed", tracked in # http://b/235566533 and http://b/208639267 DISABLED_GCSTRESS_DEBUG_TESTS = { "test.java.lang.StrictMath.HypotTests#testAgainstTranslit_shard1", "test.java.lang.StrictMath.HypotTests#testAgainstTranslit_shard2", "test.java.lang.StrictMath.HypotTests#testAgainstTranslit_shard3", "test.java.lang.StrictMath.HypotTests#testAgainstTranslit_shard4", "test.java.math.BigDecimal", "test.java.math.BigInteger#testConstructor", "test.java.util.TestFormatter", "test.java.util.Collection", } DISABLED_FUGU_TESTS = { "org.apache.harmony.luni.tests.internal.net.www.protocol.http.HttpURLConnection", "org.apache.harmony.luni.tests.internal.net.www.protocol.https.HttpsURLConnection", "test.java.awt", "test.java.io.ByteArrayInputStream", "test.java.io.ByteArrayOutputStream", "test.java.io.InputStream", "test.java.io.OutputStream", "test.java.io.PrintStream", "test.java.io.PrintWriter", "test.java.io.Reader", "test.java.io.Writer", "test.java.lang.Boolean", "test.java.lang.ClassLoader", "test.java.lang.Double", "test.java.lang.Float", "test.java.lang.Integer", "test.java.lang.Long", "test.java.lang.StrictMath.CubeRootTests", "test.java.lang.StrictMath.Expm1Tests", "test.java.lang.StrictMath.ExpTests", "test.java.lang.StrictMath.HyperbolicTests", "test.java.lang.StrictMath.HypotTests#testAgainstTranslit_shard1", "test.java.lang.StrictMath.HypotTests#testAgainstTranslit_shard2", "test.java.lang.StrictMath.HypotTests#testAgainstTranslit_shard3", "test.java.lang.StrictMath.HypotTests#testAgainstTranslit_shard4", "test.java.lang.StrictMath.HypotTests#testHypot", "test.java.lang.StrictMath.Log1pTests", "test.java.lang.StrictMath.Log10Tests", "test.java.lang.StrictMath.MultiplicationTests", "test.java.lang.StrictMath.PowTests", "test.java.lang.String", "test.java.lang.Thread", "test.java.lang.invoke", "test.java.lang.ref.SoftReference", "test.java.lang.ref.BasicTest", "test.java.lang.ref.EnqueueNullRefTest", "test.java.lang.ref.EnqueuePollRaceTest", "test.java.lang.ref.ReferenceCloneTest", "test.java.lang.ref.ReferenceEnqueuePendingTest", "test.java.math.BigDecimal", "test.java.math.BigInteger#testArithmetic", "test.java.math.BigInteger#testBitCount", "test.java.math.BigInteger#testBitLength", "test.java.math.BigInteger#testbitOps", "test.java.math.BigInteger#testBitwise", "test.java.math.BigInteger#testByteArrayConv", "test.java.math.BigInteger#testConstructor", "test.java.math.BigInteger#testDivideAndReminder", "test.java.math.BigInteger#testDivideLarge", "test.java.math.BigInteger#testModExp", "test.java.math.BigInteger#testMultiplyLarge", "test.java.math.BigInteger#testNextProbablePrime", "test.java.math.BigInteger#testPow", "test.java.math.BigInteger#testSerialize", "test.java.math.BigInteger#testShift", "test.java.math.BigInteger#testSquare", "test.java.math.BigInteger#testSquareLarge", "test.java.math.BigInteger#testSquareRootAndReminder", "test.java.math.BigInteger#testStringConv_generic", "test.java.math.RoundingMode", "test.java.net.DatagramSocket", "test.java.net.Socket", "test.java.net.SocketOptions", "test.java.net.URLDecoder", "test.java.net.URLEncoder", "test.java.nio.channels.Channels", "test.java.nio.channels.SelectionKey", "test.java.nio.channels.Selector", "test.java.nio.file", "test.java.security.cert", "test.java.security.KeyAgreement.KeyAgreementTest", "test.java.security.KeyAgreement.KeySizeTest#testECDHKeySize", "test.java.security.KeyAgreement.KeySpecTest", "test.java.security.KeyAgreement.MultiThreadTest", "test.java.security.KeyAgreement.NegativeTest", "test.java.security.KeyStore", "test.java.security.Provider", "test.java.time", "test.java.util.Arrays", "test.java.util.Collection", "test.java.util.Collections", "test.java.util.Date", "test.java.util.EnumMap", "test.java.util.EnumSet", "test.java.util.GregorianCalendar", "test.java.util.LinkedHashMap", "test.java.util.LinkedHashSet", "test.java.util.List", "test.java.util.Map", "test.java.util.Optional", "test.java.util.TestFormatter", "test.java.util.TimeZone", "test.java.util.function", "test.java.util.stream", "tck.java.time", } def get_jar_filename(classpath): base_path = (ANDROID_PRODUCT_OUT + "/../..") if ANDROID_PRODUCT_OUT else "out/target" base_path = os.path.normpath(base_path) # Normalize ".." components for readability. return f"{base_path}/common/obj/JAVA_LIBRARIES/{classpath}_intermediates/classes.jar" def get_timeout_secs(): default_timeout_secs = 600 if args.gcstress: default_timeout_secs = 1200 if args.debug: default_timeout_secs = 1800 return args.timeout or default_timeout_secs def get_expected_failures(): failures = ["art/tools/libcore_failures.txt"] if args.mode != "jvm": if args.gcstress: failures.append("art/tools/libcore_gcstress_failures.txt") if args.gcstress and args.debug: failures.append("art/tools/libcore_gcstress_debug_failures.txt") if args.debug and not args.gcstress and args.getrandom: failures.append("art/tools/libcore_debug_failures.txt") if not args.getrandom: failures.append("art/tools/libcore_fugu_failures.txt") return failures def get_test_names(): if args.tests: return args.tests test_names = list(LIBCORE_TEST_NAMES) # See b/78228743 and b/178351808. if args.gcstress or args.debug or args.mode == "jvm": test_names = list(t for t in test_names if not t.startswith("libcore.highmemorytest")) test_names = list(filter(lambda x: x not in SLOW_OJLUNI_TESTS, test_names)) if args.gcstress and args.debug: test_names = list(filter(lambda x: x not in DISABLED_GCSTRESS_DEBUG_TESTS, test_names)) if not args.getrandom: # Disable libcore.highmemorytest due to limited ram on fugu. http://b/258173036 test_names = list(filter(lambda x: x not in DISABLED_FUGU_TESTS and not x.startswith("libcore.highmemorytest"), test_names)) return test_names def get_vogar_command(test_name): cmd = ["vogar"] if args.mode == "device": cmd.append("--mode=device --vm-arg -Ximage:/system/framework/art_boot_images/boot.art") cmd.append("--vm-arg -Xbootclasspath:" + ":".join(BOOT_CLASSPATH)) if args.mode == "host": # We explicitly give a wrong path for the image, to ensure vogar # will create a boot image with the default compiler. Note that # giving an existing image on host does not work because of # classpath/resources differences when compiling the boot image. cmd.append("--mode=host --vm-arg -Ximage:/non/existent/vogar.art") if args.mode == "jvm": cmd.append("--mode=jvm") if args.variant: cmd.append("--variant=" + args.variant) if args.gcstress: cmd.append("--vm-arg -Xgc:gcstress") cmd.append('--vm-arg -Djsr166.delay.factor="1.50"') if args.debug: cmd.append("--vm-arg -XXlib:libartd.so --vm-arg -XX:SlowDebug=true") # The only device in go/art-buildbot without getrandom is fugu. We limit the amount of memory # per runtime for fugu to avoid low memory killer, fugu has 4-cores 1GB RAM (b/258171768). if not args.getrandom: cmd.append("--vm-arg -Xmx128M") if args.mode == "device": if ART_TEST_CHROOT: cmd.append(f"--chroot {ART_TEST_CHROOT} --device-dir=/tmp/vogar/test-{test_name}") else: cmd.append(f"--device-dir=/data/local/tmp/vogar/test-{test_name}") cmd.append(f"--vm-command={ART_TEST_ANDROID_ROOT}/bin/art") else: cmd.append(f"--device-dir=/tmp/vogar/test-{test_name}") if args.mode != "jvm": cmd.append("--timeout {}".format(get_timeout_secs())) cmd.append("--toolchain d8 --language CUR") if args.jit: cmd.append("--vm-arg -Xcompiler-option --vm-arg --compiler-filter=verify") cmd.append("--vm-arg -Xusejit:{}".format(str(args.jit).lower())) if args.verbose: cmd.append("--verbose") # Suppress color codes if not attached to a terminal if not sys.stdout.isatty(): cmd.append("--no-color") cmd.extend("--expectations " + f for f in get_expected_failures()) cmd.extend("--classpath " + get_jar_filename(cp) for cp in CLASSPATH) cmd.append(test_name) # vogar target options if not os.path.exists('frameworks/base'): cmd.append("--") # Skip @NonMts test in thin manifest which uses prebuilt Conscrypt and ICU. # It's similar to running libcore tests on the older platforms. # @NonMts means that the test doesn't pass on a older platform version. cmd.append("--exclude-filter libcore.test.annotation.NonMts") return cmd def get_target_cpu_count(): adb_command = 'adb shell cat /sys/devices/system/cpu/present' with subprocess.Popen(adb_command.split(), stderr=subprocess.STDOUT, stdout=subprocess.PIPE, universal_newlines=True) as proc: assert(proc.wait() == 0) # Check the exit code. match = re.match(r'\d*-(\d*)', proc.stdout.read()) assert(match) return int(match.group(1)) + 1 # Add one to convert from "last-index" to "count" def main(): global args args = parse_args() if not os.path.exists('build/envsetup.sh'): raise AssertionError("Script needs to be run at the root of the android tree") for jar in map(get_jar_filename, CLASSPATH): if not os.path.exists(jar): raise AssertionError(f"Missing {jar}. Run buildbot-build.sh first.") if not args.jobs: if args.mode == "device": args.jobs = get_target_cpu_count() else: args.jobs = multiprocessing.cpu_count() if args.gcstress: # TODO: Investigate and fix the underlying issues. args.jobs = args.jobs // 2 def run_test(test_name): cmd = " ".join(get_vogar_command(test_name)) if args.dry_run: return test_name, cmd, "Dry-run: skipping execution", 0 with subprocess.Popen(shlex.split(cmd), stderr=subprocess.STDOUT, stdout=subprocess.PIPE, universal_newlines=True) as proc: return test_name, cmd, proc.communicate()[0], proc.wait() failed_regex = re.compile(r"^.* FAIL \((?:EXEC_FAILED|ERROR)\)$", re.MULTILINE) failed_tests, max_exit_code = [], 0 with concurrent.futures.ThreadPoolExecutor(max_workers=args.jobs) as pool: futures = [pool.submit(run_test, test_name) for test_name in get_test_names()] print(f"Running {len(futures)} tasks on {args.jobs} core(s)...\n") for i, future in enumerate(concurrent.futures.as_completed(futures)): test_name, cmd, stdout, exit_code = future.result() if exit_code != 0 or args.dry_run or args.verbose: print(cmd) print(stdout.strip()) else: print(stdout.strip().split("\n")[-1]) # Vogar final summary line. failed_match = failed_regex.findall(stdout) failed_tests.extend(failed_match) max_exit_code = max(max_exit_code, exit_code) result = "PASSED" if exit_code == 0 else f"FAILED ({len(failed_match)} test(s) failed)" print(f"[{i+1}/{len(futures)}] Test set {test_name} {result}\n") print(f"Overall, {len(failed_tests)} test(s) failed:") print("\n".join(failed_tests)) sys.exit(max_exit_code) if __name__ == '__main__': main()