Skip to content

Commit fed3af8

Browse files
committedJul 11, 2022
8287809: Revisit implementation of memory session
Reviewed-by: jvernee
1 parent cb6e9cb commit fed3af8

21 files changed

+327
-382
lines changed
 

‎src/java.base/share/classes/java/nio/Buffer.java

+1-6
Original file line numberDiff line numberDiff line change
@@ -37,7 +37,6 @@
3737

3838
import java.io.FileDescriptor;
3939
import java.lang.foreign.MemorySegment;
40-
import java.lang.foreign.MemorySession;
4140
import java.util.Objects;
4241
import java.util.Spliterator;
4342

@@ -771,11 +770,7 @@ final MemorySessionImpl session() {
771770
final void checkSession() {
772771
MemorySessionImpl session = session();
773772
if (session != null) {
774-
try {
775-
session.checkValidState();
776-
} catch (ScopedMemoryAccess.ScopedAccessError e) {
777-
throw new IllegalStateException("This segment is already closed");
778-
}
773+
session.checkValidState();
779774
}
780775
}
781776

‎src/java.base/share/classes/java/nio/Direct-X-Buffer.java.template

+1-5
Original file line numberDiff line numberDiff line change
@@ -313,11 +313,7 @@ class Direct$Type$Buffer$RW$$BO$
313313
if (session.ownerThread() == null && session.isCloseable()) {
314314
throw new UnsupportedOperationException("ByteBuffer derived from closeable shared sessions not supported");
315315
}
316-
try {
317-
session.checkValidState();
318-
} catch (ScopedAccessError e) {
319-
throw new IllegalStateException("This segment is already closed");
320-
}
316+
session.checkValidState();
321317
}
322318
return address;
323319
}

‎src/java.base/share/classes/jdk/internal/foreign/AbstractMemorySegmentImpl.java

+35-60
Original file line numberDiff line numberDiff line change
@@ -70,40 +70,33 @@ public abstract non-sealed class AbstractMemorySegmentImpl implements MemorySegm
7070

7171
private static final ScopedMemoryAccess SCOPED_MEMORY_ACCESS = ScopedMemoryAccess.getScopedMemoryAccess();
7272

73-
static final int READ_ONLY = 1;
7473
static final long NONCE = new Random().nextLong();
7574

76-
static final int DEFAULT_MODES = 0;
77-
7875
static final JavaNioAccess nioAccess = SharedSecrets.getJavaNioAccess();
7976

8077
final long length;
81-
final int mask;
78+
final boolean readOnly;
8279
final MemorySession session;
8380

