63e43030ed
Reviewed-by: cjplummer, sspitsyn
139 lines
4.5 KiB
Java
139 lines
4.5 KiB
Java
/*
|
|
* Copyright (c) 2018, 2021, 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.
|
|
*/
|
|
|
|
/**
|
|
* @test
|
|
* @bug 8193369
|
|
* @summary Tests that all FieldAccess and FieldModification notifications
|
|
are generated.
|
|
* @requires vm.jvmti
|
|
* @compile FieldAccessWatch.java
|
|
* @run main/othervm/native -agentlib:FieldAccessWatch FieldAccessWatch
|
|
*/
|
|
|
|
import java.lang.reflect.Field;
|
|
import java.util.ArrayList;
|
|
import java.util.Arrays;
|
|
import java.util.List;
|
|
|
|
|
|
public class FieldAccessWatch {
|
|
|
|
private static final String agentLib = "FieldAccessWatch";
|
|
|
|
private static class MyItem {
|
|
}
|
|
|
|
private static class MyList {
|
|
public List<MyItem> items = new ArrayList<>();
|
|
}
|
|
|
|
public static void main(String[] args) throws Exception {
|
|
try {
|
|
System.loadLibrary(agentLib);
|
|
} catch (UnsatisfiedLinkError ex) {
|
|
System.err.println("Failed to load " + agentLib + " lib");
|
|
System.err.println("java.library.path: " + System.getProperty("java.library.path"));
|
|
throw ex;
|
|
}
|
|
|
|
if (!initWatchers(MyList.class, MyList.class.getDeclaredField("items"), Thread.currentThread())) {
|
|
throw new RuntimeException("Watchers initializations error");
|
|
}
|
|
|
|
MyList list = new MyList();
|
|
|
|
test("[1]items.add(0, object)",() -> list.items.add(0, new MyItem()));
|
|
test("[2]items.add(object)", () -> list.items.add(new MyItem()));
|
|
test("[3]items.add(1, object)", () -> list.items.add(1, new MyItem()));
|
|
test("[4]items.add(object)", () -> list.items.add(new MyItem()));
|
|
test("[5]items.add(1, object)", () -> list.items.add(1, new MyItem()));
|
|
}
|
|
|
|
private static void log(String msg) {
|
|
System.out.println(msg);
|
|
System.out.flush();
|
|
}
|
|
|
|
// For every access/modify notification native part tries to locate
|
|
// boolean "<field_name>_access"/"<field_name>_modify" field and sets it to true
|
|
private static class TestResult {
|
|
// MyList.items
|
|
public boolean items_access;
|
|
|
|
// AbstractList.modCount
|
|
public boolean modCount_access;
|
|
public boolean modCount_modify;
|
|
|
|
// ArrayList.size
|
|
public boolean size_access;
|
|
public boolean size_modify;
|
|
|
|
// ArrayList.elementData
|
|
public boolean elementData_access;
|
|
|
|
// verify that all fields are set to true
|
|
public void verify() {
|
|
Arrays.stream(this.getClass().getDeclaredFields()).forEach(f -> verify(f));
|
|
}
|
|
|
|
private void verify(Field f) {
|
|
try {
|
|
if (!f.getBoolean(this)) {
|
|
throw new RuntimeException(f.getName() + " notification is missed");
|
|
}
|
|
} catch (IllegalAccessException ex) {
|
|
throw new RuntimeException(ex);
|
|
}
|
|
}
|
|
}
|
|
|
|
@FunctionalInterface
|
|
private interface TestAction {
|
|
void apply();
|
|
}
|
|
|
|
private static void test(String descr, TestAction action) throws Exception {
|
|
log(descr + ": starting");
|
|
TestResult result = new TestResult();
|
|
if (!startTest(result)) {
|
|
throw new RuntimeException("startTest failed");
|
|
}
|
|
action.apply();
|
|
// wait some time to ensure all posted events are handled
|
|
Thread.sleep(500);
|
|
|
|
stopTest();
|
|
|
|
// check the results
|
|
result.verify();
|
|
|
|
log(descr + ": OK");
|
|
}
|
|
|
|
private static native boolean initWatchers(Class cls, Field field, Thread testThread);
|
|
private static native boolean startTest(TestResult results);
|
|
private static native void stopTest();
|
|
|
|
}
|