194 lines
		
	
	
		
			6.6 KiB
		
	
	
	
		
			Java
		
	
	
	
			
		
		
	
	
			194 lines
		
	
	
		
			6.6 KiB
		
	
	
	
		
			Java
		
	
	
	
| /*
 | |
|  * 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 {}
 |