Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

8294312: [lworld] Add java.util.Objects.isIdentityObject #770

Closed
Closed
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
10 changes: 9 additions & 1 deletion src/java.base/share/classes/java/lang/Class.java
Original file line number Diff line number Diff line change
@@ -72,6 +72,7 @@
import jdk.internal.loader.BootLoader;
import jdk.internal.loader.BuiltinClassLoader;
import jdk.internal.misc.Unsafe;
import jdk.internal.misc.ValhallaFeatures;
import jdk.internal.module.Resources;
import jdk.internal.reflect.CallerSensitive;
import jdk.internal.reflect.CallerSensitiveAdapter;
@@ -638,15 +639,22 @@ private static Class<?> forName(Module module, String name, Class<?> caller) {

/**
* {@return {@code true} if this class is an identity class, otherwise {@code false}}
* If this {@code Class} object represents an array type, then this method returns {@code true}.
* If this {@code Class} object represents a primitive type, or {@code void},
* then this method returns {@code false}.
*
* @since Valhalla
*/
public boolean isIdentity() {
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The javadoc for isIdentity should specify if this Class is an array class, primitive type or void.

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The primitive classes and void also do not have specified modifier bits.
At present, they are not reported to be identity or value classes. The mainline javadoc says they are "value classes" but in reality they are identity classes. JEP 204 (future) would be the authority on that.

return (this.getModifiers() & Modifier.IDENTITY) != 0;
return !ValhallaFeatures.isEnabled() || // Before Valhalla all classes are identity classes
(this.getModifiers() & Modifier.IDENTITY) != 0 ||
isArray();
}

/**
* {@return {@code true} if this class is a value class, otherwise {@code false}}
* If this {@code Class} object represents an array type, a primitive type, or
* {@code void}, then this method returns {@code false}.
*
* @since Valhalla
*/
7 changes: 7 additions & 0 deletions src/java.base/share/classes/java/lang/Object.java
Original file line number Diff line number Diff line change
@@ -35,6 +35,13 @@
* Class {@code Object} is the root of the class hierarchy.
* Every class has {@code Object} as a superclass. All objects,
* including arrays, implement the methods of this class.
* <p>
* Subclasses of {@code java.lang.Object} can be either {@linkplain Class#isIdentity() identity classes}
* or {@linkplain Class#isValue value classes}.
* The class {@code Object} itself is neither an identity class nor a value class.
* See {@jls The Java Language Specification 8.1.1.5 identity and value Classes}.
* An instance can be created with {@code new Object()}, those instances are
* {@link Objects#isIdentityObject(Object) an identity object}.
*
* @see java.lang.Class
* @since 1.0
23 changes: 18 additions & 5 deletions src/java.base/share/classes/java/util/Objects.java
Original file line number Diff line number Diff line change
@@ -25,6 +25,7 @@

package java.util;

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

/**
* {@return {@code true} if the specified object reference is an identity object, otherwise {@code false}}
*
* @param obj an object
* @throws NullPointerException if {@code obj} is {@code null}
*/
// @IntrinsicCandidate
public static boolean isIdentityObject(Object obj) {
requireNonNull(obj);
return obj.getClass().isIdentity() || // Before Valhalla all classes are identity classes
obj.getClass() == Object.class;
}

/**
* Checks that the specified object reference is an identity object.
*
@@ -202,9 +216,8 @@ public static String toIdentityString(Object o) {
@ForceInline
public static <T> T requireIdentity(T obj) {
Objects.requireNonNull(obj);
var cl = obj.getClass();
if (cl.isValue())
throw new IdentityException(cl);
if (!isIdentityObject(obj))
throw new IdentityException(obj.getClass());
return obj;
}

@@ -223,7 +236,7 @@ public static <T> T requireIdentity(T obj) {
@ForceInline
public static <T> T requireIdentity(T obj, String message) {
Objects.requireNonNull(obj);
if (obj.getClass().isValue())
if (!isIdentityObject(obj))
throw new IdentityException(message);
return obj;
}
@@ -243,7 +256,7 @@ public static <T> T requireIdentity(T obj, String message) {
@ForceInline
public static <T> T requireIdentity(T obj, Supplier<String> messageSupplier) {
Objects.requireNonNull(obj);
if (obj.getClass().isValue())
if (!isIdentityObject(obj))
throw new IdentityException(messageSupplier == null ?
null : messageSupplier.get());
return obj;
26 changes: 26 additions & 0 deletions test/jdk/valhalla/valuetypes/ObjectMethods.java
Original file line number Diff line number Diff line change
@@ -34,6 +34,7 @@
import java.lang.reflect.Modifier;
import java.util.Arrays;
import java.util.List;
import java.util.Objects;
import java.util.stream.Stream;

import org.testng.annotations.BeforeTest;
@@ -69,6 +70,31 @@ public class ObjectMethods {
.setPointRef(Point.makePoint(200, 200))
.setReference(Point.makePoint(300, 300))
.setNumber(Value.Number.intValue(20)).build();

@DataProvider(name="Identities")
Object[][] identitiesData() {
return new Object[][]{
{new Object(), true},
{"String", true},
{String.class, true},
{Object.class, true},
{new ValueType1(1), false},
{new ValueType2(2), false},
{new PrimitiveRecord(1, "A"), false},
{new ValueRecord(1,"B"), false},
{new int[0], true}, // arrays of primitive classes are identity objects
{new Object[0], true}, // arrays of identity classes are identity objects
{new String[0], true}, // arrays of identity classes are identity objects
{new ValueType1[0], true}, // arrays of value classes are identity objects
};
}

@Test(dataProvider="Identities")
void identityTests(Object obj, boolean expected) {
var actual = Objects.isIdentityObject(obj);
assertEquals(expected, actual, "Objects.isIdentityObject unexpected");
}

@DataProvider(name="equalsTests")
Object[][] equalsTests() {
return new Object[][]{
36 changes: 20 additions & 16 deletions test/jdk/valhalla/valuetypes/ValhallaFeaturesTest.java
Original file line number Diff line number Diff line change
@@ -25,34 +25,38 @@

import jdk.internal.misc.ValhallaFeatures;

import org.junit.*;
import static org.junit.Assert.*;

/*
* @test
* @modules java.base/jdk.internal.misc
* @summary Test feature flags reflect command line flags
* @run main/othervm ValhallaFeaturesTest true
* @run main/othervm -XX:+EnableValhalla ValhallaFeaturesTest true
* @run main/othervm -XX:-EnableValhalla ValhallaFeaturesTest false
* @run junit/othervm -Dexpected=true ValhallaFeaturesTest
* @run junit/othervm -XX:+EnableValhalla -Dexpected=true ValhallaFeaturesTest
* @run junit/othervm -XX:-EnableValhalla -Dexpected=false ValhallaFeaturesTest
*/

public class ValhallaFeaturesTest {

public static void main(String[] args) {
boolean expected = args.length > 0 ? args[0].equalsIgnoreCase("true") : false;
// Save the expected enable from the command line -Dexpected
private static boolean expected = Boolean.getBoolean("expected");

@Test
public void checkEnable() {
boolean enabled = ValhallaFeatures.isEnabled();
System.out.println("EnableValhalla: " + enabled);
if (expected != enabled) {
throw new RuntimeException("expected: " + expected + ", actual: " + enabled);
}
assertEquals("EnableValhalla Flag", expected, enabled);
}

try {
@Test
public void checkEnsure() {
if (expected) {
// Throwing an exception is an error
ValhallaFeatures.ensureValhallaEnabled();
if (!enabled) {
throw new RuntimeException("ensureValhallaEnabled should have thrown UOE");
}
} catch (UnsupportedOperationException uoe) {
if (enabled) {
throw new RuntimeException("UnsupportedOperationException not expected", uoe);
}
} else {
assertThrows("EnableValhalla Flag", UnsupportedOperationException.class,
() -> ValhallaFeatures.ensureValhallaEnabled());
}
}
}