Skip to content

Commit

Permalink
blame: use a fingerprint heuristic to match ignored lines
Browse files Browse the repository at this point in the history
This replaces the heuristic used to identify lines from ignored commits
with one that finds likely candidate lines in the parent's version of
the file.

The old heuristic simply assigned lines in the target to the same line
number (plus offset) in the parent.  The new function uses a
fingerprinting algorithm to detect similarity between lines.

The fingerprint code and the idea to use them for blame came from
Michael Platings <michael@platin.gs>.

For each line changed in the target, i.e. in a blame_entry touched by a
target's diff, guess_line_blames() finds the best line in the parent,
above a magic threshold.  Ties are broken by proximity of the parent
line number to the target's line.

We actually make two passes.  The first pass checks in the diff chunk
associated with the blame entry - specifically from blame_chunk().
Often times, those diff chunks are small; any 'context' in a normal diff
chunk is broken up into multiple calls to blame_chunk().  We make a
second pass over the entire parent, with a slightly higher threshold.

Here's an example of the difference the fingerprinting makes.  Consider
a file with four commits:

	commit-a 11) void new_func_1(void *x, void *y);
	commit-b 12) void new_func_2(void *x, void *y);
	commit-c 13) some_line_c
	commit-d 14) some_line_d

After a commit 'X', we have:

	commit-X 11) void new_func_1(void *x,
	commit-X 12)                 void *y);
	commit-X 13) void new_func_2(void *x,
	commit-X 14)                 void *y);
	commit-c 15) some_line_c
	commit-d 16) some_line_d

When we blame-ignored with the old algorithm, we get:

	commit-a 11) void new_func_1(void *x,
	commit-b 12)                 void *y);
	00000000 13) void new_func_2(void *x,
	00000000 14)                 void *y);
	commit-c 15) some_line_c
	commit-d 16) some_line_d

Where commit-b is blamed for 12 instead of 13.  With the fingerprint
algorithm, we get:

	commit-a 11) void new_func_1(void *x,
	commit-b 12)                 void *y);
	commit-b 13) void new_func_2(void *x,
	commit-b 14)                 void *y);
	commit-c 15) some_line_c
	commit-d 16) some_line_d

Note both lines 12 and 14 are given to commit b.  Their match is above
the FINGERPRINT_CHUNK_THRESHOLD, and they tied.  Specifically, parent
lines 11 and 12 both match these lines.  The algorithm chose parent line
12, since that was closest to the target line numbers of 12 and 14.

If we increase the threshold, say to 10, those two lines won't match,
and will be treated as 'unblamable.'

For an example of scanning the entire parent for a match, consider:

	commit-a 30) #include <sys/header_a.h>
	commit-b 31) #include <header_b.h>
	commit-c 32) #include <header_c.h>

Then commit X alphabetizes them:

	commit-X 30) #include <header_b.h>
	commit-X 31) #include <header_c.h>
	commit-X 32) #include <sys/header_a.h>

If we just check the parent's chunk (i.e. the first pass), we'd get:

	commit-b 30) #include <header_b.h>
	commit-c 31) #include <header_c.h>
	00000000 32) #include <sys/header_a.h>

That's because commit X consists of two chunks: one chunk is removing
sys/header_a.h, then some context, and the second chunk is adding
sys/header_a.h.

If we scan the entire parent file, we get:

	commit-b 30) #include <header_b.h>
	commit-c 31) #include <header_c.h>
	commit-a 32) #include <sys/header_a.h>

Suggested-by: Michael Platings <michael@platin.gs>
Signed-off-by: Barret Rhoden <brho@google.com>
Signed-off-by: Junio C Hamano <gitster@pobox.com>
  • Loading branch information
Barret Rhoden authored and gitster committed Apr 14, 2019
1 parent 84ac6e4 commit dfb4ee1
Showing 1 changed file with 131 additions and 9 deletions.
140 changes: 131 additions & 9 deletions blame.c
Expand Up @@ -339,6 +339,84 @@ static int find_line_starts(int **line_starts, const char *buf,
return num;
}

struct fingerprint {
struct hashmap map;
struct hashmap_entry *entries;
};

static void get_fingerprint(struct fingerprint *result,
const char *line_begin,
const char *line_end)
{
unsigned int hash;
char c0, c1;
const char *p;
int map_entry_count = line_end - line_begin - 1;
struct hashmap_entry *entry = xcalloc(map_entry_count,
sizeof(struct hashmap_entry));

hashmap_init(&result->map, NULL, NULL, map_entry_count);
result->entries = entry;
for (p = line_begin; p + 1 < line_end; ++p, ++entry) {
c0 = *p;
c1 = *(p + 1);
/* Ignore whitespace pairs */
if (isspace(c0) && isspace(c1))
continue;
hash = tolower(c0) | (tolower(c1) << 8);
hashmap_entry_init(entry, hash);
hashmap_put(&result->map, entry);
}
}

