Skip to content

Commit 10b6ee1

Browse files
danielhbbonzini
authored andcommitted
vl.c: do not execute trace_init_backends() before daemonizing
Commit v5.2.0-190-g0546c0609c ("vl: split various early command line options to a separate function") moved the trace backend init code to the qemu_process_early_options(). Which is now being called before os_daemonize() via qemu_maybe_daemonize(). Turns out that this change of order causes a problem when executing QEMU in daemon mode and with CONFIG_TRACE_SIMPLE. The trace thread is now being created by the parent, and the parent is left waiting for a trace file flush that was registered via st_init(). The result is that the parent process never exits. To reproduce, fire up a QEMU process with -daemonize and with CONFIG_TRACE_SIMPLE enabled. Two QEMU process will be left in the host: $ sudo ./x86_64-softmmu/qemu-system-x86_64 -S -no-user-config -nodefaults \ -nographic -machine none,accel=kvm:tcg -daemonize $ ps axf | grep qemu 529710 pts/3 S+ 0:00 | \_ grep --color=auto qemu 529697 ? Ssl 0:00 \_ ./x86_64-softmmu/qemu-system-x86_64 -S -no-user-config -nodefaults -nographic -machine none,accel=kvm:tcg -daemonize 529699 ? Sl 0:00 \_ ./x86_64-softmmu/qemu-system-x86_64 -S -no-user-config -nodefaults -nographic -machine none,accel=kvm:tcg -daemonize The parent thread is hang in flush_trace_file: $ sudo gdb ./x86_64-softmmu/qemu-system-x86_64 529697 (..) (gdb) bt #0 0x00007f9dac6a137d in syscall () at /lib64/libc.so.6 #1 0x00007f9dacc3c4f3 in g_cond_wait () at /lib64/libglib-2.0.so.0 #2 0x0000555d12f952da in flush_trace_file (wait=true) at ../trace/simple.c:140 #3 0x0000555d12f95b4c in st_flush_trace_buffer () at ../trace/simple.c:383 #4 0x00007f9dac5e43a7 in __run_exit_handlers () at /lib64/libc.so.6 #5 0x00007f9dac5e4550 in on_exit () at /lib64/libc.so.6 #6 0x0000555d12d454de in os_daemonize () at ../os-posix.c:255 #7 0x0000555d12d0bd5c in qemu_maybe_daemonize (pid_file=0x0) at ../softmmu/vl.c:2408 #8 0x0000555d12d0e566 in qemu_init (argc=8, argv=0x7fffc594d9b8, envp=0x7fffc594da00) at ../softmmu/vl.c:3459 #9 0x0000555d128edac1 in main (argc=8, argv=0x7fffc594d9b8, envp=0x7fffc594da00) at ../softmmu/main.c:49 (gdb) Aside from the 'zombie' process in the host, this is directly impacting Libvirt. Libvirt waits for the parent process to exit to be sure that the QMP monitor is available in the daemonized process to fetch QEMU capabilities, and as is now Libvirt hangs at daemon start waiting for the parent thread to exit. The fix is simple: just move the trace backend related code back to be executed after daemonizing. Fixes: 0546c06 Reviewed-by: Paolo Bonzini <[email protected]> Signed-off-by: Daniel Henrique Barboza <[email protected]> Message-Id: <[email protected]> Acked-by: Stefan Hajnoczi <[email protected]> Signed-off-by: Paolo Bonzini <[email protected]>
1 parent 0bd5a2e commit 10b6ee1

File tree

1 file changed

+13
-5
lines changed

1 file changed

+13
-5
lines changed

softmmu/vl.c

+13-5
Original file line numberDiff line numberDiff line change
@@ -2361,11 +2361,6 @@ static void qemu_process_early_options(void)
23612361
cleanup_add_fd, NULL, &error_fatal);
23622362
#endif
23632363

2364-
if (!trace_init_backends()) {
2365-
exit(1);
2366-
}
2367-
trace_init_file();
2368-
23692364
/* Open the logfile at this point and set the log mask if necessary. */
23702365
qemu_set_log_filename(log_file, &error_fatal);
23712366
if (log_mask) {
@@ -3475,6 +3470,19 @@ void qemu_init(int argc, char **argv, char **envp)
34753470
qemu_process_help_options();
34763471
qemu_maybe_daemonize(pid_file);
34773472

3473+
/*
3474+
* The trace backend must be initialized after daemonizing.
3475+
* trace_init_backends() will call st_init(), which will create the
3476+
* trace thread in the parent, and also register st_flush_trace_buffer()
3477+
* in atexit(). This function will force the parent to wait for the
3478+
* writeout thread to finish, which will not occur, and the parent
3479+
* process will be left in the host.
3480+
*/
3481+
if (!trace_init_backends()) {
3482+
exit(1);
3483+
}
3484+
trace_init_file();
3485+
34783486
qemu_init_main_loop(&error_fatal);
34793487
cpu_timers_init();
34803488

0 commit comments

Comments
 (0)