File: wait.c

package info (click to toggle)
libguestfs 1%3A1.44.0-2
  • links: PTS, VCS
  • area: main
  • in suites: bullseye
  • size: 118,932 kB
  • sloc: ansic: 458,017; ml: 51,424; sh: 13,191; java: 9,578; makefile: 7,931; cs: 6,328; haskell: 5,674; python: 3,871; perl: 3,528; erlang: 2,446; xml: 1,347; ruby: 350; pascal: 257; javascript: 157; lex: 135; yacc: 128; cpp: 10
file content (87 lines) | stat: -rw-r--r-- 2,509 bytes parent folder | download | duplicates (6)
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
87
/* libguestfs
 * Copyright (C) 2016 Red Hat Inc.
 *
 * This library is free software; you can redistribute it and/or
 * modify it under the terms of the GNU Lesser General Public
 * License as published by the Free Software Foundation; either
 * version 2 of the License, or (at your option) any later version.
 *
 * This library is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
 * Lesser General Public License for more details.
 *
 * You should have received a copy of the GNU Lesser General Public
 * License along with this library; if not, write to the Free Software
 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
 */

#include <config.h>

#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <errno.h>
#include <sys/types.h>
#include <sys/time.h>
#include <sys/resource.h>
#include <sys/wait.h>

#include "guestfs.h"
#include "guestfs-internal.h"

/**
 * A safe version of L<waitpid(3)> which retries if C<EINTR> is
 * returned.
 *
 * I<Note:> this only needs to be used in the library, or in programs
 * that install a non-restartable C<SIGCHLD> handler (which is not the
 * case for any current libguestfs virt tools).
 *
 * If the main program installs a SIGCHLD handler and sets it to be
 * non-restartable, then what can happen is the library is waiting in
 * a wait syscall, the child exits, C<SIGCHLD> is sent to the process,
 * and the wait syscall returns C<EINTR>.  Since the library cannot
 * control the signal handler, we have to instead restart the wait
 * syscall, which is the purpose of this wrapper.
 */
int
guestfs_int_waitpid (guestfs_h *g, pid_t pid, int *status, const char *errmsg)
{
 again:
  if (waitpid (pid, status, 0) == -1) {
    if (errno == EINTR)
      goto again;
    perrorf (g, "%s: waitpid", errmsg);
    return -1;
  }
  return 0;
}

/**
 * Like C<guestfs_int_waitpid>, but ignore errors.
 */
void
guestfs_int_waitpid_noerror (pid_t pid)
{
  while (waitpid (pid, NULL, 0) == -1 && errno == EINTR)
    ;
}

/**
 * A safe version of L<wait4(2)> which retries if C<EINTR> is
 * returned.
 */
int
guestfs_int_wait4 (guestfs_h *g, pid_t pid, int *status,
                   struct rusage *rusage, const char *errmsg)
{
 again:
  if (wait4 (pid, status, 0, rusage) == -1) {
    if (errno == EINTR)
      goto again;
    perrorf (g, "%s: wait4", errmsg);
    return -1;
  }
  return 0;
}