summaryrefslogtreecommitdiffstats
path: root/org.eclipse.jgit
diff options
context:
space:
mode:
authorChris Aniszczyk <caniszczyk@gmail.com>2010-10-11 17:18:05 -0400
committerCode Review <codereview-daemon@eclipse.org>2010-10-11 17:18:05 -0400
commit7429a9a5aa18ee9c7192413f5e2c91f9d997f61e (patch)
tree3646cbc77c3284c2c3a799c9f7c6659a83cbdbc2 /org.eclipse.jgit
parent033ab7f6f0f5501cc27866500f4f072dcafcc61e (diff)
parent1bd24a23f944e6e9a27eccd34109bd19a94582df (diff)
downloadjgit-7429a9a5aa18ee9c7192413f5e2c91f9d997f61e.tar.gz
jgit-7429a9a5aa18ee9c7192413f5e2c91f9d997f61e.zip
Merge "Define LowLevelDiffAlgorithm to bypass re-hashing"
Diffstat (limited to 'org.eclipse.jgit')
-rw-r--r--org.eclipse.jgit/src/org/eclipse/jgit/diff/HistogramDiff.java26
-rw-r--r--org.eclipse.jgit/src/org/eclipse/jgit/diff/LowLevelDiffAlgorithm.java92
-rw-r--r--org.eclipse.jgit/src/org/eclipse/jgit/diff/MyersDiff.java36
3 files changed, 125 insertions, 29 deletions
diff --git a/org.eclipse.jgit/src/org/eclipse/jgit/diff/HistogramDiff.java b/org.eclipse.jgit/src/org/eclipse/jgit/diff/HistogramDiff.java
index 23c0cd12e2..3941f0c560 100644
--- a/org.eclipse.jgit/src/org/eclipse/jgit/diff/HistogramDiff.java
+++ b/org.eclipse.jgit/src/org/eclipse/jgit/diff/HistogramDiff.java
@@ -89,7 +89,7 @@ package org.eclipse.jgit.diff;
* This implementation has an internal limitation that prevents it from handling
* sequences with more than 268,435,456 (2^28) elements.
*/
-public class HistogramDiff extends DiffAlgorithm {
+public class HistogramDiff extends LowLevelDiffAlgorithm {
/** Algorithm to use when there are too many element occurrences. */
private DiffAlgorithm fallback = MyersDiff.INSTANCE;
@@ -127,11 +127,10 @@ public class HistogramDiff extends DiffAlgorithm {
maxChainLength = maxLen;
}
- public <S extends Sequence> EditList diffNonCommon(
- SequenceComparator<? super S> cmp, S a, S b) {
- State<S> s = new State<S>(new HashedSequencePair<S>(cmp, a, b));
- s.diffReplace(new Edit(0, s.a.size(), 0, s.b.size()));
- return s.edits;
+ public <S extends Sequence> void diffNonCommon(EditList edits,
+ HashedSequenceComparator<S> cmp, HashedSequence<S> a,
+ HashedSequence<S> b, Edit region) {
+ new State<S>(edits, cmp, a, b).diffReplace(region);
}
private class State<S extends Sequence> {
@@ -144,11 +143,12 @@ public class HistogramDiff extends DiffAlgorithm {
/** Result edits we have determined that must be made to convert a to b. */
final EditList edits;
- State(HashedSequencePair<S> p) {
- this.cmp = p.getComparator();
- this.a = p.getA();
- this.b = p.getB();
- this.edits = new EditList();
+ State(EditList edits, HashedSequenceComparator<S> cmp,
+ HashedSequence<S> a, HashedSequence<S> b) {
+ this.cmp = cmp;
+ this.a = a;
+ this.b = b;
+ this.edits = edits;
}
void diffReplace(Edit r) {
@@ -167,6 +167,10 @@ public class HistogramDiff extends DiffAlgorithm {
diff(r.after(lcs));
}
+ } else if (fallback instanceof LowLevelDiffAlgorithm) {
+ LowLevelDiffAlgorithm fb = (LowLevelDiffAlgorithm) fallback;
+ fb.diffNonCommon(edits, cmp, a, b, r);
+
} else if (fallback != null) {
SubsequenceComparator<HashedSequence<S>> cs = subcmp();
Subsequence<HashedSequence<S>> as = Subsequence.a(a, r);
diff --git a/org.eclipse.jgit/src/org/eclipse/jgit/diff/LowLevelDiffAlgorithm.java b/org.eclipse.jgit/src/org/eclipse/jgit/diff/LowLevelDiffAlgorithm.java
new file mode 100644
index 0000000000..e3861cd65e
--- /dev/null
+++ b/org.eclipse.jgit/src/org/eclipse/jgit/diff/LowLevelDiffAlgorithm.java
@@ -0,0 +1,92 @@
+/*
+ * Copyright (C) 2010, Google Inc.
+ * and other copyright owners as documented in the project's IP log.
+ *
+ * This program and the accompanying materials are made available
+ * under the terms of the Eclipse Distribution License v1.0 which
+ * accompanies this distribution, is reproduced below, and is
+ * available at http://www.eclipse.org/org/documents/edl-v10.php
+ *
+ * All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or
+ * without modification, are permitted provided that the following
+ * conditions are met:
+ *
+ * - Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ *
+ * - Redistributions in binary form must reproduce the above
+ * copyright notice, this list of conditions and the following
+ * disclaimer in the documentation and/or other materials provided
+ * with the distribution.
+ *
+ * - Neither the name of the Eclipse Foundation, Inc. nor the
+ * names of its contributors may be used to endorse or promote
+ * products derived from this software without specific prior
+ * written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND
+ * CONTRIBUTORS "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES,
+ * INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
+ * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
+ * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR
+ * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
+ * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
+ * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
+ * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
+ * CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
+ * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
+ * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF
+ * ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+package org.eclipse.jgit.diff;
+
+/** Compares two sequences primarily based upon hash codes. */
+public abstract class LowLevelDiffAlgorithm extends DiffAlgorithm {
+ @Override
+ public <S extends Sequence> EditList diffNonCommon(
+ SequenceComparator<? super S> cmp, S a, S b) {
+ HashedSequencePair<S> p = new HashedSequencePair<S>(cmp, a, b);
+ HashedSequenceComparator<S> hc = p.getComparator();
+ HashedSequence<S> ha = p.getA();
+ HashedSequence<S> hb = p.getB();
+ p = null;
+
+ EditList res = new EditList();
+ Edit region = new Edit(0, a.size(), 0, b.size());
+ diffNonCommon(res, hc, ha, hb, region);
+ return res;
+ }
+
+ /**
+ * Compare two sequences and identify a list of edits between them.
+ *
+ * This method should be invoked only after the two sequences have been
+ * proven to have no common starting or ending elements. The expected
+ * elimination of common starting and ending elements is automatically
+ * performed by the {@link #diff(SequenceComparator, Sequence, Sequence)}
+ * method, which invokes this method using {@link Subsequence}s.
+ *
+ * @param <S>
+ * type of sequence being compared.
+ * @param edits
+ * result list to append the region's edits onto.
+ * @param cmp
+ * the comparator supplying the element equivalence function.
+ * @param a
+ * the first (also known as old or pre-image) sequence. Edits
+ * returned by this algorithm will reference indexes using the
+ * 'A' side: {@link Edit#getBeginA()}, {@link Edit#getEndA()}.
+ * @param b
+ * the second (also known as new or post-image) sequence. Edits
+ * returned by this algorithm will reference indexes using the
+ * 'B' side: {@link Edit#getBeginB()}, {@link Edit#getEndB()}.
+ * @param region
+ * the region being compared within the two sequences.
+ */
+ public abstract <S extends Sequence> void diffNonCommon(EditList edits,
+ HashedSequenceComparator<S> cmp, HashedSequence<S> a,
+ HashedSequence<S> b, Edit region);
+}
diff --git a/org.eclipse.jgit/src/org/eclipse/jgit/diff/MyersDiff.java b/org.eclipse.jgit/src/org/eclipse/jgit/diff/MyersDiff.java
index 821d06be29..3459109b6e 100644
--- a/org.eclipse.jgit/src/org/eclipse/jgit/diff/MyersDiff.java
+++ b/org.eclipse.jgit/src/org/eclipse/jgit/diff/MyersDiff.java
@@ -107,15 +107,18 @@ import org.eclipse.jgit.util.LongList;
*/
public class MyersDiff<S extends Sequence> {
/** Singleton instance of MyersDiff. */
- public static final DiffAlgorithm INSTANCE = new DiffAlgorithm() {
- public <S extends Sequence> EditList diffNonCommon(
- SequenceComparator<? super S> cmp, S a, S b) {
- return new MyersDiff<S>(cmp, a, b).edits;
+ public static final DiffAlgorithm INSTANCE = new LowLevelDiffAlgorithm() {
+ @Override
+ public <S extends Sequence> void diffNonCommon(EditList edits,
+ HashedSequenceComparator<S> cmp, HashedSequence<S> a,
+ HashedSequence<S> b, Edit region) {
+ new MyersDiff<S>(edits, cmp, a, b, region);
}
};
/**
- * The list of edits found during the last call to {@link #calculateEdits()}
+ * The list of edits found during the last call to
+ * {@link #calculateEdits(Edit)}
*/
protected EditList edits;
@@ -132,15 +135,13 @@ public class MyersDiff<S extends Sequence> {
*/
protected HashedSequence<S> b;
- private MyersDiff(SequenceComparator<? super S> cmp, S a, S b) {
- HashedSequencePair<S> pair;
-
- pair = new HashedSequencePair<S>(cmp, a, b);
- this.cmp = pair.getComparator();
- this.a = pair.getA();
- this.b = pair.getB();
-
- calculateEdits();
+ private MyersDiff(EditList edits, HashedSequenceComparator<S> cmp,
+ HashedSequence<S> a, HashedSequence<S> b, Edit region) {
+ this.edits = edits;
+ this.cmp = cmp;
+ this.a = a;
+ this.b = b;
+ calculateEdits(region);
}
// TODO: use ThreadLocal for future multi-threaded operations
@@ -149,11 +150,10 @@ public class MyersDiff<S extends Sequence> {
/**
* Entrypoint into the algorithm this class is all about. This method triggers that the
* differences between A and B are calculated in form of a list of edits.
+ * @param r portion of the sequences to examine.
*/
- protected void calculateEdits() {
- edits = new EditList();
-
- middle.initialize(0, a.size(), 0, b.size());
+ private void calculateEdits(Edit r) {
+ middle.initialize(r.beginA, r.endA, r.beginB, r.endB);
if (middle.beginA >= middle.endA &&
middle.beginB >= middle.endB)
return;