/* $Id: server-fn.c,v 1.34 2007-11-24 18:32:52 nicm Exp $ */ /* * Copyright (c) 2007 Nicholas Marriott * * 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 #include #include #include "tmux.h" /* Find session from command message. */ struct session * server_extract_session(struct msg_command_data *data, char *name, char **cause) { struct session *s; u_int i, n; if (name != NULL) { if ((s = session_find(name)) == NULL) { xasprintf(cause, "session not found: %s", name); return (NULL); } return (s); } if (data->pid != -1) { if (data->pid != getpid()) { xasprintf(cause, "wrong server: %lld", data->pid); return (NULL); } if (data->idx > ARRAY_LENGTH(&sessions)) { xasprintf(cause, "index out of range: %d", data->idx); return (NULL); } if ((s = ARRAY_ITEM(&sessions, data->idx)) == NULL) { xasprintf( cause, "session doesn't exist: %u", data->idx); return (NULL); } return (s); } s = NULL; n = 0; for (i = 0; i < ARRAY_LENGTH(&sessions); i++) { if (ARRAY_ITEM(&sessions, i) != NULL) { s = ARRAY_ITEM(&sessions, i); n++; } } if (s == NULL) { xasprintf(cause, "no sessions found"); return (NULL); } if (n != 1) { xasprintf(cause, "multiple sessions and session not specified"); return (NULL); } return (s); } void server_write_client( struct client *c, enum hdrtype type, const void *buf, size_t len) { struct hdr hdr; log_debug("writing %d to client %d", type, c->fd); hdr.type = type; hdr.size = len; buffer_write(c->out, &hdr, sizeof hdr); if (buf != NULL) buffer_write(c->out, buf, len); } void server_write_session( struct session *s, enum hdrtype type, const void *buf, size_t len) { struct client *c; u_int i; for (i = 0; i < ARRAY_LENGTH(&clients); i++) { c = ARRAY_ITEM(&clients, i); if (c != NULL && c->session == s) server_write_client(c, type, buf, len); } } void server_write_window_cur( struct window *w, enum hdrtype type, const void *buf, size_t len) { struct client *c; u_int i; for (i = 0; i < ARRAY_LENGTH(&clients); i++) { c = ARRAY_ITEM(&clients, i); if (c == NULL || c->session == NULL) continue; if (c->session->curw->window == w) server_write_client(c, type, buf, len); } } void server_write_window_all( struct window *w, enum hdrtype type, const void *buf, size_t len) { struct client *c; u_int i; for (i = 0; i < ARRAY_LENGTH(&clients); i++) { c = ARRAY_ITEM(&clients, i); if (c == NULL || c->session == NULL) continue; if (session_has(c->session, w)) server_write_client(c, type, buf, len); } } void server_status_client(struct client *c) { struct hdr hdr; size_t size; if (status_lines == 0 || c->sy <= status_lines) return; buffer_ensure(c->out, sizeof hdr); buffer_add(c->out, sizeof hdr); size = BUFFER_USED(c->out); status_write(c); size = BUFFER_USED(c->out) - size; hdr.type = MSG_DATA; hdr.size = size; memcpy(BUFFER_IN(c->out) - size - sizeof hdr, &hdr, sizeof hdr); } void server_clear_client(struct client *c) { struct screen *s = &c->session->curw->window->screen; struct hdr hdr; size_t size; u_int i; buffer_ensure(c->out, sizeof hdr); buffer_add(c->out, sizeof hdr); size = BUFFER_USED(c->out); input_store_zero(c->out, CODE_CURSOROFF); for (i = 0; i < screen_size_y(s); i++) { input_store_two(c->out, CODE_CURSORMOVE, i + 1, 1); input_store_zero(c->out, CODE_CLEARLINE); } input_store_two(c->out, CODE_CURSORMOVE, s->cy + 1, s->cx + 1); if (s->mode & MODE_BACKGROUND) { if (s->mode & MODE_BGCURSOR) input_store_zero(c->out, CODE_CURSORON); } else { if (s->mode & MODE_CURSOR) input_store_zero(c->out, CODE_CURSORON); } size = BUFFER_USED(c->out) - size; hdr.type = MSG_DATA; hdr.size = size; memcpy(BUFFER_IN(c->out) - size - sizeof hdr, &hdr, sizeof hdr); } void server_redraw_client(struct client *c) { struct window *w = c->session->curw->window; struct hdr hdr; size_t size; buffer_ensure(c->out, sizeof hdr); buffer_add(c->out, sizeof hdr); size = BUFFER_USED(c->out); window_draw(w, c->out, 0, screen_size_y(&w->screen)); size = BUFFER_USED(c->out) - size; log_debug("redrawing window, %zu bytes", size); if (size != 0) { hdr.type = MSG_DATA; hdr.size = size; memcpy(BUFFER_IN(c->out) - size - sizeof hdr, &hdr, sizeof hdr); } else buffer_reverse_add(c->out, sizeof hdr); server_status_client(c); } void server_redraw_session(struct session *s) { struct client *c; u_int i; for (i = 0; i < ARRAY_LENGTH(&clients); i++) { c = ARRAY_ITEM(&clients, i); if (c != NULL && c->session == s) server_redraw_client(c); } } void server_status_session(struct session *s) { struct client *c; u_int i; for (i = 0; i < ARRAY_LENGTH(&clients); i++) { c = ARRAY_ITEM(&clients, i); if (c != NULL && c->session == s) server_status_client(c); } } void server_clear_window_cur(struct window *w) { struct client *c; u_int i; for (i = 0; i < ARRAY_LENGTH(&clients); i++) { c = ARRAY_ITEM(&clients, i); if (c != NULL && c->session != NULL && c->session->curw->window == w) server_clear_client(c); } } void server_clear_window_all(struct window *w) { struct client *c; u_int i; for (i = 0; i < ARRAY_LENGTH(&clients); i++) { c = ARRAY_ITEM(&clients, i); if (c == NULL || c->session == NULL) continue; if (session_has(c->session, w)) server_redraw_client(c); } } void server_redraw_window_cur(struct window *w) { struct client *c; u_int i; for (i = 0; i < ARRAY_LENGTH(&clients); i++) { c = ARRAY_ITEM(&clients, i); if (c != NULL && c->session != NULL && c->session->curw->window == w) server_redraw_client(c); } } void server_redraw_window_all(struct window *w) { struct client *c; u_int i; for (i = 0; i < ARRAY_LENGTH(&clients); i++) { c = ARRAY_ITEM(&clients, i); if (c == NULL || c->session == NULL) continue; if (session_has(c->session, w)) server_redraw_client(c); } } void server_status_window_cur(struct window *w) { struct client *c; u_int i; for (i = 0; i < ARRAY_LENGTH(&clients); i++) { c = ARRAY_ITEM(&clients, i); if (c != NULL && c->session != NULL && c->session->curw->window == w) server_status_client(c); } } void server_status_window_all(struct window *w) { struct client *c; u_int i; for (i = 0; i < ARRAY_LENGTH(&clients); i++) { c = ARRAY_ITEM(&clients, i); if (c == NULL || c->session == NULL) continue; if (session_has(c->session, w)) server_status_client(c); } } void printflike2 server_write_message(struct client *c, const char *fmt, ...) { struct window *w = c->session->curw->window; struct screen_draw_ctx ctx; struct hdr hdr; va_list ap; char *msg; size_t size; u_int i; buffer_ensure(c->out, sizeof hdr); buffer_add(c->out, sizeof hdr); size = BUFFER_USED(c->out); screen_draw_start(&ctx, &w->screen, c->out, 0, 0); screen_draw_move(&ctx, 0, c->sy - 1); screen_draw_set_attributes(&ctx, ATTR_REVERSE, 0x88); va_start(ap, fmt); xvasprintf(&msg, fmt, ap); va_end(ap); if (strlen(msg) > c->sx - 1) msg[c->sx - 1] = '\0'; buffer_write(c->out, msg, strlen(msg)); for (i = strlen(msg); i < c->sx; i++) input_store8(c->out, ' '); xfree(msg); size = BUFFER_USED(c->out) - size; hdr.type = MSG_DATA; hdr.size = size; memcpy(BUFFER_IN(c->out) - size - sizeof hdr, &hdr, sizeof hdr); hdr.type = MSG_PAUSE; hdr.size = 0; buffer_write(c->out, &hdr, sizeof hdr); buffer_ensure(c->out, sizeof hdr); buffer_add(c->out, sizeof hdr); size = BUFFER_USED(c->out); screen_draw_stop(&ctx); if (status_lines == 0) window_draw(w, c->out, c->sy - 1, 1); else status_write(c); size = BUFFER_USED(c->out) - size; hdr.type = MSG_DATA; hdr.size = size; memcpy(BUFFER_IN(c->out) - size - sizeof hdr, &hdr, sizeof hdr); }