Book a Demo!
CoCalc Logo Icon
StoreFeaturesDocsShareSupportNewsAboutPoliciesSign UpSign In
PojavLauncherTeam
GitHub Repository: PojavLauncherTeam/mobile
Path: blob/master/src/java.base/share/classes/sun/invoke/util/Wrapper.java
41159 views
1
/*
2
* Copyright (c) 2008, 2018, Oracle and/or its affiliates. All rights reserved.
3
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
4
*
5
* This code is free software; you can redistribute it and/or modify it
6
* under the terms of the GNU General Public License version 2 only, as
7
* published by the Free Software Foundation. Oracle designates this
8
* particular file as subject to the "Classpath" exception as provided
9
* by Oracle in the LICENSE file that accompanied this code.
10
*
11
* This code is distributed in the hope that it will be useful, but WITHOUT
12
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
13
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
14
* version 2 for more details (a copy is included in the LICENSE file that
15
* accompanied this code).
16
*
17
* You should have received a copy of the GNU General Public License version
18
* 2 along with this work; if not, write to the Free Software Foundation,
19
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
20
*
21
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
22
* or visit www.oracle.com if you need additional information or have any
23
* questions.
24
*/
25
26
package sun.invoke.util;
27
28
public enum Wrapper {
29
// wrapperType simple primitiveType simple char emptyArray format
30
BOOLEAN( Boolean.class, "Boolean", boolean.class, "boolean", 'Z', new boolean[0], Format.unsigned( 1)),
31
// These must be in the order defined for widening primitive conversions in JLS 5.1.2
32
// Avoid boxing integral types here to defer initialization of internal caches
33
BYTE ( Byte.class, "Byte", byte.class, "byte", 'B', new byte[0], Format.signed( 8)),
34
SHORT ( Short.class, "Short", short.class, "short", 'S', new short[0], Format.signed( 16)),
35
CHAR (Character.class, "Character", char.class, "char", 'C', new char[0], Format.unsigned(16)),
36
INT ( Integer.class, "Integer", int.class, "int", 'I', new int[0], Format.signed( 32)),
37
LONG ( Long.class, "Long", long.class, "long", 'J', new long[0], Format.signed( 64)),
38
FLOAT ( Float.class, "Float", float.class, "float", 'F', new float[0], Format.floating(32)),
39
DOUBLE ( Double.class, "Double", double.class, "double", 'D', new double[0], Format.floating(64)),
40
OBJECT ( Object.class, "Object", Object.class, "Object", 'L', new Object[0], Format.other( 1)),
41
// VOID must be the last type, since it is "assignable" from any other type:
42
VOID ( Void.class, "Void", void.class, "void", 'V', null, Format.other( 0)),
43
;
44
45
public static final int COUNT = 10;
46
47
private final Class<?> wrapperType;
48
private final Class<?> primitiveType;
49
private final char basicTypeChar;
50
private final String basicTypeString;
51
private final Object emptyArray;
52
private final int format;
53
private final String wrapperSimpleName;
54
private final String primitiveSimpleName;
55
56
private Wrapper(Class<?> wtype, String wtypeName, Class<?> ptype, String ptypeName, char tchar, Object emptyArray, int format) {
57
this.wrapperType = wtype;
58
this.primitiveType = ptype;
59
this.basicTypeChar = tchar;
60
this.basicTypeString = String.valueOf(this.basicTypeChar);
61
this.emptyArray = emptyArray;
62
this.format = format;
63
this.wrapperSimpleName = wtypeName;
64
this.primitiveSimpleName = ptypeName;
65
}
66
67
/** For debugging, give the details of this wrapper. */
68
public String detailString() {
69
return wrapperSimpleName+
70
java.util.Arrays.asList(wrapperType, primitiveType,
71
basicTypeChar, zero(),
72
"0x"+Integer.toHexString(format));
73
}
74
75
private abstract static class Format {
76
static final int SLOT_SHIFT = 0, SIZE_SHIFT = 2, KIND_SHIFT = 12;
77
static final int
78
SIGNED = (-1) << KIND_SHIFT,
79
UNSIGNED = 0 << KIND_SHIFT,
80
FLOATING = 1 << KIND_SHIFT;
81
static final int
82
SLOT_MASK = ((1<<(SIZE_SHIFT-SLOT_SHIFT))-1),
83
SIZE_MASK = ((1<<(KIND_SHIFT-SIZE_SHIFT))-1);
84
static int format(int kind, int size, int slots) {
85
assert(((kind >> KIND_SHIFT) << KIND_SHIFT) == kind);
86
assert((size & (size-1)) == 0); // power of two
87
assert((kind == SIGNED) ? (size > 0) :
88
(kind == UNSIGNED) ? (size > 0) :
89
(kind == FLOATING) ? (size == 32 || size == 64) :
90
false);
91
assert((slots == 2) ? (size == 64) :
92
(slots == 1) ? (size <= 32) :
93
false);
94
return kind | (size << SIZE_SHIFT) | (slots << SLOT_SHIFT);
95
}
96
static final int
97
INT = SIGNED | (32 << SIZE_SHIFT) | (1 << SLOT_SHIFT),
98
SHORT = SIGNED | (16 << SIZE_SHIFT) | (1 << SLOT_SHIFT),
99
BOOLEAN = UNSIGNED | (1 << SIZE_SHIFT) | (1 << SLOT_SHIFT),
100
CHAR = UNSIGNED | (16 << SIZE_SHIFT) | (1 << SLOT_SHIFT),
101
FLOAT = FLOATING | (32 << SIZE_SHIFT) | (1 << SLOT_SHIFT),
102
VOID = UNSIGNED | (0 << SIZE_SHIFT) | (0 << SLOT_SHIFT),
103
NUM_MASK = (-1) << SIZE_SHIFT;
104
static int signed(int size) { return format(SIGNED, size, (size > 32 ? 2 : 1)); }
105
static int unsigned(int size) { return format(UNSIGNED, size, (size > 32 ? 2 : 1)); }
106
static int floating(int size) { return format(FLOATING, size, (size > 32 ? 2 : 1)); }
107
static int other(int slots) { return slots << SLOT_SHIFT; }
108
}
109
110
/// format queries:
111
112
/** How many bits are in the wrapped value? Returns 0 for OBJECT or VOID. */
113
public int bitWidth() { return (format >> Format.SIZE_SHIFT) & Format.SIZE_MASK; }
114
/** How many JVM stack slots occupied by the wrapped value? Returns 0 for VOID. */
115
public int stackSlots() { return (format >> Format.SLOT_SHIFT) & Format.SLOT_MASK; }
116
/** Does the wrapped value occupy a single JVM stack slot? */
117
public boolean isSingleWord() { return (format & (1 << Format.SLOT_SHIFT)) != 0; }
118
/** Does the wrapped value occupy two JVM stack slots? */
119
public boolean isDoubleWord() { return (format & (2 << Format.SLOT_SHIFT)) != 0; }
120
/** Is the wrapped type numeric (not void or object)? */
121
public boolean isNumeric() { return (format & Format.NUM_MASK) != 0; }
122
/** Is the wrapped type a primitive other than float, double, or void? */
123
public boolean isIntegral() { return isNumeric() && format < Format.FLOAT; }
124
/** Is the wrapped type one of int, boolean, byte, char, or short? */
125
public boolean isSubwordOrInt() { return isIntegral() && isSingleWord(); }
126
/* Is the wrapped value a signed integral type (one of byte, short, int, or long)? */
127
public boolean isSigned() { return format < Format.VOID; }
128
/* Is the wrapped value an unsigned integral type (one of boolean or char)? */
129
public boolean isUnsigned() { return format >= Format.BOOLEAN && format < Format.FLOAT; }
130
/** Is the wrapped type either float or double? */
131
public boolean isFloating() { return format >= Format.FLOAT; }
132
/** Is the wrapped type either void or a reference? */
133
public boolean isOther() { return (format & ~Format.SLOT_MASK) == 0; }
134
135
/** Does the JLS 5.1.2 allow a variable of this wrapper's
136
* primitive type to be assigned from a value of the given wrapper's primitive type?
137
* Cases:
138
* <ul>
139
* <li>unboxing followed by widening primitive conversion
140
* <li>any type converted to {@code void} (i.e., dropping a method call's value)
141
* <li>boxing conversion followed by widening reference conversion to {@code Object}
142
* </ul>
143
* These are the cases allowed by MethodHandle.asType.
144
*/
145
public boolean isConvertibleFrom(Wrapper source) {
146
if (this == source) return true;
147
if (this.compareTo(source) < 0) {
148
// At best, this is a narrowing conversion.
149
return false;
150
}
151
// All conversions are allowed in the enum order between floats and signed ints.
152
// First detect non-signed non-float types (boolean, char, Object, void).
153
boolean floatOrSigned = (((this.format & source.format) & Format.SIGNED) != 0);
154
if (!floatOrSigned) {
155
if (this.isOther()) return true;
156
// can convert char to int or wider, but nothing else
157
if (source.format == Format.CHAR) return true;
158
// no other conversions are classified as widening
159
return false;
160
}
161
// All signed and float conversions in the enum order are widening.
162
assert(this.isFloating() || this.isSigned());
163
assert(source.isFloating() || source.isSigned());
164
return true;
165
}
166
167
static {
168
assert(checkConvertibleFrom());
169
assert(COUNT == Wrapper.values().length);
170
}
171
private static boolean checkConvertibleFrom() {
172
// Check the matrix for correct classification of widening conversions.
173
for (Wrapper w : values()) {
174
assert(w.isConvertibleFrom(w));
175
assert(VOID.isConvertibleFrom(w));
176
if (w != VOID) {
177
assert(OBJECT.isConvertibleFrom(w));
178
assert(!w.isConvertibleFrom(VOID));
179
}
180
// check relations with unsigned integral types:
181
if (w != CHAR) {
182
assert(!CHAR.isConvertibleFrom(w));
183
if (!w.isConvertibleFrom(INT))
184
assert(!w.isConvertibleFrom(CHAR));
185
}
186
if (w != BOOLEAN) {
187
assert(!BOOLEAN.isConvertibleFrom(w));
188
if (w != VOID && w != OBJECT)
189
assert(!w.isConvertibleFrom(BOOLEAN));
190
}
191
// check relations with signed integral types:
192
if (w.isSigned()) {
193
for (Wrapper x : values()) {
194
if (w == x) continue;
195
if (x.isFloating())
196
assert(!w.isConvertibleFrom(x));
197
else if (x.isSigned()) {
198
if (w.compareTo(x) < 0)
199
assert(!w.isConvertibleFrom(x));
200
else
201
assert(w.isConvertibleFrom(x));
202
}
203
}
204
}
205
// check relations with floating types:
206
if (w.isFloating()) {
207
for (Wrapper x : values()) {
208
if (w == x) continue;
209
if (x.isSigned())
210
assert(w.isConvertibleFrom(x));
211
else if (x.isFloating()) {
212
if (w.compareTo(x) < 0)
213
assert(!w.isConvertibleFrom(x));
214
else
215
assert(w.isConvertibleFrom(x));
216
}
217
}
218
}
219
}
220
return true; // i.e., assert(true)
221
}
222
223
/** Produce a zero value for the given wrapper type.
224
* This will be a numeric zero for a number or character,
225
* false for a boolean, and null for a reference or void.
226
* The common thread is that this is what is contained
227
* in a default-initialized variable of the given primitive
228
* type. (For void, it is what a reflective method returns
229
* instead of no value at all.)
230
*/
231
public Object zero() {
232
switch (this) {
233
case BOOLEAN:
234
return Boolean.FALSE;
235
case INT:
236
return (Integer)0;
237
case BYTE:
238
return (Byte)(byte)0;
239
case CHAR:
240
return (Character)(char)0;
241
case SHORT:
242
return (Short)(short)0;
243
case LONG:
244
return (Long)(long)0;
245
case FLOAT:
246
return FLOAT_ZERO;
247
case DOUBLE:
248
return DOUBLE_ZERO;
249
case VOID:
250
case OBJECT:
251
default:
252
return null;
253
}
254
}
255
256
private static final Object DOUBLE_ZERO = (Double)(double)0;
257
private static final Object FLOAT_ZERO = (Float)(float)0;
258
259
/** Produce a zero value for the given wrapper type T.
260
* The optional argument must a type compatible with this wrapper.
261
* Equivalent to {@code this.cast(this.zero(), type)}.
262
*/
263
public <T> T zero(Class<T> type) { return convert(zero(), type); }
264
265
/** Return the wrapper that wraps values of the given type.
266
* The type may be {@code Object}, meaning the {@code OBJECT} wrapper.
267
* Otherwise, the type must be a primitive.
268
* @throws IllegalArgumentException for unexpected types
269
*/
270
public static Wrapper forPrimitiveType(Class<?> type) {
271
Wrapper w = findPrimitiveType(type);
272
if (w != null) return w;
273
if (type.isPrimitive())
274
throw new InternalError(); // redo hash function
275
throw newIllegalArgumentException("not primitive: "+type);
276
}
277
278
/** Return the wrapper that corresponds to the provided basic type char.
279
* The basic type char must be for one of the eight primitive types, or void.
280
* @throws IllegalArgumentException for unexpected types
281
*/
282
public static Wrapper forPrimitiveType(char basicTypeChar) {
283
switch (basicTypeChar) {
284
case 'I': return INT;
285
case 'J': return LONG;
286
case 'S': return SHORT;
287
case 'B': return BYTE;
288
case 'C': return CHAR;
289
case 'F': return FLOAT;
290
case 'D': return DOUBLE;
291
case 'Z': return BOOLEAN;
292
case 'V': return VOID;
293
default: throw newIllegalArgumentException("not primitive: " + basicTypeChar);
294
}
295
}
296
297
static Wrapper findPrimitiveType(Class<?> type) {
298
Wrapper w = FROM_PRIM[hashPrim(type)];
299
if (w != null && w.primitiveType == type) {
300
return w;
301
}
302
return null;
303
}
304
305
/** Return the wrapper that wraps values into the given wrapper type.
306
* If it is {@code Object}, return {@code OBJECT}.
307
* Otherwise, it must be a wrapper type.
308
* The type must not be a primitive type.
309
* @throws IllegalArgumentException for unexpected types
310
*/
311
public static Wrapper forWrapperType(Class<?> type) {
312
Wrapper w = findWrapperType(type);
313
if (w != null) return w;
314
for (Wrapper x : values())
315
if (x.wrapperType == type)
316
throw new InternalError(); // redo hash function
317
throw newIllegalArgumentException("not wrapper: "+type);
318
}
319
320
static Wrapper findWrapperType(Class<?> type) {
321
Wrapper w = FROM_WRAP[hashWrap(type)];
322
if (w != null && w.wrapperType == type) {
323
return w;
324
}
325
return null;
326
}
327
328
/** Return the wrapper that corresponds to the given bytecode
329
* signature character. Return {@code OBJECT} for the character 'L'.
330
* @throws IllegalArgumentException for any non-signature character or {@code '['}.
331
*/
332
public static Wrapper forBasicType(char type) {
333
Wrapper w = FROM_CHAR[hashChar(type)];
334
if (w != null && w.basicTypeChar == type) {
335
return w;
336
}
337
for (Wrapper x : values())
338
if (w.basicTypeChar == type)
339
throw new InternalError(); // redo hash function
340
throw newIllegalArgumentException("not basic type char: "+type);
341
}
342
343
/** Return the wrapper for the given type, if it is
344
* a primitive type, else return {@code OBJECT}.
345
*/
346
public static Wrapper forBasicType(Class<?> type) {
347
if (type.isPrimitive())
348
return forPrimitiveType(type);
349
return OBJECT; // any reference, including wrappers or arrays
350
}
351
352
// Note on perfect hashes:
353
// for signature chars c, do (c + (c >> 1)) % 16
354
// for primitive type names n, do (n[0] + n[2]) % 16
355
// The type name hash works for both primitive and wrapper names.
356
// You can add "java/lang/Object" to the primitive names.
357
// But you add the wrapper name Object, use (n[2] + (3*n[1])) % 16.
358
private static final Wrapper[] FROM_PRIM = new Wrapper[16];
359
private static final Wrapper[] FROM_WRAP = new Wrapper[16];
360
private static final Wrapper[] FROM_CHAR = new Wrapper[16];
361
private static int hashPrim(Class<?> x) {
362
String xn = x.getName();
363
if (xn.length() < 3) return 0;
364
return (xn.charAt(0) + xn.charAt(2)) % 16;
365
}
366
private static int hashWrap(Class<?> x) {
367
String xn = x.getName();
368
final int offset = 10; assert(offset == "java.lang.".length());
369
if (xn.length() < offset+3) return 0;
370
return (3*xn.charAt(offset+1) + xn.charAt(offset+2)) % 16;
371
}
372
private static int hashChar(char x) {
373
return (x + (x >> 1)) % 16;
374
}
375
static {
376
for (Wrapper w : values()) {
377
int pi = hashPrim(w.primitiveType);
378
int wi = hashWrap(w.wrapperType);
379
int ci = hashChar(w.basicTypeChar);
380
assert(FROM_PRIM[pi] == null);
381
assert(FROM_WRAP[wi] == null);
382
assert(FROM_CHAR[ci] == null);
383
FROM_PRIM[pi] = w;
384
FROM_WRAP[wi] = w;
385
FROM_CHAR[ci] = w;
386
}
387
//assert(jdk.sun.invoke.util.WrapperTest.test(false));
388
}
389
390
/** What is the primitive type wrapped by this wrapper? */
391
public Class<?> primitiveType() { return primitiveType; }
392
393
/** What is the wrapper type for this wrapper? */
394
public Class<?> wrapperType() { return wrapperType; }
395
396
/** What is the wrapper type for this wrapper?
397
* Otherwise, the example type must be the wrapper type,
398
* or the corresponding primitive type.
399
* (For {@code OBJECT}, the example type can be any non-primitive,
400
* and is normalized to {@code Object.class}.)
401
* The resulting class type has the same type parameter.
402
*/
403
public <T> Class<T> wrapperType(Class<T> exampleType) {
404
if (exampleType == wrapperType) {
405
return exampleType;
406
} else if (exampleType == primitiveType ||
407
wrapperType == Object.class ||
408
exampleType.isInterface()) {
409
return forceType(wrapperType, exampleType);
410
}
411
throw newClassCastException(exampleType, primitiveType);
412
}
413
414
private static ClassCastException newClassCastException(Class<?> actual, Class<?> expected) {
415
return new ClassCastException(actual + " is not compatible with " + expected);
416
}
417
418
/** If {@code type} is a primitive type, return the corresponding
419
* wrapper type, else return {@code type} unchanged.
420
*/
421
public static <T> Class<T> asWrapperType(Class<T> type) {
422
if (type.isPrimitive()) {
423
return forPrimitiveType(type).wrapperType(type);
424
}
425
return type;
426
}
427
428
/** If {@code type} is a wrapper type, return the corresponding
429
* primitive type, else return {@code type} unchanged.
430
*/
431
public static <T> Class<T> asPrimitiveType(Class<T> type) {
432
Wrapper w = findWrapperType(type);
433
if (w != null) {
434
return forceType(w.primitiveType(), type);
435
}
436
return type;
437
}
438
439
/** Query: Is the given type a wrapper, such as {@code Integer} or {@code Void}? */
440
public static boolean isWrapperType(Class<?> type) {
441
return findWrapperType(type) != null;
442
}
443
444
/** Query: Is the given type a primitive, such as {@code int} or {@code void}? */
445
public static boolean isPrimitiveType(Class<?> type) {
446
return type.isPrimitive();
447
}
448
449
/** What is the bytecode signature character for this type?
450
* All non-primitives, including array types, report as 'L', the signature character for references.
451
*/
452
public static char basicTypeChar(Class<?> type) {
453
if (!type.isPrimitive())
454
return 'L';
455
else
456
return forPrimitiveType(type).basicTypeChar();
457
}
458
459
/** What is the bytecode signature character for this wrapper's
460
* primitive type?
461
*/
462
public char basicTypeChar() { return basicTypeChar; }
463
464
/** What is the bytecode signature string for this wrapper's
465
* primitive type?
466
*/
467
public String basicTypeString() { return basicTypeString; }
468
469
/** What is the simple name of the wrapper type?
470
*/
471
public String wrapperSimpleName() { return wrapperSimpleName; }
472
473
/** What is the simple name of the primitive type?
474
*/
475
public String primitiveSimpleName() { return primitiveSimpleName; }
476
477
// /** Wrap a value in the given type, which may be either a primitive or wrapper type.
478
// * Performs standard primitive conversions, including truncation and float conversions.
479
// */
480
// public static <T> T wrap(Object x, Class<T> type) {
481
// return Wrapper.valueOf(type).cast(x, type);
482
// }
483
484
/** Cast a wrapped value to the given type, which may be either a primitive or wrapper type.
485
* The given target type must be this wrapper's primitive or wrapper type.
486
* If this wrapper is OBJECT, the target type may also be an interface, perform no runtime check.
487
* Performs standard primitive conversions, including truncation and float conversions.
488
* The given type must be compatible with this wrapper. That is, it must either
489
* be the wrapper type (or a subtype, in the case of {@code OBJECT}) or else
490
* it must be the wrapper's primitive type.
491
* Primitive conversions are only performed if the given type is itself a primitive.
492
* @throws ClassCastException if the given type is not compatible with this wrapper
493
*/
494
public <T> T cast(Object x, Class<T> type) {
495
return convert(x, type, true);
496
}
497
498
/** Convert a wrapped value to the given type.
499
* The given target type must be this wrapper's primitive or wrapper type.
500
* This is equivalent to {@link #cast}, except that it refuses to perform
501
* narrowing primitive conversions.
502
*/
503
public <T> T convert(Object x, Class<T> type) {
504
return convert(x, type, false);
505
}
506
507
private <T> T convert(Object x, Class<T> type, boolean isCast) {
508
if (this == OBJECT) {
509
// If the target wrapper is OBJECT, just do a reference cast.
510
// If the target type is an interface, perform no runtime check.
511
// (This loophole is safe, and is allowed by the JVM verifier.)
512
// If the target type is a primitive, change it to a wrapper.
513
assert(!type.isPrimitive());
514
if (!type.isInterface())
515
type.cast(x);
516
@SuppressWarnings("unchecked")
517
T result = (T) x; // unchecked warning is expected here
518
return result;
519
}
520
Class<T> wtype = wrapperType(type);
521
if (wtype.isInstance(x)) {
522
return wtype.cast(x);
523
}
524
if (!isCast) {
525
Class<?> sourceType = x.getClass(); // throw NPE if x is null
526
Wrapper source = findWrapperType(sourceType);
527
if (source == null || !this.isConvertibleFrom(source)) {
528
throw newClassCastException(wtype, sourceType);
529
}
530
} else if (x == null) {
531
@SuppressWarnings("unchecked")
532
T z = (T) zero();
533
return z;
534
}
535
@SuppressWarnings("unchecked")
536
T result = (T) wrap(x); // unchecked warning is expected here
537
assert (result == null ? Void.class : result.getClass()) == wtype;
538
return result;
539
}
540
541
/** Cast a reference type to another reference type.
542
* If the target type is an interface, perform no runtime check.
543
* (This loophole is safe, and is allowed by the JVM verifier.)
544
* If the target type is a primitive, change it to a wrapper.
545
*/
546
static <T> Class<T> forceType(Class<?> type, Class<T> exampleType) {
547
assert(type == exampleType ||
548
type.isPrimitive() && forPrimitiveType(type) == findWrapperType(exampleType) ||
549
exampleType.isPrimitive() && forPrimitiveType(exampleType) == findWrapperType(type) ||
550
type == Object.class && !exampleType.isPrimitive());
551
@SuppressWarnings("unchecked")
552
Class<T> result = (Class<T>) type; // unchecked warning is expected here
553
return result;
554
}
555
556
/** Wrap a value in this wrapper's type.
557
* Performs standard primitive conversions, including truncation and float conversions.
558
* Performs returns the unchanged reference for {@code OBJECT}.
559
* Returns null for {@code VOID}.
560
* Returns a zero value for a null input.
561
* @throws ClassCastException if this wrapper is numeric and the operand
562
* is not a number, character, boolean, or null
563
*/
564
public Object wrap(Object x) {
565
// do non-numeric wrappers first
566
switch (basicTypeChar) {
567
case 'L': return x;
568
case 'V': return null;
569
}
570
Number xn = numberValue(x);
571
switch (basicTypeChar) {
572
case 'I': return Integer.valueOf(xn.intValue());
573
case 'J': return Long.valueOf(xn.longValue());
574
case 'F': return Float.valueOf(xn.floatValue());
575
case 'D': return Double.valueOf(xn.doubleValue());
576
case 'S': return Short.valueOf((short) xn.intValue());
577
case 'B': return Byte.valueOf((byte) xn.intValue());
578
case 'C': return Character.valueOf((char) xn.intValue());
579
case 'Z': return Boolean.valueOf(boolValue(xn.byteValue()));
580
}
581
throw new InternalError("bad wrapper");
582
}
583
584
/** Wrap a value (an int or smaller value) in this wrapper's type.
585
* Performs standard primitive conversions, including truncation and float conversions.
586
* Produces an {@code Integer} for {@code OBJECT}, although the exact type
587
* of the operand is not known.
588
* Returns null for {@code VOID}.
589
*/
590
public Object wrap(int x) {
591
switch (basicTypeChar) {
592
case 'L': return (Integer)x;
593
case 'V': return null;
594
case 'I': return Integer.valueOf(x);
595
case 'J': return Long.valueOf(x);
596
case 'F': return Float.valueOf(x);
597
case 'D': return Double.valueOf(x);
598
case 'S': return Short.valueOf((short) x);
599
case 'B': return Byte.valueOf((byte) x);
600
case 'C': return Character.valueOf((char) x);
601
case 'Z': return Boolean.valueOf(boolValue((byte) x));
602
}
603
throw new InternalError("bad wrapper");
604
}
605
606
private static Number numberValue(Object x) {
607
if (x instanceof Number) return (Number)x;
608
if (x instanceof Character) return (int)(Character)x;
609
if (x instanceof Boolean) return (Boolean)x ? 1 : 0;
610
// Remaining allowed case of void: Must be a null reference.
611
return (Number)x;
612
}
613
614
// Parameter type of boolValue must be byte, because
615
// MethodHandles.explicitCastArguments defines boolean
616
// conversion as first converting to byte.
617
private static boolean boolValue(byte bits) {
618
bits &= 1; // simple 31-bit zero extension
619
return (bits != 0);
620
}
621
622
private static RuntimeException newIllegalArgumentException(String message, Object x) {
623
return newIllegalArgumentException(message + x);
624
}
625
private static RuntimeException newIllegalArgumentException(String message) {
626
return new IllegalArgumentException(message);
627
}
628
629
// primitive array support
630
public Object makeArray(int len) {
631
return java.lang.reflect.Array.newInstance(primitiveType, len);
632
}
633
public Class<?> arrayType() {
634
return emptyArray.getClass();
635
}
636
public void copyArrayUnboxing(Object[] values, int vpos, Object a, int apos, int length) {
637
if (a.getClass() != arrayType())
638
arrayType().cast(a); // throw NPE or CCE if bad type
639
for (int i = 0; i < length; i++) {
640
Object value = values[i+vpos];
641
value = convert(value, primitiveType);
642
java.lang.reflect.Array.set(a, i+apos, value);
643
}
644
}
645
public void copyArrayBoxing(Object a, int apos, Object[] values, int vpos, int length) {
646
if (a.getClass() != arrayType())
647
arrayType().cast(a); // throw NPE or CCE if bad type
648
for (int i = 0; i < length; i++) {
649
Object value = java.lang.reflect.Array.get(a, i+apos);
650
//Already done: value = convert(value, primitiveType);
651
assert(value.getClass() == wrapperType);
652
values[i+vpos] = value;
653
}
654
}
655
}
656
657