You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
194 lines
6.6 KiB
194 lines
6.6 KiB
/*
|
|
* 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 {}
|