8481
@ForceInline
85-
AbstractMemorySegmentImpl(long length, int mask, MemorySession session) {
82+
AbstractMemorySegmentImpl(long length, boolean readOnly, MemorySession session) {
8683
this.length = length;
87-
this.mask = mask;
84+
this.readOnly = readOnly;
8885
this.session = session;
8986
}
9087

91-
abstract long min();
92-
93-
abstract Object base();
94-
95-
abstract AbstractMemorySegmentImpl dup(long offset, long size, int mask, MemorySession session);
88+
abstract AbstractMemorySegmentImpl dup(long offset, long size, boolean readOnly, MemorySession session);
9689

9790
abstract ByteBuffer makeByteBuffer();
9891

9992
@Override
10093
public AbstractMemorySegmentImpl asReadOnly() {
101-
return dup(0, length, mask | READ_ONLY, session);
94+
return dup(0, length, true, session);
10295
}
10396

10497
@Override
10598
public boolean isReadOnly() {
106-
return isSet(READ_ONLY);
99+
return readOnly;
107100
}
108101

109102
@Override
@@ -119,7 +112,7 @@ public AbstractMemorySegmentImpl asSlice(long offset) {
119112
}
120113

121114
private AbstractMemorySegmentImpl asSliceNoCheck(long offset, long newSize) {
122-
return dup(offset, newSize, mask, session);
115+
return dup(offset, newSize, readOnly, session);
123116
}
124117

125118
@Override
@@ -147,7 +140,7 @@ public Stream<MemorySegment> elements(MemoryLayout elementLayout) {
147140
@Override
148141
public final MemorySegment fill(byte value){
149142
checkAccess(0, length, false);
150-
SCOPED_MEMORY_ACCESS.setMemory(sessionImpl(), base(), min(), length, value);
143+
SCOPED_MEMORY_ACCESS.setMemory(sessionImpl(), unsafeGetBase(), unsafeGetOffset(), length, value);
151144
return this;
152145
}
153146

@@ -176,8 +169,8 @@ public long mismatch(MemorySegment other) {
176169
return 0;
177170
}
178171
i = vectorizedMismatchLargeForBytes(sessionImpl(), that.sessionImpl(),
179-
this.base(), this.min(),
180-
that.base(), that.min(),
172+
this.unsafeGetBase(), this.unsafeGetOffset(),
173+
that.unsafeGetBase(), that.unsafeGetOffset(),
181174
length);
182175
if (i >= 0) {
183176
return i;
@@ -235,7 +228,7 @@ public MemoryAddress address() {
235228
public final ByteBuffer asByteBuffer() {
236229
checkArraySize("ByteBuffer", 1);
237230
ByteBuffer _bb = makeByteBuffer();
238-
if (isSet(READ_ONLY)) {
231+
if (readOnly) {
239232
//session is IMMUTABLE - obtain a RO byte buffer
240233
_bb = _bb.asReadOnlyBuffer();
241234
}
@@ -260,9 +253,9 @@ public boolean isNative() {
260253
@Override
261254
public final Optional<MemorySegment> asOverlappingSlice(MemorySegment other) {
262255
AbstractMemorySegmentImpl that = (AbstractMemorySegmentImpl)Objects.requireNonNull(other);
263-
if (base() == that.base()) { // both either native or heap
264-
final long thisStart = this.min();
265-
final long thatStart = that.min();
256+
if (unsafeGetBase() == that.unsafeGetBase()) { // both either native or heap
257+
final long thisStart = this.unsafeGetOffset();
258+
final long thatStart = that.unsafeGetOffset();
266259
final long thisEnd = thisStart + this.byteSize();
267260
final long thatEnd = thatStart + that.byteSize();
268261

@@ -278,8 +271,8 @@ public final Optional<MemorySegment> asOverlappingSlice(MemorySegment other) {
278271
@Override
279272
public final long segmentOffset(MemorySegment other) {
280273
AbstractMemorySegmentImpl that = (AbstractMemorySegmentImpl) Objects.requireNonNull(other);
281-
if (base() == that.base()) {
282-
return that.min() - this.min();
274+
if (unsafeGetBase() == that.unsafeGetBase()) {
275+
return that.unsafeGetOffset() - this.unsafeGetOffset();
283276
}
284277
throw new UnsupportedOperationException("Cannot compute offset from native to heap (or vice versa).");
285278
}
@@ -347,31 +340,24 @@ private <Z> Z toArray(Class<Z> arrayClass, ValueLayout elemLayout, IntFunction<Z
347340
return arr;
348341
}
349342

343+
@ForceInline
350344
public void checkAccess(long offset, long length, boolean readOnly) {
351-
if (!readOnly && isSet(READ_ONLY)) {
345+
if (!readOnly && this.readOnly) {
352346
throw new UnsupportedOperationException("Attempt to write a read-only segment");
353347
}
354348
checkBounds(offset, length);
355349
}
356350

357351
public void checkValidState() {
358-
sessionImpl().checkValidStateSlow();
352+
sessionImpl().checkValidState();
359353
}
360354

361-
public long unsafeGetOffset() {
362-
return min();
363-
}
355+
public abstract long unsafeGetOffset();
364356

365-
public Object unsafeGetBase() {
366-
return base();
367-
}
357+
public abstract Object unsafeGetBase();
368358

369359
// Helper methods
370360

371-
private boolean isSet(int mask) {
372-
return (this.mask & mask) != 0;
373-
}
374-
375361
public abstract long maxAlignMask();
376362

377363
@ForceInline
@@ -407,25 +393,19 @@ public RuntimeException apply(String s, List<Number> numbers) {
407393
return outOfBoundException(offset, length);
408394
}
409395

410-
@Override
411-
@ForceInline
412-
public MemorySessionImpl sessionImpl() {
413-
return MemorySessionImpl.toSessionImpl(session);
414-
}
415-
416396
@Override
417397
public MemorySession session() {
418398
return session;
419399
}
420400

421401
private IndexOutOfBoundsException outOfBoundException(long offset, long length) {
422-
return new IndexOutOfBoundsException(String.format("Out of bound access on segment %s; offset = %d; length = %d",
402+
return new IndexOutOfBoundsException(String.format("Out of bound access on segment %s; new offset = %d; new length = %d",
423403
this, offset, length));
424404
}
425405

426406
protected int id() {
427407
//compute a stable and random id for this memory segment
428-
return Math.abs(Objects.hash(base(), min(), NONCE));
408+
return Math.abs(Objects.hash(unsafeGetBase(), unsafeGetOffset(), NONCE));
429409
}
430410

431411
static class SegmentSplitter implements Spliterator<MemorySegment> {
@@ -541,42 +521,37 @@ public static AbstractMemorySegmentImpl ofBuffer(Buffer bb) {
541521
int size = limit - pos;
542522

543523
AbstractMemorySegmentImpl bufferSegment = (AbstractMemorySegmentImpl)nioAccess.bufferSegment(bb);
544-
final MemorySessionImpl bufferSession;
545-
int modes;
524+
final MemorySession bufferSession;
546525
if (bufferSegment != null) {
547-
bufferSession = bufferSegment.sessionImpl();
548-
modes = bufferSegment.mask;
526+
bufferSession = bufferSegment.session;
549527
} else {
550528
bufferSession = MemorySessionImpl.heapSession(bb);
551-
modes = DEFAULT_MODES;
552-
}
553-
if (bb.isReadOnly()) {
554-
modes |= READ_ONLY;
555529
}
530+
boolean readOnly = bb.isReadOnly();
556531
int scaleFactor = getScaleFactor(bb);
557532
if (base != null) {
558533
if (base instanceof byte[]) {
559-
return new HeapMemorySegmentImpl.OfByte(bbAddress + (pos << scaleFactor), base, size << scaleFactor, modes);
534+
return new HeapMemorySegmentImpl.OfByte(bbAddress + (pos << scaleFactor), base, size << scaleFactor, readOnly);
560535
} else if (base instanceof short[]) {
561-
return new HeapMemorySegmentImpl.OfShort(bbAddress + (pos << scaleFactor), base, size << scaleFactor, modes);
536+
return new HeapMemorySegmentImpl.OfShort(bbAddress + (pos << scaleFactor), base, size << scaleFactor, readOnly);
562537
} else if (base instanceof char[]) {
563-
return new HeapMemorySegmentImpl.OfChar(bbAddress + (pos << scaleFactor), base, size << scaleFactor, modes);
538+
return new HeapMemorySegmentImpl.OfChar(bbAddress + (pos << scaleFactor), base, size << scaleFactor, readOnly);
564539
} else if (base instanceof int[]) {
565-
return new HeapMemorySegmentImpl.OfInt(bbAddress + (pos << scaleFactor), base, size << scaleFactor, modes);
540+
return new HeapMemorySegmentImpl.OfInt(bbAddress + (pos << scaleFactor), base, size << scaleFactor, readOnly);
566541
} else if (base instanceof float[]) {
567-
return new HeapMemorySegmentImpl.OfFloat(bbAddress + (pos << scaleFactor), base, size << scaleFactor, modes);
542+
return new HeapMemorySegmentImpl.OfFloat(bbAddress + (pos << scaleFactor), base, size << scaleFactor, readOnly);
568543
} else if (base instanceof long[]) {
569-
return new HeapMemorySegmentImpl.OfLong(bbAddress + (pos << scaleFactor), base, size << scaleFactor, modes);
544+
return new HeapMemorySegmentImpl.OfLong(bbAddress + (pos << scaleFactor), base, size << scaleFactor, readOnly);
570545
} else if (base instanceof double[]) {
571-
return new HeapMemorySegmentImpl.OfDouble(bbAddress + (pos << scaleFactor), base, size << scaleFactor, modes);
546+
return new HeapMemorySegmentImpl.OfDouble(bbAddress + (pos << scaleFactor), base, size << scaleFactor, readOnly);
572547
} else {
573548
throw new AssertionError("Cannot get here");
574549
}
575550
} else if (unmapper == null) {
576-
return new NativeMemorySegmentImpl(bbAddress + (pos << scaleFactor), size << scaleFactor, modes, bufferSession);
551+
return new NativeMemorySegmentImpl(bbAddress + (pos << scaleFactor), size << scaleFactor, readOnly, bufferSession);
577552
} else {
578553
// we can ignore scale factor here, a mapped buffer is always a byte buffer, so scaleFactor == 0.
579-
return new MappedMemorySegmentImpl(bbAddress + pos, unmapper, size, modes, bufferSession);
554+
return new MappedMemorySegmentImpl(bbAddress + pos, unmapper, size, readOnly, bufferSession);
580555
}
581556
}
582557

‎src/java.base/share/classes/jdk/internal/foreign/ConfinedSession.java

+6-11
Original file line numberDiff line numberDiff line change
@@ -55,17 +55,12 @@ public ConfinedSession(Thread owner, Cleaner cleaner) {
5555
super(owner, new ConfinedResourceList(), cleaner);
5656
}
5757

58-
@Override
59-
public boolean isAlive() {
60-
return state != CLOSED;
61-
}
62-
6358
@Override
6459
@ForceInline
6560
public void acquire0() {
66-
checkValidStateSlow();
61+
checkValidState();
6762
if (state == MAX_FORKS) {
68-
throw new IllegalStateException("Session keep alive limit exceeded");
63+
throw tooManyAcquires();
6964
}
7065
state++;
7166
}
@@ -85,11 +80,11 @@ public void release0() {
8580
}
8681

8782
void justClose() {
88-
checkValidStateSlow();
83+
checkValidState();
8984
if (state == 0 || state - ((int)ASYNC_RELEASE_COUNT.getVolatile(this)) == 0) {
9085
state = CLOSED;
9186
} else {
92-
throw new IllegalStateException("Session is acquired by " + state + " clients");
87+
throw alreadyAcquired(state);
9388
}
9489
}
9590

@@ -103,7 +98,7 @@ void add(ResourceCleanup cleanup) {
10398
cleanup.next = fst;
10499
fst = cleanup;
105100
} else {
106-
throw new IllegalStateException("Already closed!");
101+
throw alreadyClosed();
107102
}
108103
}
109104

@@ -114,7 +109,7 @@ void cleanup() {
114109
fst = ResourceCleanup.CLOSED_LIST;
115110
cleanup(prev);
116111
} else {
117-
throw new IllegalStateException("Attempt to cleanup an already closed resource list");
112+
throw alreadyClosed();
118113
}
119114
}
120115
}

‎src/java.base/share/classes/jdk/internal/foreign/HeapMemorySegmentImpl.java

+49-52
Original file line numberDiff line numberDiff line change
@@ -40,7 +40,7 @@
4040
* a base object (typically an array). To enhance performances, the access to the base object needs to feature
4141
* sharp type information, as well as sharp null-check information. For this reason, many concrete subclasses
4242
* of {@link HeapMemorySegmentImpl} are defined (e.g. {@link OfFloat}, so that each subclass can override the
43-
* {@link HeapMemorySegmentImpl#base()} method so that it returns an array of the correct (sharp) type. Note that
43+
* {@link HeapMemorySegmentImpl#unsafeGetBase()} method so that it returns an array of the correct (sharp) type. Note that
4444
* the field type storing the 'base' coordinate is just Object; similarly, all the constructor in the subclasses
4545
* accept an Object 'base' parameter instead of a sharper type (e.g. {@code byte[]}). This is deliberate, as
4646
* using sharper types would require use of type-conversions, which in turn would inhibit some C2 optimizations,
@@ -60,54 +60,51 @@ public abstract class HeapMemorySegmentImpl extends AbstractMemorySegmentImpl {
6060
final Object base;
6161

6262
@ForceInline
63-
HeapMemorySegmentImpl(long offset, Object base, long length, int mask) {
64-
super(length, mask, MemorySessionImpl.GLOBAL);
63+
HeapMemorySegmentImpl(long offset, Object base, long length, boolean readOnly) {
64+
super(length, readOnly, MemorySessionImpl.GLOBAL);
6565
this.offset = offset;
6666
this.base = base;
6767
}
6868

6969
@Override
70-
abstract Object base();
71-
72-
@Override
73-
long min() {
70+
public long unsafeGetOffset() {
7471
return offset;
7572
}
7673

7774
@Override
78-
abstract HeapMemorySegmentImpl dup(long offset, long size, int mask, MemorySession session);
75+
abstract HeapMemorySegmentImpl dup(long offset, long size, boolean readOnly, MemorySession session);
7976

8077
@Override
8178
ByteBuffer makeByteBuffer() {
82-
if (!(base() instanceof byte[])) {
79+
if (!(base instanceof byte[])) {
8380
throw new UnsupportedOperationException("Not an address to an heap-allocated byte array");
8481
}
8582
JavaNioAccess nioAccess = SharedSecrets.getJavaNioAccess();
86-
return nioAccess.newHeapByteBuffer((byte[]) base(), (int)min() - BYTE_ARR_BASE, (int) byteSize(), null);
83+
return nioAccess.newHeapByteBuffer((byte[])base, (int)offset - BYTE_ARR_BASE, (int) byteSize(), null);
8784
}
8885

8986
// factories
9087

9188
public static class OfByte extends HeapMemorySegmentImpl {
9289

93-
OfByte(long offset, Object base, long length, int mask) {
94-
super(offset, base, length, mask);
90+
OfByte(long offset, Object base, long length, boolean readOnly) {
91+
super(offset, base, length, readOnly);
9592
}
9693

9794
@Override
98-
OfByte dup(long offset, long size, int mask, MemorySession session) {
99-
return new OfByte(this.offset + offset, base, size, mask);
95+
OfByte dup(long offset, long size, boolean readOnly, MemorySession session) {
96+
return new OfByte(this.offset + offset, base, size, readOnly);
10097
}
10198

10299
@Override
103-
byte[] base() {
100+
public byte[] unsafeGetBase() {
104101
return (byte[])Objects.requireNonNull(base);
105102
}
106103

107104
public static MemorySegment fromArray(byte[] arr) {
108105
Objects.requireNonNull(arr);
109106
long byteSize = (long)arr.length * Unsafe.ARRAY_BYTE_INDEX_SCALE;
110-
return new OfByte(Unsafe.ARRAY_BYTE_BASE_OFFSET, arr, byteSize, DEFAULT_MODES);
107+
return new OfByte(Unsafe.ARRAY_BYTE_BASE_OFFSET, arr, byteSize, false);
111108
}
112109

113110
@Override
@@ -118,24 +115,24 @@ public long maxAlignMask() {
118115

119116
public static class OfChar extends HeapMemorySegmentImpl {
120117

121-
OfChar(long offset, Object base, long length, int mask) {
122-
super(offset, base, length, mask);
118+
OfChar(long offset, Object base, long length, boolean readOnly) {
119+
super(offset, base, length, readOnly);
123120
}
124121

125122
@Override
126-
OfChar dup(long offset, long size, int mask, MemorySession session) {
127-
return new OfChar(this.offset + offset, base, size, mask);
123+
OfChar dup(long offset, long size, boolean readOnly, MemorySession session) {
124+
return new OfChar(this.offset + offset, base, size, readOnly);
128125
}
129126

130127
@Override
131-
char[] base() {
128+
public char[] unsafeGetBase() {
132129
return (char[])Objects.requireNonNull(base);
133130
}
134131

135132
public static MemorySegment fromArray(char[] arr) {
136133
Objects.requireNonNull(arr);
137134
long byteSize = (long)arr.length * Unsafe.ARRAY_CHAR_INDEX_SCALE;
138-
return new OfChar(Unsafe.ARRAY_CHAR_BASE_OFFSET, arr, byteSize, DEFAULT_MODES);
135+
return new OfChar(Unsafe.ARRAY_CHAR_BASE_OFFSET, arr, byteSize, false);
139136
}
140137

141138
@Override
@@ -146,24 +143,24 @@ public long maxAlignMask() {
146143

147144
public static class OfShort extends HeapMemorySegmentImpl {
148145

149-
OfShort(long offset, Object base, long length, int mask) {
150-
super(offset, base, length, mask);
146+
OfShort(long offset, Object base, long length, boolean readOnly) {
147+
super(offset, base, length, readOnly);
151148
}
152149

153150
@Override
154-
OfShort dup(long offset, long size, int mask, MemorySession session) {
155-
return new OfShort(this.offset + offset, base, size, mask);
151+
OfShort dup(long offset, long size, boolean readOnly, MemorySession session) {
152+
return new OfShort(this.offset + offset, base, size, readOnly);
156153
}
157154

158155
@Override
159-
short[] base() {
156+
public short[] unsafeGetBase() {
160157
return (short[])Objects.requireNonNull(base);
161158
}
162159

163160
public static MemorySegment fromArray(short[] arr) {
164161
Objects.requireNonNull(arr);
165162
long byteSize = (long)arr.length * Unsafe.ARRAY_SHORT_INDEX_SCALE;
166-
return new OfShort(Unsafe.ARRAY_SHORT_BASE_OFFSET, arr, byteSize, DEFAULT_MODES);
163+
return new OfShort(Unsafe.ARRAY_SHORT_BASE_OFFSET, arr, byteSize, false);
167164
}
168165

169166
@Override
@@ -174,24 +171,24 @@ public long maxAlignMask() {
174171

175172
public static class OfInt extends HeapMemorySegmentImpl {
176173

177-
OfInt(long offset, Object base, long length, int mask) {
178-
super(offset, base, length, mask);
174+
OfInt(long offset, Object base, long length, boolean readOnly) {
175+
super(offset, base, length, readOnly);
179176
}
180177

181178
@Override
182-
OfInt dup(long offset, long size, int mask, MemorySession session) {
183-
return new OfInt(this.offset + offset, base, size, mask);
179+
OfInt dup(long offset, long size, boolean readOnly, MemorySession session) {
180+
return new OfInt(this.offset + offset, base, size, readOnly);
184181
}
185182

186183
@Override
187-
int[] base() {
184+
public int[] unsafeGetBase() {
188185
return (int[])Objects.requireNonNull(base);
189186
}
190187

191188
public static MemorySegment fromArray(int[] arr) {
192189
Objects.requireNonNull(arr);
193190
long byteSize = (long)arr.length * Unsafe.ARRAY_INT_INDEX_SCALE;
194-
return new OfInt(Unsafe.ARRAY_INT_BASE_OFFSET, arr, byteSize, DEFAULT_MODES);
191+
return new OfInt(Unsafe.ARRAY_INT_BASE_OFFSET, arr, byteSize, false);
195192
}
196193

197194
@Override
@@ -202,24 +199,24 @@ public long maxAlignMask() {
202199

203200
public static class OfLong extends HeapMemorySegmentImpl {
204201

205-
OfLong(long offset, Object base, long length, int mask) {
206-
super(offset, base, length, mask);
202+
OfLong(long offset, Object base, long length, boolean readOnly) {
203+
super(offset, base, length, readOnly);
207204
}
208205

209206
@Override
210-
OfLong dup(long offset, long size, int mask, MemorySession session) {
211-
return new OfLong(this.offset + offset, base, size, mask);
207+
OfLong dup(long offset, long size, boolean readOnly, MemorySession session) {
208+
return new OfLong(this.offset + offset, base, size, readOnly);
212209
}
213210

214211
@Override
215-
long[] base() {
212+
public long[] unsafeGetBase() {
216213
return (long[])Objects.requireNonNull(base);
217214
}
218215

219216
public static MemorySegment fromArray(long[] arr) {
220217
Objects.requireNonNull(arr);
221218
long byteSize = (long)arr.length * Unsafe.ARRAY_LONG_INDEX_SCALE;
222-
return new OfLong(Unsafe.ARRAY_LONG_BASE_OFFSET, arr, byteSize, DEFAULT_MODES);
219+
return new OfLong(Unsafe.ARRAY_LONG_BASE_OFFSET, arr, byteSize, false);
223220
}
224221

225222
@Override
@@ -230,24 +227,24 @@ public long maxAlignMask() {
230227

231228
public static class OfFloat extends HeapMemorySegmentImpl {
232229

233-
OfFloat(long offset, Object base, long length, int mask) {
234-
super(offset, base, length, mask);
230+
OfFloat(long offset, Object base, long length, boolean readOnly) {
231+
super(offset, base, length, readOnly);
235232
}
236233

237234
@Override
238-
OfFloat dup(long offset, long size, int mask, MemorySession session) {
239-
return new OfFloat(this.offset + offset, base, size, mask);
235+
OfFloat dup(long offset, long size, boolean readOnly, MemorySession session) {
236+
return new OfFloat(this.offset + offset, base, size, readOnly);
240237
}
241238

242239
@Override
243-
float[] base() {
240+
public float[] unsafeGetBase() {
244241
return (float[])Objects.requireNonNull(base);
245242
}
246243

247244
public static MemorySegment fromArray(float[] arr) {
248245
Objects.requireNonNull(arr);
249246
long byteSize = (long)arr.length * Unsafe.ARRAY_FLOAT_INDEX_SCALE;
250-
return new OfFloat(Unsafe.ARRAY_FLOAT_BASE_OFFSET, arr, byteSize, DEFAULT_MODES);
247+
return new OfFloat(Unsafe.ARRAY_FLOAT_BASE_OFFSET, arr, byteSize, false);
251248
}
252249

253250
@Override
@@ -258,24 +255,24 @@ public long maxAlignMask() {
258255

259256
public static class OfDouble extends HeapMemorySegmentImpl {
260257

261-
OfDouble(long offset, Object base, long length, int mask) {
262-
super(offset, base, length, mask);
258+
OfDouble(long offset, Object base, long length, boolean readOnly) {
259+
super(offset, base, length, readOnly);
263260
}
264261

265262
@Override
266-
OfDouble dup(long offset, long size, int mask, MemorySession session) {
267-
return new OfDouble(this.offset + offset, base, size, mask);
263+
OfDouble dup(long offset, long size, boolean readOnly, MemorySession session) {
264+
return new OfDouble(this.offset + offset, base, size, readOnly);
268265
}
269266

270267
@Override
271-
double[] base() {
268+
public double[] unsafeGetBase() {
272269
return (double[])Objects.requireNonNull(base);
273270
}
274271

275272
public static MemorySegment fromArray(double[] arr) {
276273
Objects.requireNonNull(arr);
277274
long byteSize = (long)arr.length * Unsafe.ARRAY_DOUBLE_INDEX_SCALE;
278-
return new OfDouble(Unsafe.ARRAY_DOUBLE_BASE_OFFSET, arr, byteSize, DEFAULT_MODES);
275+
return new OfDouble(Unsafe.ARRAY_DOUBLE_BASE_OFFSET, arr, byteSize, false);
279276
}
280277

281278
@Override

‎src/java.base/share/classes/jdk/internal/foreign/MappedMemorySegmentImpl.java

+6-6
Original file line numberDiff line numberDiff line change
@@ -43,8 +43,8 @@ public class MappedMemorySegmentImpl extends NativeMemorySegmentImpl {
4343

4444
static ScopedMemoryAccess SCOPED_MEMORY_ACCESS = ScopedMemoryAccess.getScopedMemoryAccess();
4545

46-
public MappedMemorySegmentImpl(long min, UnmapperProxy unmapper, long length, int mask, MemorySession session) {
47-
super(min, length, mask, session);
46+
public MappedMemorySegmentImpl(long min, UnmapperProxy unmapper, long length, boolean readOnly, MemorySession session) {
47+
super(min, length, readOnly, session);
4848
this.unmapper = unmapper;
4949
}
5050

@@ -55,8 +55,8 @@ ByteBuffer makeByteBuffer() {
5555
}
5656

5757
@Override
58-
MappedMemorySegmentImpl dup(long offset, long size, int mask, MemorySession session) {
59-
return new MappedMemorySegmentImpl(min + offset, unmapper, size, mask, session);
58+
MappedMemorySegmentImpl dup(long offset, long size, boolean readOnly, MemorySession session) {
59+
return new MappedMemorySegmentImpl(min + offset, unmapper, size, readOnly, session);
6060
}
6161

6262
// mapped segment methods
@@ -95,8 +95,8 @@ public void force() {
9595

9696
public static class EmptyMappedMemorySegmentImpl extends MappedMemorySegmentImpl {
9797

98-
public EmptyMappedMemorySegmentImpl(int modes, MemorySession session) {
99-
super(0, null, 0, modes, session);
98+
public EmptyMappedMemorySegmentImpl(boolean readOnly, MemorySessionImpl session) {
99+
super(0, null, 0, readOnly, session);
100100
}
101101

102102
@Override

‎src/java.base/share/classes/jdk/internal/foreign/MemoryAddressImpl.java

+1-1
Original file line numberDiff line numberDiff line change
@@ -95,7 +95,7 @@ public static MemorySegment ofLongUnchecked(long value, long byteSize) {
9595
}
9696

9797
@Override
98-
public MemorySessionImpl sessionImpl() {
98+
public MemorySessionImpl session() {
9999
return MemorySessionImpl.GLOBAL;
100100
}
101101

‎src/java.base/share/classes/jdk/internal/foreign/MemorySessionImpl.java

+51-47
Original file line numberDiff line numberDiff line change
@@ -43,16 +43,16 @@
4343
* This class manages the temporal bounds associated with a memory segment as well
4444
* as thread confinement. A session has a liveness bit, which is updated when the session is closed
4545
* (this operation is triggered by {@link MemorySession#close()}). This bit is consulted prior
46-
* to memory access (see {@link #checkValidState()}).
46+
* to memory access (see {@link #checkValidStateRaw()}).
4747
* There are two kinds of memory session: confined memory session and shared memory session.
4848
* A confined memory session has an associated owner thread that confines some operations to
49-
* associated owner thread such as {@link #close()} or {@link #checkValidState()}.
49+
* associated owner thread such as {@link #close()} or {@link #checkValidStateRaw()}.
5050
* Shared sessions do not feature an owner thread - meaning their operations can be called, in a racy
5151
* manner, by multiple threads. To guarantee temporal safety in the presence of concurrent thread,
5252
* shared sessions use a more sophisticated synchronization mechanism, which guarantees that no concurrent
5353
* access is possible when a session is being closed (see {@link jdk.internal.misc.ScopedMemoryAccess}).
5454
*/
55-
public abstract non-sealed class MemorySessionImpl implements Scoped, MemorySession, SegmentAllocator {
55+
public abstract non-sealed class MemorySessionImpl implements MemorySession, SegmentAllocator {
5656
final ResourceList resourceList;
5757
final Cleaner.Cleanable cleanable;
5858
final Thread owner;
@@ -78,8 +78,7 @@ public abstract non-sealed class MemorySessionImpl implements Scoped, MemorySess
7878
@Override
7979
public void addCloseAction(Runnable runnable) {
8080
Objects.requireNonNull(runnable);
81-
addInternal(runnable instanceof ResourceList.ResourceCleanup cleanup ?
82-
cleanup : ResourceList.ResourceCleanup.ofRunnable(runnable));
81+
addInternal(ResourceList.ResourceCleanup.ofRunnable(runnable));
8382
}
8483

8584
/**
@@ -102,7 +101,7 @@ public void addOrCleanupIfFail(ResourceList.ResourceCleanup resource) {
102101
}
103102

104103
void addInternal(ResourceList.ResourceCleanup resource) {
105-
checkValidStateSlow();
104+
checkValidState();
106105
// Note: from here on we no longer check the session state. Two cases are possible: either the resource cleanup
107106
// is added to the list when the session is still open, in which case everything works ok; or the resource
108107
// cleanup is added while the session is being closed. In this latter case, what matters is whether we have already
@@ -141,13 +140,13 @@ public MemorySegment allocate(long bytesSize, long bytesAlignment) {
141140
public abstract void acquire0();
142141

143142
@Override
144-
public boolean equals(Object o) {
143+
public final boolean equals(Object o) {
145144
return (o instanceof MemorySession other) &&
146145
toSessionImpl(other) == this;
147146
}
148147

149148
@Override
150-
public int hashCode() {
149+
public final int hashCode() {
151150
return super.hashCode();
152151
}
153152

@@ -174,38 +173,37 @@ public final Thread ownerThread() {
174173
* Returns true, if this session is still open. This method may be called in any thread.
175174
* @return {@code true} if this session is not closed yet.
176175
*/
177-
public abstract boolean isAlive();
176+
public boolean isAlive() {
177+
return state >= OPEN;
178+
}
178179

179180
@Override
180181
public MemorySession asNonCloseable() {
181182
return isCloseable() ?
182183
new NonCloseableView(this) : this;
183184
}
184185

186+
@ForceInline
185187
public static MemorySessionImpl toSessionImpl(MemorySession session) {
186-
return ((Scoped)session).sessionImpl();
187-
}
188-
189-
@Override
190-
public MemorySessionImpl sessionImpl() {
191-
return this;
188+
return session instanceof MemorySessionImpl sessionImpl ?
189+
sessionImpl : ((NonCloseableView)session).session;
192190
}
193191

194192
/**
195-
* This is a faster version of {@link #checkValidStateSlow()}, which is called upon memory access, and which
193+
* This is a faster version of {@link #checkValidState()}, which is called upon memory access, and which
196194
* relies on invariants associated with the memory session implementations (volatile access
197195
* to the closed state bit is replaced with plain access). This method should be monomorphic,
198196
* to avoid virtual calls in the memory access hot path. This method is not intended as general purpose method
199197
* and should only be used in the memory access handle hot path; for liveness checks triggered by other API methods,
200-
* please use {@link #checkValidStateSlow()}.
198+
* please use {@link #checkValidState()}.
201199
*/
202200
@ForceInline
203-
public final void checkValidState() {
201+
public void checkValidStateRaw() {
204202
if (owner != null && owner != Thread.currentThread()) {
205-
throw new WrongThreadException("Attempted access outside owning thread");
203+
throw WRONG_THREAD;
206204
}
207205
if (state < OPEN) {
208-
throw ScopedMemoryAccess.ScopedAccessError.INSTANCE;
206+
throw ALREADY_CLOSED;
209207
}
210208
}
211209

@@ -214,11 +212,11 @@ public final void checkValidState() {
214212
* @throws IllegalStateException if this session is already closed or if this is
215213
* a confined session and this method is called outside of the owner thread.
216214
*/
217-
public final void checkValidStateSlow() {
218-
if (owner != null && Thread.currentThread() != owner) {
219-
throw new WrongThreadException("Attempted access outside owning thread");
220-
} else if (!isAlive()) {
221-
throw new IllegalStateException("Already closed");
215+
public void checkValidState() {
216+
try {
217+
checkValidStateRaw();
218+
} catch (ScopedMemoryAccess.ScopedAccessError error) {
219+
throw error.newRuntimeException();
222220
}
223221
}
224222

@@ -289,14 +287,9 @@ void addInternal(ResourceList.ResourceCleanup resource) {
289287
// do nothing
290288
}
291289

292-
@Override
293-
public boolean isAlive() {
294-
return true;
295-
}
296-
297290
@Override
298291
public void justClose() {
299-
throw new UnsupportedOperationException();
292+
throw nonCloseable();
300293
}
301294
}
302295

@@ -335,19 +328,9 @@ public boolean isCloseable() {
335328
return false;
336329
}
337330

338-
@Override
339-
public boolean isAlive() {
340-
return true;
341-
}
342-
343-
@Override
344-
public MemorySession asNonCloseable() {
345-
return this;
346-
}
347-
348331
@Override
349332
public void justClose() {
350-
throw new UnsupportedOperationException();
333+
throw nonCloseable();
351334
}
352335
}
353336

@@ -358,17 +341,13 @@ public void justClose() {
358341
* a strong reference to the original session, so even if the original session is dropped by the client
359342
* it would still be reachable by the GC, which is important if the session is implicitly closed.
360343
*/
361-
public final static class NonCloseableView implements MemorySession, Scoped {
344+
public final static class NonCloseableView implements MemorySession {
362345
final MemorySessionImpl session;
363346

364347
public NonCloseableView(MemorySessionImpl session) {
365348
this.session = session;
366349
}
367350

368-
public MemorySessionImpl sessionImpl() {
369-
return session;
370-
}
371-
372351
@Override
373352
public boolean isAlive() {
374353
return session.isAlive();
@@ -461,6 +440,31 @@ public void cleanup() {
461440
};
462441
}
463442
}
443+
}
444+
445+
// helper functions to centralize error handling
446+
447+
static IllegalStateException tooManyAcquires() {
448+
return new IllegalStateException("Session acquire limit exceeded");
449+
}
464450

451+
static IllegalStateException alreadyAcquired(int acquires) {
452+
return new IllegalStateException(String.format("Session is acquired by %d clients", acquires));
465453
}
454+
455+
static IllegalStateException alreadyClosed() {
456+
return new IllegalStateException("Already closed");
457+
}
458+
459+
static WrongThreadException wrongThread() {
460+
return new WrongThreadException("Attempted access outside owning thread");
461+
}
462+
463+
static UnsupportedOperationException nonCloseable() {
464+
return new UnsupportedOperationException("Attempted to close a non-closeable session");
465+
}
466+
467+
static final ScopedMemoryAccess.ScopedAccessError ALREADY_CLOSED = new ScopedMemoryAccess.ScopedAccessError(MemorySessionImpl::alreadyClosed);
468+
469+
static final ScopedMemoryAccess.ScopedAccessError WRONG_THREAD = new ScopedMemoryAccess.ScopedAccessError(MemorySessionImpl::wrongThread);
466470
}

‎src/java.base/share/classes/jdk/internal/foreign/NativeMemorySegmentImpl.java

+13-14
Original file line numberDiff line numberDiff line change
@@ -41,14 +41,14 @@
4141
*/
4242
public class NativeMemorySegmentImpl extends AbstractMemorySegmentImpl {
4343

44-
public static final MemorySegment EVERYTHING = new NativeMemorySegmentImpl(0, Long.MAX_VALUE, 0, MemorySessionImpl.GLOBAL) {
44+
public static final MemorySegment EVERYTHING = new NativeMemorySegmentImpl(0, Long.MAX_VALUE, false, MemorySessionImpl.GLOBAL) {
4545
@Override
4646
void checkBounds(long offset, long length) {
4747
// do nothing
4848
}
4949

5050
@Override
51-
NativeMemorySegmentImpl dup(long offset, long size, int mask, MemorySession scope) {
51+
NativeMemorySegmentImpl dup(long offset, long size, boolean readOnly, MemorySession session) {
5252
throw new IllegalStateException();
5353
}
5454
};
@@ -64,8 +64,8 @@ NativeMemorySegmentImpl dup(long offset, long size, int mask, MemorySession scop
6464
final long min;
6565

6666
@ForceInline
67-
NativeMemorySegmentImpl(long min, long length, int mask, MemorySession session) {
68-
super(length, mask, session);
67+
NativeMemorySegmentImpl(long min, long length, boolean readOnly, MemorySession session) {
68+
super(length, readOnly, session);
6969
this.min = min;
7070
}
7171

@@ -77,13 +77,13 @@ public MemoryAddress address() {
7777
}
7878

7979
@Override
80-
NativeMemorySegmentImpl dup(long offset, long size, int mask, MemorySession session) {
81-
return new NativeMemorySegmentImpl(min + offset, size, mask, session);
80+
NativeMemorySegmentImpl dup(long offset, long size, boolean readOnly, MemorySession session) {
81+
return new NativeMemorySegmentImpl(min + offset, size, readOnly, session);
8282
}
8383

8484
@Override
8585
ByteBuffer makeByteBuffer() {
86-
return nioAccess.newDirectByteBuffer(min(), (int) this.length, null,
86+
return nioAccess.newDirectByteBuffer(min, (int) this.length, null,
8787
session == MemorySessionImpl.GLOBAL ? null : this);
8888
}
8989

@@ -93,12 +93,12 @@ public boolean isNative() {
9393
}
9494

9595
@Override
96-
long min() {
96+
public long unsafeGetOffset() {
9797
return min;
9898
}
9999

100100
@Override
101-
Object base() {
101+
public Object unsafeGetBase() {
102102
return null;
103103
}
104104

@@ -111,7 +111,7 @@ public long maxAlignMask() {
111111

112112
public static MemorySegment makeNativeSegment(long bytesSize, long alignmentBytes, MemorySession session) {
113113
MemorySessionImpl sessionImpl = MemorySessionImpl.toSessionImpl(session);
114-
sessionImpl.checkValidStateSlow();
114+
sessionImpl.checkValidState();
115115
if (VM.isDirectMemoryPageAligned()) {
116116
alignmentBytes = Math.max(alignmentBytes, nioAccess.pageSize());
117117
}
@@ -127,7 +127,7 @@ public static MemorySegment makeNativeSegment(long bytesSize, long alignmentByte
127127
}
128128
long alignedBuf = Utils.alignUp(buf, alignmentBytes);
129129
AbstractMemorySegmentImpl segment = new NativeMemorySegmentImpl(buf, alignedSize,
130-
DEFAULT_MODES, session);
130+
false, session);
131131
sessionImpl.addOrCleanupIfFail(new MemorySessionImpl.ResourceList.ResourceCleanup() {
132132
@Override
133133
public void cleanup() {
@@ -143,9 +143,8 @@ public void cleanup() {
143143
}
144144

145145
public static MemorySegment makeNativeSegmentUnchecked(MemoryAddress min, long bytesSize, MemorySession session) {
146-
MemorySessionImpl sessionImpl = MemorySessionImpl.toSessionImpl(session);
147-
sessionImpl.checkValidStateSlow();
148-
AbstractMemorySegmentImpl segment = new NativeMemorySegmentImpl(min.toRawLongValue(), bytesSize, DEFAULT_MODES, session);
146+
MemorySessionImpl.toSessionImpl(session).checkValidState();
147+
AbstractMemorySegmentImpl segment = new NativeMemorySegmentImpl(min.toRawLongValue(), bytesSize, false, session);
149148
return segment;
150149
}
151150
}

‎src/java.base/share/classes/jdk/internal/foreign/Scoped.java

+9-1
Original file line numberDiff line numberDiff line change
@@ -25,6 +25,14 @@
2525

2626
package jdk.internal.foreign;
2727

28+
import jdk.internal.vm.annotation.ForceInline;
29+
30+
import java.lang.foreign.MemorySession;
31+
2832
public interface Scoped {
29-
MemorySessionImpl sessionImpl();
33+
@ForceInline
34+
default MemorySessionImpl sessionImpl() {
35+
return MemorySessionImpl.toSessionImpl(session());
36+
}
37+
MemorySession session();
3038
}

‎src/java.base/share/classes/jdk/internal/foreign/SharedSession.java

+8-13
Original file line numberDiff line numberDiff line change
@@ -56,10 +56,10 @@ public void acquire0() {
5656
value = (int) STATE.getVolatile(this);
5757
if (value < OPEN) {
5858
//segment is not open!
59-
throw new IllegalStateException("Already closed");
59+
throw alreadyClosed();
6060
} else if (value == MAX_FORKS) {
6161
//overflow
62-
throw new IllegalStateException("Session acquire limit exceeded");
62+
throw tooManyAcquires();
6363
}
6464
} while (!STATE.compareAndSet(this, value, value + 1));
6565
}
@@ -72,30 +72,25 @@ public void release0() {
7272
value = (int) STATE.getVolatile(this);
7373
if (value <= OPEN) {
7474
//cannot get here - we can't close segment twice
75-
throw new IllegalStateException("Already closed");
75+
throw alreadyClosed();
7676
}
7777
} while (!STATE.compareAndSet(this, value, value - 1));
7878
}
7979

8080
void justClose() {
8181
int prevState = (int) STATE.compareAndExchange(this, OPEN, CLOSING);
8282
if (prevState < 0) {
83-
throw new IllegalStateException("Already closed");
83+
throw alreadyClosed();
8484
} else if (prevState != OPEN) {
85-
throw new IllegalStateException("Session is acquired by " + prevState + " clients");
85+
throw alreadyAcquired(prevState);
8686
}
8787
boolean success = SCOPED_MEMORY_ACCESS.closeScope(this);
8888
STATE.setVolatile(this, success ? CLOSED : OPEN);
8989
if (!success) {
90-
throw new IllegalStateException("Session is acquired by 1 client");
90+
throw alreadyAcquired(1);
9191
}
9292
}
9393

94-
@Override
95-
public boolean isAlive() {
96-
return (int) STATE.getVolatile(this) != CLOSED;
97-
}
98-
9994
/**
10095
* A shared resource list; this implementation has to handle add vs. add races, as well as add vs. cleanup races.
10196
*/
@@ -117,7 +112,7 @@ void add(ResourceCleanup cleanup) {
117112
ResourceCleanup prev = (ResourceCleanup) FST.getVolatile(this);
118113
if (prev == ResourceCleanup.CLOSED_LIST) {
119114
// too late
120-
throw new IllegalStateException("Already closed");
115+
throw alreadyClosed();
121116
}
122117
cleanup.next = prev;
123118
if (FST.compareAndSet(this, prev, cleanup)) {
@@ -144,7 +139,7 @@ void cleanup() {
144139
}
145140
cleanup(prev);
146141
} else {
147-
throw new IllegalStateException("Attempt to cleanup an already closed resource list");
142+
throw alreadyClosed();
148143
}
149144
}
150145
}

‎src/java.base/share/classes/jdk/internal/foreign/abi/SharedUtils.java

-5
Original file line numberDiff line numberDiff line change
@@ -445,11 +445,6 @@ public MemorySession session() {
445445
return MemorySessionImpl.GLOBAL;
446446
}
447447

448-
@Override
449-
public MemorySessionImpl sessionImpl() {
450-
return MemorySessionImpl.GLOBAL;
451-
}
452-
453448
@Override
454449
public VaList copy() {
455450
return this;

‎src/java.base/share/classes/jdk/internal/foreign/abi/aarch64/linux/LinuxAArch64VaList.java

+1-6
Original file line numberDiff line numberDiff line change
@@ -320,7 +320,7 @@ private Object read(Class<?> carrier, MemoryLayout layout, SegmentAllocator allo
320320
@Override
321321
public void skip(MemoryLayout... layouts) {
322322
Objects.requireNonNull(layouts);
323-
MemorySessionImpl.toSessionImpl(session()).checkValidStateSlow();
323+
sessionImpl().checkValidState();
324324
for (MemoryLayout layout : layouts) {
325325
Objects.requireNonNull(layout);
326326
TypeClass typeClass = TypeClass.classifyLayout(layout);
@@ -350,11 +350,6 @@ public MemorySession session() {
350350
return segment.session();
351351
}
352352

353-
@Override
354-
public MemorySessionImpl sessionImpl() {
355-
return MemorySessionImpl.toSessionImpl(session());
356-
}
357-
358353
@Override
359354
public VaList copy() {
360355
MemorySegment copy = MemorySegment.allocateNative(LAYOUT, segment.session());

‎src/java.base/share/classes/jdk/internal/foreign/abi/aarch64/macos/MacOsAArch64VaList.java

+3-8
Original file line numberDiff line numberDiff line change
@@ -127,7 +127,7 @@ private Object read(Class<?> carrier, MemoryLayout layout, SegmentAllocator allo
127127
@Override
128128
public void skip(MemoryLayout... layouts) {
129129
Objects.requireNonNull(layouts);
130-
MemorySessionImpl.toSessionImpl(session()).checkValidStateSlow();
130+
sessionImpl().checkValidState();
131131

132132
for (MemoryLayout layout : layouts) {
133133
Objects.requireNonNull(layout);
@@ -152,14 +152,9 @@ public MemorySession session() {
152152
return session;
153153
}
154154

155-
@Override
156-
public MemorySessionImpl sessionImpl() {
157-
return MemorySessionImpl.toSessionImpl(session());
158-
}
159-
160155
@Override
161156
public VaList copy() {
162-
MemorySessionImpl.toSessionImpl(session()).checkValidStateSlow();
157+
sessionImpl().checkValidState();
163158
return new MacOsAArch64VaList(segment, session);
164159
}
165160

@@ -174,7 +169,7 @@ public static non-sealed class Builder implements VaList.Builder {
174169
private final List<SimpleVaArg> args = new ArrayList<>();
175170

176171
public Builder(MemorySession session) {
177-
MemorySessionImpl.toSessionImpl(session).checkValidStateSlow();
172+
MemorySessionImpl.toSessionImpl(session).checkValidState();
178173
this.session = session;
179174
}
180175

‎src/java.base/share/classes/jdk/internal/foreign/abi/x64/sysv/SysVVaList.java

+1-6
Original file line numberDiff line numberDiff line change
@@ -277,7 +277,7 @@ private Object read(Class<?> carrier, MemoryLayout layout, SegmentAllocator allo
277277
@Override
278278
public void skip(MemoryLayout... layouts) {
279279
Objects.requireNonNull(layouts);
280-
MemorySessionImpl.toSessionImpl(session()).checkValidStateSlow();
280+
sessionImpl().checkValidState();
281281
for (MemoryLayout layout : layouts) {
282282
Objects.requireNonNull(layout);
283283
TypeClass typeClass = TypeClass.classifyLayout(layout);
@@ -304,11 +304,6 @@ public MemorySession session() {
304304
return segment.session();
305305
}
306306

307-
@Override
308-
public MemorySessionImpl sessionImpl() {
309-
return MemorySessionImpl.toSessionImpl(session());
310-
}
311-
312307
@Override
313308
public VaList copy() {
314309
MemorySegment copy = MemorySegment.allocateNative(LAYOUT, segment.session());

‎src/java.base/share/classes/jdk/internal/foreign/abi/x64/windows/WinVaList.java

+3-8
Original file line numberDiff line numberDiff line change
@@ -133,7 +133,7 @@ private Object read(Class<?> carrier, MemoryLayout layout, SegmentAllocator allo
133133
@Override
134134
public void skip(MemoryLayout... layouts) {
135135
Objects.requireNonNull(layouts);
136-
MemorySessionImpl.toSessionImpl(session()).checkValidStateSlow();
136+
sessionImpl().checkValidState();
137137
Stream.of(layouts).forEach(Objects::requireNonNull);
138138
segment = segment.asSlice(layouts.length * VA_SLOT_SIZE_BYTES);
139139
}
@@ -152,14 +152,9 @@ public MemorySession session() {
152152
return session;
153153
}
154154

155-
@Override
156-
public MemorySessionImpl sessionImpl() {
157-
return MemorySessionImpl.toSessionImpl(session());
158-
}
159-
160155
@Override
161156
public VaList copy() {
162-
MemorySessionImpl.toSessionImpl(session).checkValidStateSlow();
157+
sessionImpl().checkValidState();
163158
return new WinVaList(segment, session);
164159
}
165160

@@ -174,7 +169,7 @@ public static non-sealed class Builder implements VaList.Builder {
174169
private final List<SimpleVaArg> args = new ArrayList<>();
175170

176171
public Builder(MemorySession session) {
177-
MemorySessionImpl.toSessionImpl(session).checkValidStateSlow();
172+
MemorySessionImpl.toSessionImpl(session).checkValidState();
178173
this.session = session;
179174
}
180175

‎src/java.base/share/classes/jdk/internal/misc/X-ScopedMemoryAccess-bin.java.template

+66-66
Large diffs are not rendered by default.

‎src/java.base/share/classes/jdk/internal/misc/X-ScopedMemoryAccess.java.template

+63-54
Original file line numberDiff line numberDiff line change
@@ -32,6 +32,7 @@ import java.lang.annotation.Target;
3232
import java.lang.foreign.MemorySegment;
3333
import java.lang.ref.Reference;
3434
import java.io.FileDescriptor;
35+
import java.util.function.Supplier;
3536

3637
import jdk.internal.access.JavaNioAccess;
3738
import jdk.internal.access.SharedSecrets;
@@ -55,10 +56,10 @@ import jdk.internal.vm.vector.VectorSupport;
5556
* a memory region while another thread is releasing it.
5657
* <p>
5758
* 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,
6061
* 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.
6263
* <p>
6364
* This synchronization strategy relies on the idea that accessing memory is atomic with respect to checking the
6465
* 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 {
9798
}
9899

99100
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;
102108
}
109+
103110
static final long serialVersionUID = 1L;
104111

105-
public static final ScopedAccessError INSTANCE = new ScopedAccessError();
112+
public final RuntimeException newRuntimeException() {
113+
return runtimeExceptionSupplier.get();
114+
}
106115
}
107116

108117
@Target({ElementType.METHOD, ElementType.CONSTRUCTOR})
@@ -112,64 +121,64 @@ public class ScopedMemoryAccess {
112121
// bulk ops
113122

114123
@ForceInline
115-
public void copyMemory(MemorySessionImpl srcScope, MemorySessionImpl dstScope,
124+
public void copyMemory(MemorySessionImpl srcSession, MemorySessionImpl dstSession,
116125
Object srcBase, long srcOffset,
117126
Object destBase, long destOffset,
118127
long bytes) {
119128
try {
120-
copyMemoryInternal(srcScope, dstScope, srcBase, srcOffset, destBase, destOffset, bytes);
129+
copyMemoryInternal(srcSession, dstSession, srcBase, srcOffset, destBase, destOffset, bytes);
121130
} catch (ScopedAccessError ex) {
122-
throw new IllegalStateException("This segment is already closed");
131+
throw ex.newRuntimeException();
123132
}
124133
}
125134

126135
@ForceInline @Scoped
127-
private void copyMemoryInternal(MemorySessionImpl srcScope, MemorySessionImpl dstScope,
136+
private void copyMemoryInternal(MemorySessionImpl srcSession, MemorySessionImpl dstSession,
128137
Object srcBase, long srcOffset,
129138
Object destBase, long destOffset,
130139
long bytes) {
131140
try {
132-
if (srcScope != null) {
133-
srcScope.checkValidState();
141+
if (srcSession != null) {
142+
srcSession.checkValidStateRaw();
134143
}
135-
if (dstScope != null) {
136-
dstScope.checkValidState();
144+
if (dstSession != null) {
145+
dstSession.checkValidStateRaw();
137146
}
138147
UNSAFE.copyMemory(srcBase, srcOffset, destBase, destOffset, bytes);
139148
} finally {
140-
Reference.reachabilityFence(srcScope);
141-
Reference.reachabilityFence(dstScope);
149+
Reference.reachabilityFence(srcSession);
150+
Reference.reachabilityFence(dstSession);
142151
}
143152
}
144153

145154
@ForceInline
146-
public void copySwapMemory(MemorySessionImpl srcScope, MemorySessionImpl dstScope,
155+
public void copySwapMemory(MemorySessionImpl srcSession, MemorySessionImpl dstSession,
147156
Object srcBase, long srcOffset,
148157
Object destBase, long destOffset,
149158
long bytes, long elemSize) {
150159
try {
151-
copySwapMemoryInternal(srcScope, dstScope, srcBase, srcOffset, destBase, destOffset, bytes, elemSize);
160+
copySwapMemoryInternal(srcSession, dstSession, srcBase, srcOffset, destBase, destOffset, bytes, elemSize);
152161
} catch (ScopedAccessError ex) {
153-
throw new IllegalStateException("This segment is already closed");
162+
throw ex.newRuntimeException();
154163
}
155164
}
156165

157166
@ForceInline @Scoped
158-
private void copySwapMemoryInternal(MemorySessionImpl srcScope, MemorySessionImpl dstScope,
167+
private void copySwapMemoryInternal(MemorySessionImpl srcSession, MemorySessionImpl dstSession,
159168
Object srcBase, long srcOffset,
160169
Object destBase, long destOffset,
161170
long bytes, long elemSize) {
162171
try {
163-
if (srcScope != null) {
164-
srcScope.checkValidState();
172+
if (srcSession != null) {
173+
srcSession.checkValidStateRaw();
165174
}
166-
if (dstScope != null) {
167-
dstScope.checkValidState();
175+
if (dstSession != null) {
176+
dstSession.checkValidStateRaw();
168177
}
169178
UNSAFE.copySwapMemory(srcBase, srcOffset, destBase, destOffset, bytes, elemSize);
170179
} finally {
171-
Reference.reachabilityFence(srcScope);
172-
Reference.reachabilityFence(dstScope);
180+
Reference.reachabilityFence(srcSession);
181+
Reference.reachabilityFence(dstSession);
173182
}
174183
}
175184

@@ -178,15 +187,15 @@ public class ScopedMemoryAccess {
178187
try {
179188
setMemoryInternal(session, o, offset, bytes, value);
180189
} catch (ScopedAccessError ex) {
181-
throw new IllegalStateException("This segment is already closed");
190+
throw ex.newRuntimeException();
182191
}
183192
}
184193

185194
@ForceInline @Scoped
186195
private void setMemoryInternal(MemorySessionImpl session, Object o, long offset, long bytes, byte value) {
187196
try {
188197
if (session != null) {
189-
session.checkValidState();
198+
session.checkValidStateRaw();
190199
}
191200
UNSAFE.setMemory(o, offset, bytes, value);
192201
} finally {
@@ -195,35 +204,35 @@ public class ScopedMemoryAccess {
195204
}
196205

197206
@ForceInline
198-
public int vectorizedMismatch(MemorySessionImpl aScope, MemorySessionImpl bScope,
207+
public int vectorizedMismatch(MemorySessionImpl aSession, MemorySessionImpl bSession,
199208
Object a, long aOffset,
200209
Object b, long bOffset,
201210
int length,
202211
int log2ArrayIndexScale) {
203212
try {
204-
return vectorizedMismatchInternal(aScope, bScope, a, aOffset, b, bOffset, length, log2ArrayIndexScale);
213+
return vectorizedMismatchInternal(aSession, bSession, a, aOffset, b, bOffset, length, log2ArrayIndexScale);
205214
} catch (ScopedAccessError ex) {
206-
throw new IllegalStateException("This segment is already closed");
215+
throw ex.newRuntimeException();
207216
}
208217
}
209218

210219
@ForceInline @Scoped
211-
private int vectorizedMismatchInternal(MemorySessionImpl aScope, MemorySessionImpl bScope,
220+
private int vectorizedMismatchInternal(MemorySessionImpl aSession, MemorySessionImpl bSession,
212221
Object a, long aOffset,
213222
Object b, long bOffset,
214223
int length,
215224
int log2ArrayIndexScale) {
216225
try {
217-
if (aScope != null) {
218-
aScope.checkValidState();
226+
if (aSession != null) {
227+
aSession.checkValidStateRaw();
219228
}
220-
if (bScope != null) {
221-
bScope.checkValidState();
229+
if (bSession != null) {
230+
bSession.checkValidStateRaw();
222231
}
223232
return ArraysSupport.vectorizedMismatch(a, aOffset, b, bOffset, length, log2ArrayIndexScale);
224233
} finally {
225-
Reference.reachabilityFence(aScope);
226-
Reference.reachabilityFence(bScope);
234+
Reference.reachabilityFence(aSession);
235+
Reference.reachabilityFence(bSession);
227236
}
228237
}
229238

@@ -232,15 +241,15 @@ public class ScopedMemoryAccess {
232241
try {
233242
return isLoadedInternal(session, address, isSync, size);
234243
} catch (ScopedAccessError ex) {
235-
throw new IllegalStateException("This segment is already closed");
244+
throw ex.newRuntimeException();
236245
}
237246
}
238247

239248
@ForceInline @Scoped
240249
public boolean isLoadedInternal(MemorySessionImpl session, long address, boolean isSync, long size) {
241250
try {
242251
if (session != null) {
243-
session.checkValidState();
252+
session.checkValidStateRaw();
244253
}
245254
return SharedSecrets.getJavaNioAccess().isLoaded(address, isSync, size);
246255
} finally {
@@ -253,15 +262,15 @@ public class ScopedMemoryAccess {
253262
try {
254263
loadInternal(session, address, isSync, size);
255264
} catch (ScopedAccessError ex) {
256-
throw new IllegalStateException("This segment is already closed");
265+
throw ex.newRuntimeException();
257266
}
258267
}
259268

260269
@ForceInline @Scoped
261270
public void loadInternal(MemorySessionImpl session, long address, boolean isSync, long size) {
262271
try {
263272
if (session != null) {
264-
session.checkValidState();
273+
session.checkValidStateRaw();
265274
}
266275
SharedSecrets.getJavaNioAccess().load(address, isSync, size);
267276
} finally {
@@ -274,15 +283,15 @@ public class ScopedMemoryAccess {
274283
try {
275284
unloadInternal(session, address, isSync, size);
276285
} catch (ScopedAccessError ex) {
277-
throw new IllegalStateException("This segment is already closed");
286+
throw ex.newRuntimeException();
278287
}
279288
}
280289

281290
@ForceInline @Scoped
282291
public void unloadInternal(MemorySessionImpl session, long address, boolean isSync, long size) {
283292
try {
284293
if (session != null) {
285-
session.checkValidState();
294+
session.checkValidStateRaw();
286295
}
287296
SharedSecrets.getJavaNioAccess().unload(address, isSync, size);
288297
} finally {
@@ -295,15 +304,15 @@ public class ScopedMemoryAccess {
295304
try {
296305
forceInternal(session, fd, address, isSync, index, length);
297306
} catch (ScopedAccessError ex) {
298-
throw new IllegalStateException("This segment is already closed");
307+
throw ex.newRuntimeException();
299308
}
300309
}
301310

302311
@ForceInline @Scoped
303312
public void forceInternal(MemorySessionImpl session, FileDescriptor fd, long address, boolean isSync, long index, long length) {
304313
try {
305314
if (session != null) {
306-
session.checkValidState();
315+
session.checkValidStateRaw();
307316
}
308317
SharedSecrets.getJavaNioAccess().force(fd, address, isSync, index, length);
309318
} finally {
@@ -333,7 +342,7 @@ public class ScopedMemoryAccess {
333342
s,
334343
defaultImpl);
335344
} catch (ScopedAccessError ex) {
336-
throw new IllegalStateException("This segment is already closed");
345+
throw ex.newRuntimeException();
337346
}
338347
}
339348

@@ -347,7 +356,7 @@ public class ScopedMemoryAccess {
347356
S s,
348357
VectorSupport.LoadOperation<AbstractMemorySegmentImpl, V, S> defaultImpl) {
349358
try {
350-
session.checkValidState();
359+
session.checkValidStateRaw();
351360

352361
return VectorSupport.load(vmClass, e, length,
353362
msp.unsafeGetBase(), msp.unsafeGetOffset() + offset,
@@ -378,7 +387,7 @@ public class ScopedMemoryAccess {
378387
s, offsetInRange,
379388
defaultImpl);
380389
} catch (ScopedAccessError ex) {
381-
throw new IllegalStateException("This segment is already closed");
390+
throw ex.newRuntimeException();
382391
}
383392
}
384393

@@ -393,7 +402,7 @@ public class ScopedMemoryAccess {
393402
S s, int offsetInRange,
394403
VectorSupport.LoadVectorMaskedOperation<AbstractMemorySegmentImpl, V, S, M> defaultImpl) {
395404
try {
396-
session.checkValidState();
405+
session.checkValidStateRaw();
397406

398407
return VectorSupport.loadMasked(vmClass, maskClass, e, length,
399408
msp.unsafeGetBase(), msp.unsafeGetOffset() + offset, m, offsetInRange,
@@ -424,7 +433,7 @@ public class ScopedMemoryAccess {
424433
msp, offset,
425434
defaultImpl);
426435
} catch (ScopedAccessError ex) {
427-
throw new IllegalStateException("This segment is already closed");
436+
throw ex.newRuntimeException();
428437
}
429438
}
430439

@@ -438,7 +447,7 @@ public class ScopedMemoryAccess {
438447
AbstractMemorySegmentImpl msp, long offset,
439448
VectorSupport.StoreVectorOperation<AbstractMemorySegmentImpl, V> defaultImpl) {
440449
try {
441-
session.checkValidState();
450+
session.checkValidStateRaw();
442451

443452
VectorSupport.store(vmClass, e, length,
444453
msp.unsafeGetBase(), msp.unsafeGetOffset() + offset,
@@ -470,7 +479,7 @@ public class ScopedMemoryAccess {
470479
msp, offset,
471480
defaultImpl);
472481
} catch (ScopedAccessError ex) {
473-
throw new IllegalStateException("This segment is already closed");
482+
throw ex.newRuntimeException();
474483
}
475484
}
476485

@@ -484,7 +493,7 @@ public class ScopedMemoryAccess {
484493
AbstractMemorySegmentImpl msp, long offset,
485494
VectorSupport.StoreVectorMaskedOperation<AbstractMemorySegmentImpl, V, M> defaultImpl) {
486495
try {
487-
session.checkValidState();
496+
session.checkValidStateRaw();
488497

489498
VectorSupport.storeMasked(vmClass, maskClass, e, length,
490499
msp.unsafeGetBase(), msp.unsafeGetOffset() + offset,

‎src/java.base/share/classes/sun/nio/ch/FileChannelImpl.java

+5-8
Original file line numberDiff line numberDiff line change
@@ -1199,9 +1199,6 @@ public MappedByteBuffer map(MapMode mode, long position, long size) throws IOExc
11991199
}
12001200
}
12011201

1202-
private static final int MAP_MEM_SEG_DEFAULT_MODES = 0;
1203-
private static final int MAP_MEM_SEG_READ_ONLY = 1;
1204-
12051202
@Override
12061203
public MemorySegment map(MapMode mode, long offset, long size,
12071204
MemorySession session)
@@ -1210,7 +1207,7 @@ public MemorySegment map(MapMode mode, long offset, long size,
12101207
Objects.requireNonNull(mode,"Mode is null");
12111208
Objects.requireNonNull(session, "Session is null");
12121209
MemorySessionImpl sessionImpl = MemorySessionImpl.toSessionImpl(session);
1213-
sessionImpl.checkValidStateSlow();
1210+
sessionImpl.checkValidState();
12141211
if (offset < 0)
12151212
throw new IllegalArgumentException("Requested bytes offset must be >= 0.");
12161213
if (size < 0)
@@ -1219,14 +1216,14 @@ public MemorySegment map(MapMode mode, long offset, long size,
12191216
boolean isSync = isSync(mode);
12201217
int prot = toProt(mode);
12211218
Unmapper unmapper = mapInternal(mode, offset, size, prot, isSync);
1222-
int modes = MAP_MEM_SEG_DEFAULT_MODES;
1219+
boolean readOnly = false;
12231220
if (mode == MapMode.READ_ONLY) {
1224-
modes |= MAP_MEM_SEG_READ_ONLY;
1221+
readOnly = true;
12251222
}
12261223
if (unmapper != null) {
12271224
AbstractMemorySegmentImpl segment =
12281225
new MappedMemorySegmentImpl(unmapper.address(), unmapper, size,
1229-
modes, session);
1226+
readOnly, session);
12301227
MemorySessionImpl.ResourceList.ResourceCleanup resource =
12311228
new MemorySessionImpl.ResourceList.ResourceCleanup() {
12321229
@Override
@@ -1237,7 +1234,7 @@ public void cleanup() {
12371234
sessionImpl.addOrCleanupIfFail(resource);
12381235
return segment;
12391236
} else {
1240-
return new MappedMemorySegmentImpl.EmptyMappedMemorySegmentImpl(modes, session);
1237+
return new MappedMemorySegmentImpl.EmptyMappedMemorySegmentImpl(readOnly, sessionImpl);
12411238
}
12421239
}
12431240

‎test/jdk/java/foreign/TestByteBuffer.java

+2-2
Original file line numberDiff line numberDiff line change
@@ -373,7 +373,7 @@ public void testScopedBuffer(Function<ByteBuffer, Buffer> bufferFactory, @NoInje
373373
Throwable cause = ex.getCause();
374374
if (cause instanceof IllegalStateException) {
375375
//all get/set buffer operation should fail because of the session check
376-
assertTrue(ex.getCause().getMessage().contains("already closed"));
376+
assertTrue(ex.getCause().getMessage().contains("Already closed"));
377377
} else {
378378
//all other exceptions were unexpected - fail
379379
fail("Unexpected exception", cause);
@@ -410,7 +410,7 @@ public void testScopedBufferAndVarHandle(VarHandle bufferHandle) {
410410
handle.invoke(e.getValue());
411411
fail();
412412
} catch (IllegalStateException ex) {
413-
assertTrue(ex.getMessage().contains("already closed"));
413+
assertTrue(ex.getMessage().contains("Already closed"));
414414
} catch (UnsupportedOperationException ex) {
415415
//skip
416416
} catch (Throwable ex) {

‎test/jdk/java/foreign/TestMemorySession.java

+3-3
Original file line numberDiff line numberDiff line change
@@ -370,8 +370,8 @@ static Object[][] sessions() {
370370
}
371371

372372
private void keepAlive(MemorySession child, MemorySession parent) {
373-
MemorySessionImpl sessionImpl = MemorySessionImpl.toSessionImpl(parent);
374-
sessionImpl.acquire0();
375-
child.addCloseAction(sessionImpl::release0);
373+
MemorySessionImpl parentImpl = MemorySessionImpl.toSessionImpl(parent);
374+
parentImpl.acquire0();
375+
child.addCloseAction(parentImpl::release0);
376376
}
377377
}

0 commit comments

Comments
 (0)
Please sign in to comment.