8333748: javap crash - Fatal error: Unmatched bit position 0x2 for location CLASS
Reviewed-by: asotona
This commit is contained in:
parent
1ff5acdaff
commit
7e55ed3b10
@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (c) 2007, 2020, Oracle and/or its affiliates. All rights reserved.
|
* Copyright (c) 2007, 2024, Oracle and/or its affiliates. All rights reserved.
|
||||||
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
||||||
*
|
*
|
||||||
* This code is free software; you can redistribute it and/or modify it
|
* This code is free software; you can redistribute it and/or modify it
|
||||||
@ -208,7 +208,7 @@ public class AttributeWriter extends BasicWriter {
|
|||||||
indent(+1);
|
indent(+1);
|
||||||
first = false;
|
first = false;
|
||||||
}
|
}
|
||||||
for (var flag : info.flags()) {
|
for (var flag : maskToAccessFlagsReportUnknown(access_flags, AccessFlag.Location.INNER_CLASS)) {
|
||||||
if (flag.sourceModifier() && (flag != AccessFlag.ABSTRACT
|
if (flag.sourceModifier() && (flag != AccessFlag.ABSTRACT
|
||||||
|| !info.has(AccessFlag.INTERFACE))) {
|
|| !info.has(AccessFlag.INTERFACE))) {
|
||||||
print(Modifier.toString(flag.mask()) + " ");
|
print(Modifier.toString(flag.mask()) + " ");
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (c) 2007, 2014, Oracle and/or its affiliates. All rights reserved.
|
* Copyright (c) 2007, 2024, Oracle and/or its affiliates. All rights reserved.
|
||||||
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
||||||
*
|
*
|
||||||
* This code is free software; you can redistribute it and/or modify it
|
* This code is free software; you can redistribute it and/or modify it
|
||||||
@ -26,6 +26,12 @@
|
|||||||
package com.sun.tools.javap;
|
package com.sun.tools.javap;
|
||||||
|
|
||||||
import java.io.PrintWriter;
|
import java.io.PrintWriter;
|
||||||
|
import java.lang.classfile.AccessFlags;
|
||||||
|
import java.lang.reflect.AccessFlag;
|
||||||
|
import java.lang.reflect.Modifier;
|
||||||
|
import java.util.EnumMap;
|
||||||
|
import java.util.Map;
|
||||||
|
import java.util.Set;
|
||||||
import java.util.function.Supplier;
|
import java.util.function.Supplier;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
@ -38,6 +44,26 @@ import java.util.function.Supplier;
|
|||||||
* deletion without notice.</b>
|
* deletion without notice.</b>
|
||||||
*/
|
*/
|
||||||
public class BasicWriter {
|
public class BasicWriter {
|
||||||
|
private static final Map<AccessFlag.Location, Integer> LOCATION_MASKS;
|
||||||
|
|
||||||
|
static {
|
||||||
|
var map = new EnumMap<AccessFlag.Location, Integer>(AccessFlag.Location.class);
|
||||||
|
for (var loc : AccessFlag.Location.values()) {
|
||||||
|
map.put(loc, 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
for (var flag : AccessFlag.values()) {
|
||||||
|
for (var loc : flag.locations()) {
|
||||||
|
map.compute(loc, (_, v) -> v | flag.mask());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Peculiarities from AccessFlag.maskToAccessFlag
|
||||||
|
map.compute(AccessFlag.Location.METHOD, (_, v) -> v | Modifier.STRICT);
|
||||||
|
|
||||||
|
LOCATION_MASKS = map;
|
||||||
|
}
|
||||||
|
|
||||||
protected BasicWriter(Context context) {
|
protected BasicWriter(Context context) {
|
||||||
lineWriter = LineWriter.instance(context);
|
lineWriter = LineWriter.instance(context);
|
||||||
out = context.get(PrintWriter.class);
|
out = context.get(PrintWriter.class);
|
||||||
@ -46,6 +72,20 @@ public class BasicWriter {
|
|||||||
throw new AssertionError();
|
throw new AssertionError();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
protected Set<AccessFlag> flagsReportUnknown(AccessFlags flags) {
|
||||||
|
return maskToAccessFlagsReportUnknown(flags.flagsMask(), flags.location());
|
||||||
|
}
|
||||||
|
|
||||||
|
protected Set<AccessFlag> maskToAccessFlagsReportUnknown(int mask, AccessFlag.Location location) {
|
||||||
|
try {
|
||||||
|
return AccessFlag.maskToAccessFlags(mask, location);
|
||||||
|
} catch (IllegalArgumentException ex) {
|
||||||
|
mask &= LOCATION_MASKS.get(location);
|
||||||
|
report("Access Flags: " + ex.getMessage());
|
||||||
|
return AccessFlag.maskToAccessFlags(mask, location);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
protected void print(String s) {
|
protected void print(String s) {
|
||||||
lineWriter.print(s);
|
lineWriter.print(s);
|
||||||
}
|
}
|
||||||
|
@ -417,7 +417,7 @@ public class ClassWriter extends BasicWriter {
|
|||||||
return;
|
return;
|
||||||
|
|
||||||
var flags = AccessFlags.ofField(f.flags().flagsMask());
|
var flags = AccessFlags.ofField(f.flags().flagsMask());
|
||||||
writeModifiers(flags.flags().stream().filter(fl -> fl.sourceModifier())
|
writeModifiers(flagsReportUnknown(flags).stream().filter(fl -> fl.sourceModifier())
|
||||||
.map(fl -> Modifier.toString(fl.mask())).toList());
|
.map(fl -> Modifier.toString(fl.mask())).toList());
|
||||||
print(() -> sigPrinter.print(
|
print(() -> sigPrinter.print(
|
||||||
f.findAttribute(Attributes.signature())
|
f.findAttribute(Attributes.signature())
|
||||||
@ -446,7 +446,7 @@ public class ClassWriter extends BasicWriter {
|
|||||||
|
|
||||||
if (options.verbose)
|
if (options.verbose)
|
||||||
writeList(String.format("flags: (0x%04x) ", flags.flagsMask()),
|
writeList(String.format("flags: (0x%04x) ", flags.flagsMask()),
|
||||||
flags.flags().stream().map(fl -> "ACC_" + fl.name()).toList(),
|
flagsReportUnknown(flags).stream().map(fl -> "ACC_" + fl.name()).toList(),
|
||||||
"\n");
|
"\n");
|
||||||
|
|
||||||
if (options.showAllAttrs) {
|
if (options.showAllAttrs) {
|
||||||
@ -478,7 +478,7 @@ public class ClassWriter extends BasicWriter {
|
|||||||
int flags = m.flags().flagsMask();
|
int flags = m.flags().flagsMask();
|
||||||
|
|
||||||
var modifiers = new ArrayList<String>();
|
var modifiers = new ArrayList<String>();
|
||||||
for (var f : AccessFlags.ofMethod(flags).flags())
|
for (var f : flagsReportUnknown(m.flags()))
|
||||||
if (f.sourceModifier()) modifiers.add(Modifier.toString(f.mask()));
|
if (f.sourceModifier()) modifiers.add(Modifier.toString(f.mask()));
|
||||||
|
|
||||||
String name = "???";
|
String name = "???";
|
||||||
@ -561,7 +561,7 @@ public class ClassWriter extends BasicWriter {
|
|||||||
StringBuilder sb = new StringBuilder();
|
StringBuilder sb = new StringBuilder();
|
||||||
String sep = "";
|
String sep = "";
|
||||||
sb.append(String.format("flags: (0x%04x) ", flags));
|
sb.append(String.format("flags: (0x%04x) ", flags));
|
||||||
for (var f : AccessFlags.ofMethod(flags).flags()) {
|
for (var f : flagsReportUnknown(m.flags())) {
|
||||||
sb.append(sep).append("ACC_").append(f.name());
|
sb.append(sep).append("ACC_").append(f.name());
|
||||||
sep = ", ";
|
sep = ", ";
|
||||||
}
|
}
|
||||||
@ -794,17 +794,9 @@ public class ClassWriter extends BasicWriter {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private static Set<String> getClassModifiers(int mask) {
|
private Set<String> getClassModifiers(int mask) {
|
||||||
return getModifiers(AccessFlags.ofClass((mask & ACC_INTERFACE) != 0
|
return getModifiers(flagsReportUnknown(AccessFlags.ofClass((mask & ACC_INTERFACE) != 0
|
||||||
? mask & ~ACC_ABSTRACT : mask).flags());
|
? mask & ~ACC_ABSTRACT : mask)));
|
||||||
}
|
|
||||||
|
|
||||||
private static Set<String> getMethodModifiers(int mask) {
|
|
||||||
return getModifiers(AccessFlags.ofMethod(mask).flags());
|
|
||||||
}
|
|
||||||
|
|
||||||
private static Set<String> getFieldModifiers(int mask) {
|
|
||||||
return getModifiers(AccessFlags.ofField(mask).flags());
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private static Set<String> getModifiers(Set<java.lang.reflect.AccessFlag> flags) {
|
private static Set<String> getModifiers(Set<java.lang.reflect.AccessFlag> flags) {
|
||||||
@ -814,16 +806,16 @@ public class ClassWriter extends BasicWriter {
|
|||||||
return s;
|
return s;
|
||||||
}
|
}
|
||||||
|
|
||||||
private static Set<String> getClassFlags(int mask) {
|
private Set<String> getClassFlags(int mask) {
|
||||||
return getFlags(mask, AccessFlags.ofClass(mask).flags());
|
return getFlags(mask, flagsReportUnknown(AccessFlags.ofClass(mask)));
|
||||||
}
|
}
|
||||||
|
|
||||||
private static Set<String> getMethodFlags(int mask) {
|
private Set<String> getMethodFlags(int mask) {
|
||||||
return getFlags(mask, AccessFlags.ofMethod(mask).flags());
|
return getFlags(mask, flagsReportUnknown(AccessFlags.ofMethod(mask)));
|
||||||
}
|
}
|
||||||
|
|
||||||
private static Set<String> getFieldFlags(int mask) {
|
private Set<String> getFieldFlags(int mask) {
|
||||||
return getFlags(mask, AccessFlags.ofField(mask).flags());
|
return getFlags(mask, flagsReportUnknown(AccessFlags.ofField(mask)));
|
||||||
}
|
}
|
||||||
|
|
||||||
private static Set<String> getFlags(int mask, Set<java.lang.reflect.AccessFlag> flags) {
|
private static Set<String> getFlags(int mask, Set<java.lang.reflect.AccessFlag> flags) {
|
||||||
@ -840,42 +832,6 @@ public class ClassWriter extends BasicWriter {
|
|||||||
return s;
|
return s;
|
||||||
}
|
}
|
||||||
|
|
||||||
public static enum AccessFlag {
|
|
||||||
ACC_PUBLIC (ClassFile.ACC_PUBLIC, "public", true, true, true, true ),
|
|
||||||
ACC_PRIVATE (ClassFile.ACC_PRIVATE, "private", false, true, true, true ),
|
|
||||||
ACC_PROTECTED (ClassFile.ACC_PROTECTED, "protected", false, true, true, true ),
|
|
||||||
ACC_STATIC (ClassFile.ACC_STATIC, "static", false, true, true, true ),
|
|
||||||
ACC_FINAL (ClassFile.ACC_FINAL, "final", true, true, true, true ),
|
|
||||||
ACC_SUPER (ClassFile.ACC_SUPER, null, true, false, false, false),
|
|
||||||
ACC_SYNCHRONIZED(ClassFile.ACC_SYNCHRONIZED, "synchronized", false, false, false, true ),
|
|
||||||
ACC_VOLATILE (ClassFile.ACC_VOLATILE, "volatile", false, false, true, false),
|
|
||||||
ACC_BRIDGE (ClassFile.ACC_BRIDGE, null, false, false, false, true ),
|
|
||||||
ACC_TRANSIENT (ClassFile.ACC_TRANSIENT, "transient", false, false, true, false),
|
|
||||||
ACC_VARARGS (ClassFile.ACC_VARARGS, null, false, false, false, true ),
|
|
||||||
ACC_NATIVE (ClassFile.ACC_NATIVE, "native", false, false, false, true ),
|
|
||||||
ACC_INTERFACE (ClassFile.ACC_INTERFACE, null, true, true, false, false),
|
|
||||||
ACC_ABSTRACT (ClassFile.ACC_ABSTRACT, "abstract", true, true, false, true ),
|
|
||||||
ACC_STRICT (ClassFile.ACC_STRICT, "strictfp", false, false, false, true ),
|
|
||||||
ACC_SYNTHETIC (ClassFile.ACC_SYNTHETIC, null, true, true, true, true ),
|
|
||||||
ACC_ANNOTATION (ClassFile.ACC_ANNOTATION, null, true, true, false, false),
|
|
||||||
ACC_ENUM (ClassFile.ACC_ENUM, null, true, true, true, false),
|
|
||||||
ACC_MODULE (ClassFile.ACC_MODULE, null, true, false, false, false);
|
|
||||||
|
|
||||||
public final int flag;
|
|
||||||
public final String modifier;
|
|
||||||
public final boolean isClass, isInnerClass, isField, isMethod;
|
|
||||||
|
|
||||||
AccessFlag(int flag, String modifier, boolean isClass,
|
|
||||||
boolean isInnerClass, boolean isField, boolean isMethod) {
|
|
||||||
this.flag = flag;
|
|
||||||
this.modifier = modifier;
|
|
||||||
this.isClass = isClass;
|
|
||||||
this.isInnerClass = isInnerClass;
|
|
||||||
this.isField = isField;
|
|
||||||
this.isMethod = isMethod;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private final Options options;
|
private final Options options;
|
||||||
private final AttributeWriter attrWriter;
|
private final AttributeWriter attrWriter;
|
||||||
private final CodeWriter codeWriter;
|
private final CodeWriter codeWriter;
|
||||||
|
128
test/langtools/tools/javap/UndefinedAccessFlagTest.java
Normal file
128
test/langtools/tools/javap/UndefinedAccessFlagTest.java
Normal file
@ -0,0 +1,128 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (c) 2024, 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 8333748
|
||||||
|
* @summary javap should not fail if reserved access flag bits are set to 1
|
||||||
|
* @library /tools/lib
|
||||||
|
* @modules jdk.jdeps/com.sun.tools.javap
|
||||||
|
* @enablePreview
|
||||||
|
* @run junit UndefinedAccessFlagTest
|
||||||
|
*/
|
||||||
|
|
||||||
|
import org.junit.jupiter.params.ParameterizedTest;
|
||||||
|
import org.junit.jupiter.params.provider.EnumSource;
|
||||||
|
import toolbox.JavapTask;
|
||||||
|
import toolbox.Task;
|
||||||
|
import toolbox.ToolBox;
|
||||||
|
|
||||||
|
import java.lang.classfile.AccessFlags;
|
||||||
|
import java.lang.classfile.ClassModel;
|
||||||
|
import java.lang.classfile.FieldModel;
|
||||||
|
import java.lang.classfile.MethodModel;
|
||||||
|
import java.lang.classfile.attribute.InnerClassInfo;
|
||||||
|
import java.lang.classfile.attribute.InnerClassesAttribute;
|
||||||
|
import java.nio.file.Files;
|
||||||
|
import java.nio.file.Path;
|
||||||
|
|
||||||
|
import static java.lang.classfile.ClassFile.*;
|
||||||
|
import static org.junit.jupiter.api.Assertions.assertTrue;
|
||||||
|
|
||||||
|
public class UndefinedAccessFlagTest {
|
||||||
|
|
||||||
|
final ToolBox toolBox = new ToolBox();
|
||||||
|
|
||||||
|
enum TestLocation {
|
||||||
|
NONE(false), CLASS, FIELD, METHOD, INNER_CLASS(false);
|
||||||
|
|
||||||
|
final boolean fails;
|
||||||
|
TestLocation() { this(true); }
|
||||||
|
TestLocation(boolean fails) { this.fails = fails; }
|
||||||
|
}
|
||||||
|
|
||||||
|
@ParameterizedTest
|
||||||
|
@EnumSource(TestLocation.class)
|
||||||
|
void test(TestLocation location) throws Throwable {
|
||||||
|
var cf = of();
|
||||||
|
ClassModel cm;
|
||||||
|
try (var is = UndefinedAccessFlagTest.class.getResourceAsStream(
|
||||||
|
"/UndefinedAccessFlagTest$SampleInnerClass.class"
|
||||||
|
)) {
|
||||||
|
cm = cf.parse(is.readAllBytes());
|
||||||
|
}
|
||||||
|
var bytes = cf.transform(cm, (cb, ce) -> {
|
||||||
|
switch (ce) {
|
||||||
|
case AccessFlags flags when location == TestLocation.CLASS -> cb
|
||||||
|
.withFlags(flags.flagsMask() | ACC_PRIVATE);
|
||||||
|
case FieldModel f when location == TestLocation.FIELD -> cb
|
||||||
|
.transformField(f, (fb, fe) -> {
|
||||||
|
if (fe instanceof AccessFlags flags) {
|
||||||
|
fb.withFlags(flags.flagsMask() | ACC_SYNCHRONIZED);
|
||||||
|
} else {
|
||||||
|
fb.with(fe);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
case MethodModel m when location == TestLocation.METHOD -> cb
|
||||||
|
.transformMethod(m, (mb, me) -> {
|
||||||
|
if (me instanceof AccessFlags flags) {
|
||||||
|
mb.withFlags(flags.flagsMask() | ACC_INTERFACE);
|
||||||
|
} else {
|
||||||
|
mb.with(me);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
case InnerClassesAttribute attr when location == TestLocation.INNER_CLASS -> cb
|
||||||
|
.with(InnerClassesAttribute.of(attr.classes().stream()
|
||||||
|
.map(ic -> InnerClassInfo.of(ic.innerClass(), ic.outerClass(), ic.innerName(), ic.flagsMask() | 0x0020))
|
||||||
|
.toList()));
|
||||||
|
default -> cb.with(ce);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
Files.write(Path.of("transformed.class"), bytes);
|
||||||
|
|
||||||
|
var lines = new JavapTask(toolBox)
|
||||||
|
.classes("transformed.class")
|
||||||
|
.options("-c", "-p", "-v")
|
||||||
|
.run(location.fails ? Task.Expect.FAIL : Task.Expect.SUCCESS)
|
||||||
|
.writeAll()
|
||||||
|
.getOutputLines(Task.OutputKind.DIRECT);
|
||||||
|
|
||||||
|
// No termination when access flag error happens
|
||||||
|
assertTrue(lines.stream().anyMatch(l -> l.contains("java.lang.String field;")));
|
||||||
|
assertTrue(lines.stream().anyMatch(l -> l.contains("UndefinedAccessFlagTest$SampleInnerClass();")));
|
||||||
|
assertTrue(lines.stream().anyMatch(l -> l.contains("void method();")));
|
||||||
|
assertTrue(lines.stream().anyMatch(l -> l.contains("SampleInnerClass=class UndefinedAccessFlagTest$SampleInnerClass of class UndefinedAccessFlagTest")));
|
||||||
|
|
||||||
|
// Remove non-error lines
|
||||||
|
assertTrue(lines.removeIf(st -> !st.startsWith("Error:")));
|
||||||
|
// Desired locations has errors
|
||||||
|
assertTrue(location == TestLocation.NONE || !lines.isEmpty());
|
||||||
|
// Access Flag errors only
|
||||||
|
assertTrue(lines.stream().allMatch(l -> l.contains("Access Flags:")), () -> String.join("\n", lines));
|
||||||
|
}
|
||||||
|
|
||||||
|
static class SampleInnerClass {
|
||||||
|
String field;
|
||||||
|
void method() {}
|
||||||
|
}
|
||||||
|
}
|
Loading…
Reference in New Issue
Block a user