From 8a877d990b6cb8c6e45c247323c2251e1d45488e Mon Sep 17 00:00:00 2001 From: Max Kellermann Date: Wed, 11 Sep 2024 12:51:19 +0200 Subject: [PATCH] LyricsPage: use posix_spawn() to launch the editor This is simpler than fork()/exec and may be more portable. --- src/LyricsPage.cxx | 37 +++++++++++++++++++++++-------------- 1 file changed, 23 insertions(+), 14 deletions(-) diff --git a/src/LyricsPage.cxx b/src/LyricsPage.cxx index ff623017..c91fcabd 100644 --- a/src/LyricsPage.cxx +++ b/src/LyricsPage.cxx @@ -20,6 +20,7 @@ #include #include +#include #include #include #include @@ -348,25 +349,33 @@ LyricsPage::Edit() noexcept def_prog_mode(); endwin(); - /* TODO: fork/exec/wait won't work on Windows, but building a command - string for system() is too tricky */ - int status; - pid_t pid = fork(); - if (pid == -1) { + posix_spawnattr_t attr; + posix_spawnattr_init(&attr); + +#ifdef USE_SIGNALFD + /* unblock all signals which may be blocked for signalfd */ + posix_spawnattr_setflags(&attr, POSIX_SPAWN_SETSIGMASK); +#endif + + char *const argv[] = { + const_cast(editor), + const_cast(path.c_str()), + nullptr + }; + + pid_t pid; + if (posix_spawn(&pid, editor, nullptr, &attr, + argv, environ) != 0) { reset_prog_mode(); FmtAlert("{} ({})"sv, _("Can't start editor"), strerror(errno)); return; - } else if (pid == 0) { - execlp(editor, editor, path.c_str(), nullptr); - /* exec failed, do what system does */ - _exit(127); - } else { - int ret; - do { - ret = waitpid(pid, &status, 0); - } while (ret == -1 && errno == EINTR); } + int ret, status; + do { + ret = waitpid(pid, &status, 0); + } while (ret == -1 && errno == EINTR); + reset_prog_mode(); /* TODO: hardly portable */