4 * Copyright (c) 2011-2012 David Herrmann <dh.herrmann@googlemail.com>
5 * Copyright (c) 2011 University of Tuebingen
7 * Permission is hereby granted, free of charge, to any person obtaining
8 * a copy of this software and associated documentation files
9 * (the "Software"), to deal in the Software without restriction, including
10 * without limitation the rights to use, copy, modify, merge, publish,
11 * distribute, sublicense, and/or sell copies of the Software, and to
12 * permit persons to whom the Software is furnished to do so, subject to
13 * the following conditions:
15 * The above copyright notice and this permission notice shall be included
16 * in all copies or substantial portions of the Software.
18 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
19 * OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
20 * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.
21 * IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY
22 * CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT,
23 * TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE
24 * SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
29 * A terminal gets assigned an input stream and several output objects and then
30 * runs a fully functional terminal emulation on it.
39 #include "kmscon_conf.h"
40 #include "kmscon_seat.h"
41 #include "kmscon_terminal.h"
43 #include "shl_dlist.h"
46 #include "tsm_screen.h"
48 #include "uterm_input.h"
49 #include "uterm_video.h"
51 #define LOG_SUBSYSTEM "terminal"
54 struct shl_dlist list;
55 struct kmscon_terminal *term;
56 struct uterm_display *disp;
57 struct kmscon_text *txt;
63 struct kmscon_terminal {
65 struct ev_eloop *eloop;
66 struct uterm_input *input;
70 struct conf_ctx *conf_ctx;
71 struct kmscon_conf_t *conf;
72 struct kmscon_session *session;
74 struct shl_dlist screens;
75 unsigned int min_cols;
76 unsigned int min_rows;
78 struct tsm_screen *console;
80 struct kmscon_pty *pty;
83 struct kmscon_font_attr font_attr;
84 struct kmscon_font *font;
85 struct kmscon_font *bold_font;
88 static void do_clear_margins(struct screen *scr)
90 unsigned int w, h, sw, sh;
91 struct uterm_mode *mode;
94 mode = uterm_display_get_current(scr->disp);
98 sw = uterm_mode_get_width(mode);
99 sh = uterm_mode_get_height(mode);
100 w = scr->txt->font->attr.width * scr->txt->cols;
101 h = scr->txt->font->attr.height * scr->txt->rows;
106 uterm_display_fill(scr->disp, 0, 0, 0,
110 uterm_display_fill(scr->disp, 0, 0, 0,
115 static void do_redraw_screen(struct screen *scr)
119 if (!scr->term->awake)
122 scr->pending = false;
123 do_clear_margins(scr);
124 tsm_screen_draw(scr->term->console, kmscon_text_prepare_cb,
125 kmscon_text_draw_cb, kmscon_text_render_cb, scr->txt);
126 ret = uterm_display_swap(scr->disp, false);
128 log_warning("cannot swap display %p", scr->disp);
132 scr->swapping = true;
135 static void redraw_screen(struct screen *scr)
137 if (!scr->term->awake)
143 do_redraw_screen(scr);
146 static void redraw_all(struct kmscon_terminal *term)
148 struct shl_dlist *iter;
154 shl_dlist_for_each(iter, &term->screens) {
155 scr = shl_dlist_entry(iter, struct screen, list);
160 static void redraw_all_test(struct kmscon_terminal *term)
162 struct shl_dlist *iter;
168 shl_dlist_for_each(iter, &term->screens) {
169 scr = shl_dlist_entry(iter, struct screen, list);
170 if (uterm_display_is_swapping(scr->disp))
171 scr->swapping = true;
176 static void display_event(struct uterm_display *disp,
177 struct uterm_display_event *ev, void *data)
179 struct screen *scr = data;
181 if (ev->action != UTERM_PAGE_FLIP)
184 scr->swapping = false;
186 do_redraw_screen(scr);
191 * We support multiple monitors per terminal. As some software-rendering
192 * backends to not support scaling, we always use the smallest cols/rows that are
193 * provided so wider displays will have black margins.
194 * This can be extended to support scaling but that would mean we need to check
195 * whether the text-renderer backend supports that, first (TODO).
197 * If @force is true, then the console/pty are notified even though the size did
198 * not changed. If @notify is false, then console/pty are not notified even
199 * though the size might have changed. force = true and notify = false doesn't
200 * make any sense, though.
202 static void terminal_resize(struct kmscon_terminal *term,
203 unsigned int cols, unsigned int rows,
204 bool force, bool notify)
208 if (!term->min_cols || (cols > 0 && cols < term->min_cols)) {
209 term->min_cols = cols;
212 if (!term->min_rows || (rows > 0 && rows < term->min_rows)) {
213 term->min_rows = rows;
217 if (!notify || (!resize && !force))
219 if (!term->min_cols || !term->min_rows)
222 tsm_screen_resize(term->console, term->min_cols, term->min_rows);
223 kmscon_pty_resize(term->pty, term->min_cols, term->min_rows);
227 static int font_set(struct kmscon_terminal *term)
230 struct kmscon_font *font, *bold_font;
231 struct shl_dlist *iter;
234 term->font_attr.bold = false;
235 ret = kmscon_font_find(&font, &term->font_attr,
236 term->conf->font_engine);
240 term->font_attr.bold = true;
241 ret = kmscon_font_find(&bold_font, &term->font_attr,
242 term->conf->font_engine);
244 log_warning("cannot create bold font: %d", ret);
246 kmscon_font_ref(bold_font);
249 kmscon_font_unref(term->bold_font);
250 kmscon_font_unref(term->font);
252 term->bold_font = bold_font;
256 shl_dlist_for_each(iter, &term->screens) {
257 ent = shl_dlist_entry(iter, struct screen, list);
259 ret = kmscon_text_set(ent->txt, font, bold_font, ent->disp);
261 log_warning("cannot change text-renderer font: %d",
264 terminal_resize(term,
265 kmscon_text_get_cols(ent->txt),
266 kmscon_text_get_rows(ent->txt),
270 terminal_resize(term, 0, 0, true, true);
274 static int add_display(struct kmscon_terminal *term, struct uterm_display *disp)
276 struct shl_dlist *iter;
282 shl_dlist_for_each(iter, &term->screens) {
283 scr = shl_dlist_entry(iter, struct screen, list);
284 if (scr->disp == disp)
288 scr = malloc(sizeof(*scr));
290 log_error("cannot allocate memory for display %p", disp);
293 memset(scr, 0, sizeof(*scr));
297 ret = uterm_display_register_cb(scr->disp, display_event, scr);
299 log_error("cannot register display callback: %d", ret);
303 ret = uterm_display_use(scr->disp, &opengl);
304 if (term->conf->render_engine)
305 be = term->conf->render_engine;
306 else if (ret >= 0 && opengl)
311 ret = kmscon_text_new(&scr->txt, be);
313 log_error("cannot create text-renderer");
317 ret = kmscon_text_set(scr->txt, term->font, term->bold_font,
320 log_error("cannot set text-renderer parameters");
324 terminal_resize(term,
325 kmscon_text_get_cols(scr->txt),
326 kmscon_text_get_rows(scr->txt),
329 shl_dlist_link(&term->screens, &scr->list);
331 log_debug("added display %p to terminal %p", disp, term);
333 uterm_display_ref(scr->disp);
337 kmscon_text_unref(scr->txt);
339 uterm_display_unregister_cb(scr->disp, display_event, scr);
345 static void free_screen(struct screen *scr, bool update)
347 struct shl_dlist *iter;
349 struct kmscon_terminal *term = scr->term;
351 log_debug("destroying terminal screen %p", scr);
352 shl_dlist_unlink(&scr->list);
353 kmscon_text_unref(scr->txt);
354 uterm_display_unregister_cb(scr->disp, display_event, scr);
355 uterm_display_unref(scr->disp);
363 shl_dlist_for_each(iter, &term->screens) {
364 ent = shl_dlist_entry(iter, struct screen, list);
365 terminal_resize(term,
366 kmscon_text_get_cols(ent->txt),
367 kmscon_text_get_rows(ent->txt),
371 terminal_resize(term, 0, 0, true, true);
374 static void rm_display(struct kmscon_terminal *term, struct uterm_display *disp)
376 struct shl_dlist *iter;
379 shl_dlist_for_each(iter, &term->screens) {
380 scr = shl_dlist_entry(iter, struct screen, list);
381 if (scr->disp == disp)
385 if (iter == &term->screens)
388 log_debug("removed display %p from terminal %p", disp, term);
389 free_screen(scr, true);
392 static void input_event(struct uterm_input *input,
393 struct uterm_input_event *ev,
396 struct kmscon_terminal *term = data;
398 if (!term->opened || !term->awake || ev->handled)
401 if (conf_grab_matches(term->conf->grab_scroll_up,
402 ev->mods, ev->num_syms, ev->keysyms)) {
403 tsm_screen_sb_up(term->console, 1);
408 if (conf_grab_matches(term->conf->grab_scroll_down,
409 ev->mods, ev->num_syms, ev->keysyms)) {
410 tsm_screen_sb_down(term->console, 1);
415 if (conf_grab_matches(term->conf->grab_page_up,
416 ev->mods, ev->num_syms, ev->keysyms)) {
417 tsm_screen_sb_page_up(term->console, 1);
422 if (conf_grab_matches(term->conf->grab_page_down,
423 ev->mods, ev->num_syms, ev->keysyms)) {
424 tsm_screen_sb_page_down(term->console, 1);
429 if (conf_grab_matches(term->conf->grab_zoom_in,
430 ev->mods, ev->num_syms, ev->keysyms)) {
432 if (term->font_attr.points + 1 < term->font_attr.points)
435 ++term->font_attr.points;
437 --term->font_attr.points;
440 if (conf_grab_matches(term->conf->grab_zoom_out,
441 ev->mods, ev->num_syms, ev->keysyms)) {
443 if (term->font_attr.points <= 1)
446 --term->font_attr.points;
448 ++term->font_attr.points;
452 /* TODO: xkbcommon supports multiple keysyms, but it is currently
453 * unclear how this feature will be used. There is no keymap, which
455 if (ev->num_syms > 1)
458 if (tsm_vte_handle_keyboard(term->vte, ev->keysyms[0], ev->ascii,
459 ev->mods, ev->codepoints[0])) {
460 tsm_screen_sb_reset(term->console);
466 static void rm_all_screens(struct kmscon_terminal *term)
468 struct shl_dlist *iter;
471 while ((iter = term->screens.next) != &term->screens) {
472 scr = shl_dlist_entry(iter, struct screen, list);
473 free_screen(scr, false);
480 static int terminal_open(struct kmscon_terminal *term)
483 unsigned short width, height;
488 tsm_vte_hard_reset(term->vte);
489 width = tsm_screen_get_width(term->console);
490 height = tsm_screen_get_height(term->console);
491 ret = kmscon_pty_open(term->pty, width, height);
500 static void terminal_close(struct kmscon_terminal *term)
502 kmscon_pty_close(term->pty);
503 term->opened = false;
506 static void terminal_destroy(struct kmscon_terminal *term)
508 log_debug("free terminal object %p", term);
510 terminal_close(term);
511 rm_all_screens(term);
512 uterm_input_unregister_cb(term->input, input_event, term);
513 ev_eloop_rm_fd(term->ptyfd);
514 kmscon_pty_unref(term->pty);
515 kmscon_font_unref(term->bold_font);
516 kmscon_font_unref(term->font);
517 tsm_vte_unref(term->vte);
518 tsm_screen_unref(term->console);
519 uterm_input_unref(term->input);
520 ev_eloop_unref(term->eloop);
524 static int session_event(struct kmscon_session *session,
525 struct kmscon_session_event *ev, void *data)
527 struct kmscon_terminal *term = data;
530 case KMSCON_SESSION_DISPLAY_NEW:
531 add_display(term, ev->disp);
533 case KMSCON_SESSION_DISPLAY_GONE:
534 rm_display(term, ev->disp);
536 case KMSCON_SESSION_ACTIVATE:
540 redraw_all_test(term);
542 case KMSCON_SESSION_DEACTIVATE:
545 case KMSCON_SESSION_UNREGISTER:
546 terminal_destroy(term);
553 static void pty_input(struct kmscon_pty *pty, const char *u8, size_t len,
556 struct kmscon_terminal *term = data;
559 terminal_close(term);
562 tsm_vte_input(term->vte, u8, len);
567 static void pty_event(struct ev_fd *fd, int mask, void *data)
569 struct kmscon_terminal *term = data;
571 kmscon_pty_dispatch(term->pty);
574 static void write_event(struct tsm_vte *vte, const char *u8, size_t len,
577 struct kmscon_terminal *term = data;
579 kmscon_pty_write(term->pty, u8, len);
582 int kmscon_terminal_register(struct kmscon_session **out,
583 struct kmscon_seat *seat)
585 struct kmscon_terminal *term;
591 term = malloc(sizeof(*term));
595 memset(term, 0, sizeof(*term));
597 term->eloop = kmscon_seat_get_eloop(seat);
598 term->input = kmscon_seat_get_input(seat);
599 shl_dlist_init(&term->screens);
601 term->conf_ctx = kmscon_seat_get_conf(seat);
602 term->conf = conf_ctx_get_mem(term->conf_ctx);
604 strncpy(term->font_attr.name, term->conf->font_name,
605 KMSCON_FONT_MAX_NAME - 1);
606 term->font_attr.ppi = term->conf->font_ppi;
607 term->font_attr.points = term->conf->font_size;
609 ret = tsm_screen_new(&term->console, log_llog, NULL);
612 tsm_screen_set_max_sb(term->console, term->conf->sb_size);
613 if (term->conf->render_timing)
614 tsm_screen_set_opts(term->console,
615 TSM_SCREEN_OPT_RENDER_TIMING);
617 ret = tsm_vte_new(&term->vte, term->console, write_event, term,
621 tsm_vte_set_palette(term->vte, term->conf->palette);
623 ret = font_set(term);
627 ret = kmscon_pty_new(&term->pty, pty_input, term);
631 kmscon_pty_set_env_reset(term->pty, term->conf->reset_env);
633 ret = kmscon_pty_set_term(term->pty, term->conf->term);
637 ret = kmscon_pty_set_colorterm(term->pty, "kmscon");
641 ret = kmscon_pty_set_argv(term->pty, term->conf->argv);
645 ret = kmscon_pty_set_seat(term->pty, kmscon_seat_get_name(seat));
649 ret = ev_eloop_new_fd(term->eloop, &term->ptyfd,
650 kmscon_pty_get_fd(term->pty),
651 EV_READABLE, pty_event, term);
655 ret = uterm_input_register_cb(term->input, input_event, term);
659 ret = kmscon_seat_register_session(seat, &term->session, session_event,
662 log_error("cannot register session for terminal: %d", ret);
666 ev_eloop_ref(term->eloop);
667 uterm_input_ref(term->input);
668 *out = term->session;
669 log_debug("new terminal object %p", term);
673 uterm_input_unregister_cb(term->input, input_event, term);
675 ev_eloop_rm_fd(term->ptyfd);
677 kmscon_pty_unref(term->pty);
679 kmscon_font_unref(term->bold_font);
680 kmscon_font_unref(term->font);
682 tsm_vte_unref(term->vte);
684 tsm_screen_unref(term->console);