plugins: Wire up rust plugin support for NBD_INFO_INIT_STATE
[nbdkit/ericb.git] / server / socket-activation.c
blob6f3b33dd649d140a786f134d63e3e23b297fe723
1 /* nbdkit
2 * Copyright (C) 2019 Red Hat Inc.
4 * Redistribution and use in source and binary forms, with or without
5 * modification, are permitted provided that the following conditions are
6 * met:
8 * * Redistributions of source code must retain the above copyright
9 * notice, this list of conditions and the following disclaimer.
11 * * Redistributions in binary form must reproduce the above copyright
12 * notice, this list of conditions and the following disclaimer in the
13 * documentation and/or other materials provided with the distribution.
15 * * Neither the name of Red Hat nor the names of its contributors may be
16 * used to endorse or promote products derived from this software without
17 * specific prior written permission.
19 * THIS SOFTWARE IS PROVIDED BY RED HAT AND CONTRIBUTORS ''AS IS'' AND
20 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO,
21 * THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A
22 * PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL RED HAT OR
23 * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
24 * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
25 * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF
26 * USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
27 * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
28 * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT
29 * OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
30 * SUCH DAMAGE.
33 #include <config.h>
35 #include <stdio.h>
36 #include <stdlib.h>
37 #include <stdarg.h>
38 #include <string.h>
39 #include <unistd.h>
40 #include <fcntl.h>
41 #include <errno.h>
43 #include "internal.h"
45 /* Handle socket activation. This is controlled through special
46 * environment variables inherited by nbdkit. Returns 0 if no socket
47 * activation. Otherwise returns the number of FDs. See also
48 * virGetListenFDs in libvirt.org:src/util/virutil.c
50 unsigned int
51 get_socket_activation (void)
53 const char *s;
54 unsigned int pid;
55 unsigned int nr_fds;
56 unsigned int i;
57 int fd;
59 s = getenv ("LISTEN_PID");
60 if (s == NULL)
61 return 0;
62 if (nbdkit_parse_unsigned ("LISTEN_PID", s, &pid) == -1)
63 return 0;
64 if (pid != getpid ()) {
65 fprintf (stderr, "%s: %s was not for us (ignored)\n",
66 program_name, "LISTEN_PID");
67 return 0;
70 s = getenv ("LISTEN_FDS");
71 if (s == NULL)
72 return 0;
73 if (nbdkit_parse_unsigned ("LISTEN_FDS", s, &nr_fds) == -1)
74 return 0;
76 /* Limit the number of fds that may be passed in to something
77 * reasonable.
79 if (nr_fds == 0 || nr_fds > 16) {
80 fprintf (stderr, "%s: socket activation: LISTEN_FDS=%s out of range\n",
81 program_name, s);
82 exit (EXIT_FAILURE);
85 /* So these are not passed to any child processes we might start. */
86 unsetenv ("LISTEN_FDS");
87 unsetenv ("LISTEN_PID");
89 /* So the file descriptors don't leak into child processes. */
90 for (i = 0; i < nr_fds; ++i) {
91 fd = FIRST_SOCKET_ACTIVATION_FD + i;
92 if (fcntl (fd, F_SETFD, FD_CLOEXEC) == -1) {
93 /* If we cannot set FD_CLOEXEC then it probably means the file
94 * descriptor is invalid, so socket activation has gone wrong
95 * and we should exit.
97 fprintf (stderr, "%s: socket activation: "
98 "invalid file descriptor fd = %d: %s\n",
99 program_name, fd, strerror(errno));
100 exit (EXIT_FAILURE);
104 return nr_fds;