зеркало из https://github.com/microsoft/git.git
Merge branch 'lt/diff-rename'
* lt/diff-rename: optimize diffcore-delta by sorting hash entries.
This commit is contained in:
Коммит
c5d236c70e
|
@ -46,22 +46,6 @@ struct spanhash_top {
|
||||||
struct spanhash data[FLEX_ARRAY];
|
struct spanhash data[FLEX_ARRAY];
|
||||||
};
|
};
|
||||||
|
|
||||||
static struct spanhash *spanhash_find(struct spanhash_top *top,
|
|
||||||
unsigned int hashval)
|
|
||||||
{
|
|
||||||
int sz = 1 << top->alloc_log2;
|
|
||||||
int bucket = hashval & (sz - 1);
|
|
||||||
while (1) {
|
|
||||||
struct spanhash *h = &(top->data[bucket++]);
|
|
||||||
if (!h->cnt)
|
|
||||||
return NULL;
|
|
||||||
if (h->hashval == hashval)
|
|
||||||
return h;
|
|
||||||
if (sz <= bucket)
|
|
||||||
bucket = 0;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
static struct spanhash_top *spanhash_rehash(struct spanhash_top *orig)
|
static struct spanhash_top *spanhash_rehash(struct spanhash_top *orig)
|
||||||
{
|
{
|
||||||
struct spanhash_top *new;
|
struct spanhash_top *new;
|
||||||
|
@ -122,6 +106,20 @@ static struct spanhash_top *add_spanhash(struct spanhash_top *top,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static int spanhash_cmp(const void *a_, const void *b_)
|
||||||
|
{
|
||||||
|
const struct spanhash *a = a_;
|
||||||
|
const struct spanhash *b = b_;
|
||||||
|
|
||||||
|
/* A count of zero compares at the end.. */
|
||||||
|
if (!a->cnt)
|
||||||
|
return !b->cnt ? 0 : 1;
|
||||||
|
if (!b->cnt)
|
||||||
|
return -1;
|
||||||
|
return a->hashval < b->hashval ? -1 :
|
||||||
|
a->hashval > b->hashval ? 1 : 0;
|
||||||
|
}
|
||||||
|
|
||||||
static struct spanhash_top *hash_chars(struct diff_filespec *one)
|
static struct spanhash_top *hash_chars(struct diff_filespec *one)
|
||||||
{
|
{
|
||||||
int i, n;
|
int i, n;
|
||||||
|
@ -158,6 +156,10 @@ static struct spanhash_top *hash_chars(struct diff_filespec *one)
|
||||||
n = 0;
|
n = 0;
|
||||||
accum1 = accum2 = 0;
|
accum1 = accum2 = 0;
|
||||||
}
|
}
|
||||||
|
qsort(hash->data,
|
||||||
|
1ul << hash->alloc_log2,
|
||||||
|
sizeof(hash->data[0]),
|
||||||
|
spanhash_cmp);
|
||||||
return hash;
|
return hash;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -169,7 +171,7 @@ int diffcore_count_changes(struct diff_filespec *src,
|
||||||
unsigned long *src_copied,
|
unsigned long *src_copied,
|
||||||
unsigned long *literal_added)
|
unsigned long *literal_added)
|
||||||
{
|
{
|
||||||
int i, ssz;
|
struct spanhash *s, *d;
|
||||||
struct spanhash_top *src_count, *dst_count;
|
struct spanhash_top *src_count, *dst_count;
|
||||||
unsigned long sc, la;
|
unsigned long sc, la;
|
||||||
|
|
||||||
|
@ -190,22 +192,26 @@ int diffcore_count_changes(struct diff_filespec *src,
|
||||||
}
|
}
|
||||||
sc = la = 0;
|
sc = la = 0;
|
||||||
|
|
||||||
ssz = 1 << src_count->alloc_log2;
|
s = src_count->data;
|
||||||
for (i = 0; i < ssz; i++) {
|
d = dst_count->data;
|
||||||
struct spanhash *s = &(src_count->data[i]);
|
for (;;) {
|
||||||
struct spanhash *d;
|
|
||||||
unsigned dst_cnt, src_cnt;
|
unsigned dst_cnt, src_cnt;
|
||||||
if (!s->cnt)
|
if (!s->cnt)
|
||||||
continue;
|
break; /* we checked all in src */
|
||||||
|
while (d->cnt) {
|
||||||
|
if (d->hashval >= s->hashval)
|
||||||
|
break;
|
||||||
|
d++;
|
||||||
|
}
|
||||||
src_cnt = s->cnt;
|
src_cnt = s->cnt;
|
||||||
d = spanhash_find(dst_count, s->hashval);
|
dst_cnt = d->hashval == s->hashval ? d->cnt : 0;
|
||||||
dst_cnt = d ? d->cnt : 0;
|
|
||||||
if (src_cnt < dst_cnt) {
|
if (src_cnt < dst_cnt) {
|
||||||
la += dst_cnt - src_cnt;
|
la += dst_cnt - src_cnt;
|
||||||
sc += src_cnt;
|
sc += src_cnt;
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
sc += dst_cnt;
|
sc += dst_cnt;
|
||||||
|
s++;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!src_count_p)
|
if (!src_count_p)
|
||||||
|
|
Загрузка…
Ссылка в новой задаче