@@ -268,6 +268,8 @@ public abstract class $Type$Buffer
268
268
extends Buffer
269
269
implements Comparable<$Type$Buffer>{#if[char]?, Appendable, CharSequence, Readable}
270
270
{
271
+ // Cached array base offset
272
+ private static final long ARRAY_BASE_OFFSET = UNSAFE.arrayBaseOffset($type$[].class);
271
273
272
274
// These fields are declared here rather than in Heap-X-Buffer in order to
273
275
// reduce the number of virtual method invocations needed to access these
@@ -791,11 +793,13 @@ public abstract class $Type$Buffer
791
793
*/
792
794
public $Type$Buffer get($type$[] dst, int offset, int length) {
793
795
Objects.checkFromIndexSize(offset, length, dst.length);
794
- if (length > remaining())
796
+ int pos = position();
797
+ if (length > limit() - pos)
795
798
throw new BufferUnderflowException();
796
- int end = offset + length;
797
- for (int i = offset; i < end; i++)
798
- dst[i] = get();
799
+
800
+ getArray(pos, dst, offset, length);
801
+
802
+ position(pos + length);
799
803
return this;
800
804
}
801
805
@@ -869,9 +873,9 @@ public abstract class $Type$Buffer
869
873
public $Type$Buffer get(int index, $type$[] dst, int offset, int length) {
870
874
Objects.checkFromIndexSize(index, length, limit());
871
875
Objects.checkFromIndexSize(offset, length, dst.length);
872
- int end = offset + length;
873
- for (int i = offset, j = index; i < end; i++, j++)
874
- dst[i] = get(j);
876
+
877
+ getArray(index, dst, offset, length);
878
+
875
879
return this;
876
880
}
877
881
@@ -906,6 +910,40 @@ public abstract class $Type$Buffer
906
910
return get(index, dst, 0, dst.length);
907
911
}
908
912
913
+ private $Type$Buffer getArray(int index, $type$[] dst, int offset, int length) {
914
+ if (
915
+ #if[char]
916
+ isAddressable() &&
917
+ #end[char]
918
+ ((long)length << $LG_BYTES_PER_VALUE$) > Bits.JNI_COPY_TO_ARRAY_THRESHOLD) {
919
+ long bufAddr = address + ((long)index << $LG_BYTES_PER_VALUE$);
920
+ long dstOffset =
921
+ ARRAY_BASE_OFFSET + ((long)offset << $LG_BYTES_PER_VALUE$);
922
+ long len = (long)length << $LG_BYTES_PER_VALUE$;
923
+
924
+ try {
925
+ #if[!byte]
926
+ if (order() != ByteOrder.nativeOrder())
927
+ SCOPED_MEMORY_ACCESS.copySwapMemory(
928
+ scope(), null, base(), bufAddr,
929
+ dst, dstOffset, len, $Fulltype$.BYTES);
930
+ else
931
+ #end[!byte]
932
+ SCOPED_MEMORY_ACCESS.copyMemory(
933
+ scope(), null, base(), bufAddr,
934
+ dst, dstOffset, len);
935
+ } finally {
936
+ Reference.reachabilityFence(this);
937
+ }
938
+ } else {
939
+ int end = offset + length;
940
+ for (int i = offset, j = index; i < end; i++, j++) {
941
+ dst[i] = get(j);
942
+ }
943
+ }
944
+ return this;
945
+ }
946
+
909
947
// -- Bulk put operations --
910
948
911
949
/**
@@ -1038,8 +1076,8 @@ public abstract class $Type$Buffer
1038
1076
}
1039
1077
1040
1078
void putBuffer(int pos, $Type$Buffer src, int srcPos, int n) {
1079
+ #if[rw]
1041
1080
Object srcBase = src.base();
1042
-
1043
1081
#if[char]
1044
1082
if (src.isAddressable()) {
1045
1083
#else[char]
@@ -1053,29 +1091,21 @@ public abstract class $Type$Buffer
1053
1091
long addr = address + ((long)pos << $LG_BYTES_PER_VALUE$);
1054
1092
long len = (long)n << $LG_BYTES_PER_VALUE$;
1055
1093
1094
+ try {
1056
1095
#if[!byte]
1057
- if (this.order() == src.order()) {
1096
+ if (this.order() != src.order())
1097
+ SCOPED_MEMORY_ACCESS.copySwapMemory(
1098
+ src.scope(), scope(), srcBase, srcAddr,
1099
+ base, addr, len, $Fulltype$.BYTES);
1100
+ else
1058
1101
#end[!byte]
1059
- try {
1060
1102
SCOPED_MEMORY_ACCESS.copyMemory(
1061
- src.scope(), scope(), srcBase,
1062
- srcAddr, base, addr, len);
1063
- } finally {
1064
- Reference.reachabilityFence(src);
1065
- Reference.reachabilityFence(this);
1066
- }
1067
- #if[!byte]
1068
- } else {
1069
- try {
1070
- SCOPED_MEMORY_ACCESS.copySwapMemory(
1071
- src.scope(), scope(), srcBase,
1072
- srcAddr, base, addr, len, (long)1 << $LG_BYTES_PER_VALUE$);
1073
- } finally {
1074
- Reference.reachabilityFence(src);
1075
- Reference.reachabilityFence(this);
1076
- }
1103
+ src.scope(), scope(), srcBase, srcAddr,
1104
+ base, addr, len);
1105
+ } finally {
1106
+ Reference.reachabilityFence(src);
1107
+ Reference.reachabilityFence(this);
1077
1108
}
1078
- #end[!byte]
1079
1109
#if[char]
1080
1110
} else { // src.isAddressable() == false
1081
1111
assert StringCharBuffer.class.isInstance(src);
@@ -1084,6 +1114,9 @@ public abstract class $Type$Buffer
1084
1114
put(i, src.get(j));
1085
1115
}
1086
1116
#end[char]
1117
+ #else[rw]
1118
+ throw new ReadOnlyBufferException();
1119
+ #end[rw]
1087
1120
}
1088
1121
1089
1122
/**
@@ -1138,12 +1171,16 @@ public abstract class $Type$Buffer
1138
1171
* If this buffer is read-only
1139
1172
*/
1140
1173
public $Type$Buffer put($type$[] src, int offset, int length) {
1174
+ if (isReadOnly())
1175
+ throw new ReadOnlyBufferException();
1141
1176
Objects.checkFromIndexSize(offset, length, src.length);
1142
- if (length > remaining())
1177
+ int pos = position();
1178
+ if (length > limit() - pos)
1143
1179
throw new BufferOverflowException();
1144
- int end = offset + length;
1145
- for (int i = offset; i < end; i++)
1146
- this.put(src[i]);
1180
+
1181
+ putArray(pos, src, offset, length);
1182
+
1183
+ position(pos + length);
1147
1184
return this;
1148
1185
}
1149
1186
@@ -1223,9 +1260,9 @@ public abstract class $Type$Buffer
1223
1260
throw new ReadOnlyBufferException();
1224
1261
Objects.checkFromIndexSize(index, length, limit());
1225
1262
Objects.checkFromIndexSize(offset, length, src.length);
1226
- int end = offset + length;
1227
- for (int i = offset, j = index; i < end; i++, j++)
1228
- this.put(j, src[i]);
1263
+
1264
+ putArray(index, src, offset, length);
1265
+
1229
1266
return this;
1230
1267
}
1231
1268
@@ -1262,6 +1299,43 @@ public abstract class $Type$Buffer
1262
1299
return put(index, src, 0, src.length);
1263
1300
}
1264
1301
1302
+ private $Type$Buffer putArray(int index, $type$[] src, int offset, int length) {
1303
+ #if[rw]
1304
+ if (
1305
+ #if[char]
1306
+ isAddressable() &&
1307
+ #end[char]
1308
+ ((long)length << $LG_BYTES_PER_VALUE$) > Bits.JNI_COPY_FROM_ARRAY_THRESHOLD) {
1309
+ long bufAddr = address + ((long)index << $LG_BYTES_PER_VALUE$);
1310
+ long srcOffset =
1311
+ ARRAY_BASE_OFFSET + ((long)offset << $LG_BYTES_PER_VALUE$);
1312
+ long len = (long)length << $LG_BYTES_PER_VALUE$;
1313
+
1314
+ try {
1315
+ #if[!byte]
1316
+ if (order() != ByteOrder.nativeOrder())
1317
+ SCOPED_MEMORY_ACCESS.copySwapMemory(
1318
+ null, scope(), src, srcOffset,
1319
+ base(), bufAddr, len, $Fulltype$.BYTES);
1320
+ else
1321
+ #end[!byte]
1322
+ SCOPED_MEMORY_ACCESS.copyMemory(
1323
+ null, scope(), src, srcOffset,
1324
+ base(), bufAddr, len);
1325
+ } finally {
1326
+ Reference.reachabilityFence(this);
1327
+ }
1328
+ } else {
1329
+ int end = offset + length;
1330
+ for (int i = offset, j = index; i < end; i++, j++)
1331
+ this.put(j, src[i]);
1332
+ }
1333
+ return this;
1334
+ #else[rw]
1335
+ throw new ReadOnlyBufferException();
1336
+ #end[rw]
1337
+ }
1338
+
1265
1339
#if[char]
1266
1340
1267
1341
/**
0 commit comments