45
45
import sun .reflect .misc .ReflectUtil ;
46
46
import sun .security .util .SecurityConstants ;
47
47
48
+ import java .lang .classfile .ClassFile ;
48
49
import java .lang .classfile .ClassModel ;
50
+ import java .lang .constant .ClassDesc ;
49
51
import java .lang .constant .ConstantDescs ;
50
52
import java .lang .invoke .LambdaForm .BasicType ;
51
53
import java .lang .invoke .MethodHandleImpl .Intrinsic ;
@@ -2248,85 +2250,70 @@ private static ClassFileDumper defaultDumper() {
2248
2250
private static final ClassFileDumper DEFAULT_DUMPER = ClassFileDumper .getInstance (
2249
2251
"jdk.invoke.MethodHandle.dumpClassFiles" , "DUMP_CLASS_FILES" );
2250
2252
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 );
2259
2265
}
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 );
2260
2270
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 );
2263
2273
}
2264
2274
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" );
2283
2292
}
2284
2293
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" );
2312
2297
}
2313
2298
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" );
2322
2305
}
2306
+ return ((bytes [offset ] & 0xFF ) << 24 )
2307
+ | ((bytes [offset + 1 ] & 0xFF ) << 16 )
2308
+ | ((bytes [offset + 2 ] & 0xFF ) << 8 )
2309
+ | (bytes [offset + 3 ] & 0xFF );
2310
+ }
2323
2311
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" );
2329
2315
}
2316
+ return ((bytes [offset ] & 0xFF ) << 8 ) | (bytes [offset + 1 ] & 0xFF );
2330
2317
}
2331
2318
2332
2319
/*
@@ -2340,23 +2327,22 @@ private static int readUnsignedShort(byte[] bytes, int offset) {
2340
2327
* {@code bytes} denotes a class in a different package than the lookup class
2341
2328
*/
2342
2329
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 ());
2345
2332
}
2346
2333
2347
2334
/**
2348
2335
* Returns a ClassDefiner that creates a {@code Class} object of a normal class
2349
2336
* from the given bytes. No package name check on the given bytes.
2350
2337
*
2351
- * @param name internal name
2338
+ * @param internalName internal name
2352
2339
* @param bytes class bytes
2353
2340
* @param dumper dumper to write the given bytes to the dumper's output directory
2354
2341
* @return ClassDefiner that defines a normal class of the given bytes.
2355
2342
*/
2356
- ClassDefiner makeClassDefiner (String name , byte [] bytes , ClassFileDumper dumper ) {
2343
+ ClassDefiner makeClassDefiner (String internalName , byte [] bytes , ClassFileDumper dumper ) {
2357
2344
// 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 );
2360
2346
}
2361
2347
2362
2348
/**
@@ -2374,8 +2360,8 @@ ClassDefiner makeClassDefiner(String name, byte[] bytes, ClassFileDumper dumper)
2374
2360
* {@code bytes} denotes a class in a different package than the lookup class
2375
2361
*/
2376
2362
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 );
2379
2365
}
2380
2366
2381
2367
/**
@@ -2397,51 +2383,53 @@ ClassDefiner makeHiddenClassDefiner(byte[] bytes, ClassFileDumper dumper) {
2397
2383
private ClassDefiner makeHiddenClassDefiner (byte [] bytes ,
2398
2384
boolean accessVmAnnotations ,
2399
2385
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 );
2402
2388
}
2403
2389
2404
2390
/**
2405
2391
* Returns a ClassDefiner that creates a {@code Class} object of a hidden class
2406
2392
* from the given bytes and the given options. No package name check on the given bytes.
2407
2393
*
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
2409
2395
* @param bytes class bytes
2410
2396
* @param dumper dumper to write the given bytes to the dumper's output directory
2411
2397
* @return ClassDefiner that defines a hidden class of the given bytes and options.
2412
2398
*/
2413
- ClassDefiner makeHiddenClassDefiner (String name , byte [] bytes , ClassFileDumper dumper ) {
2399
+ ClassDefiner makeHiddenClassDefiner (String internalName , byte [] bytes , ClassFileDumper dumper ) {
2414
2400
Objects .requireNonNull (dumper );
2415
2401
// skip name and access flags validation
2416
- return makeHiddenClassDefiner (ClassFile . newInstanceNoCheck ( name , bytes ) , false , dumper , 0 );
2402
+ return makeHiddenClassDefiner (internalName , bytes , false , dumper , 0 );
2417
2403
}
2418
2404
2419
2405
/**
2420
2406
* Returns a ClassDefiner that creates a {@code Class} object of a hidden class
2421
2407
* from the given bytes and the given options. No package name check on the given bytes.
2422
2408
*
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
2424
2410
* @param bytes class bytes
2425
2411
* @param flags class options flag mask
2426
2412
* @param dumper dumper to write the given bytes to the dumper's output directory
2427
2413
* @return ClassDefiner that defines a hidden class of the given bytes and options.
2428
2414
*/
2429
- ClassDefiner makeHiddenClassDefiner (String name , byte [] bytes , ClassFileDumper dumper , int flags ) {
2415
+ ClassDefiner makeHiddenClassDefiner (String internalName , byte [] bytes , ClassFileDumper dumper , int flags ) {
2430
2416
Objects .requireNonNull (dumper );
2431
2417
// skip name and access flags validation
2432
- return makeHiddenClassDefiner (ClassFile . newInstanceNoCheck ( name , bytes ) , false , dumper , flags );
2418
+ return makeHiddenClassDefiner (internalName , bytes , false , dumper , flags );
2433
2419
}
2434
2420
2435
2421
/**
2436
2422
* Returns a ClassDefiner that creates a {@code Class} object of a hidden class
2437
2423
* from the given class file and options.
2438
2424
*
2439
- * @param cf ClassFile
2425
+ * @param internalName internal name
2426
+ * @param bytes Class byte array
2440
2427
* @param flags class option flag mask
2441
2428
* @param accessVmAnnotations true to give the hidden class access to VM annotations
2442
2429
* @param dumper dumper to write the given bytes to the dumper's output directory
2443
2430
*/
2444
- private ClassDefiner makeHiddenClassDefiner (ClassFile cf ,
2431
+ private ClassDefiner makeHiddenClassDefiner (String internalName ,
2432
+ byte [] bytes ,
2445
2433
boolean accessVmAnnotations ,
2446
2434
ClassFileDumper dumper ,
2447
2435
int flags ) {
@@ -2452,27 +2440,12 @@ private ClassDefiner makeHiddenClassDefiner(ClassFile cf,
2452
2440
flags |= ACCESS_VM_ANNOTATIONS ;
2453
2441
}
2454
2442
2455
- return new ClassDefiner (this , cf , flags , dumper );
2443
+ return new ClassDefiner (this , internalName , bytes , flags , dumper );
2456
2444
}
2457
2445
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 );
2476
2449
}
2477
2450
2478
2451
Class <?> defineClass (boolean initialize ) {
@@ -2501,7 +2474,7 @@ Class<?> defineClass(boolean initialize, Object classData) {
2501
2474
Class <?> c = null ;
2502
2475
try {
2503
2476
c = SharedSecrets .getJavaLangAccess ()
2504
- .defineClass (loader , lookupClass , name , bytes , pd , initialize , classFlags , classData );
2477
+ .defineClass (loader , lookupClass , internalName , bytes , pd , initialize , classFlags , classData );
2505
2478
assert !isNestmate () || c .getNestHost () == lookupClass .getNestHost ();
2506
2479
return c ;
2507
2480
} finally {
0 commit comments