Skip to content

Commit 178289b

Browse files
committedOct 31, 2024
8331503: [lworld] Tearing issues in the interpreter when using the Access API
Reviewed-by: dsimms
1 parent 0c1ca7b commit 178289b

File tree

6 files changed

+152
-2
lines changed

6 files changed

+152
-2
lines changed
 

‎src/hotspot/share/oops/accessBackend.cpp

+4
Original file line numberDiff line numberDiff line change
@@ -164,6 +164,10 @@ namespace AccessInternal {
164164
Copy::conjoint_memory_atomic(src, dst, length);
165165
}
166166

167+
void value_copy_internal(void* src, void* dst, size_t length) {
168+
Copy::copy_value_content(src, dst, length);
169+
}
170+
167171
#ifdef ASSERT
168172
void check_access_thread_state() {
169173
if (VMError::is_error_reported() || DebuggingContext::is_enabled()) {

‎src/hotspot/share/oops/accessBackend.hpp

+2
Original file line numberDiff line numberDiff line change
@@ -160,6 +160,8 @@ namespace AccessInternal {
160160
void arraycopy_arrayof_conjoint(T* src, T* dst, size_t length);
161161
template<typename T>
162162
void arraycopy_conjoint_atomic(T* src, T* dst, size_t length);
163+
164+
void value_copy_internal(void* src, void* dst, size_t length);
163165
}
164166

165167
// This mask specifies what decorators are relevant for raw accesses. When passing

‎src/hotspot/share/oops/accessBackend.inline.hpp

+2-2
Original file line numberDiff line numberDiff line change
@@ -333,7 +333,7 @@ inline void RawAccessBarrier<decorators>::clone(oop src, oop dst, size_t size) {
333333

334334
template <DecoratorSet decorators>
335335
inline void RawAccessBarrier<decorators>::value_copy(void* src, void* dst, InlineKlass* md, LayoutKind lk) {
336-
assert(is_aligned(src, md->layout_alignment(lk)) && is_aligned(dst, md->layout_alignment(lk)), "Unalign value_copy");
337-
AccessInternal::arraycopy_conjoint_atomic(src, dst, static_cast<size_t>(md->layout_size_in_bytes(lk)));
336+
assert(is_aligned(src, md->layout_alignment(lk)) && is_aligned(dst, md->layout_alignment(lk)), "Unaligned value_copy");
337+
AccessInternal::value_copy_internal(src, dst, static_cast<size_t>(md->layout_size_in_bytes(lk)));
338338
}
339339
#endif // SHARE_OOPS_ACCESSBACKEND_INLINE_HPP

‎src/hotspot/share/utilities/copy.cpp

+40
Original file line numberDiff line numberDiff line change
@@ -54,6 +54,46 @@ void Copy::conjoint_memory_atomic(const void* from, void* to, size_t size) {
5454
}
5555
}
5656

57+
#define COPY_ALIGNED_SEGMENT(t) \
58+
if (bits % sizeof(t) == 0) { \
59+
size_t segment = remain / sizeof(t); \
60+
if (segment > 0) { \
61+
Copy::conjoint_##t##s_atomic((const t*) cursor_from, (t*) cursor_to, segment); \
62+
remain -= segment * sizeof(t); \
63+
cursor_from = (void*)(((char*)cursor_from) + segment * sizeof(t)); \
64+
cursor_to = (void*)(((char*)cursor_to) + segment * sizeof(t)); \
65+
} \
66+
} \
67+
68+
void Copy::copy_value_content(const void* from, void* to, size_t size) {
69+
// Simple cases first
70+
uintptr_t bits = (uintptr_t) from | (uintptr_t) to | (uintptr_t) size;
71+
if (bits % sizeof(jlong) == 0) {
72+
Copy::conjoint_jlongs_atomic((const jlong*) from, (jlong*) to, size / sizeof(jlong));
73+
return;
74+
} else if (bits % sizeof(jint) == 0) {
75+
Copy::conjoint_jints_atomic((const jint*) from, (jint*) to, size / sizeof(jint));
76+
return;
77+
} else if (bits % sizeof(jshort) == 0) {
78+
Copy::conjoint_jshorts_atomic((const jshort*) from, (jshort*) to, size / sizeof(jshort));
79+
return;
80+
}
81+
82+
// Complex cases
83+
bits = (uintptr_t) from | (uintptr_t) to;
84+
const void* cursor_from = from;
85+
void* cursor_to = to;
86+
size_t remain = size;
87+
COPY_ALIGNED_SEGMENT(jlong)
88+
COPY_ALIGNED_SEGMENT(jint)
89+
COPY_ALIGNED_SEGMENT(jshort)
90+
if (remain > 0) {
91+
Copy::conjoint_jbytes((const void*) cursor_from, (void*) cursor_to, remain);
92+
}
93+
}
94+
95+
#undef COPY_ALIGNED_SEGMENT
96+
5797
class CopySwap : AllStatic {
5898
public:
5999
/**

‎src/hotspot/share/utilities/copy.hpp

+2
Original file line numberDiff line numberDiff line change
@@ -161,6 +161,8 @@ class Copy : AllStatic {
161161
// of two which divides all of from, to, and size, whichever is smaller.
162162
static void conjoint_memory_atomic(const void* from, void* to, size_t size);
163163

164+
static void copy_value_content(const void* from, void* to, size_t size);
165+
164166
// bytes, conjoint array, atomic on each byte (not that it matters)
165167
static void arrayof_conjoint_jbytes(const HeapWord* from, HeapWord* to, size_t count) {
166168
pd_arrayof_conjoint_bytes(from, to, count);
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,102 @@
1+
/*
2+
* Copyright (c) 2024, 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.
8+
*
9+
* This code is distributed in the hope that it will be useful, but WITHOUT
10+
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
11+
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
12+
* version 2 for more details (a copy is included in the LICENSE file that
13+
* accompanied this code).
14+
*
15+
* You should have received a copy of the GNU General Public License version
16+
* 2 along with this work; if not, write to the Free Software Foundation,
17+
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
18+
*
19+
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
20+
* or visit www.oracle.com if you need additional information or have any
21+
* questions.
22+
*/
23+
24+
/**
25+
* @test ValueCopyingTest
26+
* @summary Verify that interpreter doesn't tear up primitive fields when copying a value
27+
* @library /test/lib
28+
* @modules java.base/jdk.internal.vm.annotation
29+
* @enablePreview
30+
* @compile ValueCopyingTest.java
31+
* @run main/othervm -Xint -XX:+PrintInlineLayout runtime.valhalla.inlinetypes.ValueCopyingTest
32+
*/
33+
34+
package runtime.valhalla.inlinetypes;
35+
36+
import jdk.internal.vm.annotation.ImplicitlyConstructible;
37+
import jdk.internal.vm.annotation.LooselyConsistentValue;
38+
import jdk.internal.vm.annotation.NullRestricted;
39+
import jdk.test.lib.Asserts;
40+
41+
public class ValueCopyingTest {
42+
43+
static final int NUM_WORKERS = 16;
44+
45+
static ValueCopyingTest target = new ValueCopyingTest();
46+
47+
@ImplicitlyConstructible
48+
@LooselyConsistentValue
49+
static value class TestValue {
50+
int i;
51+
byte b;
52+
TestValue(int i0) {
53+
i = i0;
54+
b = 0;
55+
}
56+
}
57+
58+
@NullRestricted
59+
TestValue tv;
60+
61+
static class Worker implements Runnable {
62+
int i;
63+
TestValue v;
64+
Worker(byte b) {
65+
i = b | (b << 8) | (b << 16) | (b << 24);
66+
v = new TestValue(i);
67+
}
68+
69+
static void checkValue(int i) {
70+
byte b = (byte)(i & 0xFF);
71+
Asserts.assertTrue(((i >> 8) & 0xFF) == b, "Tearing detected");
72+
Asserts.assertTrue(((i >> 16) & 0xFF) == b, "Tearing detected");
73+
Asserts.assertTrue(((i >> 24) & 0xFF) == b, "Tearing detected");
74+
}
75+
76+
public void run() {
77+
for (int n = 0; n < 10000000; n++) {
78+
ValueCopyingTest.target.tv = v;
79+
int ri = ValueCopyingTest.target.tv.i;
80+
checkValue(ri);
81+
}
82+
}
83+
}
84+
85+
static public void main(String[] args) throws InterruptedException {
86+
Thread[] workers = new Thread[NUM_WORKERS];
87+
for (int i = 0; i < NUM_WORKERS; i++) {
88+
workers[i] = new Thread(new Worker((byte)i));
89+
}
90+
for (int i = 0; i < NUM_WORKERS; i++) {
91+
workers[i].start();
92+
}
93+
try {
94+
for (int i = 0; i < NUM_WORKERS; i++) {
95+
workers[i].join();
96+
}
97+
} catch(InterruptedException e) {
98+
e.printStackTrace();
99+
throw e;
100+
}
101+
}
102+
}

0 commit comments

Comments
 (0)
Please sign in to comment.