]> git.ipfire.org Git - thirdparty/systemd.git/commitdiff
machined: watch leader PID's lifetime via pidfd 33424/head
authorLennart Poettering <lennart@poettering.net>
Mon, 27 May 2024 13:36:44 +0000 (15:36 +0200)
committerLennart Poettering <lennart@poettering.net>
Thu, 20 Jun 2024 20:51:24 +0000 (22:51 +0200)
If we have a pidfd, we might as well track the machine's leader PID's
lifetime, and enqueue the machine for a GC run.

(This is similar to what we are already doing for logind's session
leaders)

src/machine/machine.c
src/machine/machine.h

index 25ecc4d197bdcfdd7fec8d5934f9742f108cd1b5..a12ea7c16bbed30c5e00916ffc032fe18746e5d1 100644 (file)
@@ -118,6 +118,7 @@ Machine* machine_free(Machine *m) {
                         m->manager->host_machine = NULL;
         }
 
+        m->leader_pidfd_event_source = sd_event_source_disable_unref(m->leader_pidfd_event_source);
         if (pidref_is_set(&m->leader)) {
                 if (m->manager)
                         (void) hashmap_remove_value(m->manager->machine_leaders, PID_TO_PTR(m->leader.pid), m);
@@ -475,6 +476,38 @@ static int machine_ensure_scope(Machine *m, sd_bus_message *properties, sd_bus_e
         return 0;
 }
 
+static int machine_dispatch_leader_pidfd(sd_event_source *s, int fd, unsigned revents, void *userdata) {
+        Machine *m = ASSERT_PTR(userdata);
+
+        m->leader_pidfd_event_source = sd_event_source_disable_unref(m->leader_pidfd_event_source);
+        machine_add_to_gc_queue(m);
+
+        return 0;
+}
+
+static int machine_watch_pidfd(Machine *m) {
+        int r;
+
+        assert(m);
+        assert(m->manager);
+        assert(pidref_is_set(&m->leader));
+        assert(!m->leader_pidfd_event_source);
+
+        if (m->leader.fd < 0)
+                return 0;
+
+        /* If we have a pidfd for the leader, let's also track it for POLLIN, and GC the machine
+         * automatically if it dies */
+
+        r = sd_event_add_io(m->manager->event, &m->leader_pidfd_event_source, m->leader.fd, EPOLLIN, machine_dispatch_leader_pidfd, m);
+        if (r < 0)
+                return r;
+
+        (void) sd_event_source_set_description(m->leader_pidfd_event_source, "machine-pidfd");
+
+        return 0;
+}
+
 int machine_start(Machine *m, sd_bus_message *properties, sd_bus_error *error) {
         int r;
 
@@ -490,6 +523,10 @@ int machine_start(Machine *m, sd_bus_message *properties, sd_bus_error *error) {
         if (r < 0)
                 return r;
 
+        r = machine_watch_pidfd(m);
+        if (r < 0)
+                return r;
+
         /* Create cgroup */
         r = machine_ensure_scope(m, properties, error);
         if (r < 0)
index 12551025c7db99e3c7ea0f6ddb807d0efbfcad02..8f1feda14befb3a157ad85e3b4ca6b06785adb38 100644 (file)
@@ -49,6 +49,7 @@ struct Machine {
         char *scope_job;
 
         PidRef leader;
+        sd_event_source *leader_pidfd_event_source;
 
         dual_timestamp timestamp;