summaryrefslogtreecommitdiffstats
path: root/Source/cmUVSignalHackRAII.h
blob: 60e4ca84ec409b47ea45ba4b15c8afed7a7ebabd (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
/* Distributed under the OSI-approved BSD 3-Clause License.  See accompanying
   file Copyright.txt or https://cmake.org/licensing for details.  */
#pragma once
#include "cmConfigure.h" // IWYU pragma: keep

#include <cm3p/uv.h>

#if defined(CMAKE_USE_SYSTEM_LIBUV) && !defined(_WIN32) &&                    \
  UV_VERSION_MAJOR == 1 && UV_VERSION_MINOR < 19
#  define CMAKE_UV_SIGNAL_HACK
#  include "cmUVHandlePtr.h"
/*
   libuv does not use SA_RESTART on its signal handler, but C++ streams
   depend on it for reliable i/o operations.  This RAII helper convinces
   libuv to install its handler, and then revises the handler to add the
   SA_RESTART flag.  We use a distinct uv loop that never runs to avoid
   ever really getting a callback.  libuv may fill the hack loop's signal
   pipe and then stop writing, but that won't break any real loops.
 */
class cmUVSignalHackRAII
{
  uv_loop_t HackLoop;
  cm::uv_signal_ptr HackSignal;
  static void HackCB(uv_signal_t*, int) {}

public:
  cmUVSignalHackRAII()
  {
    uv_loop_init(&this->HackLoop);
    this->HackSignal.init(this->HackLoop);
    this->HackSignal.start(HackCB, SIGCHLD);
    struct sigaction hack_sa;
    sigaction(SIGCHLD, nullptr, &hack_sa);
    if (!(hack_sa.sa_flags & SA_RESTART)) {
      hack_sa.sa_flags |= SA_RESTART;
      sigaction(SIGCHLD, &hack_sa, nullptr);
    }
  }
  ~cmUVSignalHackRAII()
  {
    this->HackSignal.stop();
    uv_loop_close(&this->HackLoop);
  }
};
#endif