X-Git-Url: https://mattmccutchen.net/rsync/rsync.git/blobdiff_plain/4440b8aa3fb0edb9626c8eaa4856d72f78c9e176..1347d5126a4d89718544d976f64943dfa93dd8c4:/match.c diff --git a/match.c b/match.c index 29b52516..7f3212c8 100644 --- a/match.c +++ b/match.c @@ -130,12 +130,16 @@ static void hash_search(int f,struct sum_struct *s, struct map_struct *buf,OFF_T len) { OFF_T offset; - int j,k; + int j,k, last_i; int end; char sum2[SUM_LENGTH]; uint32 s1, s2, sum; schar *map; + /* last_i is used to encourage adjacent matches, allowing the RLL coding of the + output to work more efficiently */ + last_i = -1; + if (verbose > 2) rprintf(FINFO,"hash search b=%d len=%d\n",s->n,(int)len); @@ -172,16 +176,19 @@ static void hash_search(int f,struct sum_struct *s, sum = (s1 & 0xffff) | (s2 << 16); tag_hits++; for (; jcount && targets[j].t == t; j++) { - int i = targets[j].i; + int l, i = targets[j].i; if (sum != s->sums[i].sum1) continue; + /* also make sure the two blocks are the same length */ + l = MIN(s->n,len-offset); + if (l != s->sums[i].len) continue; + if (verbose > 3) rprintf(FINFO,"potential match at %d target=%d %d sum=%08x\n", (int)offset,j,i,sum); if (!done_csum2) { - int l = MIN(s->n,len-offset); map = (schar *)map_ptr(buf,offset,l); get_checksum2((char *)map,l,sum2); done_csum2 = 1; @@ -191,6 +198,22 @@ static void hash_search(int f,struct sum_struct *s, false_alarms++; continue; } + + /* we've found a match, but now check to see + if last_i can hint at a better match */ + for (j++; jcount && targets[j].t == t; j++) { + int i2 = targets[j].i; + if (i2 == last_i + 1) { + if (sum != s->sums[i2].sum1) break; + if (memcmp(sum2,s->sums[i2].sum2,csum_length) != 0) break; + /* we've found an adjacent match - the RLL coder + will be happy */ + i = i2; + break; + } + } + + last_i = i; matched(f,s,buf,offset,i); offset += s->sums[i].len - 1;