Skip to content

Commit 7f69acc

Browse files
committedJun 23, 2020
8247696: Incorrect tail computation for large segments in AbstractMemorySegmentImpl::mismatch
Reviewed-by: psandoz, mcimadamore
1 parent 6469685 commit 7f69acc

File tree

4 files changed

+99
-18
lines changed

4 files changed

+99
-18
lines changed
 

‎src/java.base/share/classes/jdk/internal/util/ArraysSupport.java

+14-9
Original file line numberDiff line numberDiff line change
@@ -163,27 +163,32 @@ public static int vectorizedMismatch(Object a, long aOffset,
163163
/**
164164
* Mismatch over long lengths.
165165
*/
166-
public static long vectorizedMismatchLarge(Object a, long aOffset,
167-
Object b, long bOffset,
168-
long length,
169-
int log2ArrayIndexScale) {
166+
public static long vectorizedMismatchLargeForBytes(Object a, long aOffset,
167+
Object b, long bOffset,
168+
long length) {
170169
long off = 0;
171170
long remaining = length;
172-
int i ;
173-
while (remaining > 7) {
174-
int size = (int) Math.min(Integer.MAX_VALUE, remaining);
171+
int i, size;
172+
boolean lastSubRange = false;
173+
while (remaining > 7 && !lastSubRange) {
174+
if (remaining > Integer.MAX_VALUE) {
175+
size = Integer.MAX_VALUE;
176+
} else {
177+
size = (int) remaining;
178+
lastSubRange = true;
179+
}
175180
i = vectorizedMismatch(
176181
a, aOffset + off,
177182
b, bOffset + off,
178-
size, log2ArrayIndexScale);
183+
size, LOG2_ARRAY_BYTE_INDEX_SCALE);
179184
if (i >= 0)
180185
return off + i;
181186

182187
i = size - ~i;
183188
off += i;
184189
remaining -= i;
185190
}
186-
return ~off;
191+
return ~remaining;
187192
}
188193

189194
// Booleans

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

+8-3
Original file line numberDiff line numberDiff line change
@@ -149,14 +149,19 @@ public long mismatch(MemorySegment other) {
149149

150150
long i = 0;
151151
if (length > 7) {
152-
i = ArraysSupport.vectorizedMismatchLarge(
152+
if ((byte) BYTE_HANDLE.get(this.baseAddress(), 0) != (byte) BYTE_HANDLE.get(that.baseAddress(), 0)) {
153+
return 0;
154+
}
155+
i = ArraysSupport.vectorizedMismatchLargeForBytes(
153156
this.base(), this.min(),
154157
that.base(), that.min(),
155-
length, ArraysSupport.LOG2_ARRAY_BYTE_INDEX_SCALE);
158+
length);
156159
if (i >= 0) {
157160
return i;
158161
}
159-
i = length - ~i;
162+
long remaining = ~i;
163+
assert remaining < 8 : "remaining greater than 7: " + remaining;
164+
i = length - remaining;
160165
}
161166
MemoryAddress thisAddress = this.baseAddress();
162167
MemoryAddress thatAddress = that.baseAddress();

‎test/jdk/java/foreign/TestMismatch.java

+22-6
Original file line numberDiff line numberDiff line change
@@ -117,12 +117,28 @@ public void testLarge() {
117117
assertEquals(s1.mismatch(s2), -1);
118118
assertEquals(s2.mismatch(s1), -1);
119119

120-
for (long i = s2.byteSize() -1 ; i >= Integer.MAX_VALUE - 10L; i--) {
121-
BYTE_HANDLE.set(s2.baseAddress().addOffset(i), (byte) 0xFF);
122-
long expectedMismatchOffset = i;
123-
assertEquals(s1.mismatch(s2), expectedMismatchOffset);
124-
assertEquals(s2.mismatch(s1), expectedMismatchOffset);
125-
}
120+
testLargeAcrossMaxBoundary(s1, s2);
121+
122+
testLargeMismatchAcrossMaxBoundary(s1, s2);
123+
}
124+
}
125+
126+
private void testLargeAcrossMaxBoundary(MemorySegment s1, MemorySegment s2) {
127+
for (long i = s2.byteSize() -1 ; i >= Integer.MAX_VALUE - 10L; i--) {
128+
var s3 = s1.asSlice(0, i);
129+
var s4 = s2.asSlice(0, i);
130+
assertEquals(s3.mismatch(s3), -1);
131+
assertEquals(s3.mismatch(s4), -1);
132+
assertEquals(s4.mismatch(s3), -1);
133+
}
134+
}
135+
136+
private void testLargeMismatchAcrossMaxBoundary(MemorySegment s1, MemorySegment s2) {
137+
for (long i = s2.byteSize() -1 ; i >= Integer.MAX_VALUE - 10L; i--) {
138+
BYTE_HANDLE.set(s2.baseAddress().addOffset(i), (byte) 0xFF);
139+
long expectedMismatchOffset = i;
140+
assertEquals(s1.mismatch(s2), expectedMismatchOffset);
141+
assertEquals(s2.mismatch(s1), expectedMismatchOffset);
126142
}
127143
}
128144

‎test/micro/org/openjdk/bench/jdk/incubator/foreign/BulkOps.java

+55
Original file line numberDiff line numberDiff line change
@@ -35,6 +35,7 @@
3535
import sun.misc.Unsafe;
3636

3737
import jdk.incubator.foreign.MemorySegment;
38+
import java.nio.ByteBuffer;
3839
import java.util.concurrent.TimeUnit;
3940

4041
import static jdk.incubator.foreign.MemoryLayouts.JAVA_INT;
@@ -60,6 +61,36 @@ public class BulkOps {
6061
static final MemorySegment bytesSegment = MemorySegment.ofArray(bytes);
6162
static final int UNSAFE_INT_OFFSET = unsafe.arrayBaseOffset(int[].class);
6263

64+
// large(ish) segments/buffers with same content, 0, for mismatch, non-multiple-of-8 sized
65+
static final int SIZE_WITH_TAIL = (1024 * 1024) + 7;
66+
static final MemorySegment mismatchSegmentLarge1 = MemorySegment.allocateNative(SIZE_WITH_TAIL);
67+
static final MemorySegment mismatchSegmentLarge2 = MemorySegment.allocateNative(SIZE_WITH_TAIL);
68+
static final ByteBuffer mismatchBufferLarge1 = ByteBuffer.allocateDirect(SIZE_WITH_TAIL);
69+
static final ByteBuffer mismatchBufferLarge2 = ByteBuffer.allocateDirect(SIZE_WITH_TAIL);
70+
71+
// mismatch at first byte
72+
static final MemorySegment mismatchSegmentSmall1 = MemorySegment.allocateNative(7);
73+
static final MemorySegment mismatchSegmentSmall2 = MemorySegment.allocateNative(7);
74+
static final ByteBuffer mismatchBufferSmall1 = ByteBuffer.allocateDirect(7);
75+
static final ByteBuffer mismatchBufferSmall2 = ByteBuffer.allocateDirect(7);
76+
static {
77+
mismatchSegmentSmall1.fill((byte) 0xFF);
78+
mismatchBufferSmall1.put((byte) 0xFF).clear();
79+
// verify expected mismatch indices
80+
long si = mismatchSegmentLarge1.mismatch(mismatchSegmentLarge2);
81+
if (si != -1)
82+
throw new AssertionError("Unexpected mismatch index:" + si);
83+
int bi = mismatchBufferLarge1.mismatch(mismatchBufferLarge2);
84+
if (bi != -1)
85+
throw new AssertionError("Unexpected mismatch index:" + bi);
86+
si = mismatchSegmentSmall1.mismatch(mismatchSegmentSmall2);
87+
if (si != 0)
88+
throw new AssertionError("Unexpected mismatch index:" + si);
89+
bi = mismatchBufferSmall1.mismatch(mismatchBufferSmall2);
90+
if (bi != 0)
91+
throw new AssertionError("Unexpected mismatch index:" + bi);
92+
}
93+
6394
static {
6495
for (int i = 0 ; i < bytes.length ; i++) {
6596
bytes[i] = i;
@@ -89,4 +120,28 @@ public void unsafe_copy() {
89120
public void segment_copy() {
90121
segment.copyFrom(bytesSegment);
91122
}
123+
124+
@Benchmark
125+
@OutputTimeUnit(TimeUnit.NANOSECONDS)
126+
public long mismatch_large_segment() {
127+
return mismatchSegmentLarge1.mismatch(mismatchSegmentLarge2);
128+
}
129+
130+
@Benchmark
131+
@OutputTimeUnit(TimeUnit.NANOSECONDS)
132+
public int mismatch_large_bytebuffer() {
133+
return mismatchBufferLarge1.mismatch(mismatchBufferLarge2);
134+
}
135+
136+
@Benchmark
137+
@OutputTimeUnit(TimeUnit.NANOSECONDS)
138+
public long mismatch_small_segment() {
139+
return mismatchSegmentSmall1.mismatch(mismatchSegmentSmall2);
140+
}
141+
142+
@Benchmark
143+
@OutputTimeUnit(TimeUnit.NANOSECONDS)
144+
public int mismatch_small_bytebuffer() {
145+
return mismatchBufferSmall1.mismatch(mismatchBufferSmall2);
146+
}
92147
}

0 commit comments

Comments
 (0)
Please sign in to comment.