Skip to content

Commit 9039a70

Browse files
author
Roger Riggs
committedSep 30, 2022
8294312: [lworld] Add java.util.Objects.isIdentityObject
Reviewed-by: mchung
1 parent b10c801 commit 9039a70

File tree

5 files changed

+80
-22
lines changed

5 files changed

+80
-22
lines changed
 

‎src/java.base/share/classes/java/lang/Class.java

+9-1
Original file line numberDiff line numberDiff line change
@@ -72,6 +72,7 @@
7272
import jdk.internal.loader.BootLoader;
7373
import jdk.internal.loader.BuiltinClassLoader;
7474
import jdk.internal.misc.Unsafe;
75+
import jdk.internal.misc.ValhallaFeatures;
7576
import jdk.internal.module.Resources;
7677
import jdk.internal.reflect.CallerSensitive;
7778
import jdk.internal.reflect.CallerSensitiveAdapter;
@@ -638,15 +639,22 @@ private static Class<?> forName(Module module, String name, Class<?> caller) {
638639

639640
/**
640641
* {@return {@code true} if this class is an identity class, otherwise {@code false}}
642+
* If this {@code Class} object represents an array type, then this method returns {@code true}.
643+
* If this {@code Class} object represents a primitive type, or {@code void},
644+
* then this method returns {@code false}.
641645
*
642646
* @since Valhalla
643647
*/
644648
public boolean isIdentity() {
645-
return (this.getModifiers() & Modifier.IDENTITY) != 0;
649+
return !ValhallaFeatures.isEnabled() || // Before Valhalla all classes are identity classes
650+
(this.getModifiers() & Modifier.IDENTITY) != 0 ||
651+
isArray();
646652
}
647653

648654
/**
649655
* {@return {@code true} if this class is a value class, otherwise {@code false}}
656+
* If this {@code Class} object represents an array type, a primitive type, or
657+
* {@code void}, then this method returns {@code false}.
650658
*
651659
* @since Valhalla
652660
*/

‎src/java.base/share/classes/java/lang/Object.java

+7
Original file line numberDiff line numberDiff line change
@@ -35,6 +35,13 @@
3535
* Class {@code Object} is the root of the class hierarchy.
3636
* Every class has {@code Object} as a superclass. All objects,
3737
* including arrays, implement the methods of this class.
38+
* <p>
39+
* Subclasses of {@code java.lang.Object} can be either {@linkplain Class#isIdentity() identity classes}
40+
* or {@linkplain Class#isValue value classes}.
41+
* The class {@code Object} itself is neither an identity class nor a value class.
42+
* See {@jls The Java Language Specification 8.1.1.5 identity and value Classes}.
43+
* An instance can be created with {@code new Object()}, those instances are
44+
* {@link Objects#isIdentityObject(Object) an identity object}.
3845
*
3946
* @see java.lang.Class
4047
* @since 1.0

‎src/java.base/share/classes/java/util/Objects.java

+18-5
Original file line numberDiff line numberDiff line change
@@ -25,6 +25,7 @@
2525

2626
package java.util;
2727

28+
import jdk.internal.misc.ValhallaFeatures;
2829
import jdk.internal.util.Preconditions;
2930
import jdk.internal.vm.annotation.ForceInline;
3031
import jdk.internal.misc.Unsafe;
@@ -189,6 +190,19 @@ public static String toIdentityString(Object o) {
189190
return o.getClass().getName() + "@" + Integer.toHexString(System.identityHashCode(o));
190191
}
191192

193+
/**
194+
* {@return {@code true} if the specified object reference is an identity object, otherwise {@code false}}
195+
*
196+
* @param obj an object
197+
* @throws NullPointerException if {@code obj} is {@code null}
198+
*/
199+
// @IntrinsicCandidate
200+
public static boolean isIdentityObject(Object obj) {
201+
requireNonNull(obj);
202+
return obj.getClass().isIdentity() || // Before Valhalla all classes are identity classes
203+
obj.getClass() == Object.class;
204+
}
205+
192206
/**
193207
* Checks that the specified object reference is an identity object.
194208
*
@@ -202,9 +216,8 @@ public static String toIdentityString(Object o) {
202216
@ForceInline
203217
public static <T> T requireIdentity(T obj) {
204218
Objects.requireNonNull(obj);
205-
var cl = obj.getClass();
206-
if (cl.isValue())
207-
throw new IdentityException(cl);
219+
if (!isIdentityObject(obj))
220+
throw new IdentityException(obj.getClass());
208221
return obj;
209222
}
210223

@@ -223,7 +236,7 @@ public static <T> T requireIdentity(T obj) {
223236
@ForceInline
224237
public static <T> T requireIdentity(T obj, String message) {
225238
Objects.requireNonNull(obj);
226-
if (obj.getClass().isValue())
239+
if (!isIdentityObject(obj))
227240
throw new IdentityException(message);
228241
return obj;
229242
}
@@ -243,7 +256,7 @@ public static <T> T requireIdentity(T obj, String message) {
243256
@ForceInline
244257
public static <T> T requireIdentity(T obj, Supplier<String> messageSupplier) {
245258
Objects.requireNonNull(obj);
246-
if (obj.getClass().isValue())
259+
if (!isIdentityObject(obj))
247260
throw new IdentityException(messageSupplier == null ?
248261
null : messageSupplier.get());
249262
return obj;

‎test/jdk/valhalla/valuetypes/ObjectMethods.java

+26
Original file line numberDiff line numberDiff line change
@@ -34,6 +34,7 @@
3434
import java.lang.reflect.Modifier;
3535
import java.util.Arrays;
3636
import java.util.List;
37+
import java.util.Objects;
3738
import java.util.stream.Stream;
3839

3940
import org.testng.annotations.BeforeTest;
@@ -69,6 +70,31 @@ public class ObjectMethods {
6970
.setPointRef(Point.makePoint(200, 200))
7071
.setReference(Point.makePoint(300, 300))
7172
.setNumber(Value.Number.intValue(20)).build();
73+
74+
@DataProvider(name="Identities")
75+
Object[][] identitiesData() {
76+
return new Object[][]{
77+
{new Object(), true},
78+
{"String", true},
79+
{String.class, true},
80+
{Object.class, true},
81+
{new ValueType1(1), false},
82+
{new ValueType2(2), false},
83+
{new PrimitiveRecord(1, "A"), false},
84+
{new ValueRecord(1,"B"), false},
85+
{new int[0], true}, // arrays of primitive classes are identity objects
86+
{new Object[0], true}, // arrays of identity classes are identity objects
87+
{new String[0], true}, // arrays of identity classes are identity objects
88+
{new ValueType1[0], true}, // arrays of value classes are identity objects
89+
};
90+
}
91+
92+
@Test(dataProvider="Identities")
93+
void identityTests(Object obj, boolean expected) {
94+
var actual = Objects.isIdentityObject(obj);
95+
assertEquals(expected, actual, "Objects.isIdentityObject unexpected");
96+
}
97+
7298
@DataProvider(name="equalsTests")
7399
Object[][] equalsTests() {
74100
return new Object[][]{

‎test/jdk/valhalla/valuetypes/ValhallaFeaturesTest.java

+20-16
Original file line numberDiff line numberDiff line change
@@ -25,34 +25,38 @@
2525

2626
import jdk.internal.misc.ValhallaFeatures;
2727

28+
import org.junit.*;
29+
import static org.junit.Assert.*;
30+
2831
/*
2932
* @test
3033
* @modules java.base/jdk.internal.misc
3134
* @summary Test feature flags reflect command line flags
32-
* @run main/othervm ValhallaFeaturesTest true
33-
* @run main/othervm -XX:+EnableValhalla ValhallaFeaturesTest true
34-
* @run main/othervm -XX:-EnableValhalla ValhallaFeaturesTest false
35+
* @run junit/othervm -Dexpected=true ValhallaFeaturesTest
36+
* @run junit/othervm -XX:+EnableValhalla -Dexpected=true ValhallaFeaturesTest
37+
* @run junit/othervm -XX:-EnableValhalla -Dexpected=false ValhallaFeaturesTest
3538
*/
3639

3740
public class ValhallaFeaturesTest {
3841

39-
public static void main(String[] args) {
40-
boolean expected = args.length > 0 ? args[0].equalsIgnoreCase("true") : false;
42+
// Save the expected enable from the command line -Dexpected
43+
private static boolean expected = Boolean.getBoolean("expected");
44+
45+
@Test
46+
public void checkEnable() {
4147
boolean enabled = ValhallaFeatures.isEnabled();
4248
System.out.println("EnableValhalla: " + enabled);
43-
if (expected != enabled) {
44-
throw new RuntimeException("expected: " + expected + ", actual: " + enabled);
45-
}
49+
assertEquals("EnableValhalla Flag", expected, enabled);
50+
}
4651

47-
try {
52+
@Test
53+
public void checkEnsure() {
54+
if (expected) {
55+
// Throwing an exception is an error
4856
ValhallaFeatures.ensureValhallaEnabled();
49-
if (!enabled) {
50-
throw new RuntimeException("ensureValhallaEnabled should have thrown UOE");
51-
}
52-
} catch (UnsupportedOperationException uoe) {
53-
if (enabled) {
54-
throw new RuntimeException("UnsupportedOperationException not expected", uoe);
55-
}
57+
} else {
58+
assertThrows("EnableValhalla Flag", UnsupportedOperationException.class,
59+
() -> ValhallaFeatures.ensureValhallaEnabled());
5660
}
5761
}
5862
}

0 commit comments

Comments
 (0)
Please sign in to comment.