mirror of
https://github.com/tmux/tmux.git
synced 2025-01-07 16:28:48 +00:00
Add a couple of helper functions, and flush imsgs on exit.
This commit is contained in:
parent
79e1984962
commit
2b58c226db
15
client.c
15
client.c
@ -223,20 +223,7 @@ client_exit_message(void)
|
|||||||
static void
|
static void
|
||||||
client_exit(void)
|
client_exit(void)
|
||||||
{
|
{
|
||||||
struct client_file *cf;
|
if (!file_write_left(&client_files))
|
||||||
size_t left;
|
|
||||||
int waiting = 0;
|
|
||||||
|
|
||||||
RB_FOREACH (cf, client_files, &client_files) {
|
|
||||||
if (cf->event == NULL)
|
|
||||||
continue;
|
|
||||||
left = EVBUFFER_LENGTH(cf->event->output);
|
|
||||||
if (left != 0) {
|
|
||||||
waiting++;
|
|
||||||
log_debug("file %u %zu bytes left", cf->stream, left);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if (waiting == 0)
|
|
||||||
proc_exit(client_proc);
|
proc_exit(client_proc);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
20
file.c
20
file.c
@ -477,6 +477,26 @@ file_push(struct client_file *cf)
|
|||||||
free(msg);
|
free(msg);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* Check if any files have data left to write. */
|
||||||
|
int
|
||||||
|
file_write_left(struct client_files *files)
|
||||||
|
{
|
||||||
|
struct client_file *cf;
|
||||||
|
size_t left;
|
||||||
|
int waiting = 0;
|
||||||
|
|
||||||
|
RB_FOREACH (cf, client_files, files) {
|
||||||
|
if (cf->event == NULL)
|
||||||
|
continue;
|
||||||
|
left = EVBUFFER_LENGTH(cf->event->output);
|
||||||
|
if (left != 0) {
|
||||||
|
waiting++;
|
||||||
|
log_debug("file %u %zu bytes left", cf->stream, left);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return (waiting != 0);
|
||||||
|
}
|
||||||
|
|
||||||
/* Client file write error callback. */
|
/* Client file write error callback. */
|
||||||
static void
|
static void
|
||||||
file_write_error_callback(__unused struct bufferevent *bev, __unused short what,
|
file_write_error_callback(__unused struct bufferevent *bev, __unused short what,
|
||||||
|
36
proc.c
36
proc.c
@ -18,6 +18,7 @@
|
|||||||
|
|
||||||
#include <sys/types.h>
|
#include <sys/types.h>
|
||||||
#include <sys/queue.h>
|
#include <sys/queue.h>
|
||||||
|
#include <sys/socket.h>
|
||||||
#include <sys/uio.h>
|
#include <sys/uio.h>
|
||||||
#include <sys/utsname.h>
|
#include <sys/utsname.h>
|
||||||
|
|
||||||
@ -45,6 +46,8 @@ struct tmuxproc {
|
|||||||
struct event ev_sigusr1;
|
struct event ev_sigusr1;
|
||||||
struct event ev_sigusr2;
|
struct event ev_sigusr2;
|
||||||
struct event ev_sigwinch;
|
struct event ev_sigwinch;
|
||||||
|
|
||||||
|
TAILQ_HEAD(, tmuxpeer) peers;
|
||||||
};
|
};
|
||||||
|
|
||||||
struct tmuxpeer {
|
struct tmuxpeer {
|
||||||
@ -58,6 +61,8 @@ struct tmuxpeer {
|
|||||||
|
|
||||||
void (*dispatchcb)(struct imsg *, void *);
|
void (*dispatchcb)(struct imsg *, void *);
|
||||||
void *arg;
|
void *arg;
|
||||||
|
|
||||||
|
TAILQ_ENTRY(tmuxpeer) entry;
|
||||||
};
|
};
|
||||||
|
|
||||||
static int peer_check_version(struct tmuxpeer *, struct imsg *);
|
static int peer_check_version(struct tmuxpeer *, struct imsg *);
|
||||||
@ -190,6 +195,7 @@ proc_start(const char *name)
|
|||||||
|
|
||||||
tp = xcalloc(1, sizeof *tp);
|
tp = xcalloc(1, sizeof *tp);
|
||||||
tp->name = xstrdup(name);
|
tp->name = xstrdup(name);
|
||||||
|
TAILQ_INIT(&tp->peers);
|
||||||
|
|
||||||
return (tp);
|
return (tp);
|
||||||
}
|
}
|
||||||
@ -207,6 +213,10 @@ proc_loop(struct tmuxproc *tp, int (*loopcb)(void))
|
|||||||
void
|
void
|
||||||
proc_exit(struct tmuxproc *tp)
|
proc_exit(struct tmuxproc *tp)
|
||||||
{
|
{
|
||||||
|
struct tmuxpeer *peer;
|
||||||
|
|
||||||
|
TAILQ_FOREACH(peer, &tp->peers, entry)
|
||||||
|
imsg_flush(&peer->ibuf);
|
||||||
tp->exit = 1;
|
tp->exit = 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -297,6 +307,7 @@ proc_add_peer(struct tmuxproc *tp, int fd,
|
|||||||
event_set(&peer->event, fd, EV_READ, proc_event_cb, peer);
|
event_set(&peer->event, fd, EV_READ, proc_event_cb, peer);
|
||||||
|
|
||||||
log_debug("add peer %p: %d (%p)", peer, fd, arg);
|
log_debug("add peer %p: %d (%p)", peer, fd, arg);
|
||||||
|
TAILQ_INSERT_TAIL(&tp->peers, peer, entry);
|
||||||
|
|
||||||
proc_update_event(peer);
|
proc_update_event(peer);
|
||||||
return (peer);
|
return (peer);
|
||||||
@ -305,6 +316,7 @@ proc_add_peer(struct tmuxproc *tp, int fd,
|
|||||||
void
|
void
|
||||||
proc_remove_peer(struct tmuxpeer *peer)
|
proc_remove_peer(struct tmuxpeer *peer)
|
||||||
{
|
{
|
||||||
|
TAILQ_REMOVE(&peer->parent->peers, peer, entry);
|
||||||
log_debug("remove peer %p", peer);
|
log_debug("remove peer %p", peer);
|
||||||
|
|
||||||
event_del(&peer->event);
|
event_del(&peer->event);
|
||||||
@ -325,3 +337,27 @@ proc_toggle_log(struct tmuxproc *tp)
|
|||||||
{
|
{
|
||||||
log_toggle(tp->name);
|
log_toggle(tp->name);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pid_t
|
||||||
|
proc_fork_and_daemon(int *fd)
|
||||||
|
{
|
||||||
|
pid_t pid;
|
||||||
|
int pair[2];
|
||||||
|
|
||||||
|
if (socketpair(AF_UNIX, SOCK_STREAM, PF_UNSPEC, pair) != 0)
|
||||||
|
fatal("socketpair failed");
|
||||||
|
switch (pid = fork()) {
|
||||||
|
case -1:
|
||||||
|
fatal("fork failed");
|
||||||
|
case 0:
|
||||||
|
close(pair[0]);
|
||||||
|
*fd = pair[1];
|
||||||
|
if (daemon(1, 0) != 0)
|
||||||
|
fatal("daemon failed");
|
||||||
|
return (0);
|
||||||
|
default:
|
||||||
|
close(pair[1]);
|
||||||
|
*fd = pair[0];
|
||||||
|
return (pid);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
29
server.c
29
server.c
@ -156,35 +156,22 @@ int
|
|||||||
server_start(struct tmuxproc *client, int flags, struct event_base *base,
|
server_start(struct tmuxproc *client, int flags, struct event_base *base,
|
||||||
int lockfd, char *lockfile)
|
int lockfd, char *lockfile)
|
||||||
{
|
{
|
||||||
int pair[2];
|
int fd;
|
||||||
sigset_t set, oldset;
|
sigset_t set, oldset;
|
||||||
struct client *c = NULL;
|
struct client *c = NULL;
|
||||||
char *cause = NULL;
|
char *cause = NULL;
|
||||||
|
|
||||||
sigfillset(&set);
|
sigfillset(&set);
|
||||||
sigprocmask(SIG_BLOCK, &set, &oldset);
|
sigprocmask(SIG_BLOCK, &set, &oldset);
|
||||||
|
|
||||||
if (~flags & CLIENT_NOFORK) {
|
if (~flags & CLIENT_NOFORK) {
|
||||||
if (socketpair(AF_UNIX, SOCK_STREAM, PF_UNSPEC, pair) != 0)
|
if (proc_fork_and_daemon(&fd) != 0) {
|
||||||
fatal("socketpair failed");
|
|
||||||
|
|
||||||
switch (fork()) {
|
|
||||||
case -1:
|
|
||||||
fatal("fork failed");
|
|
||||||
case 0:
|
|
||||||
break;
|
|
||||||
default:
|
|
||||||
sigprocmask(SIG_SETMASK, &oldset, NULL);
|
sigprocmask(SIG_SETMASK, &oldset, NULL);
|
||||||
close(pair[1]);
|
return (fd);
|
||||||
return (pair[0]);
|
|
||||||
}
|
}
|
||||||
close(pair[0]);
|
|
||||||
if (daemon(1, 0) != 0)
|
|
||||||
fatal("daemon failed");
|
|
||||||
}
|
}
|
||||||
|
|
||||||
server_client_flags = flags;
|
|
||||||
proc_clear_signals(client, 0);
|
proc_clear_signals(client, 0);
|
||||||
|
server_client_flags = flags;
|
||||||
|
|
||||||
if (event_reinit(base) != 0)
|
if (event_reinit(base) != 0)
|
||||||
fatalx("event_reinit failed");
|
fatalx("event_reinit failed");
|
||||||
@ -213,7 +200,7 @@ server_start(struct tmuxproc *client, int flags, struct event_base *base,
|
|||||||
if (server_fd != -1)
|
if (server_fd != -1)
|
||||||
server_update_socket();
|
server_update_socket();
|
||||||
if (~flags & CLIENT_NOFORK)
|
if (~flags & CLIENT_NOFORK)
|
||||||
c = server_client_create(pair[1]);
|
c = server_client_create(fd);
|
||||||
else
|
else
|
||||||
options_set_number(global_options, "exit-empty", 0);
|
options_set_number(global_options, "exit-empty", 0);
|
||||||
|
|
||||||
|
2
tmux.h
2
tmux.h
@ -1899,6 +1899,7 @@ struct tmuxpeer *proc_add_peer(struct tmuxproc *, int,
|
|||||||
void proc_remove_peer(struct tmuxpeer *);
|
void proc_remove_peer(struct tmuxpeer *);
|
||||||
void proc_kill_peer(struct tmuxpeer *);
|
void proc_kill_peer(struct tmuxpeer *);
|
||||||
void proc_toggle_log(struct tmuxproc *);
|
void proc_toggle_log(struct tmuxproc *);
|
||||||
|
pid_t proc_fork_and_daemon(int *);
|
||||||
|
|
||||||
/* cfg.c */
|
/* cfg.c */
|
||||||
extern int cfg_finished;
|
extern int cfg_finished;
|
||||||
@ -2389,6 +2390,7 @@ void file_write(struct client *, const char *, int, const void *, size_t,
|
|||||||
client_file_cb, void *);
|
client_file_cb, void *);
|
||||||
void file_read(struct client *, const char *, client_file_cb, void *);
|
void file_read(struct client *, const char *, client_file_cb, void *);
|
||||||
void file_push(struct client_file *);
|
void file_push(struct client_file *);
|
||||||
|
int file_write_left(struct client_files *);
|
||||||
void file_write_open(struct client_files *, struct tmuxpeer *,
|
void file_write_open(struct client_files *, struct tmuxpeer *,
|
||||||
struct imsg *, int, int, client_file_cb, void *);
|
struct imsg *, int, int, client_file_cb, void *);
|
||||||
void file_write_data(struct client_files *, struct imsg *);
|
void file_write_data(struct client_files *, struct imsg *);
|
||||||
|
Loading…
Reference in New Issue
Block a user