Skip to content

Commit 0a12605

Browse files
Jason ZauggLance Andersen
Jason Zaugg
authored and
Lance Andersen
committedMay 11, 2021
8265448: (zipfs): Reduce read contention in ZipFileSystem
Reviewed-by: alanb, lancea
1 parent acf02ed commit 0a12605

File tree

3 files changed

+101
-13
lines changed

3 files changed

+101
-13
lines changed
 

‎src/jdk.zipfs/share/classes/jdk/nio/zipfs/ZipFileSystem.java

+9-12
Original file line numberDiff line numberDiff line change
@@ -1227,8 +1227,12 @@ final long readFullyAt(byte[] buf, int off, long len, long pos)
12271227
}
12281228

12291229
private long readFullyAt(ByteBuffer bb, long pos) throws IOException {
1230-
synchronized(ch) {
1231-
return ch.position(pos).read(bb);
1230+
if (ch instanceof FileChannel fch) {
1231+
return fch.read(bb, pos);
1232+
} else {
1233+
synchronized(ch) {
1234+
return ch.position(pos).read(bb);
1235+
}
12321236
}
12331237
}
12341238

@@ -2116,7 +2120,7 @@ else if (e.file != null)
21162120
// streams.add(eis);
21172121
return eis;
21182122
} else { // untouched CEN or COPY
2119-
eis = new EntryInputStream(e, ch);
2123+
eis = new EntryInputStream(e);
21202124
}
21212125
if (e.method == METHOD_DEFLATED) {
21222126
// MORE: Compute good size for inflater stream:
@@ -2173,15 +2177,12 @@ public int available() {
21732177
// Inner class implementing the input stream used to read
21742178
// a (possibly compressed) zip file entry.
21752179
private class EntryInputStream extends InputStream {
2176-
private final SeekableByteChannel zfch; // local ref to zipfs's "ch". zipfs.ch might
2177-
// point to a new channel after sync()
21782180
private long pos; // current position within entry data
21792181
private long rem; // number of remaining bytes within entry
21802182

2181-
EntryInputStream(Entry e, SeekableByteChannel zfch)
2183+
EntryInputStream(Entry e)
21822184
throws IOException
21832185
{
2184-
this.zfch = zfch;
21852186
rem = e.csize;
21862187
pos = e.locoff;
21872188
if (pos == -1) {
@@ -2206,14 +2207,10 @@ public int read(byte[] b, int off, int len) throws IOException {
22062207
if (len > rem) {
22072208
len = (int) rem;
22082209
}
2209-
// readFullyAt()
2210-
long n;
22112210
ByteBuffer bb = ByteBuffer.wrap(b);
22122211
bb.position(off);
22132212
bb.limit(off + len);
2214-
synchronized(zfch) {
2215-
n = zfch.position(pos).read(bb);
2216-
}
2213+
long n = readFullyAt(bb, pos);
22172214
if (n > 0) {
22182215
pos += n;
22192216
rem -= n;

‎test/jdk/jdk/nio/zipfs/ZipFSTester.java

+3-1
Original file line numberDiff line numberDiff line change
@@ -499,10 +499,12 @@ private static void checkRead(Path path, byte[] expected) throws IOException {
499499
}
500500
// System.out.printf(" --> %d, %d%n", pos, len);
501501
bb.position(0).limit(len); // bb.flip().limit(len);
502+
int expectedReadResult = sbc.size() == 0 ? -1 : len;
502503
if (sbc.position(pos).position() != pos ||
503-
sbc.read(bb) != len ||
504+
sbc.read(bb) != expectedReadResult ||
504505
!Arrays.equals(buf, 0, bb.position(), expected, pos, pos + len)) {
505506
System.out.printf("read()/position() failed%n");
507+
throw new RuntimeException("CHECK FAILED!");
506508
}
507509
}
508510
} catch (IOException x) {
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,89 @@
1+
/*
2+
* Copyright (c) 2020, 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.nio.zipfs;
24+
25+
import org.openjdk.jmh.annotations.*;
26+
import org.openjdk.jmh.infra.Blackhole;
27+
28+
import java.io.IOException;
29+
import java.io.InputStream;
30+
import java.nio.file.FileSystem;
31+
import java.nio.file.Files;
32+
import java.nio.file.Path;
33+
import java.nio.file.spi.FileSystemProvider;
34+
import java.util.Map;
35+
import java.util.Random;
36+
import java.util.concurrent.TimeUnit;
37+
38+
@BenchmarkMode(Mode.AverageTime)
39+
@Warmup(iterations = 5, time = 500, timeUnit = TimeUnit.MILLISECONDS)
40+
@Measurement(iterations = 10, time = 500, timeUnit = TimeUnit.MILLISECONDS)
41+
@State(Scope.Benchmark)
42+
@Threads(Threads.MAX)
43+
@OutputTimeUnit(TimeUnit.MILLISECONDS)
44+
@Fork(value = 3)
45+
public class ZipFileSystemBenchmark {
46+
private static final String FILE_NAME = "filename";
47+
private FileSystemProvider jarFsProvider;
48+
private Path readPath;
49+
private FileSystem fileSystem;
50+
private Path zip;
51+
52+
@Setup(Level.Trial) public void setup() throws IOException {
53+
jarFsProvider = FileSystemProvider.installedProviders().stream().filter(x -> x.getScheme().equals("jar")).findFirst().get();
54+
zip = Files.createTempFile("zipfs-benchmark", ".jar");
55+
createTestZip();
56+
fileSystem = jarFsProvider.newFileSystem(zip, Map.of());
57+
Path rootRead = fileSystem.getRootDirectories().iterator().next();
58+
readPath = rootRead.resolve(FILE_NAME);
59+
}
60+
61+
private void createTestZip() throws IOException {
62+
Files.delete(zip);
63+
FileSystem writableFileSystem = jarFsProvider.newFileSystem(zip, Map.of("create", "true"));
64+
byte[] data = new byte[16 * 1024 * 1024];
65+
new Random(31).nextBytes(data);
66+
Path root = writableFileSystem.getRootDirectories().iterator().next();
67+
Files.write(root.resolve(FILE_NAME), data);
68+
writableFileSystem.close();
69+
}
70+
71+
@TearDown public void tearDown() throws IOException {
72+
if (fileSystem != null) {
73+
fileSystem.close();
74+
}
75+
Files.deleteIfExists(zip);
76+
}
77+
78+
// Performance should remain constant when varying the number of threads up to the
79+
// number of physical cores if the NIO implementation on the platform supports
80+
// concurrent reads to a single FileChannel instance. At the time of writing, NIO on Windows
81+
// serializes access.
82+
@Benchmark public void read(Blackhole bh) throws IOException {
83+
InputStream inputStream = Files.newInputStream(readPath);
84+
byte[] buffer = new byte[8192];
85+
while (inputStream.read(buffer) != -1) {
86+
bh.consume(buffer);
87+
}
88+
}
89+
}

0 commit comments

Comments
 (0)
Please sign in to comment.