/*
|
* Copyright (C) 2017 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.lang.reflect.Constructor;
|
import java.lang.reflect.Method;
|
|
public class Main {
|
public static String TEST_NAME = "155-java-set-resolved-type";
|
|
public static void main(String[] args) {
|
try {
|
Class<?> class_loader_class = Class.forName("dalvik.system.PathClassLoader");
|
System.loadLibrary(args[0]);
|
} catch (ClassNotFoundException e) {
|
usingRI = true;
|
// Add expected JNI_OnLoad log line to match expected.txt.
|
System.out.println("JNI_OnLoad called");
|
}
|
try {
|
String dex_location = System.getenv("DEX_LOCATION");
|
ClassLoader systemLoader = ClassLoader.getSystemClassLoader().getParent();
|
ClassLoader exLoader = getClassLoaderFor(dex_location, systemLoader, /* ex */ true);
|
ClassLoader mainLoader = getClassLoaderFor(dex_location, exLoader, /* ex */ false);
|
|
// Resolve TestParameter class. It shall be defined by mainLoader.
|
// This does not resolve method parameter types.
|
Class<?> tpc = Class.forName("TestParameter", false, mainLoader);
|
// Get declared methods of TestParameter.
|
// This still does not resolve method parameter types.
|
Method[] ms = tpc.getDeclaredMethods();
|
if (ms == null || ms.length != 1) { throw new Error("Unexpected methods"); };
|
// Call getParameterTypes() to resolve parameter types. The parameter type
|
// TestInterface shall be defined by the exLoader. This used to store the
|
// TestInterface class in the dex cache resolved types for the mainLoader
|
// but not in the mainLoader's class table. This discrepancy used to cause
|
// a crash further down.
|
ms[0].getParameterTypes();
|
|
// Resolve but do not initialize TestImplementation. During the resolution,
|
// we see the TestInterface in the dex cache, so we do not try to look it up
|
// or resolve it using the mainLoader.
|
Class<?> timpl = Class.forName("TestImplementation", false, mainLoader);
|
// Clear the dex cache resolved types to force a proper lookup the next time
|
// we need to find TestInterface.
|
clearResolvedTypes(timpl);
|
|
// Force intialization of TestImplementation. This expects the interface type
|
// to be resolved and found through simple lookup.
|
timpl.newInstance();
|
} catch (Throwable t) {
|
t.printStackTrace(System.out);
|
}
|
}
|
|
public static ClassLoader getClassLoaderFor(String location, ClassLoader parent, boolean ex)
|
throws Exception {
|
try {
|
Class<?> class_loader_class = Class.forName("dalvik.system.PathClassLoader");
|
Constructor<?> ctor =
|
class_loader_class.getConstructor(String.class, ClassLoader.class);
|
/* on Dalvik, this is a DexFile; otherwise, it's null */
|
String path = location + "/" + TEST_NAME + (ex ? "-ex.jar" : ".jar");
|
return (ClassLoader)ctor.newInstance(path, parent);
|
} catch (ClassNotFoundException e) {
|
// Running on RI. Use URLClassLoader.
|
String url = "file://" + location + (ex ? "/classes-ex/" : "/classes/");
|
return new java.net.URLClassLoader(
|
new java.net.URL[] { new java.net.URL(url) }, parent);
|
}
|
}
|
|
public static void clearResolvedTypes(Class<?> c) {
|
if (!usingRI) {
|
nativeClearResolvedTypes(c);
|
}
|
}
|
|
private static boolean usingRI = false;
|
|
public static native void nativeClearResolvedTypes(Class<?> c);
|
}
|