@@ -32,6 +32,7 @@ import java.lang.annotation.Target;
32
32
import java.lang.foreign.MemorySegment;
33
33
import java.lang.ref.Reference;
34
34
import java.io.FileDescriptor;
35
+ import java.util.function.Supplier;
35
36
36
37
import jdk.internal.access.JavaNioAccess;
37
38
import jdk.internal.access.SharedSecrets;
@@ -55,10 +56,10 @@ import jdk.internal.vm.vector.VectorSupport;
55
56
* a memory region while another thread is releasing it.
56
57
* <p>
57
58
* This class provides tools to manage races when multiple threads are accessing and/or releasing the same memory
58
- * region concurrently. More specifically, when a thread wants to release a memory region , it should call the
59
- * {@link MemorySessionImpl#close( )} method. This method initiates thread-local handshakes with all the other VM threads,
59
+ * session concurrently. More specifically, when a thread wants to release a memory session , it should call the
60
+ * {@link ScopedMemoryAccess#closeScope(MemorySessionImpl )} method. This method initiates thread-local handshakes with all the other VM threads,
60
61
* which are then stopped one by one. If any thread is found accessing a resource associated to the very memory session
61
- * being closed, the handshake fails, and the session cannot be closed.
62
+ * being closed, the handshake fails, and the session will not be closed.
62
63
* <p>
63
64
* This synchronization strategy relies on the idea that accessing memory is atomic with respect to checking the
64
65
* validity of the session associated with that memory region - that is, a thread that wants to perform memory access will be
@@ -97,12 +98,20 @@ public class ScopedMemoryAccess {
97
98
}
98
99
99
100
public static final class ScopedAccessError extends Error {
100
- private ScopedAccessError() {
101
- super("Attempt to access an already released memory resource", null, false, false);
101
+
102
+ @SuppressWarnings("serial")
103
+ private final Supplier<RuntimeException> runtimeExceptionSupplier;
104
+
105
+ public ScopedAccessError(Supplier<RuntimeException> runtimeExceptionSupplier) {
106
+ super("Invalid memory access", null, false, false);
107
+ this.runtimeExceptionSupplier = runtimeExceptionSupplier;
102
108
}
109
+
103
110
static final long serialVersionUID = 1L;
104
111
105
- public static final ScopedAccessError INSTANCE = new ScopedAccessError();
112
+ public final RuntimeException newRuntimeException() {
113
+ return runtimeExceptionSupplier.get();
114
+ }
106
115
}
107
116
108
117
@Target({ElementType.METHOD, ElementType.CONSTRUCTOR})
@@ -112,64 +121,64 @@ public class ScopedMemoryAccess {
112
121
// bulk ops
113
122
114
123
@ForceInline
115
- public void copyMemory(MemorySessionImpl srcScope , MemorySessionImpl dstScope ,
124
+ public void copyMemory(MemorySessionImpl srcSession , MemorySessionImpl dstSession ,
116
125
Object srcBase, long srcOffset,
117
126
Object destBase, long destOffset,
118
127
long bytes) {
119
128
try {
120
- copyMemoryInternal(srcScope, dstScope , srcBase, srcOffset, destBase, destOffset, bytes);
129
+ copyMemoryInternal(srcSession, dstSession , srcBase, srcOffset, destBase, destOffset, bytes);
121
130
} catch (ScopedAccessError ex) {
122
- throw new IllegalStateException("This segment is already closed" );
131
+ throw ex.newRuntimeException( );
123
132
}
124
133
}
125
134
126
135
@ForceInline @Scoped
127
- private void copyMemoryInternal(MemorySessionImpl srcScope , MemorySessionImpl dstScope ,
136
+ private void copyMemoryInternal(MemorySessionImpl srcSession , MemorySessionImpl dstSession ,
128
137
Object srcBase, long srcOffset,
129
138
Object destBase, long destOffset,
130
139
long bytes) {
131
140
try {
132
- if (srcScope != null) {
133
- srcScope.checkValidState ();
141
+ if (srcSession != null) {
142
+ srcSession.checkValidStateRaw ();
134
143
}
135
- if (dstScope != null) {
136
- dstScope.checkValidState ();
144
+ if (dstSession != null) {
145
+ dstSession.checkValidStateRaw ();
137
146
}
138
147
UNSAFE.copyMemory(srcBase, srcOffset, destBase, destOffset, bytes);
139
148
} finally {
140
- Reference.reachabilityFence(srcScope );
141
- Reference.reachabilityFence(dstScope );
149
+ Reference.reachabilityFence(srcSession );
150
+ Reference.reachabilityFence(dstSession );
142
151
}
143
152
}
144
153
145
154
@ForceInline
146
- public void copySwapMemory(MemorySessionImpl srcScope , MemorySessionImpl dstScope ,
155
+ public void copySwapMemory(MemorySessionImpl srcSession , MemorySessionImpl dstSession ,
147
156
Object srcBase, long srcOffset,
148
157
Object destBase, long destOffset,
149
158
long bytes, long elemSize) {
150
159
try {
151
- copySwapMemoryInternal(srcScope, dstScope , srcBase, srcOffset, destBase, destOffset, bytes, elemSize);
160
+ copySwapMemoryInternal(srcSession, dstSession , srcBase, srcOffset, destBase, destOffset, bytes, elemSize);
152
161
} catch (ScopedAccessError ex) {
153
- throw new IllegalStateException("This segment is already closed" );
162
+ throw ex.newRuntimeException( );
154
163
}
155
164
}
156
165
157
166
@ForceInline @Scoped
158
- private void copySwapMemoryInternal(MemorySessionImpl srcScope , MemorySessionImpl dstScope ,
167
+ private void copySwapMemoryInternal(MemorySessionImpl srcSession , MemorySessionImpl dstSession ,
159
168
Object srcBase, long srcOffset,
160
169
Object destBase, long destOffset,
161
170
long bytes, long elemSize) {
162
171
try {
163
- if (srcScope != null) {
164
- srcScope.checkValidState ();
172
+ if (srcSession != null) {
173
+ srcSession.checkValidStateRaw ();
165
174
}
166
- if (dstScope != null) {
167
- dstScope.checkValidState ();
175
+ if (dstSession != null) {
176
+ dstSession.checkValidStateRaw ();
168
177
}
169
178
UNSAFE.copySwapMemory(srcBase, srcOffset, destBase, destOffset, bytes, elemSize);
170
179
} finally {
171
- Reference.reachabilityFence(srcScope );
172
- Reference.reachabilityFence(dstScope );
180
+ Reference.reachabilityFence(srcSession );
181
+ Reference.reachabilityFence(dstSession );
173
182
}
174
183
}
175
184
@@ -178,15 +187,15 @@ public class ScopedMemoryAccess {
178
187
try {
179
188
setMemoryInternal(session, o, offset, bytes, value);
180
189
} catch (ScopedAccessError ex) {
181
- throw new IllegalStateException("This segment is already closed" );
190
+ throw ex.newRuntimeException( );
182
191
}
183
192
}
184
193
185
194
@ForceInline @Scoped
186
195
private void setMemoryInternal(MemorySessionImpl session, Object o, long offset, long bytes, byte value) {
187
196
try {
188
197
if (session != null) {
189
- session.checkValidState ();
198
+ session.checkValidStateRaw ();
190
199
}
191
200
UNSAFE.setMemory(o, offset, bytes, value);
192
201
} finally {
@@ -195,35 +204,35 @@ public class ScopedMemoryAccess {
195
204
}
196
205
197
206
@ForceInline
198
- public int vectorizedMismatch(MemorySessionImpl aScope , MemorySessionImpl bScope ,
207
+ public int vectorizedMismatch(MemorySessionImpl aSession , MemorySessionImpl bSession ,
199
208
Object a, long aOffset,
200
209
Object b, long bOffset,
201
210
int length,
202
211
int log2ArrayIndexScale) {
203
212
try {
204
- return vectorizedMismatchInternal(aScope, bScope , a, aOffset, b, bOffset, length, log2ArrayIndexScale);
213
+ return vectorizedMismatchInternal(aSession, bSession , a, aOffset, b, bOffset, length, log2ArrayIndexScale);
205
214
} catch (ScopedAccessError ex) {
206
- throw new IllegalStateException("This segment is already closed" );
215
+ throw ex.newRuntimeException( );
207
216
}
208
217
}
209
218
210
219
@ForceInline @Scoped
211
- private int vectorizedMismatchInternal(MemorySessionImpl aScope , MemorySessionImpl bScope ,
220
+ private int vectorizedMismatchInternal(MemorySessionImpl aSession , MemorySessionImpl bSession ,
212
221
Object a, long aOffset,
213
222
Object b, long bOffset,
214
223
int length,
215
224
int log2ArrayIndexScale) {
216
225
try {
217
- if (aScope != null) {
218
- aScope.checkValidState ();
226
+ if (aSession != null) {
227
+ aSession.checkValidStateRaw ();
219
228
}
220
- if (bScope != null) {
221
- bScope.checkValidState ();
229
+ if (bSession != null) {
230
+ bSession.checkValidStateRaw ();
222
231
}
223
232
return ArraysSupport.vectorizedMismatch(a, aOffset, b, bOffset, length, log2ArrayIndexScale);
224
233
} finally {
225
- Reference.reachabilityFence(aScope );
226
- Reference.reachabilityFence(bScope );
234
+ Reference.reachabilityFence(aSession );
235
+ Reference.reachabilityFence(bSession );
227
236
}
228
237
}
229
238
@@ -232,15 +241,15 @@ public class ScopedMemoryAccess {
232
241
try {
233
242
return isLoadedInternal(session, address, isSync, size);
234
243
} catch (ScopedAccessError ex) {
235
- throw new IllegalStateException("This segment is already closed" );
244
+ throw ex.newRuntimeException( );
236
245
}
237
246
}
238
247
239
248
@ForceInline @Scoped
240
249
public boolean isLoadedInternal(MemorySessionImpl session, long address, boolean isSync, long size) {
241
250
try {
242
251
if (session != null) {
243
- session.checkValidState ();
252
+ session.checkValidStateRaw ();
244
253
}
245
254
return SharedSecrets.getJavaNioAccess().isLoaded(address, isSync, size);
246
255
} finally {
@@ -253,15 +262,15 @@ public class ScopedMemoryAccess {
253
262
try {
254
263
loadInternal(session, address, isSync, size);
255
264
} catch (ScopedAccessError ex) {
256
- throw new IllegalStateException("This segment is already closed" );
265
+ throw ex.newRuntimeException( );
257
266
}
258
267
}
259
268
260
269
@ForceInline @Scoped
261
270
public void loadInternal(MemorySessionImpl session, long address, boolean isSync, long size) {
262
271
try {
263
272
if (session != null) {
264
- session.checkValidState ();
273
+ session.checkValidStateRaw ();
265
274
}
266
275
SharedSecrets.getJavaNioAccess().load(address, isSync, size);
267
276
} finally {
@@ -274,15 +283,15 @@ public class ScopedMemoryAccess {
274
283
try {
275
284
unloadInternal(session, address, isSync, size);
276
285
} catch (ScopedAccessError ex) {
277
- throw new IllegalStateException("This segment is already closed" );
286
+ throw ex.newRuntimeException( );
278
287
}
279
288
}
280
289
281
290
@ForceInline @Scoped
282
291
public void unloadInternal(MemorySessionImpl session, long address, boolean isSync, long size) {
283
292
try {
284
293
if (session != null) {
285
- session.checkValidState ();
294
+ session.checkValidStateRaw ();
286
295
}
287
296
SharedSecrets.getJavaNioAccess().unload(address, isSync, size);
288
297
} finally {
@@ -295,15 +304,15 @@ public class ScopedMemoryAccess {
295
304
try {
296
305
forceInternal(session, fd, address, isSync, index, length);
297
306
} catch (ScopedAccessError ex) {
298
- throw new IllegalStateException("This segment is already closed" );
307
+ throw ex.newRuntimeException( );
299
308
}
300
309
}
301
310
302
311
@ForceInline @Scoped
303
312
public void forceInternal(MemorySessionImpl session, FileDescriptor fd, long address, boolean isSync, long index, long length) {
304
313
try {
305
314
if (session != null) {
306
- session.checkValidState ();
315
+ session.checkValidStateRaw ();
307
316
}
308
317
SharedSecrets.getJavaNioAccess().force(fd, address, isSync, index, length);
309
318
} finally {
@@ -333,7 +342,7 @@ public class ScopedMemoryAccess {
333
342
s,
334
343
defaultImpl);
335
344
} catch (ScopedAccessError ex) {
336
- throw new IllegalStateException("This segment is already closed" );
345
+ throw ex.newRuntimeException( );
337
346
}
338
347
}
339
348
@@ -347,7 +356,7 @@ public class ScopedMemoryAccess {
347
356
S s,
348
357
VectorSupport.LoadOperation<AbstractMemorySegmentImpl, V, S> defaultImpl) {
349
358
try {
350
- session.checkValidState ();
359
+ session.checkValidStateRaw ();
351
360
352
361
return VectorSupport.load(vmClass, e, length,
353
362
msp.unsafeGetBase(), msp.unsafeGetOffset() + offset,
@@ -378,7 +387,7 @@ public class ScopedMemoryAccess {
378
387
s, offsetInRange,
379
388
defaultImpl);
380
389
} catch (ScopedAccessError ex) {
381
- throw new IllegalStateException("This segment is already closed" );
390
+ throw ex.newRuntimeException( );
382
391
}
383
392
}
384
393
@@ -393,7 +402,7 @@ public class ScopedMemoryAccess {
393
402
S s, int offsetInRange,
394
403
VectorSupport.LoadVectorMaskedOperation<AbstractMemorySegmentImpl, V, S, M> defaultImpl) {
395
404
try {
396
- session.checkValidState ();
405
+ session.checkValidStateRaw ();
397
406
398
407
return VectorSupport.loadMasked(vmClass, maskClass, e, length,
399
408
msp.unsafeGetBase(), msp.unsafeGetOffset() + offset, m, offsetInRange,
@@ -424,7 +433,7 @@ public class ScopedMemoryAccess {
424
433
msp, offset,
425
434
defaultImpl);
426
435
} catch (ScopedAccessError ex) {
427
- throw new IllegalStateException("This segment is already closed" );
436
+ throw ex.newRuntimeException( );
428
437
}
429
438
}
430
439
@@ -438,7 +447,7 @@ public class ScopedMemoryAccess {
438
447
AbstractMemorySegmentImpl msp, long offset,
439
448
VectorSupport.StoreVectorOperation<AbstractMemorySegmentImpl, V> defaultImpl) {
440
449
try {
441
- session.checkValidState ();
450
+ session.checkValidStateRaw ();
442
451
443
452
VectorSupport.store(vmClass, e, length,
444
453
msp.unsafeGetBase(), msp.unsafeGetOffset() + offset,
@@ -470,7 +479,7 @@ public class ScopedMemoryAccess {
470
479
msp, offset,
471
480
defaultImpl);
472
481
} catch (ScopedAccessError ex) {
473
- throw new IllegalStateException("This segment is already closed" );
482
+ throw ex.newRuntimeException( );
474
483
}
475
484
}
476
485
@@ -484,7 +493,7 @@ public class ScopedMemoryAccess {
484
493
AbstractMemorySegmentImpl msp, long offset,
485
494
VectorSupport.StoreVectorMaskedOperation<AbstractMemorySegmentImpl, V, M> defaultImpl) {
486
495
try {
487
- session.checkValidState ();
496
+ session.checkValidStateRaw ();
488
497
489
498
VectorSupport.storeMasked(vmClass, maskClass, e, length,
490
499
msp.unsafeGetBase(), msp.unsafeGetOffset() + offset,
0 commit comments