diff options
author | Terry Parker <tparker@google.com> | 2019-01-10 19:01:57 -0500 |
---|---|---|
committer | Gerrit Code Review @ Eclipse.org <gerrit@eclipse.org> | 2019-01-10 19:01:57 -0500 |
commit | 01f8e53b26fd71a83620bcf9732cdd2b0528de1b (patch) | |
tree | 330170d5c25b0ffa1c3efede59c6c397db81d728 | |
parent | 8ed59c511cf61a23393fabf252a315371a1438f0 (diff) | |
parent | 513d80db1066beaa284c97b1515cd069049f6f05 (diff) | |
download | jgit-01f8e53b26fd71a83620bcf9732cdd2b0528de1b.tar.gz jgit-01f8e53b26fd71a83620bcf9732cdd2b0528de1b.zip |
Merge "Lazily open ReadableChannel in BlockBasedFile.getOrLoadBlock"
4 files changed, 42 insertions, 7 deletions
diff --git a/org.eclipse.jgit/src/org/eclipse/jgit/internal/storage/dfs/BlockBasedFile.java b/org.eclipse.jgit/src/org/eclipse/jgit/internal/storage/dfs/BlockBasedFile.java index 600b1a44d1..4d14742440 100644 --- a/org.eclipse.jgit/src/org/eclipse/jgit/internal/storage/dfs/BlockBasedFile.java +++ b/org.eclipse.jgit/src/org/eclipse/jgit/internal/storage/dfs/BlockBasedFile.java @@ -129,8 +129,8 @@ abstract class BlockBasedFile { } DfsBlock getOrLoadBlock(long pos, DfsReader ctx) throws IOException { - try (ReadableChannel rc = ctx.db.openFile(desc, ext)) { - return cache.getOrLoad(this, pos, ctx, () -> rc); + try (LazyChannel c = new LazyChannel(ctx, desc, ext)) { + return cache.getOrLoad(this, pos, ctx, c); } } @@ -203,4 +203,41 @@ abstract class BlockBasedFile { static long elapsedMicros(long start) { return (System.nanoTime() - start) / 1000L; } + + /** + * A supplier of readable channel that opens the channel lazily. + */ + private static class LazyChannel + implements AutoCloseable, DfsBlockCache.ReadableChannelSupplier { + private final DfsReader ctx; + private final DfsPackDescription desc; + private final PackExt ext; + + private ReadableChannel rc = null; + + LazyChannel(DfsReader ctx, DfsPackDescription desc, PackExt ext) { + this.ctx = ctx; + this.desc = desc; + this.ext = ext; + } + + @Override + public ReadableChannel get() throws IOException { + if (rc == null) { + synchronized (this) { + if (rc == null) { + rc = ctx.db.openFile(desc, ext); + } + } + } + return rc; + } + + @Override + public void close() throws IOException { + if (rc != null) { + rc.close(); + } + } + } } diff --git a/org.eclipse.jgit/src/org/eclipse/jgit/internal/storage/dfs/DfsBlockCache.java b/org.eclipse.jgit/src/org/eclipse/jgit/internal/storage/dfs/DfsBlockCache.java index d9e9327d3b..c6e2fae42f 100644 --- a/org.eclipse.jgit/src/org/eclipse/jgit/internal/storage/dfs/DfsBlockCache.java +++ b/org.eclipse.jgit/src/org/eclipse/jgit/internal/storage/dfs/DfsBlockCache.java @@ -754,7 +754,7 @@ public final class DfsBlockCache { * Supplier for readable channel */ @FunctionalInterface - public interface ReadableChannelSupplier { + interface ReadableChannelSupplier { /** * @return ReadableChannel * @throws IOException diff --git a/org.eclipse.jgit/src/org/eclipse/jgit/internal/storage/dfs/DfsBlockCacheConfig.java b/org.eclipse.jgit/src/org/eclipse/jgit/internal/storage/dfs/DfsBlockCacheConfig.java index eea9ef4e0b..cd1fa5f78f 100644 --- a/org.eclipse.jgit/src/org/eclipse/jgit/internal/storage/dfs/DfsBlockCacheConfig.java +++ b/org.eclipse.jgit/src/org/eclipse/jgit/internal/storage/dfs/DfsBlockCacheConfig.java @@ -212,7 +212,7 @@ public class DfsBlockCacheConfig { * consumer of wait time in milliseconds. * @return {@code this} */ - public DfsBlockCacheConfig setReflockWaitTimeConsumer(Consumer<Long> c) { + public DfsBlockCacheConfig setRefLockWaitTimeConsumer(Consumer<Long> c) { refLock = c; return this; } diff --git a/org.eclipse.jgit/src/org/eclipse/jgit/internal/storage/dfs/DfsReftable.java b/org.eclipse.jgit/src/org/eclipse/jgit/internal/storage/dfs/DfsReftable.java index a80797730f..4853298012 100644 --- a/org.eclipse.jgit/src/org/eclipse/jgit/internal/storage/dfs/DfsReftable.java +++ b/org.eclipse.jgit/src/org/eclipse/jgit/internal/storage/dfs/DfsReftable.java @@ -128,9 +128,7 @@ public class DfsReftable extends BlockBasedFile { open().setReadAheadBytes(readAhead); } - DfsBlock block = cache.getOrLoad(file, pos, ctx, () -> { - return open(); - }); + DfsBlock block = cache.getOrLoad(file, pos, ctx, () -> open()); if (block.start == pos && block.size() >= cnt) { return block.zeroCopyByteBuffer(cnt); } |