2018-05-22 13:05:22 -07:00
|
|
|
/*
|
2022-07-08 15:55:14 +00:00
|
|
|
* Copyright (c) 2013, 2022, Oracle and/or its affiliates. All rights reserved.
|
2018-05-22 13:05:22 -07: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
|
2020-05-05 09:54:51 -07:00
|
|
|
* @key randomness
|
2018-05-22 13:05:22 -07:00
|
|
|
* @modules java.base/jdk.internal.misc:+open
|
|
|
|
*
|
|
|
|
* @summary converted from VM Testbase metaspace/staticReferences.
|
|
|
|
* VM Testbase keywords: [nonconcurrent, javac, no_cds]
|
|
|
|
*
|
2018-06-05 15:56:14 +02:00
|
|
|
* @requires vm.opt.final.ClassUnloading
|
2018-05-22 13:05:22 -07:00
|
|
|
* @library /vmTestbase /test/lib
|
2022-07-08 15:55:14 +00:00
|
|
|
* @build jdk.test.whitebox.WhiteBox
|
|
|
|
* @run driver jdk.test.lib.helpers.ClassFileInstaller jdk.test.whitebox.WhiteBox
|
2018-05-22 13:05:22 -07:00
|
|
|
* @run main/othervm
|
|
|
|
* -Xmx800m
|
|
|
|
* -Xbootclasspath/a:.
|
|
|
|
* -XX:+UnlockDiagnosticVMOptions
|
|
|
|
* -XX:+WhiteBoxAPI
|
2021-05-13 12:46:54 +00:00
|
|
|
* StaticReferences
|
2018-05-22 13:05:22 -07:00
|
|
|
*/
|
|
|
|
|
2021-05-13 12:46:54 +00:00
|
|
|
import java.lang.invoke.MethodHandles;
|
|
|
|
import java.lang.invoke.MethodHandles.Lookup;
|
2018-05-22 13:05:22 -07:00
|
|
|
import java.lang.ref.WeakReference;
|
|
|
|
import java.lang.ref.Reference;
|
|
|
|
import java.lang.reflect.Field;
|
|
|
|
import java.lang.reflect.Modifier;
|
|
|
|
import java.util.HashMap;
|
|
|
|
import java.util.LinkedList;
|
|
|
|
import java.util.List;
|
|
|
|
import java.util.Map;
|
|
|
|
import java.util.Random;
|
|
|
|
|
|
|
|
import vm.share.InMemoryJavaCompiler;
|
|
|
|
import nsk.share.gc.GCTestBase;
|
|
|
|
import nsk.share.test.ExecutionController;
|
|
|
|
import nsk.share.test.Stresser;
|
|
|
|
import nsk.share.test.TestBase;
|
|
|
|
import nsk.share.test.Tests;
|
|
|
|
import vm.share.gc.TriggerUnloadingHelper;
|
|
|
|
import vm.share.gc.TriggerUnloadingWithWhiteBox;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Test checks that static fields will be initialized in new loaded class. Test performs in loop the following routine:
|
2021-05-13 12:46:54 +00:00
|
|
|
* 1.) Load class either by regular classloader or by defineHiddenClass.
|
2018-07-20 18:03:23 -04:00
|
|
|
* 2.) Trigger unloading. Class must be alive. Next step will check that static fields were not lost.
|
2018-05-22 13:05:22 -07:00
|
|
|
* 3.) Change static fields.
|
|
|
|
* 4.) Unload class.
|
|
|
|
* 5.) Load class again as in step 1.
|
|
|
|
* 6.) Check that static fields were initialized.
|
|
|
|
*/
|
|
|
|
@SuppressWarnings("rawtypes")
|
|
|
|
public class StaticReferences extends GCTestBase {
|
|
|
|
|
|
|
|
private static final int UNLOADING_ATTEMPTS_LIMIT = 50;
|
|
|
|
|
|
|
|
private static final Object[] NO_CP_PATCHES = new Object[0];
|
|
|
|
|
|
|
|
private static String[] args;
|
|
|
|
|
2018-07-20 18:03:23 -04:00
|
|
|
private static final int LIMIT = 20;
|
2018-05-22 13:05:22 -07:00
|
|
|
|
|
|
|
private List<Object> keepAlive = new LinkedList<Object>();
|
|
|
|
|
|
|
|
private Random random;
|
|
|
|
|
|
|
|
private TriggerUnloadingHelper triggerUnloadingHelper = new TriggerUnloadingWithWhiteBox();
|
|
|
|
|
|
|
|
private String[] typesArray = new String[] {"Object object", "boolean boolean", "byte byte", "char char", "double double", "float float", "int int", "long long", "short short"};
|
|
|
|
|
|
|
|
public static void main(String[] args) {
|
|
|
|
StaticReferences.args = args;
|
|
|
|
Tests.runTest(new StaticReferences(), args);
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public void run() {
|
2018-07-20 18:03:23 -04:00
|
|
|
random = new Random(runParams.getSeed());
|
2018-05-22 13:05:22 -07:00
|
|
|
ExecutionController stresser = new Stresser(args);
|
|
|
|
stresser.start(1);
|
|
|
|
|
|
|
|
// Generate and compile classes
|
|
|
|
List<byte[]> bytecodeList = new LinkedList<byte[]>();
|
|
|
|
int[] fieldQuantities = new int[9];
|
|
|
|
long startTimeStamp = System.currentTimeMillis();
|
|
|
|
for (int i = 0; i < LIMIT; i++) {
|
|
|
|
if (!stresser.continueExecution()) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
for (int j = 0; j < fieldQuantities.length; j++) {
|
2018-07-20 18:03:23 -04:00
|
|
|
fieldQuantities[j] = 1 + random.nextInt(20);
|
2018-05-22 13:05:22 -07:00
|
|
|
}
|
|
|
|
bytecodeList.add(generateAndCompile(fieldQuantities));
|
|
|
|
}
|
|
|
|
log.info("Compilation finished in " + ((System.currentTimeMillis() - startTimeStamp)/1000/60.0) + " minutes ");
|
|
|
|
|
|
|
|
// Core of test
|
|
|
|
for (byte[] classBytecode : bytecodeList) {
|
2021-05-13 12:46:54 +00:00
|
|
|
boolean hidden = random.nextBoolean();
|
2018-05-22 13:05:22 -07:00
|
|
|
|
|
|
|
log.info("Load class first time");
|
2021-05-13 12:46:54 +00:00
|
|
|
Class clazz = loadClass(classBytecode, hidden);
|
2018-05-22 13:05:22 -07:00
|
|
|
|
|
|
|
log.info("Trigger unloading");
|
|
|
|
triggerUnloadingHelper.triggerUnloading(stresser);
|
|
|
|
if (!stresser.continueExecution()) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
log.info("Set up static fields. This will check that static fields are reachable.");
|
|
|
|
setupFields(clazz);
|
|
|
|
|
|
|
|
log.info("Cleanup references");
|
|
|
|
Reference<Class> weakReference = new WeakReference<Class>(clazz);
|
|
|
|
clazz = null;
|
|
|
|
|
|
|
|
log.info("Trigger unloading again");
|
|
|
|
int numberOfAttemps = 0;
|
|
|
|
while (weakReference.get() != null && numberOfAttemps < UNLOADING_ATTEMPTS_LIMIT) {
|
|
|
|
if (!stresser.continueExecution()) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
triggerUnloadingHelper.triggerUnloading(stresser);
|
|
|
|
}
|
|
|
|
if (numberOfAttemps >= UNLOADING_ATTEMPTS_LIMIT) {
|
|
|
|
setFailed(true);
|
|
|
|
throw new RuntimeException("Test failed: was unable to unload class with " + UNLOADING_ATTEMPTS_LIMIT + " attempts.");
|
|
|
|
}
|
|
|
|
|
|
|
|
log.info("Load class second time");
|
2021-05-13 12:46:54 +00:00
|
|
|
clazz = loadClass(classBytecode, hidden);
|
2018-05-22 13:05:22 -07:00
|
|
|
|
|
|
|
log.info("check fields reinitialized");
|
|
|
|
checkStaticFields(clazz);
|
|
|
|
|
|
|
|
keepAlive.add(clazz);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-05-13 12:46:54 +00:00
|
|
|
private Class loadClass(byte[] classBytecode, boolean hidden) {
|
2018-05-22 13:05:22 -07:00
|
|
|
Class clazz;
|
2021-05-13 12:46:54 +00:00
|
|
|
if (hidden) {
|
|
|
|
Lookup lookup = MethodHandles.lookup();
|
|
|
|
try {
|
|
|
|
clazz = lookup.defineHiddenClass(classBytecode, false).lookupClass();
|
|
|
|
} catch (IllegalAccessException e) {
|
|
|
|
e.printStackTrace();
|
|
|
|
throw new RuntimeException(
|
|
|
|
"Lookup.defineHiddenClass failed: " + e.getMessage());
|
|
|
|
}
|
2018-05-22 13:05:22 -07:00
|
|
|
} else {
|
|
|
|
OneUsageClassloader classloader = new OneUsageClassloader();
|
|
|
|
clazz = classloader.define(classBytecode);
|
|
|
|
}
|
|
|
|
return clazz;
|
|
|
|
}
|
|
|
|
|
|
|
|
private void checkStaticFields(Class clazz) {
|
|
|
|
for (Field field : clazz.getFields()) {
|
|
|
|
try {
|
|
|
|
if (Modifier.isStatic(field.getModifiers())) {
|
|
|
|
Class fieldType = field.getType();
|
|
|
|
if ((fieldType.equals(Object.class) && field.get(null) != null )
|
|
|
|
|| (fieldType.equals(int.class) && field.getInt(null) != 0)
|
|
|
|
|| (fieldType.equals(boolean.class) && field.getBoolean(null) != false)
|
|
|
|
|| (fieldType.equals(char.class) && field.getChar(null) != 0)
|
|
|
|
|| (fieldType.equals(long.class) && field.getLong(null) != 0)
|
|
|
|
|| (fieldType.equals(short.class) && field.getShort(null) != 0)
|
|
|
|
|| (fieldType.equals(float.class) && field.getFloat(null) != 0.0f)
|
|
|
|
|| (fieldType.equals(double.class) && field.getDouble(null) != 0.0)
|
|
|
|
|| (fieldType.equals(byte.class) && field.getByte(null) != 0)) {
|
|
|
|
setFailed(true);
|
|
|
|
throw new RuntimeException("Failing test: field "
|
|
|
|
+ field.getName() + " of type "
|
|
|
|
+ field.getType() + " in class "
|
|
|
|
+ field.getDeclaringClass().getName()
|
|
|
|
+ " was not cleared");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} catch (IllegalArgumentException | IllegalAccessException e) {
|
|
|
|
e.printStackTrace();
|
|
|
|
throw new RuntimeException("Was unable to set static field "
|
|
|
|
+ field.getName() + " of type "
|
|
|
|
+ field.getType().getName() + " in class "
|
|
|
|
+ field.getDeclaringClass().getName(), e);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-07-20 18:03:23 -04:00
|
|
|
private byte[] generateAndCompile(int[] fieldQuantities) {
|
2018-05-22 13:05:22 -07:00
|
|
|
Map<String, CharSequence> sources = new HashMap<String, CharSequence>();
|
2018-07-20 18:03:23 -04:00
|
|
|
sources.put("A", generateSource(fieldQuantities));
|
2018-05-22 13:05:22 -07:00
|
|
|
return InMemoryJavaCompiler.compile(sources).values().iterator().next();
|
|
|
|
}
|
|
|
|
|
|
|
|
private StringBuffer generateSource(int[] fieldQuantities) {
|
|
|
|
StringBuffer result = new StringBuffer("public class A { \n");
|
|
|
|
int fieldsCounter = 0;
|
|
|
|
for (int i = 0; i < typesArray.length; i++) {
|
|
|
|
for (int j = 0; j < fieldQuantities[i]; j++) {
|
|
|
|
result.append(" public static " + typesArray[i] + fieldsCounter++ + ";\n");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
result.append(" } ");
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
private void setupFields(Class clazz) {
|
|
|
|
for (Field field : clazz.getFields()) {
|
|
|
|
try {
|
|
|
|
if (Modifier.isStatic(field.getModifiers())) {
|
|
|
|
Class fieldType = field.getType();
|
|
|
|
if (fieldType.equals(Object.class)) {
|
|
|
|
field.set(null, this);
|
|
|
|
} else if (fieldType.equals(int.class)) {
|
|
|
|
field.setInt(null, 42);
|
|
|
|
} else if (fieldType.equals(boolean.class)) {
|
|
|
|
field.setBoolean(null, true);
|
|
|
|
} else if (fieldType.equals(char.class)) {
|
|
|
|
field.setChar(null, 'c');
|
|
|
|
} else if (fieldType.equals(long.class)) {
|
|
|
|
field.setLong(null, (long) 42);
|
|
|
|
} else if (fieldType.equals(short.class)) {
|
|
|
|
field.setShort(null, (short) 42);
|
|
|
|
} else if (fieldType.equals(float.class)) {
|
|
|
|
field.setFloat(null, 42.42f);
|
|
|
|
} else if (fieldType.equals(double.class)) {
|
|
|
|
field.setDouble(null, 42.42);
|
|
|
|
} else if (fieldType.equals(byte.class)) {
|
|
|
|
field.setByte(null, (byte) 42);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} catch (IllegalArgumentException | IllegalAccessException e) {
|
|
|
|
e.printStackTrace();
|
|
|
|
throw new RuntimeException(
|
|
|
|
"Was unable to set static field " + field.getName()
|
|
|
|
+ " of type " + field.getType().getName()
|
|
|
|
+ " in class "
|
|
|
|
+ field.getDeclaringClass().getName(), e);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|