8321220: JFR: RecordedClass reports incorrect modifiers

Reviewed-by: mgronlun
This commit is contained in:
Erik Gahlin 2023-12-04 15:51:22 +00:00
parent 9769dfecb5
commit d5f59cf0a8
2 changed files with 116 additions and 1 deletions

View File

@ -219,7 +219,7 @@ static int write_klass(JfrCheckpointWriter* writer, KlassPtr klass, bool leakp)
writer->write(cld != nullptr ? cld_id(cld, leakp) : 0);
writer->write(mark_symbol(klass, leakp));
writer->write(package_id(klass, leakp));
writer->write(get_flags(klass));
writer->write(klass->modifier_flags());
writer->write<bool>(klass->is_hidden());
return 1;
}

View File

@ -0,0 +1,115 @@
/*
* Copyright (c) 2023, 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.
*/
package jdk.jfr.api.consumer;
import java.lang.reflect.Modifier;
import java.util.List;
import jdk.jfr.Event;
import jdk.jfr.Recording;
import jdk.jfr.consumer.RecordedClass;
import jdk.jfr.consumer.RecordedEvent;
import jdk.test.lib.jfr.Events;
/**
* @test
* @summary Verifies methods of RecordedClass
* @key jfr
* @requires vm.hasJFR
* @library /test/lib
* @run main/othervm jdk.jfr.api.consumer.TestRecordedClass
*/
public class TestRecordedClass {
static class TestEvent extends Event {
Class<?> typeA;
Class<?> typeB;
}
private static class TypeA {
}
public final static class TypeB {
}
public static void main(String[] args) throws Exception {
try (Recording recording = new Recording()) {
recording.start();
TestEvent event = new TestEvent();
event.typeA = TypeA.class;
event.typeB = TypeB.class;
event.commit();
recording.stop();
List<RecordedEvent> events = Events.fromRecording(recording);
Events.hasEvents(events);
for (RecordedEvent recordedEvent : events) {
RecordedClass typeA = recordedEvent.getClass("typeA");
RecordedClass typeB = recordedEvent.getClass("typeB");
assertModifiers(typeA, TypeA.class);
assertModifiers(typeB, TypeB.class);
assertName(typeA, TypeA.class);
assertName(typeB, TypeB.class);
assertClassLoader(typeA, TypeA.class.getClassLoader());
assertClassLoader(typeB, TypeB.class.getClassLoader());
assertId(typeA);
assertId(typeB);
if (typeA.getId() == typeB.getId()) {
throw new Exception("Same ID for different classes");
}
}
}
}
private static void assertId(RecordedClass recordedClass) throws Exception {
long id = recordedClass.getId();
if (id < 1 || id >= 1024 * 1024) {
throw new Exception("Expected class ID to be above 1 and below 1 M");
}
}
private static void assertClassLoader(RecordedClass recordedClass, ClassLoader classLoader) throws Exception {
String expected = classLoader.getClass().getName();
String actual = recordedClass.getClassLoader().getType().getName();
if (!expected.equals(actual)) {
throw new Exception("Expected class loader to be " + expected + ", was " + actual);
}
}
private static void assertName(RecordedClass recordedClass, Class<?> clazz) throws Exception {
String className = clazz.getClass().getName();
if (className.equals(recordedClass.getName())) {
throw new Exception("Expected class to be named " + className);
}
}
private static void assertModifiers(RecordedClass recordedClass, Class<?> clazz) throws Exception {
int modifiers = clazz.getModifiers();
if (modifiers != recordedClass.getModifiers()) {
String expected = Modifier.toString(modifiers);
String actual = Modifier.toString(recordedClass.getModifiers());
throw new Exception("Expected modifier to be '" + expected + "', was '" + actual + "'");
}
}
}