1
0
mirror of https://github.com/tmux/tmux.git synced 2025-04-04 23:28:51 +00:00
This commit is contained in:
danieldg 2025-03-29 08:44:06 +05:30 committed by GitHub
commit f2ea9309f4
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
3 changed files with 105 additions and 15 deletions

View File

@ -450,7 +450,7 @@ void *recallocarray(void *, size_t, size_t, size_t);
/* systemd.c */
int systemd_activated(void);
int systemd_create_socket(int, char **);
int systemd_move_pid_to_new_cgroup(pid_t, char **);
int systemd_move_to_new_cgroup(char **);
#endif
#ifdef HAVE_UTF8PROC

View File

@ -75,16 +75,53 @@ fail:
return (-1);
}
struct job_watch {
const char* path;
int done;
};
static int
job_removed_handler(sd_bus_message *m, void *userdata, sd_bus_error *ret_error)
{
struct job_watch *watch = userdata;
const char* path = NULL;
uint32_t id;
int r;
(void)ret_error;
/* This handler could be called during sd_bus_call. */
if (watch->path == NULL) {
return 0;
}
r = sd_bus_message_read(m, "uo", &id, &path);
if (r < 0) {
return (r);
}
if (strcmp(path, watch->path) == 0) {
watch->done = 1;
}
return (0);
}
int
systemd_move_pid_to_new_cgroup(pid_t pid, char **cause)
systemd_move_to_new_cgroup(char **cause)
{
sd_bus_error error = SD_BUS_ERROR_NULL;
sd_bus_message *m = NULL, *reply = NULL;
sd_bus *bus = NULL;
sd_bus_slot *slot = NULL;
char *name, *desc, *slice;
sd_id128_t uuid;
int r;
pid_t parent_pid;
uint64_t elapsed_usec;
pid_t pid, parent_pid;
struct job_watch watch = {};
struct timeval start, now;
gettimeofday(&start, NULL);
/* Connect to the session bus. */
r = sd_bus_default_user(&bus);
@ -94,6 +131,20 @@ systemd_move_pid_to_new_cgroup(pid_t pid, char **cause)
goto finish;
}
/* Start watching for JobRemoved events */
r = sd_bus_match_signal(bus, &slot,
"org.freedesktop.systemd1",
"/org/freedesktop/systemd1",
"org.freedesktop.systemd1.Manager",
"JobRemoved",
job_removed_handler,
&watch);
if (r < 0) {
xasprintf(cause, "failed to create match signal: %s",
strerror(-r));
goto finish;
}
/* Start building the method call. */
r = sd_bus_message_new_method_call(bus, &m,
"org.freedesktop.systemd1",
@ -138,7 +189,8 @@ systemd_move_pid_to_new_cgroup(pid_t pid, char **cause)
goto finish;
}
parent_pid = getpid();
pid = getpid();
parent_pid = getppid();
xasprintf(&desc, "tmux child pane %ld launched by process %ld",
(long)pid, (long)parent_pid);
r = sd_bus_message_append(m, "(sv)", "Description", "s", desc);
@ -223,10 +275,49 @@ systemd_move_pid_to_new_cgroup(pid_t pid, char **cause)
goto finish;
}
/* Get the job (object path) from the reply */
r = sd_bus_message_read(reply, "o", &watch.path);
if (r < 0) {
xasprintf(cause, "failed to parse method reply: %s",
strerror(-r));
goto finish;
}
while (!watch.done) {
/* Process events, invoking callbacks that may set watch.done */
r = sd_bus_process(bus, NULL);
if (r < 0) {
xasprintf(cause, "failed waiting for cgroup allocation: %s",
strerror(-r));
goto finish;
}
/* A zero return means we should wait for events */
if (r != 0) {
continue;
}
gettimeofday(&now, NULL);
elapsed_usec = (now.tv_sec - start.tv_sec) * 1000000 + now.tv_usec - start.tv_usec;
if (elapsed_usec >= 1000000) {
xasprintf(cause, "timeout waiting for cgroup allocation");
goto finish;
}
r = sd_bus_wait(bus, 1000000 - elapsed_usec);
if (r < 0) {
xasprintf(cause, "failed waiting for cgroup allocation: %s",
strerror(-r));
goto finish;
}
}
finish:
sd_bus_error_free(&error);
sd_bus_message_unref(m);
sd_bus_message_unref(reply);
sd_bus_slot_unref(slot);
sd_bus_unref(bus);
return (r);

21
spawn.c
View File

@ -382,20 +382,19 @@ spawn_pane(struct spawn_context *sc, char **cause)
/* In the parent process, everything is done now. */
if (new_wp->pid != 0) {
#if defined(HAVE_SYSTEMD) && defined(ENABLE_CGROUPS)
/*
* Move the child process into a new cgroup for systemd-oomd
* isolation.
*/
if (systemd_move_pid_to_new_cgroup(new_wp->pid, cause) < 0) {
log_debug("%s: moving pane to new cgroup failed: %s",
__func__, *cause);
free (*cause);
}
#endif
goto complete;
}
#if defined(HAVE_SYSTEMD) && defined(ENABLE_CGROUPS)
/*
* Move the child process into a new cgroup for systemd-oomd isolation.
*/
if (systemd_move_to_new_cgroup(cause) < 0) {
log_debug("%s: moving pane to new cgroup failed: %s",
__func__, *cause);
free (*cause);
}
#endif
/*
* Child process. Change to the working directory or home if that
* fails.