mirror of
https://github.com/morgan9e/systemd
synced 2026-04-15 17:06:39 +09:00
pidref: add structure that can reference a pid via both pidfd and pid_t
Let's start with the conversion of PID 1 to pidfds. Let's add a simple structure with just two fields that can be used to maintain a reference to arbitrary processes via both pid_t and pidfd. This is an embeddable struct, to keep it in line with where we previously used a pid_t directly to track a process. Of course, since this might contain an fd on systems where we have pidfd this structure has a proper lifecycle. (Note that this is quite different from sd_event_add_child() event source objects as that one is only for child processes and collects process results, while this infra is much simpler and more generic and can be used to reference any process, anywhere in the tree.)
This commit is contained in:
145
src/basic/pidref.c
Normal file
145
src/basic/pidref.c
Normal file
@@ -0,0 +1,145 @@
|
||||
/* SPDX-License-Identifier: LGPL-2.1-or-later */
|
||||
|
||||
#include "errno-util.h"
|
||||
#include "fd-util.h"
|
||||
#include "missing_syscall.h"
|
||||
#include "parse-util.h"
|
||||
#include "pidref.h"
|
||||
#include "process-util.h"
|
||||
|
||||
int pidref_set_pid(PidRef *pidref, pid_t pid) {
|
||||
int fd;
|
||||
|
||||
assert(pidref);
|
||||
|
||||
if (pid < 0)
|
||||
return -ESRCH;
|
||||
if (pid == 0)
|
||||
pid = getpid_cached();
|
||||
|
||||
fd = pidfd_open(pid, 0);
|
||||
if (fd < 0) {
|
||||
/* Graceful fallback in case the kernel doesn't support pidfds or is out of fds */
|
||||
if (!ERRNO_IS_NOT_SUPPORTED(errno) && !ERRNO_IS_PRIVILEGE(errno) && !ERRNO_IS_RESOURCE(errno))
|
||||
return -errno;
|
||||
|
||||
fd = -EBADF;
|
||||
}
|
||||
|
||||
*pidref = (PidRef) {
|
||||
.fd = fd,
|
||||
.pid = pid,
|
||||
};
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
int pidref_set_pidstr(PidRef *pidref, const char *pid) {
|
||||
pid_t nr;
|
||||
int r;
|
||||
|
||||
assert(pidref);
|
||||
|
||||
r = parse_pid(pid, &nr);
|
||||
if (r < 0)
|
||||
return r;
|
||||
|
||||
return pidref_set_pid(pidref, nr);
|
||||
}
|
||||
|
||||
int pidref_set_pidfd(PidRef *pidref, int fd) {
|
||||
int r;
|
||||
|
||||
assert(pidref);
|
||||
|
||||
if (fd < 0)
|
||||
return -EBADF;
|
||||
|
||||
int fd_copy = fcntl(fd, F_DUPFD_CLOEXEC, 3);
|
||||
if (fd_copy < 0) {
|
||||
pid_t pid;
|
||||
|
||||
if (!ERRNO_IS_RESOURCE(errno))
|
||||
return -errno;
|
||||
|
||||
/* Graceful fallback if we are out of fds */
|
||||
r = pidfd_get_pid(fd, &pid);
|
||||
if (r < 0)
|
||||
return r;
|
||||
|
||||
*pidref = (PidRef) {
|
||||
.fd = -EBADF,
|
||||
.pid = pid,
|
||||
};
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
return pidref_set_pidfd_consume(pidref, fd_copy);
|
||||
}
|
||||
|
||||
int pidref_set_pidfd_take(PidRef *pidref, int fd) {
|
||||
pid_t pid;
|
||||
int r;
|
||||
|
||||
assert(pidref);
|
||||
|
||||
if (fd < 0)
|
||||
return -EBADF;
|
||||
|
||||
r = pidfd_get_pid(fd, &pid);
|
||||
if (r < 0)
|
||||
return r;
|
||||
|
||||
*pidref = (PidRef) {
|
||||
.fd = fd,
|
||||
.pid = pid,
|
||||
};
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
int pidref_set_pidfd_consume(PidRef *pidref, int fd) {
|
||||
int r;
|
||||
|
||||
r = pidref_set_pidfd_take(pidref, fd);
|
||||
if (r < 0)
|
||||
safe_close(fd);
|
||||
|
||||
return r;
|
||||
}
|
||||
|
||||
void pidref_done(PidRef *pidref) {
|
||||
assert(pidref);
|
||||
|
||||
*pidref = (PidRef) {
|
||||
.fd = safe_close(pidref->fd),
|
||||
};
|
||||
}
|
||||
|
||||
int pidref_kill(PidRef *pidref, int sig) {
|
||||
|
||||
if (!pidref)
|
||||
return -ESRCH;
|
||||
|
||||
if (pidref->fd >= 0)
|
||||
return RET_NERRNO(pidfd_send_signal(pidref->fd, sig, NULL, 0));
|
||||
|
||||
if (pidref->pid > 0)
|
||||
return RET_NERRNO(kill(pidref->pid, sig));
|
||||
|
||||
return -ESRCH;
|
||||
}
|
||||
|
||||
int pidref_kill_and_sigcont(PidRef *pidref, int sig) {
|
||||
int r;
|
||||
|
||||
r = pidref_kill(pidref, sig);
|
||||
if (r < 0)
|
||||
return r;
|
||||
|
||||
if (!IN_SET(sig, SIGCONT, SIGKILL))
|
||||
(void) pidref_kill(pidref, SIGCONT);
|
||||
|
||||
return 0;
|
||||
}
|
||||
Reference in New Issue
Block a user