8177530: Module system implementation refresh (4/2017)
Reviewed-by: mchung, alanb
Contributed-by: alan.bateman@oracle.com, mandy.chung@oracle.com
/*
* Copyright (c) 2005, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
*
* This code is free software; you can redistribute it and/or modify it
* under the terms of the GNU General Public License version 2 only, as
* published by the Free Software Foundation.
*
* This code is distributed in the hope that it will be useful, but WITHOUT
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
* version 2 for more details (a copy is included in the LICENSE file that
* accompanied this code).
*
* You should have received a copy of the GNU General Public License version
* 2 along with this work; if not, write to the Free Software Foundation,
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
*
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
* or visit www.oracle.com if you need additional information or have any
* questions.
*/
/*
*
*
* Unit test for Instrumentation appendToSystemClassLoaderSearch. This
* test does the following:
*
* 1. Creates a class loader to load class Foo. Execute Foo.doSomething
* which references a missing class Bar. The doSomething method
* should fail with NoClassDefFoundError.
*
* 2. Add Bar.jar to the system class path. Bar.jar contains Bar.
*
* 3. Create another class loader to load Foo. Execute Foo.doSomething.
* doSomething will load Bar.
*
* 4. Re-execute the first Foo's doSomething - it should fail a second
* time because the attempt to resolve Bar must fail with the same
* error as the first attempt.
*
* 5. De-reference both class loaders and execute System.gc(). We can't
* assert that the Foo classes will be unloaded but it serves to
* exercise the unload code path in HotSpot.
*/
import java.lang.instrument.Instrumentation;
import java.io.File;
import java.net.URL;
import java.net.URLClassLoader;
import java.lang.reflect.Method;
import java.lang.reflect.InvocationTargetException;
import java.util.jar.JarFile;
public class ClassUnloadTest {
static Instrumentation ins;
public static void main(String args[]) throws Exception {
String dir = args[0] + File.separator;
String jar = dir + args[1];
System.out.println(jar);
URL u = (new File(dir)).toURL();
URL urls[] = { u };
// This should fail as Bar is not available
Invoker i1 = new Invoker(urls, "Foo", "doSomething");
Boolean result = (Boolean)i1.invoke((Object)null);
if (result.booleanValue()) {
throw new RuntimeException("Test configuration error - doSomething should not succeed");
}
// put Bar on the system class path
ins.appendToSystemClassLoaderSearch( new JarFile(jar) );
// This should fail even though Bar is now available
result = (Boolean)i1.invoke((Object)null);
if (result.booleanValue()) {
throw new RuntimeException("Test configuration error - doSomething should not succeed");
}
// This should succeed because this is a different Foo
Invoker i2 = new Invoker(urls, "Foo", "doSomething");
result = (Boolean)i2.invoke((Object)null);
if (!result.booleanValue()) {
throw new RuntimeException("Test configuration error - doSomething did not succeed");
}
// Exercise some class unloading
i1 = i2 = null;
System.gc();
}
static class Invoker {
URLClassLoader cl;
Method m;
public Invoker(URL urls[], String cn, String mn, Class ... params)
throws ClassNotFoundException, NoSuchMethodException
{
cl = new URLClassLoader(urls);
Class c = Class.forName("Foo", true, cl);
m = c.getDeclaredMethod(mn, params);
}
public Object invoke(Object ... args)
throws IllegalAccessException, InvocationTargetException
{
return m.invoke(args);
}
}
public static void premain(String args, Instrumentation i) {
ins = i;
}
}