Note that this drops a lot of "const" qualifiers on PidRef arguments.
That's because pidref_is_self() suddenly might end changing the PidRef
because it acquires the pidfd ID.
We had this previously already with pidfd_equal(), but this amplifies
the problem.
I guess we C's "const" doesn't really work for stuff that contains
caches, that is just conceptually constant, but not actually.
}
int pidref_set_pid(PidRef *pidref, pid_t pid) {
+ uint64_t pidfdid = 0;
int fd;
assert(pidref);
if (pid < 0)
return -ESRCH;
- if (pid == 0)
+ if (pid == 0) {
pid = getpid_cached();
+ (void) pidfd_get_inode_id_self_cached(&pidfdid);
+ }
fd = pidfd_open(pid, 0);
if (fd < 0) {
*pidref = (PidRef) {
.fd = fd,
.pid = pid,
+ .fd_id = pidfdid,
};
return 0;
return 1; /* We have a pidfd and it still points to the PID we have, hence all is *really* OK → return 1 */
}
-bool pidref_is_self(const PidRef *pidref) {
- if (!pidref)
+bool pidref_is_self(PidRef *pidref) {
+ if (!pidref_is_set(pidref))
return false;
if (pidref_is_remote(pidref))
return false;
- return pidref->pid == getpid_cached();
+ if (pidref->pid != getpid_cached())
+ return false;
+
+ /* PID1 cannot exit, hence no point in comparing pidfd IDs, they can never change */
+ if (pidref->pid == 1)
+ return true;
+
+ /* Also compare pidfd ID if we can get it */
+ if (pidref_acquire_pidfd_id(pidref) < 0)
+ return true;
+
+ uint64_t self_id;
+ if (pidfd_get_inode_id_self_cached(&self_id) < 0)
+ return true;
+
+ return pidref->fd_id == self_id;
}
-int pidref_wait(const PidRef *pidref, siginfo_t *ret, int options) {
+int pidref_wait(PidRef *pidref, siginfo_t *ret, int options) {
int r;
if (!pidref_is_set(pidref))
return 0;
}
-int pidref_wait_for_terminate(const PidRef *pidref, siginfo_t *ret) {
+int pidref_wait_for_terminate(PidRef *pidref, siginfo_t *ret) {
int r;
for (;;) {
their own pidfs and each process comes with a unique inode number */
};
-#define PIDREF_NULL (const PidRef) { .fd = -EBADF }
+#define PIDREF_NULL (PidRef) { .fd = -EBADF }
/* A special pidref value that we are using when a PID shall be automatically acquired from some surrounding
* context, for example connection peer. Much like PIDREF_NULL it will be considered unset by
return pidref_set_pid(pidref, 0);
}
-bool pidref_is_self(const PidRef *pidref);
+bool pidref_is_self(PidRef *pidref);
void pidref_done(PidRef *pidref);
PidRef* pidref_free(PidRef *pidref);
int pidref_kill_and_sigcont(const PidRef *pidref, int sig);
int pidref_sigqueue(const PidRef *pidref, int sig, int value);
-int pidref_wait(const PidRef *pidref, siginfo_t *siginfo, int options);
-int pidref_wait_for_terminate(const PidRef *pidref, siginfo_t *ret);
+int pidref_wait(PidRef *pidref, siginfo_t *siginfo, int options);
+int pidref_wait_for_terminate(PidRef *pidref, siginfo_t *ret);
static inline void pidref_done_sigkill_wait(PidRef *pidref) {
if (!pidref_is_set(pidref))
return 0;
}
-int pidref_is_my_child(const PidRef *pid) {
+int pidref_is_my_child(PidRef *pid) {
int r;
if (!pidref_is_set(pid))
return pidref_is_my_child(&PIDREF_MAKE_FROM_PID(pid));
}
-int pidref_is_unwaited(const PidRef *pid) {
+int pidref_is_unwaited(PidRef *pid) {
int r;
/* Checks whether a PID is still valid at all, including a zombie */
int pid_is_alive(pid_t pid);
int pidref_is_alive(const PidRef *pidref);
int pid_is_unwaited(pid_t pid);
-int pidref_is_unwaited(const PidRef *pidref);
+int pidref_is_unwaited(PidRef *pidref);
int pid_is_my_child(pid_t pid);
-int pidref_is_my_child(const PidRef *pidref);
+int pidref_is_my_child(PidRef *pidref);
int pidref_from_same_root_fs(PidRef *a, PidRef *b);
bool is_main_thread(void);
return NULL;
}
-Unit *manager_get_unit_by_pidref(Manager *m, const PidRef *pid) {
+Unit* manager_get_unit_by_pidref(Manager *m, PidRef *pid) {
Unit *u;
assert(m);
Unit *manager_get_unit_by_cgroup(Manager *m, const char *cgroup);
Unit *manager_get_unit_by_pidref_cgroup(Manager *m, const PidRef *pid);
Unit *manager_get_unit_by_pidref_watching(Manager *m, const PidRef *pid);
-Unit* manager_get_unit_by_pidref(Manager *m, const PidRef *pid);
+Unit* manager_get_unit_by_pidref(Manager *m, PidRef *pid);
Unit* manager_get_unit_by_pid(Manager *m, pid_t pid);
uint64_t unit_get_ancestor_memory_min(Unit *u);
return exec_context_may_touch_console(ec);
}
-int unit_pid_attachable(Unit *u, const PidRef *pid, sd_bus_error *error) {
+int unit_pid_attachable(Unit *u, PidRef *pid, sd_bus_error *error) {
int r;
assert(u);
bool unit_needs_console(Unit *u);
-int unit_pid_attachable(Unit *unit, const PidRef *pid, sd_bus_error *error);
+int unit_pid_attachable(Unit *unit, PidRef *pid, sd_bus_error *error);
static inline bool unit_has_job_type(Unit *u, JobType type) {
return u && u->job && u->job->type == type;
#include "stdio-util.h"
#include "tests.h"
-#define PIDREF_NULL_NONCONST (PidRef) { .fd = -EBADF }
-
TEST(pidref_is_set) {
assert_se(!pidref_is_set(NULL));
assert_se(!pidref_is_set(&PIDREF_NULL));
TEST(pidref_equal) {
assert_se(pidref_equal(NULL, NULL));
- assert_se(pidref_equal(NULL, &PIDREF_NULL_NONCONST));
- assert_se(pidref_equal(&PIDREF_NULL_NONCONST, NULL));
- assert_se(pidref_equal(&PIDREF_NULL_NONCONST, &PIDREF_NULL_NONCONST));
+ assert_se(pidref_equal(NULL, &PIDREF_NULL));
+ assert_se(pidref_equal(&PIDREF_NULL, NULL));
+ assert_se(pidref_equal(&PIDREF_NULL, &PIDREF_NULL));
assert_se(!pidref_equal(NULL, &PIDREF_MAKE_FROM_PID(1)));
assert_se(!pidref_equal(&PIDREF_MAKE_FROM_PID(1), NULL));
- assert_se(!pidref_equal(&PIDREF_NULL_NONCONST, &PIDREF_MAKE_FROM_PID(1)));
- assert_se(!pidref_equal(&PIDREF_MAKE_FROM_PID(1), &PIDREF_NULL_NONCONST));
+ assert_se(!pidref_equal(&PIDREF_NULL, &PIDREF_MAKE_FROM_PID(1)));
+ assert_se(!pidref_equal(&PIDREF_MAKE_FROM_PID(1), &PIDREF_NULL));
assert_se(pidref_equal(&PIDREF_MAKE_FROM_PID(1), &PIDREF_MAKE_FROM_PID(1)));
assert_se(!pidref_equal(&PIDREF_MAKE_FROM_PID(1), &PIDREF_MAKE_FROM_PID(2)));
}
assert_se(!pidref_is_remote(&PIDREF_MAKE_FROM_PID(getpid_cached())));
assert_se(!pidref_is_remote(&PIDREF_AUTOMATIC));
- static const PidRef p = {
+ PidRef p = {
.pid = 1,
.fd = -EREMOTE,
.fd_id = 4711,