diff options
3 files changed, 40 insertions, 144 deletions
diff --git a/org.eclipse.jgit/src/org/eclipse/jgit/internal/storage/dfs/DfsPackFile.java b/org.eclipse.jgit/src/org/eclipse/jgit/internal/storage/dfs/DfsPackFile.java index b9e40cd9ff..9be3df3b17 100644 --- a/org.eclipse.jgit/src/org/eclipse/jgit/internal/storage/dfs/DfsPackFile.java +++ b/org.eclipse.jgit/src/org/eclipse/jgit/internal/storage/dfs/DfsPackFile.java @@ -59,8 +59,12 @@ public final class DfsPackFile extends BlockBasedFile { private static final int REC_SIZE = Constants.OBJECT_ID_LENGTH + 8; private static final long REF_POSITION = 0; - /** Lock for initialization of {@link #index} and {@link #reverseIndex}. */ - private final Object indexLock = new Object(); + /** + * Lock for initialization of {@link #index} and {@link #corruptObjects}. + * <p> + * This lock ensures only one thread can perform the initialization work. + */ + private final Object initLock = new Object(); /** Index mapping {@link ObjectId} to position within the pack stream. */ private volatile PackIndex index; @@ -68,15 +72,9 @@ public final class DfsPackFile extends BlockBasedFile { /** Reverse version of {@link #index} mapping position to {@link ObjectId}. */ private volatile PackReverseIndex reverseIndex; - /** Lock for initialization of {@link #bitmapIndex}. */ - private final Object bitmapIndexLock = new Object(); - /** Index of compressed bitmap mapping entire object graph. */ private volatile PackBitmapIndex bitmapIndex; - /** Lock for {@link #corruptObjects}. */ - private final Object corruptObjectsLock = new Object(); - /** * Objects we have tried to read, and discovered to be corrupt. * <p> @@ -158,7 +156,7 @@ public final class DfsPackFile extends BlockBasedFile { Repository.getGlobalListenerList() .dispatch(new BeforeDfsPackIndexLoadedEvent(this)); - synchronized (indexLock) { + synchronized (initLock) { if (index != null) { return index; } @@ -193,17 +191,7 @@ public final class DfsPackFile extends BlockBasedFile { return desc.getPackSource() == UNREACHABLE_GARBAGE; } - /** - * Get the BitmapIndex for this PackFile. - * - * @param ctx - * reader context to support reading from the backing store if - * the index is not already loaded in memory. - * @return the BitmapIndex. - * @throws java.io.IOException - * the bitmap index is not available, or is corrupt. - */ - public PackBitmapIndex getBitmapIndex(DfsReader ctx) throws IOException { + PackBitmapIndex getBitmapIndex(DfsReader ctx) throws IOException { if (invalid || isGarbage() || !desc.hasFileExt(BITMAP_INDEX)) { return null; } @@ -212,11 +200,13 @@ public final class DfsPackFile extends BlockBasedFile { return bitmapIndex; } - synchronized (bitmapIndexLock) { + synchronized (initLock) { if (bitmapIndex != null) { return bitmapIndex; } + PackIndex idx = idx(ctx); + PackReverseIndex revidx = getReverseIdx(ctx); DfsStreamKey bitmapKey = desc.getStreamKey(BITMAP_INDEX); AtomicBoolean cacheHit = new AtomicBoolean(true); DfsBlockCache.Ref<PackBitmapIndex> idxref = cache.getOrLoadRef( @@ -224,7 +214,7 @@ public final class DfsPackFile extends BlockBasedFile { REF_POSITION, () -> { cacheHit.set(false); - return loadBitmapIndex(ctx, bitmapKey); + return loadBitmapIndex(ctx, bitmapKey, idx, revidx); }); if (cacheHit.get()) { ctx.stats.bitmapCacheHit++; @@ -242,7 +232,7 @@ public final class DfsPackFile extends BlockBasedFile { return reverseIndex; } - synchronized (indexLock) { + synchronized (initLock) { if (reverseIndex != null) { return reverseIndex; } @@ -1013,7 +1003,7 @@ public final class DfsPackFile extends BlockBasedFile { private void setCorrupt(long offset) { LongList list = corruptObjects; if (list == null) { - synchronized (corruptObjectsLock) { + synchronized (initLock) { list = corruptObjects; if (list == null) { list = new LongList(); @@ -1076,8 +1066,11 @@ public final class DfsPackFile extends BlockBasedFile { revidx); } - private DfsBlockCache.Ref<PackBitmapIndex> loadBitmapIndex(DfsReader ctx, - DfsStreamKey bitmapKey) throws IOException { + private DfsBlockCache.Ref<PackBitmapIndex> loadBitmapIndex( + DfsReader ctx, + DfsStreamKey bitmapKey, + PackIndex idx, + PackReverseIndex revidx) throws IOException { ctx.stats.readBitmap++; long start = System.nanoTime(); try (ReadableChannel rc = ctx.db.openFile(desc, BITMAP_INDEX)) { @@ -1093,8 +1086,7 @@ public final class DfsPackFile extends BlockBasedFile { bs = wantSize; } in = new BufferedInputStream(in, bs); - bmidx = PackBitmapIndex.read(in, () -> idx(ctx), - () -> getReverseIdx(ctx)); + bmidx = PackBitmapIndex.read(in, idx, revidx); } finally { size = rc.position(); ctx.stats.readBitmapIdxBytes += size; diff --git a/org.eclipse.jgit/src/org/eclipse/jgit/internal/storage/file/PackBitmapIndex.java b/org.eclipse.jgit/src/org/eclipse/jgit/internal/storage/file/PackBitmapIndex.java index b7439f9c5b..beb51dc2eb 100644 --- a/org.eclipse.jgit/src/org/eclipse/jgit/internal/storage/file/PackBitmapIndex.java +++ b/org.eclipse.jgit/src/org/eclipse/jgit/internal/storage/file/PackBitmapIndex.java @@ -97,37 +97,7 @@ public abstract class PackBitmapIndex { public static PackBitmapIndex read( InputStream fd, PackIndex packIndex, PackReverseIndex reverseIndex) throws IOException { - return new PackBitmapIndexV1(fd, () -> packIndex, () -> reverseIndex); - } - - /** - * Read an existing pack bitmap index file from a buffered stream. - * <p> - * The format of the file will be automatically detected and a proper access - * implementation for that format will be constructed and returned to the - * caller. The file may or may not be held open by the returned instance. - * - * @param fd - * stream to read the bitmap index file from. The stream must be - * buffered as some small IOs are performed against the stream. - * The caller is responsible for closing the stream. - * @param packIndexSupplier - * the supplier for pack index for the corresponding pack file. - * @param reverseIndexSupplier - * the supplier for pack reverse index for the corresponding pack - * file. - * @return a copy of the index in-memory. - * @throws java.io.IOException - * the stream cannot be read. - * @throws CorruptObjectException - * the stream does not contain a valid pack bitmap index. - */ - public static PackBitmapIndex read(InputStream fd, - SupplierWithIOException<PackIndex> packIndexSupplier, - SupplierWithIOException<PackReverseIndex> reverseIndexSupplier) - throws IOException { - return new PackBitmapIndexV1(fd, packIndexSupplier, - reverseIndexSupplier); + return new PackBitmapIndexV1(fd, packIndex, reverseIndex); } /** Footer checksum applied on the bottom of the pack file. */ @@ -191,19 +161,4 @@ public abstract class PackBitmapIndex { * @return the number of bitmaps in this bitmap index. */ public abstract int getBitmapCount(); - - /** - * Supplier that propagates IOException. - * - * @param <T> - * the return type which is expected from {@link #get()} - */ - @FunctionalInterface - public interface SupplierWithIOException<T> { - /** - * @return result - * @throws IOException - */ - T get() throws IOException; - } } diff --git a/org.eclipse.jgit/src/org/eclipse/jgit/internal/storage/file/PackBitmapIndexV1.java b/org.eclipse.jgit/src/org/eclipse/jgit/internal/storage/file/PackBitmapIndexV1.java index c7864f4e70..b7d241f3f8 100644 --- a/org.eclipse.jgit/src/org/eclipse/jgit/internal/storage/file/PackBitmapIndexV1.java +++ b/org.eclipse.jgit/src/org/eclipse/jgit/internal/storage/file/PackBitmapIndexV1.java @@ -14,11 +14,8 @@ import java.io.DataInput; import java.io.IOException; import java.io.InputStream; import java.text.MessageFormat; -import java.util.ArrayList; import java.util.Arrays; -import java.util.List; -import org.eclipse.jgit.annotations.Nullable; import org.eclipse.jgit.internal.JGitText; import org.eclipse.jgit.lib.AnyObjectId; import org.eclipse.jgit.lib.Constants; @@ -49,13 +46,11 @@ class PackBitmapIndexV1 extends BasePackBitmapIndex { private final ObjectIdOwnerMap<StoredBitmap> bitmaps; - PackBitmapIndexV1(final InputStream fd, - SupplierWithIOException<PackIndex> packIndexSupplier, - SupplierWithIOException<PackReverseIndex> reverseIndexSupplier) - throws IOException { - // An entry is object id, xor offset, flag byte, and a length encoded - // bitmap. The object id is an int32 of the nth position sorted by name. + PackBitmapIndexV1(final InputStream fd, PackIndex packIndex, + PackReverseIndex reverseIndex) throws IOException { super(new ObjectIdOwnerMap<StoredBitmap>()); + this.packIndex = packIndex; + this.reverseIndex = reverseIndex; this.bitmaps = getBitmaps(); final byte[] scratch = new byte[32]; @@ -102,10 +97,10 @@ class PackBitmapIndexV1 extends BasePackBitmapIndex { this.blobs = readBitmap(dataInput); this.tags = readBitmap(dataInput); - // Read full bitmap from storage first. - List<IdxPositionBitmap> idxPositionBitmapList = new ArrayList<>(); + // An entry is object id, xor offset, flag byte, and a length encoded + // bitmap. The object id is an int32 of the nth position sorted by name. // The xor offset is a single byte offset back in the list of entries. - IdxPositionBitmap[] recentBitmaps = new IdxPositionBitmap[MAX_XOR_OFFSET]; + StoredBitmap[] recentBitmaps = new StoredBitmap[MAX_XOR_OFFSET]; for (int i = 0; i < (int) numEntries; i++) { IO.readFully(fd, scratch, 0, 6); int nthObjectId = NB.decodeInt32(scratch, 0); @@ -113,58 +108,38 @@ class PackBitmapIndexV1 extends BasePackBitmapIndex { int flags = scratch[5]; EWAHCompressedBitmap bitmap = readBitmap(dataInput); - if (nthObjectId < 0) { + if (nthObjectId < 0) throw new IOException(MessageFormat.format( JGitText.get().invalidId, String.valueOf(nthObjectId))); - } - if (xorOffset < 0) { + if (xorOffset < 0) throw new IOException(MessageFormat.format( JGitText.get().invalidId, String.valueOf(xorOffset))); - } - if (xorOffset > MAX_XOR_OFFSET) { + if (xorOffset > MAX_XOR_OFFSET) throw new IOException(MessageFormat.format( JGitText.get().expectedLessThanGot, String.valueOf(MAX_XOR_OFFSET), String.valueOf(xorOffset))); - } - if (xorOffset > i) { + if (xorOffset > i) throw new IOException(MessageFormat.format( JGitText.get().expectedLessThanGot, String.valueOf(i), String.valueOf(xorOffset))); - } - IdxPositionBitmap xorIdxPositionBitmap = null; + + ObjectId objectId = packIndex.getObjectId(nthObjectId); + StoredBitmap xorBitmap = null; if (xorOffset > 0) { int index = (i - xorOffset); - xorIdxPositionBitmap = recentBitmaps[index - % recentBitmaps.length]; - if (xorIdxPositionBitmap == null) { + xorBitmap = recentBitmaps[index % recentBitmaps.length]; + if (xorBitmap == null) throw new IOException(MessageFormat.format( JGitText.get().invalidId, String.valueOf(xorOffset))); - } } - IdxPositionBitmap idxPositionBitmap = new IdxPositionBitmap( - nthObjectId, xorIdxPositionBitmap, bitmap, flags); - idxPositionBitmapList.add(idxPositionBitmap); - recentBitmaps[i % recentBitmaps.length] = idxPositionBitmap; - } - this.packIndex = packIndexSupplier.get(); - for (int i = 0; i < idxPositionBitmapList.size(); ++i) { - IdxPositionBitmap idxPositionBitmap = idxPositionBitmapList.get(i); - ObjectId objectId = packIndex - .getObjectId(idxPositionBitmap.nthObjectId); - StoredBitmap sb = new StoredBitmap(objectId, - idxPositionBitmap.bitmap, - idxPositionBitmap.getXorStoredBitmap(), - idxPositionBitmap.flags); - // Save the StoredBitmap for a possible future XorStoredBitmap - // reference. - idxPositionBitmap.sb = sb; + StoredBitmap sb = new StoredBitmap( + objectId, bitmap, xorBitmap, flags); bitmaps.add(sb); + recentBitmaps[i % recentBitmaps.length] = sb; } - - this.reverseIndex = reverseIndexSupplier.get(); } /** {@inheritDoc} */ @@ -239,30 +214,4 @@ class PackBitmapIndexV1 extends BasePackBitmapIndex { bitmap.deserialize(dataInput); return bitmap; } - - /** - * Temporary holder of object position in pack index and other metadata for - * {@code StoredBitmap}. - */ - private static final class IdxPositionBitmap { - int nthObjectId; - IdxPositionBitmap xorIdxPositionBitmap; - EWAHCompressedBitmap bitmap; - int flags; - StoredBitmap sb; - - IdxPositionBitmap(int nthObjectId, @Nullable - IdxPositionBitmap xorIdxPositionBitmap, EWAHCompressedBitmap bitmap, - int flags) { - this.nthObjectId = nthObjectId; - this.xorIdxPositionBitmap = xorIdxPositionBitmap; - this.bitmap = bitmap; - this.flags = flags; - } - - StoredBitmap getXorStoredBitmap() { - return xorIdxPositionBitmap == null ? null - : xorIdxPositionBitmap.sb; - } - } } |