View Javadoc
1   /*
2    * Copyright (C) 2010, Google Inc.
3    * and other copyright owners as documented in the project's IP log.
4    *
5    * This program and the accompanying materials are made available
6    * under the terms of the Eclipse Distribution License v1.0 which
7    * accompanies this distribution, is reproduced below, and is
8    * available at http://www.eclipse.org/org/documents/edl-v10.php
9    *
10   * All rights reserved.
11   *
12   * Redistribution and use in source and binary forms, with or
13   * without modification, are permitted provided that the following
14   * conditions are met:
15   *
16   * - Redistributions of source code must retain the above copyright
17   *   notice, this list of conditions and the following disclaimer.
18   *
19   * - Redistributions in binary form must reproduce the above
20   *   copyright notice, this list of conditions and the following
21   *   disclaimer in the documentation and/or other materials provided
22   *   with the distribution.
23   *
24   * - Neither the name of the Eclipse Foundation, Inc. nor the
25   *   names of its contributors may be used to endorse or promote
26   *   products derived from this software without specific prior
27   *   written permission.
28   *
29   * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND
30   * CONTRIBUTORS "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES,
31   * INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
32   * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
33   * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR
34   * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
35   * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
36   * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
37   * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
38   * CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
39   * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
40   * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF
41   * ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
42   */
43  
44  package org.eclipse.jgit.diff;
45  
46  import java.io.EOFException;
47  import java.io.IOException;
48  import java.io.InputStream;
49  import java.util.Arrays;
50  
51  import org.eclipse.jgit.errors.MissingObjectException;
52  import org.eclipse.jgit.lib.ObjectLoader;
53  import org.eclipse.jgit.lib.ObjectStream;
54  
55  /**
56   * Index structure of lines/blocks in one file.
57   * <p>
58   * This structure can be used to compute an approximation of the similarity
59   * between two files. The index is used by {@link SimilarityRenameDetector} to
60   * compute scores between files.
61   * <p>
62   * To save space in memory, this index uses a space efficient encoding which
63   * will not exceed 1 MiB per instance. The index starts out at a smaller size
64   * (closer to 2 KiB), but may grow as more distinct blocks within the scanned
65   * file are discovered.
66   *
67   * @since 4.0
68   */
69  public class SimilarityIndex {
70  	/** A special {@link TableFullException} used in place of OutOfMemoryError. */
71  	public static final TableFullException
72  			TABLE_FULL_OUT_OF_MEMORY = new TableFullException();
73  
74  	/**
75  	 * Shift to apply before storing a key.
76  	 * <p>
77  	 * Within the 64 bit table record space, we leave the highest bit unset so
78  	 * all values are positive. The lower 32 bits to count bytes.
79  	 */
80  	private static final int KEY_SHIFT = 32;
81  
82  	/** Maximum value of the count field, also mask to extract the count. */
83  	private static final long MAX_COUNT = (1L << KEY_SHIFT) - 1;
84  
85  	/**
86  	 * Total amount of bytes hashed into the structure, including \n. This is
87  	 * usually the size of the file minus number of CRLF encounters.
88  	 */
89  	private long hashedCnt;
90  
91  	/** Number of non-zero entries in {@link #idHash}. */
92  	private int idSize;
93  
94  	/** {@link #idSize} that triggers {@link #idHash} to double in size. */
95  	private int idGrowAt;
96  
97  	/**
98  	 * Pairings of content keys and counters.
99  	 * <p>
100 	 * Slots in the table are actually two ints wedged into a single long. The
101 	 * upper 32 bits stores the content key, and the remaining lower bits stores
102 	 * the number of bytes associated with that key. Empty slots are denoted by
103 	 * 0, which cannot occur because the count cannot be 0. Values can only be
104 	 * positive, which we enforce during key addition.
105 	 */
106 	private long[] idHash;
107 
108 	/** {@code idHash.length == 1 << idHashBits}. */
109 	private int idHashBits;
110 
111 	/**
112 	 * Create a new similarity index for the given object
113 	 *
114 	 * @param obj
115 	 *            the object to hash
116 	 * @return similarity index for this object
117 	 * @throws IOException
118 	 *             file contents cannot be read from the repository.
119 	 * @throws TableFullException
120 	 *             object hashing overflowed the storage capacity of the
121 	 *             SimilarityIndex.
122 	 */
123 	public static SimilarityIndex create(ObjectLoader obj) throws IOException,
124 			TableFullException {
125 		SimilarityIndex idx = new SimilarityIndex();
126 		idx.hash(obj);
127 		idx.sort();
128 		return idx;
129 	}
130 
131 	SimilarityIndex() {
132 		idHashBits = 8;
133 		idHash = new long[1 << idHashBits];
134 		idGrowAt = growAt(idHashBits);
135 	}
136 
137 	void hash(ObjectLoader obj) throws MissingObjectException, IOException,
138 			TableFullException {
139 		if (obj.isLarge()) {
140 			hashLargeObject(obj);
141 		} else {
142 			byte[] raw = obj.getCachedBytes();
143 			hash(raw, 0, raw.length);
144 		}
145 	}
146 
147 	private void hashLargeObject(ObjectLoader obj) throws IOException,
148 			TableFullException {
149 		ObjectStream in1 = obj.openStream();
150 		boolean text;
151 		try {
152 			text = !RawText.isBinary(in1);
153 		} finally {
154 			in1.close();
155 		}
156 
157 		ObjectStream in2 = obj.openStream();
158 		try {
159 			hash(in2, in2.getSize(), text);
160 		} finally {
161 			in2.close();
162 		}
163 	}
164 
165 	void hash(byte[] raw, int ptr, final int end) throws TableFullException {
166 		final boolean text = !RawText.isBinary(raw);
167 		hashedCnt = 0;
168 		while (ptr < end) {
169 			int hash = 5381;
170 			int blockHashedCnt = 0;
171 			int start = ptr;
172 
173 			// Hash one line, or one block, whichever occurs first.
174 			do {
175 				int c = raw[ptr++] & 0xff;
176 				// Ignore CR in CRLF sequence if text
177 				if (text && c == '\r' && ptr < end && raw[ptr] == '\n')
178 					continue;
179 				blockHashedCnt++;
180 				if (c == '\n')
181 					break;
182 				hash = (hash << 5) + hash + c;
183 			} while (ptr < end && ptr - start < 64);
184 			hashedCnt += blockHashedCnt;
185 			add(hash, blockHashedCnt);
186 		}
187 	}
188 
189 	void hash(InputStream in, long remaining, boolean text) throws IOException,
190 			TableFullException {
191 		byte[] buf = new byte[4096];
192 		int ptr = 0;
193 		int cnt = 0;
194 
195 		while (0 < remaining) {
196 			int hash = 5381;
197 			int blockHashedCnt = 0;
198 
199 			// Hash one line, or one block, whichever occurs first.
200 			int n = 0;
201 			do {
202 				if (ptr == cnt) {
203 					ptr = 0;
204 					cnt = in.read(buf, 0, buf.length);
205 					if (cnt <= 0)
206 						throw new EOFException();
207 				}
208 
209 				n++;
210 				int c = buf[ptr++] & 0xff;
211 				// Ignore CR in CRLF sequence if text
212 				if (text && c == '\r' && ptr < cnt && buf[ptr] == '\n')
213 					continue;
214 				blockHashedCnt++;
215 				if (c == '\n')
216 					break;
217 				hash = (hash << 5) + hash + c;
218 			} while (n < 64 && n < remaining);
219 			hashedCnt += blockHashedCnt;
220 			add(hash, blockHashedCnt);
221 			remaining -= n;
222 		}
223 	}
224 
225 	/**
226 	 * Sort the internal table so it can be used for efficient scoring.
227 	 * <p>
228 	 * Once sorted, additional lines/blocks cannot be added to the index.
229 	 */
230 	void sort() {
231 		// Sort the array. All of the empty space will wind up at the front,
232 		// because we forced all of the keys to always be positive. Later
233 		// we only work with the back half of the array.
234 		//
235 		Arrays.sort(idHash);
236 	}
237 
238 	/**
239 	 * Compute the similarity score between this index and another.
240 	 * <p>
241 	 * A region of a file is defined as a line in a text file or a fixed-size
242 	 * block in a binary file. To prepare an index, each region in the file is
243 	 * hashed; the values and counts of hashes are retained in a sorted table.
244 	 * Define the similarity fraction F as the the count of matching regions
245 	 * between the two files divided between the maximum count of regions in
246 	 * either file. The similarity score is F multiplied by the maxScore
247 	 * constant, yielding a range [0, maxScore]. It is defined as maxScore for
248 	 * the degenerate case of two empty files.
249 	 * <p>
250 	 * The similarity score is symmetrical; i.e. a.score(b) == b.score(a).
251 	 *
252 	 * @param dst
253 	 *            the other index
254 	 * @param maxScore
255 	 *            the score representing a 100% match
256 	 * @return the similarity score
257 	 */
258 	public int score(SimilarityIndex dst, int maxScore) {
259 		long max = Math.max(hashedCnt, dst.hashedCnt);
260 		if (max == 0)
261 			return maxScore;
262 		return (int) ((common(dst) * maxScore) / max);
263 	}
264 
265 	long common(SimilarityIndex dst) {
266 		return common(this, dst);
267 	}
268 
269 	private static long common(SimilarityIndex src, SimilarityIndex dst) {
270 		int srcIdx = src.packedIndex(0);
271 		int dstIdx = dst.packedIndex(0);
272 		long[] srcHash = src.idHash;
273 		long[] dstHash = dst.idHash;
274 		return common(srcHash, srcIdx, dstHash, dstIdx);
275 	}
276 
277 	private static long common(long[] srcHash, int srcIdx, //
278 			long[] dstHash, int dstIdx) {
279 		if (srcIdx == srcHash.length || dstIdx == dstHash.length)
280 			return 0;
281 
282 		long common = 0;
283 		int srcKey = keyOf(srcHash[srcIdx]);
284 		int dstKey = keyOf(dstHash[dstIdx]);
285 
286 		for (;;) {
287 			if (srcKey == dstKey) {
288 				common += Math.min(countOf(srcHash[srcIdx]),
289 						countOf(dstHash[dstIdx]));
290 
291 				if (++srcIdx == srcHash.length)
292 					break;
293 				srcKey = keyOf(srcHash[srcIdx]);
294 
295 				if (++dstIdx == dstHash.length)
296 					break;
297 				dstKey = keyOf(dstHash[dstIdx]);
298 
299 			} else if (srcKey < dstKey) {
300 				// Regions of src which do not appear in dst.
301 				if (++srcIdx == srcHash.length)
302 					break;
303 				srcKey = keyOf(srcHash[srcIdx]);
304 
305 			} else /* if (dstKey < srcKey) */{
306 				// Regions of dst which do not appear in src.
307 				if (++dstIdx == dstHash.length)
308 					break;
309 				dstKey = keyOf(dstHash[dstIdx]);
310 			}
311 		}
312 
313 		return common;
314 	}
315 
316 	// Testing only
317 	int size() {
318 		return idSize;
319 	}
320 
321 	// Testing only
322 	int key(int idx) {
323 		return keyOf(idHash[packedIndex(idx)]);
324 	}
325 
326 	// Testing only
327 	long count(int idx) {
328 		return countOf(idHash[packedIndex(idx)]);
329 	}
330 
331 	// Brute force approach only for testing.
332 	int findIndex(int key) {
333 		for (int i = 0; i < idSize; i++)
334 			if (key(i) == key)
335 				return i;
336 		return -1;
337 	}
338 
339 	private int packedIndex(int idx) {
340 		return (idHash.length - idSize) + idx;
341 	}
342 
343 	void add(int key, int cnt) throws TableFullException {
344 		key = (key * 0x9e370001) >>> 1; // Mix bits and ensure not negative.
345 
346 		int j = slot(key);
347 		for (;;) {
348 			long v = idHash[j];
349 			if (v == 0) {
350 				// Empty slot in the table, store here.
351 				if (idGrowAt <= idSize) {
352 					grow();
353 					j = slot(key);
354 					continue;
355 				}
356 				idHash[j] = pair(key, cnt);
357 				idSize++;
358 				return;
359 
360 			} else if (keyOf(v) == key) {
361 				// Same key, increment the counter. If it overflows, fail
362 				// indexing to prevent the key from being impacted.
363 				//
364 				idHash[j] = pair(key, countOf(v) + cnt);
365 				return;
366 
367 			} else if (++j >= idHash.length) {
368 				j = 0;
369 			}
370 		}
371 	}
372 
373 	private static long pair(int key, long cnt) throws TableFullException {
374 		if (MAX_COUNT < cnt)
375 			throw new TableFullException();
376 		return (((long) key) << KEY_SHIFT) | cnt;
377 	}
378 
379 	private int slot(int key) {
380 		// We use 31 - idHashBits because the upper bit was already forced
381 		// to be 0 and we want the remaining high bits to be used as the
382 		// table slot.
383 		//
384 		return key >>> (31 - idHashBits);
385 	}
386 
387 	private static int growAt(int idHashBits) {
388 		return (1 << idHashBits) * (idHashBits - 3) / idHashBits;
389 	}
390 
391 	private void grow() throws TableFullException {
392 		if (idHashBits == 30)
393 			throw new TableFullException();
394 
395 		long[] oldHash = idHash;
396 		int oldSize = idHash.length;
397 
398 		idHashBits++;
399 		idGrowAt = growAt(idHashBits);
400 
401 		try {
402 			idHash = new long[1 << idHashBits];
403 		} catch (OutOfMemoryError noMemory) {
404 			throw TABLE_FULL_OUT_OF_MEMORY;
405 		}
406 
407 		for (int i = 0; i < oldSize; i++) {
408 			long v = oldHash[i];
409 			if (v != 0) {
410 				int j = slot(keyOf(v));
411 				while (idHash[j] != 0)
412 					if (++j >= idHash.length)
413 						j = 0;
414 				idHash[j] = v;
415 			}
416 		}
417 	}
418 
419 	private static int keyOf(long v) {
420 		return (int) (v >>> KEY_SHIFT);
421 	}
422 
423 	private static long countOf(long v) {
424 		return v & MAX_COUNT;
425 	}
426 
427 	/** Thrown by {@code create()} when file is too large. */
428 	public static class TableFullException extends Exception {
429 		private static final long serialVersionUID = 1L;
430 	}
431 }