summaryrefslogtreecommitdiff
path: root/src/daemontools-extras/s6-notifywhenup.c
blob: 61388df6ed4876d3832c5284ceaf8a81e69f21a3 (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
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
/* ISC license. */

#include <unistd.h>
#include <errno.h>
#include <skalibs/uint.h>
#include <skalibs/bytestr.h>
#include <skalibs/sgetopt.h>
#include <skalibs/strerr2.h>
#include <skalibs/allreadwrite.h>
#include <skalibs/tai.h>
#include <skalibs/iopause.h>
#include <skalibs/djbunix.h>
#include <s6/ftrigw.h>

#define USAGE "s6-notifywhenup [ -d fd ] [ -e fifodir ] [ -f ] [ -t timeout ] prog..."
#define dieusage() strerr_dieusage(100, USAGE)

static int run_child (int fd, char const *fifodir, unsigned int timeout)
{
  char dummy[4096] ;
  iopause_fd x = { .fd = fd, .events = IOPAUSE_READ } ;
  tain_t deadline ;
  if (!tain_now_g()) strerr_diefu1sys(111, "tain_now") ;
  if (timeout) tain_from_millisecs(&deadline, timeout) ;
  else deadline = tain_infinite_relative ;
  tain_add_g(&deadline, &deadline) ;
  for (;;)
  {
    register int r = iopause_g(&x, 1, &deadline) ;
    if (r < 0) strerr_diefu1sys(111, "iopause") ;
    if (!r) return 99 ;
    r = sanitize_read(fd_read(fd, dummy, 4096)) ;
    if (r < 0)
      if (errno == EPIPE) return 1 ;
      else strerr_diefu1sys(111, "read from parent") ;
    else if (r)
      if (byte_chr(dummy, r, '\n') < r) break ;
  }
  ftrigw_notify(fifodir, 'U') ;
  return 0 ;
}

int main (int argc, char const *const *argv, char const *const *envp)
{
  unsigned int fd = 1 ;
  char const *fifodir = "event" ;
  int df = 1 ;
  unsigned int timeout = 0 ;
  PROG = "s6-notifywhenup" ;
  {
    subgetopt_t l = SUBGETOPT_ZERO ;
    for (;;)
    {
      register int opt = subgetopt_r(argc, argv, "d:e:ft:", &l) ;
      if (opt == -1) break ;
      switch (opt)
      {
        case 'd' : if (!uint0_scan(l.arg, &fd)) dieusage() ; break ;
        case 'e' : fifodir = l.arg ; break ;
        case 'f' : df = 0 ; break ;
        case 't' : if (!uint0_scan(l.arg, &timeout)) dieusage() ; break ;
        default : dieusage() ;
      }
    }
    argc -= l.ind ; argv += l.ind ;
  }
  if (!argc) dieusage() ;

  {
    int p[2] ;
    pid_t pid ;
    if (pipe(p) < 0) strerr_diefu1sys(111, "pipe") ;
    pid = df ? doublefork() : fork() ;
    if (pid < 0) strerr_diefu1sys(111, df ? "doublefork" : "fork") ;
    else if (!pid)
    {
      PROG = "s6-notifywhenup (child)" ;
      fd_close(p[1]) ;
      return run_child(p[0], fifodir, timeout) ;
    }
    fd_close(p[0]) ;
    if (fd_move((int)fd, p[1]) < 0) strerr_diefu1sys(111, "fd_move") ;
  }
  pathexec_run(argv[0], argv, envp) ;
  strerr_dieexec(111, argv[1]) ;
}