static void free_fingerprint(struct fingerprint *f)
{
hashmap_free(&f->map, 0);
free(f->entries);
}

static int fingerprint_similarity(struct fingerprint *a,
struct fingerprint *b)
{
int intersection = 0;
struct hashmap_iter iter;
struct hashmap_entry *entry;

hashmap_iter_init(&b->map, &iter);

while ((entry = hashmap_iter_next(&iter))) {
if (hashmap_get(&a->map, entry, NULL))
++intersection;
}
return intersection;
}

static void get_line_fingerprints(struct fingerprint *fingerprints,
const char *content,
const int *line_starts,
int first_line,
int nr_lines)
{
int i;

line_starts += first_line;
for (i = 0; i < nr_lines; ++i) {
const char *linestart = content + line_starts[i];
const char *lineend = content + line_starts[i + 1];

get_fingerprint(fingerprints + i, linestart, lineend);
}
}

static void free_chunk_fingerprints(struct fingerprint *fingerprints,
int nr_fingerprints)
{
int i;

for (i = 0; i < nr_fingerprints; i++)
free_fingerprint(&fingerprints[i]);
}

static void fill_origin_fingerprints(struct blame_origin *o, mmfile_t *file)
{
int *line_starts;
Expand All @@ -347,14 +425,16 @@ static void fill_origin_fingerprints(struct blame_origin *o, mmfile_t *file)
return;
o->num_lines = find_line_starts(&line_starts, o->file.ptr,
o->file.size);
/* TODO: Will fill in fingerprints in a future commit */
o->fingerprints = xcalloc(sizeof(struct fingerprint), o->num_lines);
get_line_fingerprints(o->fingerprints, o->file.ptr, line_starts,
0, o->num_lines);
free(line_starts);
}

static void drop_origin_fingerprints(struct blame_origin *o)
{
if (o->fingerprints) {
free_chunk_fingerprints(o->fingerprints, o->num_lines);
o->num_lines = 0;
FREE_AND_NULL(o->fingerprints);
}
Expand Down Expand Up @@ -934,27 +1014,69 @@ static int are_lines_adjacent(struct blame_line_tracker *first,
first->s_lno + 1 == second->s_lno;
}

static void scan_parent_range(struct fingerprint *p_fps,
struct fingerprint *t_fps, int t_idx,
int from, int nr_lines,
int *best_sim_val, int *best_sim_idx)
{
int sim, p_idx;

for (p_idx = from; p_idx < from + nr_lines; p_idx++) {
sim = fingerprint_similarity(&t_fps[t_idx], &p_fps[p_idx]);
if (sim < *best_sim_val)
continue;
/* Break ties with the closest-to-target line number */
if (sim == *best_sim_val && *best_sim_idx != -1 &&
abs(*best_sim_idx - t_idx) < abs(p_idx - t_idx))
continue;
*best_sim_val = sim;
*best_sim_idx = p_idx;
}
}

/*
* This cheap heuristic assigns lines in the chunk to their relative location in
* the parent's chunk. Any additional lines are left with the target.
* The CHUNK threshold is for how similar we must be within a diff chunk, which
* is typically the adjacent '-' and '+' sections in a diff, separated by the
* ' ' context.
*
* We have a greater threshold for similarity for lines in any part of the
* parent's file. If no line in the parent meets the appropriate threshold,
* then the blame_entry will stay with the target and be considered
* 'unblamable'.
*/
#define FINGERPRINT_CHUNK_THRESHOLD 1
#define FINGERPRINT_FILE_THRESHOLD 10

static void guess_line_blames(struct blame_entry *e,
struct blame_origin *parent,
struct blame_origin *target,
int offset, int parent_slno, int parent_len,
struct blame_line_tracker *line_blames)
{
int i, parent_idx;
int i, target_idx;

for (i = 0; i < e->num_lines; i++) {
parent_idx = e->s_lno + i + offset;
if (parent_slno <= parent_idx &&
parent_idx < parent_slno + parent_len) {
int best_val = FINGERPRINT_CHUNK_THRESHOLD;
int best_idx = -1;

target_idx = e->s_lno + i;
scan_parent_range(parent->fingerprints,
target->fingerprints, target_idx,
parent_slno, parent_len,
&best_val, &best_idx);
if (best_idx == -1) {
best_val = FINGERPRINT_FILE_THRESHOLD;
scan_parent_range(parent->fingerprints,
target->fingerprints, target_idx,
0, parent->num_lines,
&best_val, &best_idx);
}
if (best_idx >= 0) {
line_blames[i].is_parent = 1;
line_blames[i].s_lno = parent_idx;
line_blames[i].s_lno = best_idx;
} else {
line_blames[i].is_parent = 0;
line_blames[i].s_lno = e->s_lno + i;
line_blames[i].s_lno = target_idx;
}
}
}
Expand Down

0 comments on commit dfb4ee1

Please sign in to comment.