Skip to content

Commit 9a6e4f0

Browse files
peffgitster
authored andcommitted
add_submodule_odb: initialize alt_odb list earlier
The add_submodule_odb function tries to add a submodule's object store as an "alternate". It needs the existing list to be initialized (from the objects/info/alternates file) for two reasons: 1. We look for duplicates with the existing alternate stores, but obviously this doesn't work if we haven't loaded any yet. 2. We link our new entry into the list by prepending it to alt_odb_list. But we do _not_ modify alt_odb_tail. This variable starts as NULL, and is a signal to the alt_odb code that the list has not yet been initialized. We then call read_info_alternates on the submodule (to recursively load its alternates), which will try to append to that tail, assuming it has been initialized. This causes us to segfault if it is NULL. This rarely comes up in practice, because we will have initialized the alt_odb any time we do an object lookup. So you can trigger this only when: - you try to access a submodule (e.g., a diff with diff.submodule=log) - the access happens before any other object has been accessed (e.g., because the diff is between the working tree and the index) - the submodule contains an alternates file (so we try to add an entry to the NULL alt_odb_tail) To fix this, we just need to call prepare_alt_odb at the start of the function (and if we have already initialized, it is a noop). Note that we can remove the prepare_alt_odb call from the end. It is guaranteed to be a noop, since we will have called it earlier. Signed-off-by: Jeff King <[email protected]> Signed-off-by: Junio C Hamano <[email protected]>
1 parent 5e73633 commit 9a6e4f0

File tree

1 file changed

+1
-1
lines changed

1 file changed

+1
-1
lines changed

submodule.c

Lines changed: 1 addition & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -49,6 +49,7 @@ static int add_submodule_odb(const char *path)
4949
goto done;
5050
}
5151
/* avoid adding it twice */
52+
prepare_alt_odb();
5253
for (alt_odb = alt_odb_list; alt_odb; alt_odb = alt_odb->next)
5354
if (alt_odb->name - alt_odb->base == objects_directory.len &&
5455
!strncmp(alt_odb->base, objects_directory.buf,
@@ -66,7 +67,6 @@ static int add_submodule_odb(const char *path)
6667

6768
/* add possible alternates from the submodule */
6869
read_info_alternates(objects_directory.buf, 0);
69-
prepare_alt_odb();
7070
done:
7171
strbuf_release(&objects_directory);
7272
return ret;

0 commit comments

Comments
 (0)