Skip to content

Commit 0f7ac90

Browse files
committed
Merge branch 'ds/reachable'
Recent update broke the reachability algorithm when refs (e.g. tags) that point at objects that are not commit were involved, which has been fixed. * ds/reachable: commit-reach: fix memory and flag leaks commit-reach: properly peel tags
2 parents faadedb + 4067a64 commit 0f7ac90

File tree

3 files changed

+79
-14
lines changed

3 files changed

+79
-14
lines changed

commit-reach.c

Lines changed: 34 additions & 7 deletions
Original file line numberDiff line numberDiff line change
@@ -547,20 +547,42 @@ int can_all_from_reach_with_flag(struct object_array *from,
547547
{
548548
struct commit **list = NULL;
549549
int i;
550+
int nr_commits;
550551
int result = 1;
551552

552553
ALLOC_ARRAY(list, from->nr);
554+
nr_commits = 0;
553555
for (i = 0; i < from->nr; i++) {
554-
list[i] = (struct commit *)from->objects[i].item;
556+
struct object *from_one = from->objects[i].item;
555557

556-
if (parse_commit(list[i]) ||
557-
list[i]->generation < min_generation)
558-
return 0;
558+
if (!from_one || from_one->flags & assign_flag)
559+
continue;
560+
561+
from_one = deref_tag(the_repository, from_one,
562+
"a from object", 0);
563+
if (!from_one || from_one->type != OBJ_COMMIT) {
564+
/* no way to tell if this is reachable by
565+
* looking at the ancestry chain alone, so
566+
* leave a note to ourselves not to worry about
567+
* this object anymore.
568+
*/
569+
from->objects[i].item->flags |= assign_flag;
570+
continue;
571+
}
572+
573+
list[nr_commits] = (struct commit *)from_one;
574+
if (parse_commit(list[nr_commits]) ||
575+
list[nr_commits]->generation < min_generation) {
576+
result = 0;
577+
goto cleanup;
578+
}
579+
580+
nr_commits++;
559581
}
560582

561-
QSORT(list, from->nr, compare_commits_by_gen);
583+
QSORT(list, nr_commits, compare_commits_by_gen);
562584

563-
for (i = 0; i < from->nr; i++) {
585+
for (i = 0; i < nr_commits; i++) {
564586
/* DFS from list[i] */
565587
struct commit_list *stack = NULL;
566588

@@ -603,10 +625,15 @@ int can_all_from_reach_with_flag(struct object_array *from,
603625
}
604626

605627
cleanup:
606-
for (i = 0; i < from->nr; i++) {
628+
for (i = 0; i < nr_commits; i++) {
607629
clear_commit_marks(list[i], RESULT);
608630
clear_commit_marks(list[i], assign_flag);
609631
}
632+
free(list);
633+
634+
for (i = 0; i < from->nr; i++)
635+
from->objects[i].item->flags &= ~assign_flag;
636+
610637
return result;
611638
}
612639

t/helper/test-reach.c

Lines changed: 17 additions & 5 deletions
Original file line numberDiff line numberDiff line change
@@ -31,6 +31,7 @@ int cmd__reach(int ac, const char **av)
3131
struct object_id oid_A, oid_B;
3232
struct commit *A, *B;
3333
struct commit_list *X, *Y;
34+
struct object_array X_obj = OBJECT_ARRAY_INIT;
3435
struct commit **X_array;
3536
int X_nr, X_alloc;
3637
struct strbuf buf = STRBUF_INIT;
@@ -49,22 +50,23 @@ int cmd__reach(int ac, const char **av)
4950

5051
while (strbuf_getline(&buf, stdin) != EOF) {
5152
struct object_id oid;
52-
struct object *o;
53+
struct object *orig;
54+
struct object *peeled;
5355
struct commit *c;
5456
if (buf.len < 3)
5557
continue;
5658

5759
if (get_oid_committish(buf.buf + 2, &oid))
5860
die("failed to resolve %s", buf.buf + 2);
5961

60-
o = parse_object(r, &oid);
61-
o = deref_tag_noverify(o);
62+
orig = parse_object(r, &oid);
63+
peeled = deref_tag_noverify(orig);
6264

63-
if (!o)
65+
if (!peeled)
6466
die("failed to load commit for input %s resulting in oid %s\n",
6567
buf.buf, oid_to_hex(&oid));
6668

67-
c = object_as_type(r, o, OBJ_COMMIT, 0);
69+
c = object_as_type(r, peeled, OBJ_COMMIT, 0);
6870

6971
if (!c)
7072
die("failed to load commit for input %s resulting in oid %s\n",
@@ -85,6 +87,7 @@ int cmd__reach(int ac, const char **av)
8587
commit_list_insert(c, &X);
8688
ALLOC_GROW(X_array, X_nr + 1, X_alloc);
8789
X_array[X_nr++] = c;
90+
add_object_array(orig, NULL, &X_obj);
8891
break;
8992

9093
case 'Y':
@@ -113,6 +116,15 @@ int cmd__reach(int ac, const char **av)
113116
print_sorted_commit_ids(list);
114117
} else if (!strcmp(av[1], "can_all_from_reach")) {
115118
printf("%s(X,Y):%d\n", av[1], can_all_from_reach(X, Y, 1));
119+
} else if (!strcmp(av[1], "can_all_from_reach_with_flag")) {
120+
struct commit_list *iter = Y;
121+
122+
while (iter) {
123+
iter->item->object.flags |= 2;
124+
iter = iter->next;
125+
}
126+
127+
printf("%s(X,_,_,0,0):%d\n", av[1], can_all_from_reach_with_flag(&X_obj, 2, 4, 0, 0));
116128
} else if (!strcmp(av[1], "commit_contains")) {
117129
struct ref_filter filter;
118130
struct contains_cache cache;

t/t6600-test-reach.sh

Lines changed: 28 additions & 2 deletions
Original file line numberDiff line numberDiff line change
@@ -31,19 +31,22 @@ test_expect_success 'setup' '
3131
for i in $(test_seq 1 10)
3232
do
3333
test_commit "1-$i" &&
34-
git branch -f commit-1-$i
34+
git branch -f commit-1-$i &&
35+
git tag -a -m "1-$i" tag-1-$i commit-1-$i
3536
done &&
3637
for j in $(test_seq 1 9)
3738
do
3839
git reset --hard commit-$j-1 &&
3940
x=$(($j + 1)) &&
4041
test_commit "$x-1" &&
4142
git branch -f commit-$x-1 &&
43+
git tag -a -m "$x-1" tag-$x-1 commit-$x-1 &&
4244
4345
for i in $(test_seq 2 10)
4446
do
4547
git merge commit-$j-$i -m "$x-$i" &&
46-
git branch -f commit-$x-$i
48+
git branch -f commit-$x-$i &&
49+
git tag -a -m "$x-$i" tag-$x-$i commit-$x-$i
4750
done
4851
done &&
4952
git commit-graph write --reachable &&
@@ -205,6 +208,29 @@ test_expect_success 'can_all_from_reach:miss' '
205208
test_three_modes can_all_from_reach
206209
'
207210

211+
test_expect_success 'can_all_from_reach_with_flag: tags case' '
212+
cat >input <<-\EOF &&
213+
X:tag-2-10
214+
X:tag-3-9
215+
X:tag-4-8
216+
X:commit-5-7
217+
X:commit-6-6
218+
X:commit-7-5
219+
X:commit-8-4
220+
X:commit-9-3
221+
Y:tag-1-9
222+
Y:tag-2-8
223+
Y:tag-3-7
224+
Y:commit-4-6
225+
Y:commit-5-5
226+
Y:commit-6-4
227+
Y:commit-7-3
228+
Y:commit-8-1
229+
EOF
230+
echo "can_all_from_reach_with_flag(X,_,_,0,0):1" >expect &&
231+
test_three_modes can_all_from_reach_with_flag
232+
'
233+
208234
test_expect_success 'commit_contains:hit' '
209235
cat >input <<-\EOF &&
210236
A:commit-7-7

0 commit comments

Comments
 (0)