sd_bus_message_unref(m->create_message);
+ m->cgroup_empty_event_source = sd_event_source_disable_unref(m->cgroup_empty_event_source);
+
free(m->name);
free(m->state_file);
free(m->unit);
free(m->subgroup);
free(m->scope_job);
+ free(m->cgroup);
free(m->netif);
free(m->ssh_address);
return 0;
}
+static int machine_dispatch_cgroup_empty(sd_event_source *s, const struct inotify_event *event, void *userdata) {
+ Machine *m = ASSERT_PTR(userdata);
+ int r;
+
+ assert(m->cgroup);
+
+ r = cg_is_empty(SYSTEMD_CGROUP_CONTROLLER, m->cgroup);
+ if (r < 0)
+ return log_error_errno(r, "Failed to determine if cgroup '%s' is empty: %m", m->cgroup);
+
+ if (r > 0)
+ machine_add_to_gc_queue(m);
+
+ return 0;
+}
+
+static int machine_watch_cgroup(Machine *m) {
+ int r;
+
+ assert(m);
+ assert(!m->cgroup_empty_event_source);
+
+ if (!m->cgroup)
+ return 0;
+
+ _cleanup_free_ char *p = NULL;
+ r = cg_get_path(SYSTEMD_CGROUP_CONTROLLER, m->cgroup, "cgroup.events", &p);
+ if (r < 0)
+ return log_error_errno(r, "Failed to get cgroup path for cgroup '%s': %m", m->cgroup);
+
+ r = sd_event_add_inotify(m->manager->event, &m->cgroup_empty_event_source, p, IN_MODIFY, machine_dispatch_cgroup_empty, m);
+ if (r < 0)
+ return log_error_errno(r, "Failed to watch %s events: %m", p);
+
+ return 0;
+}
+
int machine_start(Machine *m, sd_bus_message *properties, sd_bus_error *error) {
int r;
if (r < 0)
return r;
+ r = machine_watch_cgroup(m);
+ if (r < 0)
+ return r;
+
/* Create cgroup */
r = machine_ensure_scope(m, properties, error);
if (r < 0)
return false;
}
+ if (m->cgroup) {
+ r = cg_is_empty(SYSTEMD_CGROUP_CONTROLLER, m->cgroup);
+ if (IN_SET(r, 0, -ENOENT))
+ return true;
+ if (r < 0)
+ log_debug_errno(r, "Failed to determine if cgroup '%s' is empty, ignoring: %m", m->cgroup);
+ }
+
return true;
}
goto fail;
}
+ if (!empty_or_root(m->subgroup)) {
+ /* If this is not a top-level cgroup, then we need the cgroup path to be able to watch when
+ * it empties */
+
+ r = cg_pidref_get_path(SYSTEMD_CGROUP_CONTROLLER, &m->leader, &m->cgroup);
+ if (r < 0) {
+ r = sd_bus_error_set_errnof(error, r,
+ "Failed to determine cgroup of process "PID_FMT" : %m",
+ m->leader.pid);
+ goto fail;
+ }
+ }
+
r = machine_start(m, NULL, error);
if (r < 0)
goto fail;