/* * Copyright (C) 2016 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 java.io.File; import java.io.IOException; import java.lang.reflect.Method; import java.time.Duration; public class Main { public static void main(String[] args) throws Exception { System.loadLibrary(args[0]); if (!hasJit()) { // Test requires JIT for creating profiling infos. return; } // Register `file2` with an empty jar. Even though `file2` is registered before `file`, the // runtime should not write bootclasspath methods to `file2`, and it should not even create // `file2`. File file2 = createTempFile(); file2.deleteOnExit(); String emptyJarPath = System.getenv("DEX_LOCATION") + "/res/art-gtest-jars-MainEmptyUncompressed.jar"; VMRuntime.registerAppInfo("test.app", file2.getPath(), file2.getPath(), new String[] {emptyJarPath}, VMRuntime.CODE_PATH_TYPE_SPLIT_APK); File file = createTempFile(); file.deleteOnExit(); String codePath = System.getenv("DEX_LOCATION") + "/595-profile-saving.jar"; VMRuntime.registerAppInfo("test.app", file.getPath(), file.getPath(), new String[] {codePath}, VMRuntime.CODE_PATH_TYPE_PRIMARY_APK); File file3 = createTempFile(); file3.deleteOnExit(); String dexPath = System.getenv("DEX_LOCATION") + "/res/art-gtest-jars-Main.dex"; VMRuntime.registerAppInfo("test.app", file3.getPath(), file3.getPath(), new String[] {dexPath}, VMRuntime.CODE_PATH_TYPE_SPLIT_APK); // Delete the files so that we can check if the runtime creates them. The runtime should // create `file` and `file3` but not `file2`. file.delete(); file2.delete(); file3.delete(); // Test that the runtime saves the profiling info of an app method in a .jar file. Method appMethod = Main.class.getDeclaredMethod("testAddMethodToProfile", File.class, Method.class); testAddMethodToProfile(file, appMethod); // Test that the runtime saves the profiling info of an app method in a .dex file. ClassLoader dexClassLoader = (ClassLoader) Class.forName("dalvik.system.PathClassLoader") .getDeclaredConstructor(String.class, ClassLoader.class) .newInstance(dexPath, null /* parent */); Class c = Class.forName("Main", true /* initialize */, dexClassLoader); Method methodInDex = c.getMethod("main", (new String[0]).getClass()); testAddMethodToProfile(file3, methodInDex); // Test that the runtime saves the profiling info of a bootclasspath method. Method bootMethod = File.class.getDeclaredMethod("exists"); if (bootMethod.getDeclaringClass().getClassLoader() != Object.class.getClassLoader()) { System.out.println("Class loader does not match boot class"); } testAddMethodToProfile(file, bootMethod); // We never expect System.console to be executed before Main.main gets invoked, and therefore // it should never be in a profile. Method bootNotInProfileMethod = System.class.getDeclaredMethod("console"); testMethodNotInProfile(file, bootNotInProfileMethod); testProfileNotExist(file2); if (!isForBootImage(file.getPath())) { throw new Error("Expected profile to be for boot image"); } // Test that: // 1. The runtime always writes to disk upon a forced save, even if there is nothing to update. // 2. The profile for the primary APK is always the last one to write. // The checks may yield false negatives. Repeat multiple times to reduce the chance of false // negatives. for (int i = 0; i < 10; i++) { Duration primaryTimestampBefore = getMTime(file.getPath()); Duration splitTimestampBefore = getMTime(file3.getPath()); ensureProfileProcessing(); Duration primaryTimestampAfter = getMTime(file.getPath()); Duration splitTimestampAfter = getMTime(file3.getPath()); if (primaryTimestampAfter.compareTo(primaryTimestampBefore) <= 0) { throw new Error( String.format("Profile for primary APK not updated (before: %d, after: %d)", primaryTimestampBefore.toNanos(), primaryTimestampAfter.toNanos())); } if (splitTimestampAfter.compareTo(splitTimestampBefore) <= 0) { throw new Error( String.format("Profile for split APK not updated (before: %d, after: %d)", primaryTimestampBefore.toNanos(), primaryTimestampAfter.toNanos())); } if (primaryTimestampAfter.compareTo(splitTimestampAfter) < 0) { throw new Error(String.format( "Profile for primary APK is unexpected updated before profile for " + "split APK (primary: %d, split: %d)", primaryTimestampAfter.toNanos(), splitTimestampAfter.toNanos())); } } } static void testAddMethodToProfile(File file, Method m) { // Make sure we have a profile info for this method without the need to loop. ensureProfilingInfo(m); // Make sure the profile gets saved. ensureProfileProcessing(); // Verify that the profile was saved and contains the method. if (!presentInProfile(file.getPath(), m)) { throw new RuntimeException("Expected method " + m + " to be in the profile"); } } static void testMethodNotInProfile(File file, Method m) { // Make sure the profile gets saved. ensureProfileProcessing(); // Verify that the profile was saved and contains the method. if (presentInProfile(file.getPath(), m)) { throw new RuntimeException("Did not expect method " + m + " to be in the profile"); } } static void testProfileNotExist(File file) { // Make sure the profile saving has been attempted. ensureProfileProcessing(); // Verify that the profile does not exist. if (file.exists()) { throw new RuntimeException("Did not expect " + file + " to exist"); } } // Ensure a method has a profiling info. public static void ensureProfilingInfo(Method method) { ensureJitBaselineCompiled(method.getDeclaringClass(), method.getName()); } public static native void ensureJitBaselineCompiled(Class cls, String methodName); // Ensures the profile saver does its usual processing. public static native void ensureProfileProcessing(); // Checks if the profiles saver knows about the method. public static native boolean presentInProfile(String profile, Method method); // Returns true if the profile is for the boot image. public static native boolean isForBootImage(String profile); public static native boolean hasJit(); private static final String TEMP_FILE_NAME_PREFIX = "temp"; private static final String TEMP_FILE_NAME_SUFFIX = "-file"; static native String getProfileInfoDump( String filename); private static File createTempFile() throws Exception { try { return File.createTempFile(TEMP_FILE_NAME_PREFIX, TEMP_FILE_NAME_SUFFIX); } catch (IOException e) { System.setProperty("java.io.tmpdir", "/data/local/tmp"); try { return File.createTempFile(TEMP_FILE_NAME_PREFIX, TEMP_FILE_NAME_SUFFIX); } catch (IOException e2) { System.setProperty("java.io.tmpdir", "/sdcard"); return File.createTempFile(TEMP_FILE_NAME_PREFIX, TEMP_FILE_NAME_SUFFIX); } } } private static Duration getMTime(String path) throws Exception { // We cannot use `Files.getLastModifiedTime` because it doesn't have nanosecond precision. StructTimespec st_mtim = Os.stat(path).st_mtim; return Duration.ofSeconds(st_mtim.tv_sec).plus(Duration.ofNanos(st_mtim.tv_nsec)); } private static class VMRuntime { public static final int CODE_PATH_TYPE_PRIMARY_APK = 1 << 0; public static final int CODE_PATH_TYPE_SPLIT_APK = 1 << 1; private static final Method registerAppInfoMethod; static { try { Class c = Class.forName("dalvik.system.VMRuntime"); registerAppInfoMethod = c.getDeclaredMethod("registerAppInfo", String.class, String.class, String.class, String[].class, int.class); } catch (Exception e) { throw new RuntimeException(e); } } public static void registerAppInfo( String packageName, String curProfile, String refProfile, String[] codePaths, int codePathsType) throws Exception { registerAppInfoMethod.invoke( null, packageName, curProfile, refProfile, codePaths, codePathsType); } } private static class Os { public static StructStat stat(String path) throws Exception { return new StructStat(Class.forName("android.system.Os") .getMethod("stat", String.class) .invoke(null, path)); } } private static class StructStat { public final StructTimespec st_mtim; public StructStat(Object instance) throws Exception { st_mtim = new StructTimespec(instance.getClass().getField("st_mtim").get(instance)); } } private static class StructTimespec { public final long tv_nsec; public final long tv_sec; public StructTimespec(Object instance) throws Exception { tv_nsec = (long) instance.getClass().getField("tv_nsec").get(instance); tv_sec = (long) instance.getClass().getField("tv_sec").get(instance); } } }