1 /*-*- Mode: C; c-basic-offset: 8; indent-tabs-mode: nil -*-*/
4 Copyright 2010 Lennart Poettering
6 Permission is hereby granted, free of charge, to any person
7 obtaining a copy of this software and associated documentation files
8 (the "Software"), to deal in the Software without restriction,
9 including without limitation the rights to use, copy, modify, merge,
10 publish, distribute, sublicense, and/or sell copies of the Software,
11 and to permit persons to whom the Software is furnished to do so,
12 subject to the following conditions:
14 The above copyright notice and this permission notice shall be
15 included in all copies or substantial portions of the Software.
17 THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
18 EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
19 MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
20 NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS
21 BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN
22 ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
23 CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
31 #include <sys/types.h>
33 #include <sys/socket.h>
35 #include <sys/fcntl.h>
36 #include <netinet/in.h>
44 #include "sd-daemon.h"
46 int sd_listen_fds(int unset_environment) {
48 #if defined(DISABLE_SYSTEMD) || !defined(__linux__)
56 if (!(e = getenv("LISTEN_PID"))) {
62 l = strtoul(e, &p, 10);
69 if (!p || *p || l <= 0) {
75 if (getpid() != (pid_t) l) {
80 if (!(e = getenv("LISTEN_FDS"))) {
86 l = strtoul(e, &p, 10);
98 for (fd = SD_LISTEN_FDS_START; fd < SD_LISTEN_FDS_START + (int) l; fd ++) {
101 if ((flags = fcntl(fd, F_GETFD)) < 0) {
106 if (flags & FD_CLOEXEC)
109 if (fcntl(fd, F_SETFD, flags | FD_CLOEXEC) < 0) {
118 if (unset_environment) {
119 unsetenv("LISTEN_PID");
120 unsetenv("LISTEN_FDS");
127 int sd_is_fifo(int fd, const char *path) {
133 memset(&st_fd, 0, sizeof(st_fd));
134 if (fstat(fd, &st_fd) < 0)
137 if (!S_ISFIFO(st_fd.st_mode))
143 memset(&st_path, 0, sizeof(st_path));
144 if (stat(path, &st_path) < 0) {
146 if (errno == ENOENT || errno == ENOTDIR)
153 st_path.st_dev == st_fd.st_dev &&
154 st_path.st_ino == st_fd.st_ino;
160 static int sd_is_socket_internal(int fd, int type, int listening) {
163 if (fd < 0 || type < 0)
166 if (fstat(fd, &st_fd) < 0)
169 if (!S_ISSOCK(st_fd.st_mode))
174 socklen_t l = sizeof(other_type);
176 if (getsockopt(fd, SOL_SOCKET, SO_TYPE, &other_type, &l) < 0)
179 if (l != sizeof(other_type))
182 if (other_type != type)
186 if (listening >= 0) {
188 socklen_t l = sizeof(accepting);
190 if (getsockopt(fd, SOL_SOCKET, SO_ACCEPTCONN, &accepting, &l) < 0)
193 if (l != sizeof(accepting))
196 if (!accepting != !listening)
203 union sockaddr_union {
205 struct sockaddr_in in4;
206 struct sockaddr_in6 in6;
207 struct sockaddr_un un;
208 struct sockaddr_storage storage;
211 int sd_is_socket(int fd, int family, int type, int listening) {
217 if ((r = sd_is_socket_internal(fd, type, listening)) <= 0)
221 union sockaddr_union sockaddr;
224 memset(&sockaddr, 0, sizeof(sockaddr));
225 l = sizeof(sockaddr);
227 if (getsockname(fd, &sockaddr.sa, &l) < 0)
230 if (l < sizeof(sa_family_t))
233 return sockaddr.sa.sa_family == family;
239 int sd_is_socket_inet(int fd, int family, int type, int listening, uint16_t port) {
240 union sockaddr_union sockaddr;
244 if (family != 0 && family != AF_INET && family != AF_INET6)
247 if ((r = sd_is_socket_internal(fd, type, listening)) <= 0)
250 memset(&sockaddr, 0, sizeof(sockaddr));
251 l = sizeof(sockaddr);
253 if (getsockname(fd, &sockaddr.sa, &l) < 0)
256 if (l < sizeof(sa_family_t))
259 if (sockaddr.sa.sa_family != AF_INET &&
260 sockaddr.sa.sa_family != AF_INET6)
264 if (sockaddr.sa.sa_family != family)
268 if (sockaddr.sa.sa_family == AF_INET) {
269 if (l < sizeof(struct sockaddr_in))
272 return htons(port) == sockaddr.in4.sin_port;
274 if (l < sizeof(struct sockaddr_in6))
277 return htons(port) == sockaddr.in6.sin6_port;
284 int sd_is_socket_unix(int fd, int type, int listening, const char *path, size_t length) {
285 union sockaddr_union sockaddr;
289 if ((r = sd_is_socket_internal(fd, type, listening)) <= 0)
292 memset(&sockaddr, 0, sizeof(sockaddr));
293 l = sizeof(sockaddr);
295 if (getsockname(fd, &sockaddr.sa, &l) < 0)
298 if (l < sizeof(sa_family_t))
301 if (sockaddr.sa.sa_family != AF_UNIX)
306 length = strlen(path);
310 return l == sizeof(sa_family_t);
313 /* Normal path socket */
315 (l >= sizeof(sa_family_t) + length + 1) &&
316 memcmp(path, sockaddr.un.sun_path, length+1) == 0;
318 /* Abstract namespace socket */
320 (l == sizeof(sa_family_t) + length) &&
321 memcmp(path, sockaddr.un.sun_path, length) == 0;
327 int sd_notify(int unset_environment, const char *state) {
328 #if defined(DISABLE_SYSTEMD) || !defined(__linux__) || !defined(SOCK_CLOEXEC)
332 struct msghdr msghdr;
334 union sockaddr_union sockaddr;
337 struct cmsghdr cmsghdr;
338 uint8_t buf[CMSG_SPACE(sizeof(struct ucred))];
347 if (!(e = getenv("NOTIFY_SOCKET")))
350 /* Must be an abstract socket, or an absolute path */
351 if ((e[0] != '@' && e[0] != '/') || e[1] == 0) {
356 if ((fd = socket(AF_UNIX, SOCK_DGRAM|SOCK_CLOEXEC, 0)) < 0) {
361 memset(&sockaddr, 0, sizeof(sockaddr));
362 sockaddr.sa.sa_family = AF_UNIX;
363 strncpy(sockaddr.un.sun_path, e, sizeof(sockaddr.un.sun_path));
365 if (sockaddr.un.sun_path[0] == '@')
366 sockaddr.un.sun_path[0] = 0;
368 memset(&iovec, 0, sizeof(iovec));
369 iovec.iov_base = (char*) state;
370 iovec.iov_len = strlen(state);
372 memset(&control, 0, sizeof(control));
373 control.cmsghdr.cmsg_level = SOL_SOCKET;
374 control.cmsghdr.cmsg_type = SCM_CREDENTIALS;
375 control.cmsghdr.cmsg_len = CMSG_LEN(sizeof(struct ucred));
377 ucred = (struct ucred*) CMSG_DATA(&control.cmsghdr);
378 ucred->pid = getpid();
379 ucred->uid = getuid();
380 ucred->gid = getgid();
382 memset(&msghdr, 0, sizeof(msghdr));
383 msghdr.msg_name = &sockaddr;
384 msghdr.msg_namelen = sizeof(sa_family_t) + strlen(e);
386 if (msghdr.msg_namelen > sizeof(struct sockaddr_un))
387 msghdr.msg_namelen = sizeof(struct sockaddr_un);
389 msghdr.msg_iov = &iovec;
390 msghdr.msg_iovlen = 1;
391 msghdr.msg_control = &control;
392 msghdr.msg_controllen = control.cmsghdr.cmsg_len;
394 if (sendmsg(fd, &msghdr, MSG_NOSIGNAL) < 0) {
402 if (unset_environment)
403 unsetenv("NOTIFY_SOCKET");
412 int sd_notifyf(int unset_environment, const char *format, ...) {
413 #if defined(DISABLE_SYSTEMD) || !defined(__linux__)
420 va_start(ap, format);
421 r = vasprintf(&p, format, ap);
427 r = sd_notify(unset_environment, p);
434 int sd_booted(void) {
435 #if defined(DISABLE_SYSTEMD) || !defined(__linux__)
441 /* We simply test whether the systemd cgroup hierarchy is
444 if (lstat("/cgroup", &a) < 0)
447 if (lstat("/cgroup/systemd", &b) < 0)
450 return a.st_dev != b.st_dev;