mirror of
https://github.com/tmate-io/tmate.git
synced 2024-12-26 08:38:51 +01:00
abf3a5d50e
This moves the client-side loops are pretty much fully over to event-based only (tmux.c and client.c) but server-side (server.c and friends) treats libevent as a sort of clever poll, waking up after every event to run various things. Moving the server stuff over to bufferevents and timers and so on will come later.
200 lines
4.1 KiB
C
200 lines
4.1 KiB
C
/* $OpenBSD$ */
|
|
|
|
/*
|
|
* Copyright (c) 2009 Nicholas Marriott <nicm@users.sourceforge.net>
|
|
*
|
|
* Permission to use, copy, modify, and distribute this software for any
|
|
* purpose with or without fee is hereby granted, provided that the above
|
|
* copyright notice and this permission notice appear in all copies.
|
|
*
|
|
* THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
|
|
* WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
|
|
* MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
|
|
* ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
|
|
* WHATSOEVER RESULTING FROM LOSS OF MIND, USE, DATA OR PROFITS, WHETHER
|
|
* IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING
|
|
* OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
|
|
*/
|
|
|
|
#include <sys/types.h>
|
|
|
|
#include <fcntl.h>
|
|
#include <paths.h>
|
|
#include <string.h>
|
|
#include <unistd.h>
|
|
|
|
#include "tmux.h"
|
|
|
|
/*
|
|
* Job scheduling. Run queued commands in the background and record their
|
|
* output.
|
|
*/
|
|
|
|
/* All jobs list. */
|
|
struct joblist all_jobs = SLIST_HEAD_INITIALIZER(&all_jobs);
|
|
|
|
RB_GENERATE(jobs, job, entry, job_cmp);
|
|
|
|
int
|
|
job_cmp(struct job *job1, struct job *job2)
|
|
{
|
|
return (strcmp(job1->cmd, job2->cmd));
|
|
}
|
|
|
|
/* Initialise job tree. */
|
|
void
|
|
job_tree_init(struct jobs *jobs)
|
|
{
|
|
RB_INIT(jobs);
|
|
}
|
|
|
|
/* Destroy a job tree. */
|
|
void
|
|
job_tree_free(struct jobs *jobs)
|
|
{
|
|
struct job *job;
|
|
|
|
while (!RB_EMPTY(jobs)) {
|
|
job = RB_ROOT(jobs);
|
|
RB_REMOVE(jobs, jobs, job);
|
|
job_free(job);
|
|
}
|
|
}
|
|
|
|
/* Find a job and return it. */
|
|
struct job *
|
|
job_get(struct jobs *jobs, const char *cmd)
|
|
{
|
|
struct job job;
|
|
|
|
job.cmd = (char *) cmd;
|
|
return (RB_FIND(jobs, jobs, &job));
|
|
}
|
|
|
|
/* Add a job. */
|
|
struct job *
|
|
job_add(struct jobs *jobs, int flags, struct client *c, const char *cmd,
|
|
void (*callbackfn)(struct job *), void (*freefn)(void *), void *data)
|
|
{
|
|
struct job *job;
|
|
|
|
job = xmalloc(sizeof *job);
|
|
job->cmd = xstrdup(cmd);
|
|
job->pid = -1;
|
|
job->status = 0;
|
|
|
|
job->client = c;
|
|
|
|
job->fd = -1;
|
|
job->out = buffer_create(BUFSIZ);
|
|
memset(&job->event, 0, sizeof job->event);
|
|
|
|
job->callbackfn = callbackfn;
|
|
job->freefn = freefn;
|
|
job->data = data;
|
|
|
|
job->flags = flags|JOB_DONE;
|
|
|
|
if (jobs != NULL)
|
|
RB_INSERT(jobs, jobs, job);
|
|
SLIST_INSERT_HEAD(&all_jobs, job, lentry);
|
|
|
|
return (job);
|
|
}
|
|
|
|
/* Remove job from tree and free. */
|
|
void
|
|
job_remove(struct jobs *jobs, struct job *job)
|
|
{
|
|
if (jobs != NULL)
|
|
RB_REMOVE(jobs, jobs, job);
|
|
job_free(job);
|
|
}
|
|
|
|
/* Kill and free an individual job. */
|
|
void
|
|
job_free(struct job *job)
|
|
{
|
|
job_kill(job);
|
|
|
|
SLIST_REMOVE(&all_jobs, job, job, lentry);
|
|
xfree(job->cmd);
|
|
|
|
if (job->freefn != NULL && job->data != NULL)
|
|
job->freefn(job->data);
|
|
|
|
if (job->fd != -1)
|
|
close(job->fd);
|
|
if (job->out != NULL)
|
|
buffer_destroy(job->out);
|
|
event_del(&job->event);
|
|
|
|
xfree(job);
|
|
}
|
|
|
|
/* Start a job running, if it isn't already. */
|
|
int
|
|
job_run(struct job *job)
|
|
{
|
|
int nullfd, out[2], mode;
|
|
|
|
if (!(job->flags & JOB_DONE))
|
|
return (0);
|
|
job->flags &= ~JOB_DONE;
|
|
|
|
if (pipe(out) != 0)
|
|
return (-1);
|
|
|
|
switch (job->pid = fork()) {
|
|
case -1:
|
|
return (-1);
|
|
case 0: /* child */
|
|
server_signal_clear();
|
|
/* XXX environ? */
|
|
|
|
if (dup2(out[1], STDOUT_FILENO) == -1)
|
|
fatal("dup2 failed");
|
|
if (out[1] != STDOUT_FILENO)
|
|
close(out[1]);
|
|
close(out[0]);
|
|
|
|
nullfd = open(_PATH_DEVNULL, O_RDWR, 0);
|
|
if (nullfd < 0)
|
|
fatal("open failed");
|
|
if (dup2(nullfd, STDIN_FILENO) == -1)
|
|
fatal("dup2 failed");
|
|
if (dup2(nullfd, STDERR_FILENO) == -1)
|
|
fatal("dup2 failed");
|
|
if (nullfd != STDIN_FILENO && nullfd != STDERR_FILENO)
|
|
close(nullfd);
|
|
|
|
execl(_PATH_BSHELL, "sh", "-c", job->cmd, (char *) NULL);
|
|
fatal("execl failed");
|
|
default: /* parent */
|
|
close(out[1]);
|
|
|
|
job->fd = out[0];
|
|
if ((mode = fcntl(job->fd, F_GETFL)) == -1)
|
|
fatal("fcntl failed");
|
|
if (fcntl(job->fd, F_SETFL, mode|O_NONBLOCK) == -1)
|
|
fatal("fcntl failed");
|
|
if (fcntl(job->fd, F_SETFD, FD_CLOEXEC) == -1)
|
|
fatal("fcntl failed");
|
|
|
|
if (BUFFER_USED(job->out) != 0)
|
|
buffer_remove(job->out, BUFFER_USED(job->out));
|
|
|
|
return (0);
|
|
}
|
|
}
|
|
|
|
/* Kill a job. */
|
|
void
|
|
job_kill(struct job *job)
|
|
{
|
|
if (job->pid == -1)
|
|
return;
|
|
kill(job->pid, SIGTERM);
|
|
job->pid = -1;
|
|
}
|