Skip to content

Commit 39f140a

Browse files
committedJan 18, 2022
Merge
2 parents 37143c0 + 4d9b3f4 commit 39f140a

File tree

9 files changed

+399
-42
lines changed

9 files changed

+399
-42
lines changed
 

‎make/autoconf/toolchain.m4

+2-2
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
#
2-
# Copyright (c) 2011, 2021, Oracle and/or its affiliates. All rights reserved.
2+
# Copyright (c) 2011, 2022, Oracle and/or its affiliates. All rights reserved.
33
# DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
44
#
55
# This code is free software; you can redistribute it and/or modify it
@@ -234,7 +234,7 @@ AC_DEFUN_ONCE([TOOLCHAIN_DETERMINE_TOOLCHAIN_TYPE],
234234
if test "x$OPENJDK_TARGET_OS" = xmacosx; then
235235
if test -n "$XCODEBUILD"; then
236236
# On Mac OS X, default toolchain to clang after Xcode 5
237-
XCODE_VERSION_OUTPUT=`"$XCODEBUILD" -version 2>&1 | $HEAD -n 1`
237+
XCODE_VERSION_OUTPUT=`"$XCODEBUILD" -version | $HEAD -n 1`
238238
$ECHO "$XCODE_VERSION_OUTPUT" | $GREP "Xcode " > /dev/null
239239
if test $? -ne 0; then
240240
AC_MSG_NOTICE([xcodebuild output: $XCODE_VERSION_OUTPUT])

‎src/hotspot/cpu/ppc/c1_CodeStubs_ppc.cpp

-2
Original file line numberDiff line numberDiff line change
@@ -81,8 +81,6 @@ void RangeCheckStub::emit_code(LIR_Assembler* ce) {
8181

8282
if (_info->deoptimize_on_exception()) {
8383
address a = Runtime1::entry_for(Runtime1::predicate_failed_trap_id);
84-
// May be used by optimizations like LoopInvariantCodeMotion or RangeCheckEliminator.
85-
DEBUG_ONLY( __ untested("RangeCheckStub: predicate_failed_trap_id"); )
8684
//__ load_const_optimized(R0, a);
8785
__ add_const_optimized(R0, R29_TOC, MacroAssembler::offset_to_global_toc(a));
8886
__ mtctr(R0);

‎src/hotspot/cpu/ppc/frame_ppc.cpp

+51-4
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,6 @@
11
/*
2-
* Copyright (c) 2000, 2021, Oracle and/or its affiliates. All rights reserved.
3-
* Copyright (c) 2012, 2021 SAP SE. All rights reserved.
2+
* Copyright (c) 2000, 2022, Oracle and/or its affiliates. All rights reserved.
3+
* Copyright (c) 2012, 2022 SAP SE. All rights reserved.
44
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
55
*
66
* This code is free software; you can redistribute it and/or modify it
@@ -294,9 +294,56 @@ void frame::patch_pc(Thread* thread, address pc) {
294294
}
295295

296296
bool frame::is_interpreted_frame_valid(JavaThread* thread) const {
297-
// Is there anything to do?
298297
assert(is_interpreted_frame(), "Not an interpreted frame");
299-
return true;
298+
// These are reasonable sanity checks
299+
if (fp() == 0 || (intptr_t(fp()) & (wordSize-1)) != 0) {
300+
return false;
301+
}
302+
if (sp() == 0 || (intptr_t(sp()) & (wordSize-1)) != 0) {
303+
return false;
304+
}
305+
if (fp() - (abi_minframe_size + ijava_state_size) < sp()) {
306+
return false;
307+
}
308+
// These are hacks to keep us out of trouble.
309+
// The problem with these is that they mask other problems
310+
if (fp() <= sp()) { // this attempts to deal with unsigned comparison above
311+
return false;
312+
}
313+
314+
// do some validation of frame elements
315+
316+
// first the method
317+
318+
Method* m = *interpreter_frame_method_addr();
319+
320+
// validate the method we'd find in this potential sender
321+
if (!Method::is_valid_method(m)) return false;
322+
323+
// stack frames shouldn't be much larger than max_stack elements
324+
// this test requires the use of unextended_sp which is the sp as seen by
325+
// the current frame, and not sp which is the "raw" pc which could point
326+
// further because of local variables of the callee method inserted after
327+
// method arguments
328+
if (fp() - unextended_sp() > 1024 + m->max_stack()*Interpreter::stackElementSize) {
329+
return false;
330+
}
331+
332+
// validate bci/bcx
333+
334+
address bcp = interpreter_frame_bcp();
335+
if (m->validate_bci_from_bcp(bcp) < 0) {
336+
return false;
337+
}
338+
339+
// validate constantPoolCache*
340+
ConstantPoolCache* cp = *interpreter_frame_cache_addr();
341+
if (MetaspaceObj::is_valid(cp) == false) return false;
342+
343+
// validate locals
344+
345+
address locals = (address) *interpreter_frame_locals_addr();
346+
return thread->is_in_stack_range_incl(locals, (address)fp());
300347
}
301348

302349
BasicType frame::interpreter_frame_result(oop* oop_result, jvalue* value_result) {

‎src/hotspot/cpu/s390/frame_s390.cpp

+51-4
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,6 @@
11
/*
2-
* Copyright (c) 2016, 2021, Oracle and/or its affiliates. All rights reserved.
3-
* Copyright (c) 2016, 2019 SAP SE. All rights reserved.
2+
* Copyright (c) 2016, 2022, Oracle and/or its affiliates. All rights reserved.
3+
* Copyright (c) 2016, 2022 SAP SE. All rights reserved.
44
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
55
*
66
* This code is free software; you can redistribute it and/or modify it
@@ -298,9 +298,56 @@ void frame::patch_pc(Thread* thread, address pc) {
298298
}
299299

300300
bool frame::is_interpreted_frame_valid(JavaThread* thread) const {
301-
// Is there anything to do?
302301
assert(is_interpreted_frame(), "Not an interpreted frame");
303-
return true;
302+
// These are reasonable sanity checks
303+
if (fp() == 0 || (intptr_t(fp()) & (wordSize-1)) != 0) {
304+
return false;
305+
}
306+
if (sp() == 0 || (intptr_t(sp()) & (wordSize-1)) != 0) {
307+
return false;
308+
}
309+
if (fp() - (z_abi_16_size + z_ijava_state_size) < sp()) {
310+
return false;
311+
}
312+
// These are hacks to keep us out of trouble.
313+
// The problem with these is that they mask other problems
314+
if (fp() <= sp()) { // this attempts to deal with unsigned comparison above
315+
return false;
316+
}
317+
318+
// do some validation of frame elements
319+
320+
// first the method
321+
322+
Method* m = *interpreter_frame_method_addr();
323+
324+
// validate the method we'd find in this potential sender
325+
if (!Method::is_valid_method(m)) return false;
326+
327+
// stack frames shouldn't be much larger than max_stack elements
328+
// this test requires the use of unextended_sp which is the sp as seen by
329+
// the current frame, and not sp which is the "raw" pc which could point
330+
// further because of local variables of the callee method inserted after
331+
// method arguments
332+
if (fp() - unextended_sp() > 1024 + m->max_stack()*Interpreter::stackElementSize) {
333+
return false;
334+
}
335+
336+
// validate bci/bcx
337+
338+
address bcp = interpreter_frame_bcp();
339+
if (m->validate_bci_from_bcp(bcp) < 0) {
340+
return false;
341+
}
342+
343+
// validate constantPoolCache*
344+
ConstantPoolCache* cp = *interpreter_frame_cache_addr();
345+
if (MetaspaceObj::is_valid(cp) == false) return false;
346+
347+
// validate locals
348+
349+
address locals = (address) *interpreter_frame_locals_addr();
350+
return thread->is_in_stack_range_incl(locals, (address)fp());
304351
}
305352

306353
BasicType frame::interpreter_frame_result(oop* oop_result, jvalue* value_result) {

‎src/jdk.incubator.foreign/share/classes/jdk/incubator/foreign/MemorySegment.java

+1-1
Original file line numberDiff line numberDiff line change
@@ -266,7 +266,7 @@ public sealed interface MemorySegment extends Addressable permits AbstractMemory
266266
* <p>
267267
* The returned spliterator splits this segment according to the specified element layout; that is,
268268
* if the supplied layout has size N, then calling {@link Spliterator#trySplit()} will result in a spliterator serving
269-
* approximately {@code S/N/2} elements (depending on whether N is even or not), where {@code S} is the size of
269+
* approximately {@code S/N} elements (depending on whether N is even or not), where {@code S} is the size of
270270
* this segment. As such, splitting is possible as long as {@code S/N >= 2}. The spliterator returns segments that
271271
* are associated with the same scope as this segment.
272272
* <p>

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

+32-28
Original file line numberDiff line numberDiff line change
@@ -36,13 +36,17 @@
3636
import java.util.Objects;
3737

3838
/**
39-
* Implementation for heap memory segments. An heap memory segment is composed by an offset and
39+
* Implementation for heap memory segments. A heap memory segment is composed by an offset and
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.
43+
* {@link HeapMemorySegmentImpl#base()} method so that it returns an array of the correct (sharp) type. Note that
44+
* the field type storing the 'base' coordinate is just Object; similarly, all the constructor in the subclasses
45+
* accept an Object 'base' parameter instead of a sharper type (e.g. {@code byte[]}). This is deliberate, as
46+
* using sharper types would require use of type-conversions, which in turn would inhibit some C2 optimizations,
47+
* such as the elimination of store barriers in methods like {@link HeapMemorySegmentImpl#dup(long, long, int, ResourceScopeImpl)}.
4448
*/
45-
public abstract class HeapMemorySegmentImpl<H> extends AbstractMemorySegmentImpl {
49+
public abstract class HeapMemorySegmentImpl extends AbstractMemorySegmentImpl {
4650

4751
private static final Unsafe UNSAFE = Unsafe.getUnsafe();
4852
private static final int BYTE_ARR_BASE = UNSAFE.arrayBaseOffset(byte[].class);
@@ -53,25 +57,25 @@ public abstract class HeapMemorySegmentImpl<H> extends AbstractMemorySegmentImpl
5357
private static final long MAX_ALIGN_8 = 8;
5458

5559
final long offset;
56-
final H base;
60+
final Object base;
5761

5862
@ForceInline
59-
HeapMemorySegmentImpl(long offset, H base, long length, int mask) {
63+
HeapMemorySegmentImpl(long offset, Object base, long length, int mask) {
6064
super(length, mask, ResourceScopeImpl.GLOBAL);
6165
this.offset = offset;
6266
this.base = base;
6367
}
6468

6569
@Override
66-
abstract H base();
70+
abstract Object base();
6771

6872
@Override
6973
long min() {
7074
return offset;
7175
}
7276

7377
@Override
74-
abstract HeapMemorySegmentImpl<H> dup(long offset, long size, int mask, ResourceScopeImpl scope);
78+
abstract HeapMemorySegmentImpl dup(long offset, long size, int mask, ResourceScopeImpl scope);
7579

7680
@Override
7781
ByteBuffer makeByteBuffer() {
@@ -84,9 +88,9 @@ ByteBuffer makeByteBuffer() {
8488

8589
// factories
8690

87-
public static class OfByte extends HeapMemorySegmentImpl<byte[]> {
91+
public static class OfByte extends HeapMemorySegmentImpl {
8892

89-
OfByte(long offset, byte[] base, long length, int mask) {
93+
OfByte(long offset, Object base, long length, int mask) {
9094
super(offset, base, length, mask);
9195
}
9296

@@ -97,7 +101,7 @@ OfByte dup(long offset, long size, int mask, ResourceScopeImpl scope) {
97101

98102
@Override
99103
byte[] base() {
100-
return Objects.requireNonNull(base);
104+
return (byte[])Objects.requireNonNull(base);
101105
}
102106

103107
public static MemorySegment fromArray(byte[] arr) {
@@ -112,9 +116,9 @@ public long maxAlignMask() {
112116
}
113117
}
114118

115-
public static class OfChar extends HeapMemorySegmentImpl<char[]> {
119+
public static class OfChar extends HeapMemorySegmentImpl {
116120

117-
OfChar(long offset, char[] base, long length, int mask) {
121+
OfChar(long offset, Object base, long length, int mask) {
118122
super(offset, base, length, mask);
119123
}
120124

@@ -125,7 +129,7 @@ OfChar dup(long offset, long size, int mask, ResourceScopeImpl scope) {
125129

126130
@Override
127131
char[] base() {
128-
return Objects.requireNonNull(base);
132+
return (char[])Objects.requireNonNull(base);
129133
}
130134

131135
public static MemorySegment fromArray(char[] arr) {
@@ -140,9 +144,9 @@ public long maxAlignMask() {
140144
}
141145
}
142146

143-
public static class OfShort extends HeapMemorySegmentImpl<short[]> {
147+
public static class OfShort extends HeapMemorySegmentImpl {
144148

145-
OfShort(long offset, short[] base, long length, int mask) {
149+
OfShort(long offset, Object base, long length, int mask) {
146150
super(offset, base, length, mask);
147151
}
148152

@@ -153,7 +157,7 @@ OfShort dup(long offset, long size, int mask, ResourceScopeImpl scope) {
153157

154158
@Override
155159
short[] base() {
156-
return Objects.requireNonNull(base);
160+
return (short[])Objects.requireNonNull(base);
157161
}
158162

159163
public static MemorySegment fromArray(short[] arr) {
@@ -168,9 +172,9 @@ public long maxAlignMask() {
168172
}
169173
}
170174

171-
public static class OfInt extends HeapMemorySegmentImpl<int[]> {
175+
public static class OfInt extends HeapMemorySegmentImpl {
172176

173-
OfInt(long offset, int[] base, long length, int mask) {
177+
OfInt(long offset, Object base, long length, int mask) {
174178
super(offset, base, length, mask);
175179
}
176180

@@ -181,7 +185,7 @@ OfInt dup(long offset, long size, int mask, ResourceScopeImpl scope) {
181185

182186
@Override
183187
int[] base() {
184-
return Objects.requireNonNull(base);
188+
return (int[])Objects.requireNonNull(base);
185189
}
186190

187191
public static MemorySegment fromArray(int[] arr) {
@@ -196,9 +200,9 @@ public long maxAlignMask() {
196200
}
197201
}
198202

199-
public static class OfLong extends HeapMemorySegmentImpl<long[]> {
203+
public static class OfLong extends HeapMemorySegmentImpl {
200204

201-
OfLong(long offset, long[] base, long length, int mask) {
205+
OfLong(long offset, Object base, long length, int mask) {
202206
super(offset, base, length, mask);
203207
}
204208

@@ -209,7 +213,7 @@ OfLong dup(long offset, long size, int mask, ResourceScopeImpl scope) {
209213

210214
@Override
211215
long[] base() {
212-
return Objects.requireNonNull(base);
216+
return (long[])Objects.requireNonNull(base);
213217
}
214218

215219
public static MemorySegment fromArray(long[] arr) {
@@ -224,9 +228,9 @@ public long maxAlignMask() {
224228
}
225229
}
226230

227-
public static class OfFloat extends HeapMemorySegmentImpl<float[]> {
231+
public static class OfFloat extends HeapMemorySegmentImpl {
228232

229-
OfFloat(long offset, float[] base, long length, int mask) {
233+
OfFloat(long offset, Object base, long length, int mask) {
230234
super(offset, base, length, mask);
231235
}
232236

@@ -237,7 +241,7 @@ OfFloat dup(long offset, long size, int mask, ResourceScopeImpl scope) {
237241

238242
@Override
239243
float[] base() {
240-
return Objects.requireNonNull(base);
244+
return (float[])Objects.requireNonNull(base);
241245
}
242246

243247
public static MemorySegment fromArray(float[] arr) {
@@ -252,9 +256,9 @@ public long maxAlignMask() {
252256
}
253257
}
254258

255-
public static class OfDouble extends HeapMemorySegmentImpl<double[]> {
259+
public static class OfDouble extends HeapMemorySegmentImpl {
256260

257-
OfDouble(long offset, double[] base, long length, int mask) {
261+
OfDouble(long offset, Object base, long length, int mask) {
258262
super(offset, base, length, mask);
259263
}
260264

@@ -265,7 +269,7 @@ OfDouble dup(long offset, long size, int mask, ResourceScopeImpl scope) {
265269

266270
@Override
267271
double[] base() {
268-
return Objects.requireNonNull(base);
272+
return (double[])Objects.requireNonNull(base);
269273
}
270274

271275
public static MemorySegment fromArray(double[] arr) {

‎test/hotspot/jtreg/compiler/codecache/stress/ReturnBlobToWrongHeapTest.java

+2-1
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 2017, 2021, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 2017, 2022, Oracle and/or its affiliates. All rights reserved.
33
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
44
*
55
* This code is free software; you can redistribute it and/or modify it
@@ -37,6 +37,7 @@
3737
* -XX:+SegmentedCodeCache
3838
* -XX:ReservedCodeCacheSize=16M
3939
* -XX:CodeCacheMinBlockLength=1
40+
* -XX:CICompilerCount=2
4041
* compiler.codecache.stress.ReturnBlobToWrongHeapTest
4142
*/
4243

‎test/jdk/ProblemList.txt

+1
Original file line numberDiff line numberDiff line change
@@ -826,6 +826,7 @@ jdk/jfr/event/os/TestThreadContextSwitches.java 8247776 windows-
826826
jdk/jfr/startupargs/TestStartName.java 8214685 windows-x64
827827
jdk/jfr/startupargs/TestStartDuration.java 8214685 windows-x64
828828
jdk/jfr/event/oldobject/TestLargeRootSet.java 8276333 macosx-x64,windows-x64
829+
jdk/jfr/api/consumer/recordingstream/TestOnEvent.java 8255404 linux-x64
829830

830831
############################################################################
831832

Original file line numberDiff line numberDiff line change
@@ -0,0 +1,259 @@
1+
/*
2+
* Copyright (c) 2022, 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+
package org.openjdk.bench.jdk.incubator.foreign;
24+
25+
import jdk.incubator.foreign.MemorySegment;
26+
import jdk.incubator.foreign.ResourceScope;
27+
import org.openjdk.jmh.annotations.Benchmark;
28+
import org.openjdk.jmh.annotations.BenchmarkMode;
29+
import org.openjdk.jmh.annotations.Fork;
30+
import org.openjdk.jmh.annotations.Measurement;
31+
import org.openjdk.jmh.annotations.Mode;
32+
import org.openjdk.jmh.annotations.OutputTimeUnit;
33+
import org.openjdk.jmh.annotations.Setup;
34+
import org.openjdk.jmh.annotations.State;
35+
import org.openjdk.jmh.annotations.TearDown;
36+
import org.openjdk.jmh.annotations.Warmup;
37+
38+
import java.nio.ByteBuffer;
39+
import java.nio.ByteOrder;
40+
import java.nio.IntBuffer;
41+
import java.util.Iterator;
42+
import java.util.concurrent.TimeUnit;
43+
44+
import static jdk.incubator.foreign.MemoryLayout.PathElement.sequenceElement;
45+
import static jdk.incubator.foreign.ValueLayout.JAVA_INT;
46+
47+
@BenchmarkMode(Mode.AverageTime)
48+
@Warmup(iterations = 5, time = 500, timeUnit = TimeUnit.MILLISECONDS)
49+
@Measurement(iterations = 10, time = 500, timeUnit = TimeUnit.MILLISECONDS)
50+
@State(org.openjdk.jmh.annotations.Scope.Thread)
51+
@OutputTimeUnit(TimeUnit.MILLISECONDS)
52+
@Fork(value = 3, jvmArgsAppend = { "--add-modules=jdk.incubator.foreign", "--enable-native-access=ALL-UNNAMED" })
53+
54+
public class LoopOverSlice {
55+
56+
static final int ELEM_SIZE = 1_000_000;
57+
static final int CARRIER_SIZE = (int)JAVA_INT.byteSize();
58+
static final int ALLOC_SIZE = ELEM_SIZE * CARRIER_SIZE;
59+
60+
MemorySegment nativeSegment, heapSegment;
61+
IntBuffer nativeBuffer, heapBuffer;
62+
ResourceScope scope;
63+
64+
@Setup
65+
public void setup() {
66+
scope = ResourceScope.newConfinedScope();
67+
nativeSegment = MemorySegment.allocateNative(ALLOC_SIZE, scope);
68+
heapSegment = MemorySegment.ofArray(new int[ELEM_SIZE]);
69+
nativeBuffer = ByteBuffer.allocateDirect(ALLOC_SIZE).order(ByteOrder.LITTLE_ENDIAN).asIntBuffer();
70+
heapBuffer = IntBuffer.wrap(new int[ELEM_SIZE]);
71+
}
72+
73+
@TearDown
74+
public void tearDown() {
75+
scope.close();
76+
}
77+
78+
@Benchmark
79+
public void native_segment_slice_loop() {
80+
new NativeSegmentWrapper(nativeSegment).forEach(NativeSegmentWrapper.Element::get);
81+
}
82+
83+
@Benchmark
84+
public void native_buffer_slice_loop() {
85+
new NativeBufferWrapper(nativeBuffer).forEach(NativeBufferWrapper.Element::get);
86+
}
87+
88+
@Benchmark
89+
public void heap_segment_slice_loop() {
90+
new HeapSegmentWrapper(heapSegment).forEach(HeapSegmentWrapper.Element::get);
91+
}
92+
93+
@Benchmark
94+
public void heap_buffer_slice_loop() {
95+
new HeapBufferWrapper(heapBuffer).forEach(HeapBufferWrapper.Element::get);
96+
}
97+
98+
class HeapSegmentWrapper implements Iterable<HeapSegmentWrapper.Element> {
99+
final MemorySegment segment;
100+
101+
public HeapSegmentWrapper(MemorySegment segment) {
102+
this.segment = segment;
103+
}
104+
105+
@Override
106+
public Iterator<Element> iterator() {
107+
return new Iterator<Element>() {
108+
109+
MemorySegment current = segment;
110+
111+
@Override
112+
public boolean hasNext() {
113+
return current.byteSize() > 4;
114+
}
115+
116+
@Override
117+
public Element next() {
118+
Element element = new Element(current);
119+
current = current.asSlice(4);
120+
return element;
121+
}
122+
};
123+
}
124+
125+
static class Element {
126+
final MemorySegment segment;
127+
128+
public Element(MemorySegment segment) {
129+
this.segment = segment;
130+
}
131+
132+
int get() {
133+
return segment.getAtIndex(JAVA_INT, 0);
134+
}
135+
}
136+
}
137+
138+
class NativeSegmentWrapper implements Iterable<NativeSegmentWrapper.Element> {
139+
final MemorySegment segment;
140+
141+
public NativeSegmentWrapper(MemorySegment segment) {
142+
this.segment = segment;
143+
}
144+
145+
@Override
146+
public Iterator<Element> iterator() {
147+
return new Iterator<Element>() {
148+
149+
MemorySegment current = segment;
150+
151+
@Override
152+
public boolean hasNext() {
153+
return current.byteSize() > 4;
154+
}
155+
156+
@Override
157+
public Element next() {
158+
Element element = new Element(current);
159+
current = current.asSlice(4);
160+
return element;
161+
}
162+
};
163+
}
164+
165+
static class Element {
166+
final MemorySegment segment;
167+
168+
public Element(MemorySegment segment) {
169+
this.segment = segment;
170+
}
171+
172+
int get() {
173+
return segment.getAtIndex(JAVA_INT, 0);
174+
}
175+
}
176+
}
177+
178+
class NativeBufferWrapper implements Iterable<NativeBufferWrapper.Element> {
179+
final IntBuffer buffer;
180+
181+
public NativeBufferWrapper(IntBuffer buffer) {
182+
this.buffer = buffer;
183+
}
184+
185+
@Override
186+
public Iterator<Element> iterator() {
187+
return new Iterator<Element>() {
188+
189+
IntBuffer current = buffer;
190+
191+
@Override
192+
public boolean hasNext() {
193+
return current.position() < current.limit();
194+
}
195+
196+
@Override
197+
public Element next() {
198+
Element element = new Element(current);
199+
int lim = current.limit();
200+
current = current.slice(1, lim - 1);
201+
return element;
202+
}
203+
};
204+
}
205+
206+
static class Element {
207+
final IntBuffer buffer;
208+
209+
public Element(IntBuffer segment) {
210+
this.buffer = segment;
211+
}
212+
213+
int get() {
214+
return buffer.get( 0);
215+
}
216+
}
217+
}
218+
219+
class HeapBufferWrapper implements Iterable<HeapBufferWrapper.Element> {
220+
final IntBuffer buffer;
221+
222+
public HeapBufferWrapper(IntBuffer buffer) {
223+
this.buffer = buffer;
224+
}
225+
226+
@Override
227+
public Iterator<Element> iterator() {
228+
return new Iterator<Element>() {
229+
230+
IntBuffer current = buffer;
231+
232+
@Override
233+
public boolean hasNext() {
234+
return current.position() < current.limit();
235+
}
236+
237+
@Override
238+
public Element next() {
239+
Element element = new Element(current);
240+
int lim = current.limit();
241+
current = current.slice(1, lim - 1);
242+
return element;
243+
}
244+
};
245+
}
246+
247+
static class Element {
248+
final IntBuffer buffer;
249+
250+
public Element(IntBuffer segment) {
251+
this.buffer = segment;
252+
}
253+
254+
int get() {
255+
return buffer.get( 0);
256+
}
257+
}
258+
}
259+
}

0 commit comments

Comments
 (0)
Please sign in to comment.