| /* |
| * Copyright 2020 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 dalvik.system.PathClassLoader; |
| import java.lang.reflect.Field; |
| import java.lang.reflect.Constructor; |
| import java.lang.reflect.Method; |
| |
| class Main { |
| static final String DEX_FILE = System.getenv("DEX_LOCATION") + "/596-app-images.jar"; |
| static final String SECONDARY_DEX_FILE = |
| System.getenv("DEX_LOCATION") + "/596-app-images-ex.jar"; |
| static final String LIBRARY_SEARCH_PATH = System.getProperty("java.library.path"); |
| |
| static class Inner { |
| final public static int abc = 10; |
| } |
| |
| static class Nested { |
| |
| } |
| |
| public static void main(String[] args) throws Exception { |
| System.loadLibrary(args[0]); |
| |
| testAppImageLoaded(); |
| testInitializedClasses(); |
| testInternedStrings(); |
| testReloadInternedString(); |
| testClassesOutsideAppImage(); |
| testLoadingSecondaryAppImage(); |
| } |
| |
| public static native boolean checkAppImageLoaded(String name); |
| public static native boolean checkAppImageContains(Class<?> klass); |
| public static native boolean checkInitialized(Class<?> klass); |
| |
| public static void testAppImageLoaded() throws Exception { |
| assertTrue("App image is loaded", checkAppImageLoaded("596-app-images")); |
| assertTrue("App image contains Inner", checkAppImageContains(Inner.class)); |
| } |
| |
| public static void testInitializedClasses() throws Exception { |
| assertInitialized(Inner.class); |
| assertInitialized(Nested.class); |
| assertInitialized(StaticFields.class); |
| assertInitialized(StaticFieldsInitSub.class); |
| assertInitialized(StaticFieldsInit.class); |
| assertInitialized(StaticInternString.class); |
| } |
| |
| private static void assertInitialized(Class<?> klass) { |
| assertTrue(klass.toString() + " is preinitialized", checkInitialized(klass)); |
| } |
| |
| public static void testInternedStrings() throws Exception { |
| StringBuffer sb = new StringBuffer(); |
| sb.append("java."); |
| sb.append("abc."); |
| sb.append("Action"); |
| |
| String tmp = sb.toString(); |
| String intern = tmp.intern(); |
| |
| assertNotSame("Dynamically constructed string is not interned", tmp, intern); |
| assertEquals("Static string on initialized class is matches runtime interned string", intern, |
| StaticInternString.intent); |
| assertEquals("Static string on initialized class is pre-interned", BootInternedString.boot, |
| BootInternedString.boot.intern()); |
| |
| // TODO: Does this next check really provide us anything? |
| Field f = StaticInternString.class.getDeclaredField("intent"); |
| assertEquals("String literals are interned properly", intern, f.get(null)); |
| |
| assertEquals("String literals are interned properly across classes", |
| StaticInternString.getIntent(), StaticInternString2.getIntent()); |
| } |
| |
| public static void testReloadInternedString() throws Exception { |
| // reload the class StaticInternString, check whether static strings interned properly |
| PathClassLoader loader = new PathClassLoader(DEX_FILE, LIBRARY_SEARCH_PATH, null); |
| Class<?> staticInternString = loader.loadClass("StaticInternString"); |
| assertTrue("Class in app image isn't loaded a second time after loading dex file again", |
| checkAppImageContains(staticInternString)); |
| |
| Method getIntent = staticInternString.getDeclaredMethod("getIntent"); |
| assertEquals("Interned strings are still interned after multiple dex loads", |
| StaticInternString.getIntent(), getIntent.invoke(staticInternString)); |
| } |
| |
| public static void testClassesOutsideAppImage() { |
| assertFalse("App image doesn't contain non-optimized class", |
| checkAppImageContains(NonOptimizedClass.class)); |
| assertFalse("App image didn't pre-initialize non-optimized class", |
| checkInitialized(NonOptimizedClass.class)); |
| } |
| |
| public static void testLoadingSecondaryAppImage() throws Exception { |
| final ClassLoader parent = Main.class.getClassLoader(); |
| |
| // Initial check that the image isn't already loaded so we don't get bogus results below |
| assertFalse("Secondary app image isn't already loaded", |
| checkAppImageLoaded("596-app-images-ex")); |
| |
| PathClassLoader pcl = new PathClassLoader(SECONDARY_DEX_FILE, parent); |
| |
| assertTrue("Ensure app image is loaded if it should be", |
| checkAppImageLoaded("596-app-images-ex")); |
| |
| Class<?> secondaryCls = pcl.loadClass("Secondary"); |
| assertTrue("Ensure Secondary class is in the app image if the CLC is correct", |
| checkAppImageContains(secondaryCls)); |
| assertTrue("Ensure Secondary class is preinitialized if the CLC is correct", |
| checkInitialized(secondaryCls)); |
| |
| secondaryCls.getDeclaredMethod("go").invoke(null); |
| } |
| |
| private static void assertTrue(String message, boolean flag) { |
| if (flag) { |
| return; |
| } |
| throw new AssertionError(message); |
| } |
| |
| private static void assertEquals(String message, Object a, Object b) { |
| StringBuilder sb = new StringBuilder(message != null ? message : ""); |
| if (sb.length() > 0) { |
| sb.append(" "); |
| } |
| sb.append("expected:<").append(a).append("> but was:<").append(b).append(">"); |
| assertTrue(sb.toString(), (a == null && b == null) || (a != null && a.equals(b))); |
| } |
| |
| private static void assertFalse(String message, boolean flag) { |
| assertTrue(message, !flag); |
| } |
| |
| private static void assertNotSame(String message, Object a, Object b) { |
| StringBuilder sb = new StringBuilder(message != null ? message : ""); |
| if (sb.length() > 0) { |
| sb.append(" "); |
| } |
| sb.append("unexpected sameness, found:<").append(a).append("> and:<").append(b).append(">"); |
| assertTrue(sb.toString(), a != b); |
| } |
| } |
| |
| class StaticFields { |
| public static int abc; |
| } |
| |
| class StaticFieldsInitSub extends StaticFieldsInit { |
| final public static int def = 10; |
| } |
| |
| class StaticFieldsInit { |
| final public static int abc = 10; |
| } |
| |
| class StaticInternString { |
| final public static String intent = "java.abc.Action"; |
| static public String getIntent() { |
| return intent; |
| } |
| } |
| |
| class BootInternedString { |
| final public static String boot = "double"; |
| } |
| |
| class StaticInternString2 { |
| final public static String intent = "java.abc.Action"; |
| |
| static String getIntent() { |
| return intent; |
| } |
| } |
| |
| class NonOptimizedClass {} |