Skip to content

Commit 1e9d794

Browse files
committed
tmp-objdir: skip clean up when handling a signal
In the tmp-objdir api, tmp_objdir_create will create a temporary directory but also register signal handlers responsible for removing the directory's contents and the directory itself. However, the function responsible for recursively removing the contents and directory, remove_dir_recurse() calls opendir(3) and closedir(3). This can be problematic because these functions allocate and free memory, which are not async-signal-safe functions. This can lead to deadlocks. One place we call tmp_objdir_create() is in git-receive-pack, where we create a temporary quarantine directory "incoming". Incoming objects will be written to this directory before they get moved to the object directory. We have observed this code leading to a deadlock: Thread 1 (Thread 0x7f621ba0b200 (LWP 326305)): #0 __lll_lock_wait_private (futex=futex@entry=0x7f621bbf8b80 <main_arena>) at ./lowlevellock.c:35 #1 0x00007f621baa635b in __GI___libc_malloc (bytes=bytes@entry=32816) at malloc.c:3064 #2 0x00007f621bae9f49 in __alloc_dir (statp=0x7fff2ea7ed60, flags=0, close_fd=true, fd=5) at ../sysdeps/posix/opendir.c:118 #3 opendir_tail (fd=5) at ../sysdeps/posix/opendir.c:69 #4 __opendir (name=<optimized out>) at ../sysdeps/posix/opendir.c:92 #5 0x0000557c19c77de1 in remove_dir_recurse () git#6 0x0000557c19d81a4f in remove_tmp_objdir_on_signal () #7 <signal handler called> git#8 _int_malloc (av=av@entry=0x7f621bbf8b80 <main_arena>, bytes=bytes@entry=7160) at malloc.c:4116 git#9 0x00007f621baa62c9 in __GI___libc_malloc (bytes=7160) at malloc.c:3066 git#10 0x00007f621bd1e987 in inflateInit2_ () from /opt/gitlab/embedded/lib/libz.so.1 git#11 0x0000557c19dbe5f4 in git_inflate_init () git#12 0x0000557c19cee02a in unpack_compressed_entry () git#13 0x0000557c19cf08cb in unpack_entry () git#14 0x0000557c19cf0f32 in packed_object_info () git#15 0x0000557c19cd68cd in do_oid_object_info_extended () git#16 0x0000557c19cd6e2b in read_object_file_extended () git#17 0x0000557c19cdec2f in parse_object () git#18 0x0000557c19c34977 in lookup_commit_reference_gently () git#19 0x0000557c19d69309 in mark_uninteresting () git#20 0x0000557c19d2d180 in do_for_each_repo_ref_iterator () git#21 0x0000557c19d21678 in for_each_ref () git#22 0x0000557c19d6a94f in assign_shallow_commits_to_refs () git#23 0x0000557c19bc02b2 in cmd_receive_pack () git#24 0x0000557c19b29fdd in handle_builtin () git#25 0x0000557c19b2a526 in cmd_main () git#26 0x0000557c19b28ea2 in main () Since we can't do the cleanup in a portable and signal-safe way, skip the cleanup when we're handling a signal. This means that when signal handling, the temporary directory may not get cleaned up properly. This is mitigated by b3cecf4 (tmp-objdir: new API for creating temporary writable databases, 2021-12-06) which changed the default name and allows gc to clean up these temporary directories. Signed-off-by: John Cai <[email protected]>
1 parent 4fd6c5e commit 1e9d794

File tree

1 file changed

+4
-13
lines changed

1 file changed

+4
-13
lines changed

tmp-objdir.c

Lines changed: 4 additions & 13 deletions
Original file line numberDiff line numberDiff line change
@@ -31,7 +31,7 @@ static void tmp_objdir_free(struct tmp_objdir *t)
3131
free(t);
3232
}
3333

34-
static int tmp_objdir_destroy_1(struct tmp_objdir *t, int on_signal)
34+
static int tmp_objdir_destroy_1(struct tmp_objdir *t)
3535
{
3636
int err;
3737

@@ -41,7 +41,7 @@ static int tmp_objdir_destroy_1(struct tmp_objdir *t, int on_signal)
4141
if (t == the_tmp_objdir)
4242
the_tmp_objdir = NULL;
4343

44-
if (!on_signal && t->prev_odb)
44+
if (t->prev_odb)
4545
restore_primary_odb(t->prev_odb, t->path.buf);
4646

4747
/*
@@ -56,29 +56,21 @@ static int tmp_objdir_destroy_1(struct tmp_objdir *t, int on_signal)
5656
* freeing memory; it may cause a deadlock if the signal
5757
* arrived while libc's allocator lock is held.
5858
*/
59-
if (!on_signal)
60-
tmp_objdir_free(t);
59+
tmp_objdir_free(t);
6160

6261
return err;
6362
}
6463

6564
int tmp_objdir_destroy(struct tmp_objdir *t)
6665
{
67-
return tmp_objdir_destroy_1(t, 0);
66+
return tmp_objdir_destroy_1(t);
6867
}
6968

7069
static void remove_tmp_objdir(void)
7170
{
7271
tmp_objdir_destroy(the_tmp_objdir);
7372
}
7473

75-
static void remove_tmp_objdir_on_signal(int signo)
76-
{
77-
tmp_objdir_destroy_1(the_tmp_objdir, 1);
78-
sigchain_pop(signo);
79-
raise(signo);
80-
}
81-
8274
void tmp_objdir_discard_objects(struct tmp_objdir *t)
8375
{
8476
remove_dir_recursively(&t->path, REMOVE_DIR_KEEP_TOPLEVEL);
@@ -169,7 +161,6 @@ struct tmp_objdir *tmp_objdir_create(const char *prefix)
169161
the_tmp_objdir = t;
170162
if (!installed_handlers) {
171163
atexit(remove_tmp_objdir);
172-
sigchain_push_common(remove_tmp_objdir_on_signal);
173164
installed_handlers++;
174165
}
175166

0 commit comments

Comments
 (0)