Skip to content

Commit e4f5d23

Browse files
committed
Merge master HEAD into openj9-staging
Signed-off-by: J9 Build <[email protected]>
2 parents 0555d75 + 30f7e78 commit e4f5d23

File tree

26 files changed

+1040
-198
lines changed

26 files changed

+1040
-198
lines changed

SECURITY.md

+3
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,3 @@
1+
# JDK Vulnerabilities
2+
3+
Please follow the process outlined in the [OpenJDK Vulnerability Policy](https://openjdk.org/groups/vulnerability/report) to disclose vulnerabilities in the JDK.

src/java.base/share/classes/java/lang/invoke/MethodHandles.java

+79-106
Original file line numberDiff line numberDiff line change
@@ -45,7 +45,9 @@
4545
import sun.reflect.misc.ReflectUtil;
4646
import sun.security.util.SecurityConstants;
4747

48+
import java.lang.classfile.ClassFile;
4849
import java.lang.classfile.ClassModel;
50+
import java.lang.constant.ClassDesc;
4951
import java.lang.constant.ConstantDescs;
5052
import java.lang.invoke.LambdaForm.BasicType;
5153
import java.lang.invoke.MethodHandleImpl.Intrinsic;
@@ -2248,85 +2250,70 @@ private static ClassFileDumper defaultDumper() {
22482250
private static final ClassFileDumper DEFAULT_DUMPER = ClassFileDumper.getInstance(
22492251
"jdk.invoke.MethodHandle.dumpClassFiles", "DUMP_CLASS_FILES");
22502252

2251-
static class ClassFile {
2252-
final String name; // internal name
2253-
final int accessFlags;
2254-
final byte[] bytes;
2255-
ClassFile(String name, int accessFlags, byte[] bytes) {
2256-
this.name = name;
2257-
this.accessFlags = accessFlags;
2258-
this.bytes = bytes;
2253+
/**
2254+
* This method checks the class file version and the structure of `this_class`.
2255+
* and checks if the bytes is a class or interface (ACC_MODULE flag not set)
2256+
* that is in the named package.
2257+
*
2258+
* @throws IllegalArgumentException if ACC_MODULE flag is set in access flags
2259+
* or the class is not in the given package name.
2260+
*/
2261+
static String validateAndFindInternalName(byte[] bytes, String pkgName) {
2262+
int magic = readInt(bytes, 0);
2263+
if (magic != ClassFile.MAGIC_NUMBER) {
2264+
throw new ClassFormatError("Incompatible magic value: " + magic);
22592265
}
2266+
// We have to read major and minor this way as ClassFile API throws IAE
2267+
// yet we want distinct ClassFormatError and UnsupportedClassVersionError
2268+
int minor = readUnsignedShort(bytes, 4);
2269+
int major = readUnsignedShort(bytes, 6);
22602270

2261-
static ClassFile newInstanceNoCheck(String name, byte[] bytes) {
2262-
return new ClassFile(name, 0, bytes);
2271+
if (!VM.isSupportedClassFileVersion(major, minor)) {
2272+
throw new UnsupportedClassVersionError("Unsupported class file version " + major + "." + minor);
22632273
}
22642274

2265-
/**
2266-
* This method checks the class file version and the structure of `this_class`.
2267-
* and checks if the bytes is a class or interface (ACC_MODULE flag not set)
2268-
* that is in the named package.
2269-
*
2270-
* @throws IllegalArgumentException if ACC_MODULE flag is set in access flags
2271-
* or the class is not in the given package name.
2272-
*/
2273-
static ClassFile newInstance(byte[] bytes, String pkgName) {
2274-
var cf = readClassFile(bytes);
2275-
2276-
// check if it's in the named package
2277-
int index = cf.name.lastIndexOf('/');
2278-
String pn = (index == -1) ? "" : cf.name.substring(0, index).replace('/', '.');
2279-
if (!pn.equals(pkgName)) {
2280-
throw newIllegalArgumentException(cf.name + " not in same package as lookup class");
2281-
}
2282-
return cf;
2275+
String name;
2276+
ClassDesc sym;
2277+
int accessFlags;
2278+
try {
2279+
ClassModel cm = ClassFile.of().parse(bytes);
2280+
var thisClass = cm.thisClass();
2281+
name = thisClass.asInternalName();
2282+
sym = thisClass.asSymbol();
2283+
accessFlags = cm.flags().flagsMask();
2284+
} catch (IllegalArgumentException e) {
2285+
ClassFormatError cfe = new ClassFormatError();
2286+
cfe.initCause(e);
2287+
throw cfe;
2288+
}
2289+
// must be a class or interface
2290+
if ((accessFlags & ACC_MODULE) != 0) {
2291+
throw newIllegalArgumentException("Not a class or interface: ACC_MODULE flag is set");
22832292
}
22842293

2285-
private static ClassFile readClassFile(byte[] bytes) {
2286-
int magic = readInt(bytes, 0);
2287-
if (magic != 0xCAFEBABE) {
2288-
throw new ClassFormatError("Incompatible magic value: " + magic);
2289-
}
2290-
int minor = readUnsignedShort(bytes, 4);
2291-
int major = readUnsignedShort(bytes, 6);
2292-
if (!VM.isSupportedClassFileVersion(major, minor)) {
2293-
throw new UnsupportedClassVersionError("Unsupported class file version " + major + "." + minor);
2294-
}
2295-
2296-
String name;
2297-
int accessFlags;
2298-
try {
2299-
ClassModel cm = java.lang.classfile.ClassFile.of().parse(bytes);
2300-
name = cm.thisClass().asInternalName();
2301-
accessFlags = cm.flags().flagsMask();
2302-
} catch (IllegalArgumentException e) {
2303-
ClassFormatError cfe = new ClassFormatError();
2304-
cfe.initCause(e);
2305-
throw cfe;
2306-
}
2307-
// must be a class or interface
2308-
if ((accessFlags & ACC_MODULE) != 0) {
2309-
throw newIllegalArgumentException("Not a class or interface: ACC_MODULE flag is set");
2310-
}
2311-
return new ClassFile(name, accessFlags, bytes);
2294+
String pn = sym.packageName();
2295+
if (!pn.equals(pkgName)) {
2296+
throw newIllegalArgumentException(name + " not in same package as lookup class");
23122297
}
23132298

2314-
private static int readInt(byte[] bytes, int offset) {
2315-
if ((offset+4) > bytes.length) {
2316-
throw new ClassFormatError("Invalid ClassFile structure");
2317-
}
2318-
return ((bytes[offset] & 0xFF) << 24)
2319-
| ((bytes[offset + 1] & 0xFF) << 16)
2320-
| ((bytes[offset + 2] & 0xFF) << 8)
2321-
| (bytes[offset + 3] & 0xFF);
2299+
return name;
2300+
}
2301+
2302+
private static int readInt(byte[] bytes, int offset) {
2303+
if ((offset + 4) > bytes.length) {
2304+
throw new ClassFormatError("Invalid ClassFile structure");
23222305
}
2306+
return ((bytes[offset] & 0xFF) << 24)
2307+
| ((bytes[offset + 1] & 0xFF) << 16)
2308+
| ((bytes[offset + 2] & 0xFF) << 8)
2309+
| (bytes[offset + 3] & 0xFF);
2310+
}
23232311

2324-
private static int readUnsignedShort(byte[] bytes, int offset) {
2325-
if ((offset+2) > bytes.length) {
2326-
throw new ClassFormatError("Invalid ClassFile structure");
2327-
}
2328-
return ((bytes[offset] & 0xFF) << 8) | (bytes[offset + 1] & 0xFF);
2312+
private static int readUnsignedShort(byte[] bytes, int offset) {
2313+
if ((offset+2) > bytes.length) {
2314+
throw new ClassFormatError("Invalid ClassFile structure");
23292315
}
2316+
return ((bytes[offset] & 0xFF) << 8) | (bytes[offset + 1] & 0xFF);
23302317
}
23312318

23322319
/*
@@ -2340,23 +2327,22 @@ private static int readUnsignedShort(byte[] bytes, int offset) {
23402327
* {@code bytes} denotes a class in a different package than the lookup class
23412328
*/
23422329
private ClassDefiner makeClassDefiner(byte[] bytes) {
2343-
ClassFile cf = ClassFile.newInstance(bytes, lookupClass().getPackageName());
2344-
return new ClassDefiner(this, cf, STRONG_LOADER_LINK, defaultDumper());
2330+
var internalName = validateAndFindInternalName(bytes, lookupClass().getPackageName());
2331+
return new ClassDefiner(this, internalName, bytes, STRONG_LOADER_LINK, defaultDumper());
23452332
}
23462333

23472334
/**
23482335
* Returns a ClassDefiner that creates a {@code Class} object of a normal class
23492336
* from the given bytes. No package name check on the given bytes.
23502337
*
2351-
* @param name internal name
2338+
* @param internalName internal name
23522339
* @param bytes class bytes
23532340
* @param dumper dumper to write the given bytes to the dumper's output directory
23542341
* @return ClassDefiner that defines a normal class of the given bytes.
23552342
*/
2356-
ClassDefiner makeClassDefiner(String name, byte[] bytes, ClassFileDumper dumper) {
2343+
ClassDefiner makeClassDefiner(String internalName, byte[] bytes, ClassFileDumper dumper) {
23572344
// skip package name validation
2358-
ClassFile cf = ClassFile.newInstanceNoCheck(name, bytes);
2359-
return new ClassDefiner(this, cf, STRONG_LOADER_LINK, dumper);
2345+
return new ClassDefiner(this, internalName, bytes, STRONG_LOADER_LINK, dumper);
23602346
}
23612347

23622348
/**
@@ -2374,8 +2360,8 @@ ClassDefiner makeClassDefiner(String name, byte[] bytes, ClassFileDumper dumper)
23742360
* {@code bytes} denotes a class in a different package than the lookup class
23752361
*/
23762362
ClassDefiner makeHiddenClassDefiner(byte[] bytes, ClassFileDumper dumper) {
2377-
ClassFile cf = ClassFile.newInstance(bytes, lookupClass().getPackageName());
2378-
return makeHiddenClassDefiner(cf, false, dumper, 0);
2363+
var internalName = validateAndFindInternalName(bytes, lookupClass().getPackageName());
2364+
return makeHiddenClassDefiner(internalName, bytes, false, dumper, 0);
23792365
}
23802366

23812367
/**
@@ -2397,51 +2383,53 @@ ClassDefiner makeHiddenClassDefiner(byte[] bytes, ClassFileDumper dumper) {
23972383
private ClassDefiner makeHiddenClassDefiner(byte[] bytes,
23982384
boolean accessVmAnnotations,
23992385
int flags) {
2400-
ClassFile cf = ClassFile.newInstance(bytes, lookupClass().getPackageName());
2401-
return makeHiddenClassDefiner(cf, accessVmAnnotations, defaultDumper(), flags);
2386+
var internalName = validateAndFindInternalName(bytes, lookupClass().getPackageName());
2387+
return makeHiddenClassDefiner(internalName, bytes, accessVmAnnotations, defaultDumper(), flags);
24022388
}
24032389

24042390
/**
24052391
* Returns a ClassDefiner that creates a {@code Class} object of a hidden class
24062392
* from the given bytes and the given options. No package name check on the given bytes.
24072393
*
2408-
* @param name internal name that specifies the prefix of the hidden class
2394+
* @param internalName internal name that specifies the prefix of the hidden class
24092395
* @param bytes class bytes
24102396
* @param dumper dumper to write the given bytes to the dumper's output directory
24112397
* @return ClassDefiner that defines a hidden class of the given bytes and options.
24122398
*/
2413-
ClassDefiner makeHiddenClassDefiner(String name, byte[] bytes, ClassFileDumper dumper) {
2399+
ClassDefiner makeHiddenClassDefiner(String internalName, byte[] bytes, ClassFileDumper dumper) {
24142400
Objects.requireNonNull(dumper);
24152401
// skip name and access flags validation
2416-
return makeHiddenClassDefiner(ClassFile.newInstanceNoCheck(name, bytes), false, dumper, 0);
2402+
return makeHiddenClassDefiner(internalName, bytes, false, dumper, 0);
24172403
}
24182404

24192405
/**
24202406
* Returns a ClassDefiner that creates a {@code Class} object of a hidden class
24212407
* from the given bytes and the given options. No package name check on the given bytes.
24222408
*
2423-
* @param name internal name that specifies the prefix of the hidden class
2409+
* @param internalName internal name that specifies the prefix of the hidden class
24242410
* @param bytes class bytes
24252411
* @param flags class options flag mask
24262412
* @param dumper dumper to write the given bytes to the dumper's output directory
24272413
* @return ClassDefiner that defines a hidden class of the given bytes and options.
24282414
*/
2429-
ClassDefiner makeHiddenClassDefiner(String name, byte[] bytes, ClassFileDumper dumper, int flags) {
2415+
ClassDefiner makeHiddenClassDefiner(String internalName, byte[] bytes, ClassFileDumper dumper, int flags) {
24302416
Objects.requireNonNull(dumper);
24312417
// skip name and access flags validation
2432-
return makeHiddenClassDefiner(ClassFile.newInstanceNoCheck(name, bytes), false, dumper, flags);
2418+
return makeHiddenClassDefiner(internalName, bytes, false, dumper, flags);
24332419
}
24342420

24352421
/**
24362422
* Returns a ClassDefiner that creates a {@code Class} object of a hidden class
24372423
* from the given class file and options.
24382424
*
2439-
* @param cf ClassFile
2425+
* @param internalName internal name
2426+
* @param bytes Class byte array
24402427
* @param flags class option flag mask
24412428
* @param accessVmAnnotations true to give the hidden class access to VM annotations
24422429
* @param dumper dumper to write the given bytes to the dumper's output directory
24432430
*/
2444-
private ClassDefiner makeHiddenClassDefiner(ClassFile cf,
2431+
private ClassDefiner makeHiddenClassDefiner(String internalName,
2432+
byte[] bytes,
24452433
boolean accessVmAnnotations,
24462434
ClassFileDumper dumper,
24472435
int flags) {
@@ -2452,27 +2440,12 @@ private ClassDefiner makeHiddenClassDefiner(ClassFile cf,
24522440
flags |= ACCESS_VM_ANNOTATIONS;
24532441
}
24542442

2455-
return new ClassDefiner(this, cf, flags, dumper);
2443+
return new ClassDefiner(this, internalName, bytes, flags, dumper);
24562444
}
24572445

2458-
static class ClassDefiner {
2459-
private final Lookup lookup;
2460-
private final String name; // internal name
2461-
private final byte[] bytes;
2462-
private final int classFlags;
2463-
private final ClassFileDumper dumper;
2464-
2465-
private ClassDefiner(Lookup lookup, ClassFile cf, int flags, ClassFileDumper dumper) {
2466-
assert ((flags & HIDDEN_CLASS) != 0 || (flags & STRONG_LOADER_LINK) == STRONG_LOADER_LINK);
2467-
this.lookup = lookup;
2468-
this.bytes = cf.bytes;
2469-
this.name = cf.name;
2470-
this.classFlags = flags;
2471-
this.dumper = dumper;
2472-
}
2473-
2474-
String internalName() {
2475-
return name;
2446+
record ClassDefiner(Lookup lookup, String internalName, byte[] bytes, int classFlags, ClassFileDumper dumper) {
2447+
ClassDefiner {
2448+
assert ((classFlags & HIDDEN_CLASS) != 0 || (classFlags & STRONG_LOADER_LINK) == STRONG_LOADER_LINK);
24762449
}
24772450

24782451
Class<?> defineClass(boolean initialize) {
@@ -2501,7 +2474,7 @@ Class<?> defineClass(boolean initialize, Object classData) {
25012474
Class<?> c = null;
25022475
try {
25032476
c = SharedSecrets.getJavaLangAccess()
2504-
.defineClass(loader, lookupClass, name, bytes, pd, initialize, classFlags, classData);
2477+
.defineClass(loader, lookupClass, internalName, bytes, pd, initialize, classFlags, classData);
25052478
assert !isNestmate() || c.getNestHost() == lookupClass.getNestHost();
25062479
return c;
25072480
} finally {

src/java.base/share/classes/java/lang/invoke/MutableCallSite.java

+4-4
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 2008, 2022, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 2008, 2024, Oracle and/or its affiliates. All rights reserved.
33
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
44
*
55
* This code is free software; you can redistribute it and/or modify it
@@ -26,7 +26,8 @@
2626
package java.lang.invoke;
2727

2828
import java.util.Objects;
29-
import java.util.concurrent.atomic.AtomicInteger;
29+
30+
import static java.lang.invoke.MethodHandleStatics.UNSAFE;
3031

3132
/**
3233
* A {@code MutableCallSite} is a {@link CallSite} whose target variable
@@ -274,11 +275,10 @@ public final MethodHandle dynamicInvoker() {
274275
*/
275276
public static void syncAll(MutableCallSite[] sites) {
276277
if (sites.length == 0) return;
277-
STORE_BARRIER.lazySet(0);
278+
UNSAFE.storeFence();
278279
for (MutableCallSite site : sites) {
279280
Objects.requireNonNull(site); // trigger NPE on first null
280281
}
281282
// FIXME: NYI
282283
}
283-
private static final AtomicInteger STORE_BARRIER = new AtomicInteger();
284284
}

0 commit comments

Comments
 (0)