2014-06-04 11:05:30 +02:00
|
|
|
/*
|
2015-03-26 16:36:56 +01:00
|
|
|
* Copyright (c) 2014, 2015, Oracle and/or its affiliates. All rights reserved.
|
2014-06-04 11:05:30 +02:00
|
|
|
* 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.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/*
|
|
|
|
* @test
|
2015-01-30 20:00:57 +01:00
|
|
|
* @summary Test of diagnostic command VM.classloader_stats
|
|
|
|
* @library /testlibrary
|
2015-03-26 16:36:56 +01:00
|
|
|
* @modules java.base/sun.misc
|
|
|
|
* java.compiler
|
|
|
|
* java.management
|
|
|
|
* jdk.jvmstat/sun.jvmstat.monitor
|
2015-01-30 20:00:57 +01:00
|
|
|
* @build com.oracle.java.testlibrary.*
|
|
|
|
* @build com.oracle.java.testlibrary.dcmd.*
|
|
|
|
* @run testng ClassLoaderStatsTest
|
2014-06-04 11:05:30 +02:00
|
|
|
*/
|
|
|
|
|
2015-01-30 20:00:57 +01:00
|
|
|
import org.testng.annotations.Test;
|
|
|
|
import org.testng.Assert;
|
|
|
|
|
|
|
|
import com.oracle.java.testlibrary.OutputAnalyzer;
|
|
|
|
import com.oracle.java.testlibrary.dcmd.CommandExecutor;
|
|
|
|
import com.oracle.java.testlibrary.dcmd.JMXExecutor;
|
|
|
|
|
2014-06-04 11:05:30 +02:00
|
|
|
import java.io.File;
|
|
|
|
import java.io.FileInputStream;
|
|
|
|
import java.io.IOException;
|
|
|
|
import java.nio.ByteBuffer;
|
|
|
|
import java.nio.channels.FileChannel;
|
2015-01-30 20:00:57 +01:00
|
|
|
import java.util.Iterator;
|
2014-06-04 11:05:30 +02:00
|
|
|
import java.util.regex.Matcher;
|
|
|
|
import java.util.regex.Pattern;
|
|
|
|
|
|
|
|
public class ClassLoaderStatsTest {
|
|
|
|
|
|
|
|
// ClassLoader Parent CLD* Classes ChunkSz BlockSz Type
|
|
|
|
// 0x00000007c0215928 0x0000000000000000 0x0000000000000000 0 0 0 org.eclipse.osgi.baseadaptor.BaseAdaptor$1
|
|
|
|
// 0x00000007c0009868 0x0000000000000000 0x00007fc52aebcc80 1 6144 3768 sun.reflect.DelegatingClassLoader
|
|
|
|
// 0x00000007c0009868 0x0000000000000000 0x00007fc52b8916d0 1 6144 3688 sun.reflect.DelegatingClassLoader
|
|
|
|
// 0x00000007c0009868 0x00000007c0038ba8 0x00007fc52afb8760 1 6144 3688 sun.reflect.DelegatingClassLoader
|
|
|
|
// 0x00000007c0009868 0x0000000000000000 0x00007fc52afbb1a0 1 6144 3688 sun.reflect.DelegatingClassLoader
|
|
|
|
// 0x0000000000000000 0x0000000000000000 0x00007fc523416070 5019 30060544 29956216 <boot classloader>
|
|
|
|
// 455 1210368 672848 + unsafe anonymous classes
|
|
|
|
// 0x00000007c016b5c8 0x00000007c0038ba8 0x00007fc52a995000 5 8192 5864 org.netbeans.StandardModule$OneModuleClassLoader
|
|
|
|
// 0x00000007c0009868 0x00000007c016b5c8 0x00007fc52ac13640 1 6144 3896 sun.reflect.DelegatingClassLoader
|
|
|
|
// ...
|
|
|
|
|
|
|
|
static Pattern clLine = Pattern.compile("0x\\p{XDigit}*\\s*0x\\p{XDigit}*\\s*0x\\p{XDigit}*\\s*(\\d*)\\s*(\\d*)\\s*(\\d*)\\s*(.*)");
|
|
|
|
static Pattern anonLine = Pattern.compile("\\s*(\\d*)\\s*(\\d*)\\s*(\\d*)\\s*.*");
|
|
|
|
|
|
|
|
public static DummyClassLoader dummyloader;
|
|
|
|
|
2015-01-30 20:00:57 +01:00
|
|
|
public void run(CommandExecutor executor) throws ClassNotFoundException {
|
2014-06-04 11:05:30 +02:00
|
|
|
|
|
|
|
// create a classloader and load our special class
|
|
|
|
dummyloader = new DummyClassLoader();
|
|
|
|
Class<?> c = Class.forName("TestClass", true, dummyloader);
|
|
|
|
if (c.getClassLoader() != dummyloader) {
|
2015-01-30 20:00:57 +01:00
|
|
|
Assert.fail("TestClass defined by wrong classloader: " + c.getClassLoader());
|
2014-06-04 11:05:30 +02:00
|
|
|
}
|
|
|
|
|
2015-01-30 20:00:57 +01:00
|
|
|
OutputAnalyzer output = executor.execute("VM.classloader_stats");
|
|
|
|
Iterator<String> lines = output.asLines().iterator();
|
|
|
|
while (lines.hasNext()) {
|
|
|
|
String line = lines.next();
|
2014-06-04 11:05:30 +02:00
|
|
|
Matcher m = clLine.matcher(line);
|
|
|
|
if (m.matches()) {
|
|
|
|
// verify that DummyClassLoader has loaded 1 class and 1 anonymous class
|
|
|
|
if (m.group(4).equals("ClassLoaderStatsTest$DummyClassLoader")) {
|
|
|
|
System.out.println("line: " + line);
|
|
|
|
if (!m.group(1).equals("1")) {
|
2015-01-30 20:00:57 +01:00
|
|
|
Assert.fail("Should have loaded 1 class: " + line);
|
2014-06-04 11:05:30 +02:00
|
|
|
}
|
|
|
|
checkPositiveInt(m.group(2));
|
|
|
|
checkPositiveInt(m.group(3));
|
|
|
|
|
2015-01-30 20:00:57 +01:00
|
|
|
String next = lines.next();
|
2014-06-04 11:05:30 +02:00
|
|
|
System.out.println("next: " + next);
|
|
|
|
Matcher m1 = anonLine.matcher(next);
|
|
|
|
m1.matches();
|
|
|
|
if (!m1.group(1).equals("1")) {
|
2015-01-30 20:00:57 +01:00
|
|
|
Assert.fail("Should have loaded 1 anonymous class, but found : " + m1.group(1));
|
2014-06-04 11:05:30 +02:00
|
|
|
}
|
|
|
|
checkPositiveInt(m1.group(2));
|
|
|
|
checkPositiveInt(m1.group(3));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-01-30 20:00:57 +01:00
|
|
|
private static void checkPositiveInt(String s) {
|
2014-06-04 11:05:30 +02:00
|
|
|
if (Integer.parseInt(s) <= 0) {
|
2015-01-30 20:00:57 +01:00
|
|
|
Assert.fail("Value should have been > 0: " + s);
|
2014-06-04 11:05:30 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public static class DummyClassLoader extends ClassLoader {
|
|
|
|
|
|
|
|
public static final String CLASS_NAME = "TestClass";
|
|
|
|
|
|
|
|
static ByteBuffer readClassFile(String name)
|
|
|
|
{
|
|
|
|
File f = new File(System.getProperty("test.classes", "."),
|
|
|
|
name);
|
|
|
|
try (FileInputStream fin = new FileInputStream(f);
|
|
|
|
FileChannel fc = fin.getChannel())
|
|
|
|
{
|
|
|
|
return fc.map(FileChannel.MapMode.READ_ONLY, 0, fc.size());
|
|
|
|
} catch (IOException e) {
|
2015-01-30 20:00:57 +01:00
|
|
|
Assert.fail("Can't open file: " + name, e);
|
2014-06-04 11:05:30 +02:00
|
|
|
}
|
2015-01-30 20:00:57 +01:00
|
|
|
|
|
|
|
/* Will not reach here as Assert.fail() throws exception */
|
|
|
|
return null;
|
2014-06-04 11:05:30 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
protected Class<?> loadClass(String name, boolean resolve)
|
|
|
|
throws ClassNotFoundException
|
|
|
|
{
|
|
|
|
Class<?> c;
|
|
|
|
if (!"TestClass".equals(name)) {
|
|
|
|
c = super.loadClass(name, resolve);
|
|
|
|
} else {
|
|
|
|
// should not delegate to the system class loader
|
|
|
|
c = findClass(name);
|
|
|
|
if (resolve) {
|
|
|
|
resolveClass(c);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return c;
|
|
|
|
}
|
|
|
|
|
|
|
|
protected Class<?> findClass(String name)
|
|
|
|
throws ClassNotFoundException
|
|
|
|
{
|
|
|
|
if (!"TestClass".equals(name)) {
|
|
|
|
throw new ClassNotFoundException("Unexpected class: " + name);
|
|
|
|
}
|
|
|
|
return defineClass(name, readClassFile(name + ".class"), null);
|
|
|
|
}
|
|
|
|
} /* DummyClassLoader */
|
|
|
|
|
2015-01-30 20:00:57 +01:00
|
|
|
@Test
|
|
|
|
public void jmx() throws ClassNotFoundException {
|
|
|
|
run(new JMXExecutor());
|
|
|
|
}
|
2014-06-04 11:05:30 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
class TestClass {
|
|
|
|
static {
|
|
|
|
// force creation of anonymous class (for the lambdaform)
|
|
|
|
Runnable r = () -> System.out.println("Hello");
|
|
|
|
r.run();
|
|
|
|
}
|
|
|
|
}
|