Before this patch the signal handler called `logmsg()` which in turn
called `printf()` variants (internal implementations), and `FILE *`
functions, `localtime()`. Some of these called `malloc`/`free`, which
isn't supported in s signal handler. Replace them with `write` calls,
losing some logging functionality.
Also:
- De-dupe and move `STD*_FILENO` macros to `lib/curl_setup.h`. Revert
the `src` definition to point to `stderr`, instead of `tool_stderr`.
Follow-up to
e5bb88b8f824ed87620bd923552534c83c2a516e #11958
POSIX specs with list of functions allowed in a signal handler:
2004: https://pubs.opengroup.org/onlinepubs/
009695399/functions/xsh_chap02_04.html#tag_02_04_03
2017: https://pubs.opengroup.org/onlinepubs/
9699919799/functions/V2_chap02.html#tag_15_04_03
2024: https://pubs.opengroup.org/onlinepubs/
9799919799/functions/V2_chap02.html#tag_16_04_03
Linux CI run with the thread sanitizer going crazy when
hitting the signal handler in test 1238 and 1242 (TFTP):
```
WARNING: ThreadSanitizer: signal-unsafe call inside of a signal (pid=12582)
#0 malloc <null> (servers+0x5ed70)
#1 _IO_file_doallocate <null> (libc.so.6+0x851b4)
#2 formatf /home/runner/work/curl/curl/bld/tests/server/../../lib/../../lib/mprintf.c:886:9 (servers+0xdff77)
[...]
WARNING: ThreadSanitizer: signal-unsafe call inside of a signal (pid=12582)
#0 free <null> (servers+0x5f453)
#1 fclose <null> (libc.so.6+0x8532f)
#2 logmsg /home/runner/work/curl/curl/bld/tests/server/../../../tests/server/util.c:134:5 (servers+0xe684d)
```
Ref: https://github.com/curl/curl/actions/runs/
14118903372/job/
39555309490?pr=16851
Closes #16852
#define EINVAL 22
#define ENOSPC 28
#define strerror(x) "?"
+#undef STDIN_FILENO
+#define STDIN_FILENO 0
#else
#define CURL_SETERRNO(x) (errno = (x))
#endif
#define S_ISDIR(m) (((m) & S_IFMT) == S_IFDIR)
#endif
+/* For MSVC (all versions as of VS2022) */
+#ifndef STDIN_FILENO
+#define STDIN_FILENO fileno(stdin)
+#endif
+#ifndef STDOUT_FILENO
+#define STDOUT_FILENO fileno(stdout)
+#endif
+#ifndef STDERR_FILENO
+#define STDERR_FILENO fileno(stderr)
+#endif
+
/* Since O_BINARY is used in bitmasks, setting it to zero makes it usable in
source code but yet it does not ruin anything */
#ifdef O_BINARY
#define MAX_PARALLEL 300 /* conservative */
#define PARALLEL_DEFAULT 50
-#ifndef STDIN_FILENO
-# define STDIN_FILENO fileno(stdin)
-#endif
-
-#ifndef STDOUT_FILENO
-# define STDOUT_FILENO fileno(stdout)
-#endif
-
-#ifndef STDERR_FILENO
-# define STDERR_FILENO fileno(tool_stderr)
-#endif
-
#endif /* HEADER_CURL_TOOL_MAIN_H */
# define _get_osfhandle(fd) (fd)
# undef _getch
# define _getch() 0
-# undef STDIN_FILENO
-# define STDIN_FILENO 0
#endif
#ifndef HAVE_FTRUNCATE
#include "warnless.h"
#include "memdebug.h"
-/* For Windows, mainly (may be moved in a config file?) */
-#ifndef STDIN_FILENO
-#define STDIN_FILENO 0
-#endif
-#ifndef STDOUT_FILENO
-#define STDOUT_FILENO 1
-#endif
-#ifndef STDERR_FILENO
-#define STDERR_FILENO 2
-#endif
-
CURLcode test(char *URL)
{
CURLcode res;
* store in exit_signal the signal that triggered its execution.
*/
#ifndef UNDER_CE
+/*
+ * Only call signal-safe functions from the signal handler, as required by
+ * the POSIX specification:
+ * https://pubs.opengroup.org/onlinepubs/9699919799/functions/V2_chap02.html
+ * Hence, do not call 'logmsg()', and instead use 'open/write/close' to
+ * log errors.
+ */
static void exit_signal_handler(int signum)
{
int old_errno = errno;
- logmsg("exit_signal_handler (%d)", signum);
+ if(!serverlogfile) {
+ static const char msg[] = "exit_signal_handler: serverlogfile not set\n";
+ (void)!write(STDERR_FILENO, msg, sizeof(msg) - 1);
+ }
+ else {
+#ifdef _WIN32
+#define OPENMODE S_IREAD | S_IWRITE
+#else
+#define OPENMODE S_IRUSR | S_IWUSR
+#endif
+ int fd = open(serverlogfile, O_WRONLY|O_CREAT|O_APPEND, OPENMODE);
+ if(fd != -1) {
+ static const char msg[] = "exit_signal_handler: called\n";
+ (void)!write(fd, msg, sizeof(msg) - 1);
+ close(fd);
+ }
+ else {
+ static const char msg[] = "exit_signal_handler: failed opening ";
+ (void)!write(STDERR_FILENO, msg, sizeof(msg) - 1);
+ (void)!write(STDERR_FILENO, serverlogfile, strlen(serverlogfile));
+ (void)!write(STDERR_FILENO, "\n", 1);
+ }
+ }
if(got_exit_signal == 0) {
got_exit_signal = 1;
exit_signal = signum;
if(hwnd == hidden_main_window) {
switch(uMsg) {
#ifdef SIGTERM
- case WM_CLOSE:
- signum = SIGTERM;
- break;
+ case WM_CLOSE:
+ signum = SIGTERM;
+ break;
#endif
case WM_DESTROY:
PostQuitMessage(0);