mirror of
https://github.com/openjdk/jdk.git
synced 2025-09-22 12:04:39 +02:00
8161720: Better byte behavior for off-heap data
Normalize boolean values read with Unsafe. Reviewed-by: aph, simonis, jrose, psandoz
This commit is contained in:
parent
7a6ffc8b52
commit
00c9b389f6
6 changed files with 319 additions and 7 deletions
|
@ -2410,6 +2410,15 @@ void LIRGenerator::do_UnsafeGetObject(UnsafeGetObject* x) {
|
||||||
#endif // INCLUDE_ALL_GCS
|
#endif // INCLUDE_ALL_GCS
|
||||||
|
|
||||||
if (x->is_volatile() && os::is_MP()) __ membar_acquire();
|
if (x->is_volatile() && os::is_MP()) __ membar_acquire();
|
||||||
|
|
||||||
|
/* Normalize boolean value returned by unsafe operation, i.e., value != 0 ? value = true : value false. */
|
||||||
|
if (type == T_BOOLEAN) {
|
||||||
|
LabelObj* equalZeroLabel = new LabelObj();
|
||||||
|
__ cmp(lir_cond_equal, value, 0);
|
||||||
|
__ branch(lir_cond_equal, T_BOOLEAN, equalZeroLabel->label());
|
||||||
|
__ move(LIR_OprFact::intConst(1), value);
|
||||||
|
__ branch_destination(equalZeroLabel->label());
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
|
@ -2471,6 +2471,28 @@ bool LibraryCallKit::inline_unsafe_access(bool is_store, const BasicType type, c
|
||||||
// load value
|
// load value
|
||||||
switch (type) {
|
switch (type) {
|
||||||
case T_BOOLEAN:
|
case T_BOOLEAN:
|
||||||
|
{
|
||||||
|
// Normalize the value returned by getBoolean in the following cases
|
||||||
|
if (mismatched ||
|
||||||
|
heap_base_oop == top() || // - heap_base_oop is NULL or
|
||||||
|
(can_access_non_heap && alias_type->field() == NULL) // - heap_base_oop is potentially NULL
|
||||||
|
// and the unsafe access is made to large offset
|
||||||
|
// (i.e., larger than the maximum offset necessary for any
|
||||||
|
// field access)
|
||||||
|
) {
|
||||||
|
IdealKit ideal = IdealKit(this);
|
||||||
|
#define __ ideal.
|
||||||
|
IdealVariable normalized_result(ideal);
|
||||||
|
__ declarations_done();
|
||||||
|
__ set(normalized_result, p);
|
||||||
|
__ if_then(p, BoolTest::ne, ideal.ConI(0));
|
||||||
|
__ set(normalized_result, ideal.ConI(1));
|
||||||
|
ideal.end_if();
|
||||||
|
final_sync(ideal);
|
||||||
|
p = __ value(normalized_result);
|
||||||
|
#undef __
|
||||||
|
}
|
||||||
|
}
|
||||||
case T_CHAR:
|
case T_CHAR:
|
||||||
case T_BYTE:
|
case T_BYTE:
|
||||||
case T_SHORT:
|
case T_SHORT:
|
||||||
|
|
|
@ -150,14 +150,23 @@ class MemoryAccess : StackObj {
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename T>
|
template <typename T>
|
||||||
T normalize(T x) {
|
T normalize_for_write(T x) {
|
||||||
return x;
|
return x;
|
||||||
}
|
}
|
||||||
|
|
||||||
jboolean normalize(jboolean x) {
|
jboolean normalize_for_write(jboolean x) {
|
||||||
return x & 1;
|
return x & 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
template <typename T>
|
||||||
|
T normalize_for_read(T x) {
|
||||||
|
return x;
|
||||||
|
}
|
||||||
|
|
||||||
|
jboolean normalize_for_read(jboolean x) {
|
||||||
|
return x != 0;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Helper class to wrap memory accesses in JavaThread::doing_unsafe_access()
|
* Helper class to wrap memory accesses in JavaThread::doing_unsafe_access()
|
||||||
*/
|
*/
|
||||||
|
@ -196,7 +205,7 @@ public:
|
||||||
|
|
||||||
T* p = (T*)addr();
|
T* p = (T*)addr();
|
||||||
|
|
||||||
T x = *p;
|
T x = normalize_for_read(*p);
|
||||||
|
|
||||||
return x;
|
return x;
|
||||||
}
|
}
|
||||||
|
@ -207,7 +216,7 @@ public:
|
||||||
|
|
||||||
T* p = (T*)addr();
|
T* p = (T*)addr();
|
||||||
|
|
||||||
*p = normalize(x);
|
*p = normalize_for_write(x);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -223,7 +232,7 @@ public:
|
||||||
|
|
||||||
T x = OrderAccess::load_acquire((volatile T*)p);
|
T x = OrderAccess::load_acquire((volatile T*)p);
|
||||||
|
|
||||||
return x;
|
return normalize_for_read(x);
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename T>
|
template <typename T>
|
||||||
|
@ -232,7 +241,7 @@ public:
|
||||||
|
|
||||||
T* p = (T*)addr();
|
T* p = (T*)addr();
|
||||||
|
|
||||||
OrderAccess::release_store_fence((volatile T*)p, normalize(x));
|
OrderAccess::release_store_fence((volatile T*)p, normalize_for_write(x));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -256,7 +265,7 @@ public:
|
||||||
|
|
||||||
jlong* p = (jlong*)addr();
|
jlong* p = (jlong*)addr();
|
||||||
|
|
||||||
Atomic::store(normalize(x), p);
|
Atomic::store(normalize_for_write(x), p);
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
};
|
};
|
||||||
|
|
92
hotspot/test/compiler/unsafe/UnsafeOffHeapBooleanTest.java
Normal file
92
hotspot/test/compiler/unsafe/UnsafeOffHeapBooleanTest.java
Normal file
|
@ -0,0 +1,92 @@
|
||||||
|
/*
|
||||||
|
* Copyright (c) 2016, Oracle and/or its affiliates. All rights reserved.
|
||||||
|
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
||||||
|
*
|
||||||
|
* This code is free software; you can redistribute it and/or modify it
|
||||||
|
* under the terms of the GNU General Public License version 2 only, as
|
||||||
|
* published by the Free Software Foundation.
|
||||||
|
*
|
||||||
|
* This code is distributed in the hope that it will be useful, but WITHOUT
|
||||||
|
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
||||||
|
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
|
||||||
|
* version 2 for more details (a copy is included in the LICENSE file that
|
||||||
|
* accompanied this code).
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License version
|
||||||
|
* 2 along with this work; if not, write to the Free Software Foundation,
|
||||||
|
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
|
||||||
|
*
|
||||||
|
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
|
||||||
|
* or visit www.oracle.com if you need additional information or have any
|
||||||
|
* questions.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/*
|
||||||
|
* @test
|
||||||
|
* @bug 8161720
|
||||||
|
* @modules java.base/jdk.internal.misc
|
||||||
|
* @run main/othervm -Xint UnsafeOffHeapBooleanTest 1
|
||||||
|
* @run main/othervm -XX:+TieredCompilation -XX:TieredStopAtLevel=3 -Xbatch UnsafeOffHeapBooleanTest 20000
|
||||||
|
* @run main/othervm -XX:-TieredCompilation -Xbatch UnsafeOffHeapBooleanTest 20000
|
||||||
|
*/
|
||||||
|
|
||||||
|
|
||||||
|
import java.lang.reflect.Field;
|
||||||
|
import jdk.internal.misc.Unsafe;
|
||||||
|
|
||||||
|
public class UnsafeOffHeapBooleanTest {
|
||||||
|
static boolean bool0 = false, bool1 = false, result = false;
|
||||||
|
static Unsafe UNSAFE = Unsafe.getUnsafe();
|
||||||
|
static long offHeapMemory;
|
||||||
|
|
||||||
|
public static void test() {
|
||||||
|
// Write two bytes to the off-heap memory location, both
|
||||||
|
// bytes correspond to the boolean value 'true'.
|
||||||
|
UNSAFE.putShort(null, offHeapMemory, (short)0x0204);
|
||||||
|
|
||||||
|
// Read two bytes from the storage allocated above (as booleans).
|
||||||
|
bool0 = UNSAFE.getBoolean(null, offHeapMemory + 0);
|
||||||
|
bool1 = UNSAFE.getBoolean(null, offHeapMemory + 1);
|
||||||
|
result = bool0 & bool1;
|
||||||
|
}
|
||||||
|
|
||||||
|
public static void main(String args[]) {
|
||||||
|
System.out.println("### Test started");
|
||||||
|
|
||||||
|
if (args.length != 1) {
|
||||||
|
throw new RuntimeException("### Test failure: test called with incorrect number of arguments");
|
||||||
|
}
|
||||||
|
|
||||||
|
// Allocate two bytes of storage.
|
||||||
|
offHeapMemory = UNSAFE.allocateMemory(2);
|
||||||
|
|
||||||
|
try {
|
||||||
|
for (int i = 0; i < Integer.parseInt(args[0]); i++) {
|
||||||
|
test();
|
||||||
|
}
|
||||||
|
|
||||||
|
// Check if the two 'true' boolean values were normalized
|
||||||
|
// (i.e., reduced from the range 1...255 to 1).
|
||||||
|
if (!bool0 || !bool1 || !result) {
|
||||||
|
System.out.println("Some of the results below are wrong");
|
||||||
|
System.out.println("bool0 is: " + bool0);
|
||||||
|
System.out.println("bool1 is: " + bool1);
|
||||||
|
System.out.println("bool0 & bool1 is: " + result);
|
||||||
|
System.out.println("===================================");
|
||||||
|
throw new RuntimeException("### Test failed");
|
||||||
|
} else {
|
||||||
|
System.out.println("Test generated correct results");
|
||||||
|
System.out.println("bool0 is: " + bool0);
|
||||||
|
System.out.println("bool1 is: " + bool1);
|
||||||
|
System.out.println("bool0 & bool1 is: " + result);
|
||||||
|
System.out.println("===================================");
|
||||||
|
}
|
||||||
|
} catch (NumberFormatException e) {
|
||||||
|
throw new RuntimeException("### Test failure: test called with incorrectly formatted parameter");
|
||||||
|
}
|
||||||
|
|
||||||
|
UNSAFE.freeMemory(offHeapMemory);
|
||||||
|
|
||||||
|
System.out.println("### Test passed");
|
||||||
|
}
|
||||||
|
}
|
95
hotspot/test/compiler/unsafe/UnsafeOnHeapBooleanTest.java
Normal file
95
hotspot/test/compiler/unsafe/UnsafeOnHeapBooleanTest.java
Normal file
|
@ -0,0 +1,95 @@
|
||||||
|
/*
|
||||||
|
* Copyright (c) 2016, Oracle and/or its affiliates. All rights reserved.
|
||||||
|
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
||||||
|
*
|
||||||
|
* This code is free software; you can redistribute it and/or modify it
|
||||||
|
* under the terms of the GNU General Public License version 2 only, as
|
||||||
|
* published by the Free Software Foundation.
|
||||||
|
*
|
||||||
|
* This code is distributed in the hope that it will be useful, but WITHOUT
|
||||||
|
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
||||||
|
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
|
||||||
|
* version 2 for more details (a copy is included in the LICENSE file that
|
||||||
|
* accompanied this code).
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License version
|
||||||
|
* 2 along with this work; if not, write to the Free Software Foundation,
|
||||||
|
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
|
||||||
|
*
|
||||||
|
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
|
||||||
|
* or visit www.oracle.com if you need additional information or have any
|
||||||
|
* questions.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/*
|
||||||
|
* @test
|
||||||
|
* @bug 8161720
|
||||||
|
* @modules java.base/jdk.internal.misc
|
||||||
|
* @run main/othervm -Xint UnsafeOnHeapBooleanTest 1
|
||||||
|
* @run main/othervm -XX:-UseOnStackReplacement -XX:+TieredCompilation -XX:TieredStopAtLevel=3 -Xbatch UnsafeOnHeapBooleanTest 20000
|
||||||
|
* @run main/othervm -XX:-UseOnStackReplacement -XX:-TieredCompilation -Xbatch UnsafeOnHeapBooleanTest 20000
|
||||||
|
*/
|
||||||
|
|
||||||
|
import java.lang.reflect.Field;
|
||||||
|
import jdk.internal.misc.Unsafe;
|
||||||
|
|
||||||
|
public class UnsafeOnHeapBooleanTest {
|
||||||
|
static short static_v;
|
||||||
|
static boolean bool0 = false, bool1 = false, result = false;
|
||||||
|
static Unsafe UNSAFE = Unsafe.getUnsafe();
|
||||||
|
|
||||||
|
public static void test() {
|
||||||
|
try {
|
||||||
|
// Write two bytes into the static field
|
||||||
|
// UnsafeOnHeapBooleanTest.static_v write two values. Both
|
||||||
|
// bytes correspond to the boolean value 'true'.
|
||||||
|
Field staticVField = UnsafeOnHeapBooleanTest.class.getDeclaredField("static_v");
|
||||||
|
Object base = UNSAFE.staticFieldBase(staticVField);
|
||||||
|
long offset = UNSAFE.staticFieldOffset(staticVField);
|
||||||
|
UNSAFE.putShort(base, offset, (short)0x0204);
|
||||||
|
|
||||||
|
// Read two bytes from the static field
|
||||||
|
// UnsafeOnHeapBooleanTest.static_v (as booleans).
|
||||||
|
bool0 = UNSAFE.getBoolean(base, offset + 0);
|
||||||
|
bool1 = UNSAFE.getBoolean(base, offset + 1);
|
||||||
|
result = bool0 & bool1;
|
||||||
|
} catch (NoSuchFieldException e) {
|
||||||
|
throw new RuntimeException("### Test failure: static field UnsafeOnHeapBooleanTest.static_v was not found");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public static void main(String args[]) {
|
||||||
|
System.out.println("### Test started");
|
||||||
|
|
||||||
|
if (args.length != 1) {
|
||||||
|
throw new RuntimeException("### Test failure: test called with incorrect number of arguments");
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
for (int i = 0; i < Integer.parseInt(args[0]); i++) {
|
||||||
|
test();
|
||||||
|
}
|
||||||
|
|
||||||
|
// Check if the two 'true' boolean values were normalized
|
||||||
|
// (i.e., reduced from the range 1...255 to 1).
|
||||||
|
if (!bool0 || !bool1 || !result) {
|
||||||
|
System.out.println("Some of the results below are wrong");
|
||||||
|
System.out.println("bool0 is: " + bool0);
|
||||||
|
System.out.println("bool1 is: " + bool1);
|
||||||
|
System.out.println("bool0 & bool1 is: " + result);
|
||||||
|
System.out.println("===================================");
|
||||||
|
throw new RuntimeException("### Test failed");
|
||||||
|
} else {
|
||||||
|
System.out.println("Test generated correct results");
|
||||||
|
System.out.println("bool0 is: " + bool0);
|
||||||
|
System.out.println("bool1 is: " + bool1);
|
||||||
|
System.out.println("bool0 & bool1 is: " + result);
|
||||||
|
System.out.println("===================================");
|
||||||
|
}
|
||||||
|
} catch (NumberFormatException e) {
|
||||||
|
throw new RuntimeException("### Test failure: test called with incorrectly formatted parameter");
|
||||||
|
}
|
||||||
|
|
||||||
|
System.out.println("### Test passed");
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,85 @@
|
||||||
|
/*
|
||||||
|
* Copyright (c) 2016, Oracle and/or its affiliates. All rights reserved.
|
||||||
|
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
||||||
|
*
|
||||||
|
* This code is free software; you can redistribute it and/or modify it
|
||||||
|
* under the terms of the GNU General Public License version 2 only, as
|
||||||
|
* published by the Free Software Foundation.
|
||||||
|
*
|
||||||
|
* This code is distributed in the hope that it will be useful, but WITHOUT
|
||||||
|
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
||||||
|
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
|
||||||
|
* version 2 for more details (a copy is included in the LICENSE file that
|
||||||
|
* accompanied this code).
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License version
|
||||||
|
* 2 along with this work; if not, write to the Free Software Foundation,
|
||||||
|
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
|
||||||
|
*
|
||||||
|
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
|
||||||
|
* or visit www.oracle.com if you need additional information or have any
|
||||||
|
* questions.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/*
|
||||||
|
* @test
|
||||||
|
* @bug 8161720
|
||||||
|
* @modules java.base/jdk.internal.misc
|
||||||
|
* @run main/othervm -XX:+IgnoreUnrecognizedVMOptions -Xbatch -XX:-TieredCompilation UnsafeSmallOffsetBooleanAccessTest
|
||||||
|
* @run main/othervm -Xbatch UnsafeSmallOffsetBooleanAccessTest
|
||||||
|
*/
|
||||||
|
|
||||||
|
import java.util.Random;
|
||||||
|
import jdk.internal.misc.Unsafe;
|
||||||
|
|
||||||
|
public class UnsafeSmallOffsetBooleanAccessTest {
|
||||||
|
static final Unsafe UNSAFE = Unsafe.getUnsafe();
|
||||||
|
static final long F_OFFSET;
|
||||||
|
static final Random random = new Random();
|
||||||
|
|
||||||
|
static {
|
||||||
|
try {
|
||||||
|
F_OFFSET = UNSAFE.objectFieldOffset(T.class.getDeclaredField("f"));
|
||||||
|
System.out.println("The offset is: " + F_OFFSET);
|
||||||
|
} catch (Exception e) {
|
||||||
|
throw new Error(e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static class T {
|
||||||
|
boolean f;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Always return false in a way that is not obvious to the compiler.
|
||||||
|
public static boolean myRandom() {
|
||||||
|
if (random.nextInt(101) > 134) {
|
||||||
|
return true;
|
||||||
|
} else {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public static boolean test(T t) {
|
||||||
|
boolean result = false;
|
||||||
|
for (int i = 0; i < 20000; i++) {
|
||||||
|
boolean random = myRandom();
|
||||||
|
// If myRandom() returns false, access t.f.
|
||||||
|
//
|
||||||
|
// If myRandom() returns true, access virtual address
|
||||||
|
// F_OFFSET. That address is most likely not mapped,
|
||||||
|
// therefore the access will most likely cause a
|
||||||
|
// crash. We're not concerned about that, though, because
|
||||||
|
// myRandom() always returns false. However, the C2
|
||||||
|
// compiler avoids normalization of the value returned by
|
||||||
|
// getBoolean in this case.
|
||||||
|
result = UNSAFE.getBoolean(myRandom() ? null : t, F_OFFSET);
|
||||||
|
}
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
public static void main(String[] args) {
|
||||||
|
T t = new T();
|
||||||
|
UNSAFE.putBoolean(t, F_OFFSET, true);
|
||||||
|
System.out.println("The result for t is: " + test(t));
|
||||||
|
}
|
||||||
|
}
|
Loading…
Add table
Add a link
Reference in a new issue