ObjectLoaders can defer the bulk of their data access to the getCachedBytes() method invocation, rather than at the time of their construction. That means there's code paths into the mmap allocator which aren't protected by the "try GC and try again" within openObject, resulting in some random crashes due to mmap failures. The right place to check for mmap failure and retry is within WindowedFile.loadWindow, where we actually invoke the mmap call. If the call fails, we need can immediately retry it, and if it fails again, we can temporarily degrade to the byte[] variant until the JVM is able to evict and GC enough space. Signed-off-by: Shawn O. Pearce <spearce@xxxxxxxxxxx> --- .../src/org/spearce/jgit/lib/Repository.java | 27 +----------- .../src/org/spearce/jgit/lib/WindowedFile.java | 43 ++++++++++++++----- 2 files changed, 34 insertions(+), 36 deletions(-) diff --git a/org.spearce.jgit/src/org/spearce/jgit/lib/Repository.java b/org.spearce.jgit/src/org/spearce/jgit/lib/Repository.java index 02f8103..e1c4049 100644 --- a/org.spearce.jgit/src/org/spearce/jgit/lib/Repository.java +++ b/org.spearce.jgit/src/org/spearce/jgit/lib/Repository.java @@ -290,30 +290,9 @@ public ObjectLoader openObject(final WindowCursor curs, final AnyObjectId id) int k = packs.length; if (k > 0) { do { - try { - final ObjectLoader ol = packs[--k].get(curs, id); - if (ol != null) - return ol; - } catch (IOException ioe) { - // This shouldn't happen unless the pack was corrupted - // after we opened it or the VM runs out of memory. This is - // a know problem with memory mapped I/O in java and have - // been noticed with JDK < 1.6. Tell the gc that now is a good - // time to collect and try once more. - try { - curs.release(); - System.gc(); - final ObjectLoader ol = packs[k].get(curs, id); - if (ol != null) - return ol; - } catch (IOException ioe2) { - ioe2.printStackTrace(); - ioe.printStackTrace(); - // Still fails.. that's BAD, maybe the pack has - // been corrupted after all, or the gc didn't manage - // to release enough previously mmaped areas. - } - } + final ObjectLoader ol = packs[--k].get(curs, id); + if (ol != null) + return ol; } while (k > 0); } try { diff --git a/org.spearce.jgit/src/org/spearce/jgit/lib/WindowedFile.java b/org.spearce.jgit/src/org/spearce/jgit/lib/WindowedFile.java index f640c42..dd94f24 100644 --- a/org.spearce.jgit/src/org/spearce/jgit/lib/WindowedFile.java +++ b/org.spearce.jgit/src/org/spearce/jgit/lib/WindowedFile.java @@ -299,22 +299,41 @@ void cacheClose() { } void loadWindow(final WindowCursor curs, final int windowId, - final long pos, final int windowSize) throws IOException { + final long pos, final int size) throws IOException { if (WindowCache.mmap) { - final MappedByteBuffer map = fd.getChannel().map(MapMode.READ_ONLY, - pos, windowSize); - if (map.hasArray()) { - final byte[] b = map.array(); - curs.window = new ByteArrayWindow(this, pos, windowId, b); - curs.handle = b; - } else { - curs.window = new ByteBufferWindow(this, pos, windowId, map); - curs.handle = map; + MappedByteBuffer map; + try { + map = fd.getChannel().map(MapMode.READ_ONLY, pos, size); + } catch (IOException e) { + // The most likely reason this failed is the JVM has run out + // of virtual memory. We need to discard quickly, and try to + // force the GC to finalize and release any existing mappings. + try { + curs.release(); + System.gc(); + System.runFinalization(); + map = fd.getChannel().map(MapMode.READ_ONLY, pos, size); + } catch (IOException ioe2) { + // Temporarily disable mmap and do buffered disk IO. + // + map = null; + System.err.println("warning: mmap failure: "+ioe2); + } + } + if (map != null) { + if (map.hasArray()) { + final byte[] b = map.array(); + curs.window = new ByteArrayWindow(this, pos, windowId, b); + curs.handle = b; + } else { + curs.window = new ByteBufferWindow(this, pos, windowId, map); + curs.handle = map; + } + return; } - return; } - final byte[] b = new byte[windowSize]; + final byte[] b = new byte[size]; synchronized (fd) { fd.seek(pos); fd.readFully(b); -- 1.6.1.rc4.301.g5497a -- To unsubscribe from this list: send the line "unsubscribe git" in the body of a message to majordomo@xxxxxxxxxxxxxxx More majordomo info at http://vger.kernel.org/majordomo-info.html