Skip to content

Commit 2478158

Browse files
author
Brian Burkhalter
committedDec 8, 2021
8277361: java/nio/channels/Channels/ReadXBytes.java fails with OOM error
Reviewed-by: alanb, lancea
1 parent 8af3b27 commit 2478158

File tree

1 file changed

+38
-16
lines changed

1 file changed

+38
-16
lines changed
 

‎test/jdk/java/nio/channels/Channels/ReadXBytes.java

+38-16
Original file line numberDiff line numberDiff line change
@@ -25,32 +25,33 @@
2525
* @test
2626
* @bug 8268435 8274780
2727
* @summary Verify ChannelInputStream methods readAllBytes and readNBytes
28-
* @requires vm.bits == 64
28+
* @requires (sun.arch.data.model == "64" & os.maxMemory >= 16g)
2929
* @library ..
3030
* @library /test/lib
3131
* @build jdk.test.lib.RandomFactory
3232
* @modules java.base/jdk.internal.util
33-
* @run testng/othervm/timeout=900 -Xmx8G ReadXBytes
33+
* @run testng/othervm/timeout=900 -Xmx12G ReadXBytes
3434
* @key randomness
3535
*/
3636
import java.io.File;
3737
import java.io.FileInputStream;
3838
import java.io.FilterInputStream;
3939
import java.io.InputStream;
4040
import java.io.IOException;
41-
import java.io.RandomAccessFile;
41+
import java.nio.ByteBuffer;
4242
import java.nio.channels.Channel;
4343
import java.nio.channels.Channels;
4444
import java.nio.channels.FileChannel;
4545
import java.nio.channels.ReadableByteChannel;
4646
import java.nio.channels.SeekableByteChannel;
4747
import java.nio.file.Files;
4848
import java.nio.file.Path;
49-
import static java.nio.file.StandardOpenOption.READ;
5049
import java.util.List;
5150
import java.util.Random;
5251
import jdk.internal.util.ArraysSupport;
5352

53+
import static java.nio.file.StandardOpenOption.*;
54+
5455
import jdk.test.lib.RandomFactory;
5556

5657
import org.testng.Assert;
@@ -72,30 +73,51 @@ public class ReadXBytes {
7273
// A length greater than a 32-bit integer can accommodate
7374
private static final long HUGE_LENGTH = Integer.MAX_VALUE + 27L;
7475

76+
// Current directory
77+
private static final Path DIR = Path.of(System.getProperty("test.dir", "."));
78+
7579
// --- Framework ---
7680

81+
// Create a temporary file path
82+
static Path createFilePath() {
83+
String name = String.format("ReadXBytes%d.tmp", System.nanoTime());
84+
return DIR.resolve(name);
85+
}
86+
7787
// Creates a temporary file of a specified length with undefined content
7888
static Path createFile(long length) throws IOException {
79-
File file = File.createTempFile("foo", ".bar");
80-
file.deleteOnExit();
81-
try (RandomAccessFile raf = new RandomAccessFile(file, "rw")) {
82-
raf.setLength(length);
89+
Path path = createFilePath();
90+
path.toFile().deleteOnExit();
91+
try (FileChannel fc = FileChannel.open(path, CREATE_NEW, SPARSE, WRITE)) {
92+
if (length > 0) {
93+
fc.position(length - 1);
94+
fc.write(ByteBuffer.wrap(new byte[] {27}));
95+
}
8396
}
84-
return file.toPath();
97+
return path;
8598
}
8699

87100
// Creates a temporary file of a specified length with random content
88101
static Path createFileWithRandomContent(long length) throws IOException {
89102
Path file = createFile(length);
90-
try (RandomAccessFile raf = new RandomAccessFile(file.toFile(), "rw")) {
91-
long written = 0L;
92-
int bufLength = Math.min(32768, (int)Math.min(length, BIG_LENGTH));
103+
try (FileChannel fc = FileChannel.open(file, WRITE);) {
104+
long pos = 0L;
105+
// if the length exceeds 2 GB, skip the first 2 GB - 1 MB bytes
106+
if (length >= 2L*1024*1024*1024) {
107+
// write the last (length - 2GB - 1MB) bytes
108+
pos = 2047L*1024*1024;
109+
} else if (length > 0) {
110+
// write either the first or last bytes only
111+
long p = Math.min(Math.abs(RAND.nextLong()), length - 1);
112+
pos = RAND.nextBoolean() ? p : length - 1 - p;
113+
}
114+
fc.position(pos);
115+
int bufLength = Math.min(32768, (int)Math.min(length - pos, BIG_LENGTH));
93116
byte[] buf = new byte[bufLength];
94-
while (written < length) {
117+
while (pos < length) {
95118
RAND.nextBytes(buf);
96-
int len = (int)Math.min(bufLength, length - written);
97-
raf.write(buf, 0, len);
98-
written += len;
119+
int len = (int)Math.min(bufLength, length - pos);
120+
pos += fc.write(ByteBuffer.wrap(buf, 0, len));
99121
}
100122
}
101123
return file;

0 commit comments

Comments
 (0)
Please sign in to comment.