Zbigniew Jędrzejewski-Szmek 62fe94
From ce7b9f50c3fadbad22feeb28e4429ad9bee02bcc Mon Sep 17 00:00:00 2001
Zbigniew Jędrzejewski-Szmek 62fe94
From: David Herrmann <dh.herrmann@gmail.com>
Zbigniew Jędrzejewski-Szmek 62fe94
Date: Fri, 3 Oct 2014 15:58:44 +0200
Zbigniew Jędrzejewski-Szmek 62fe94
Subject: [PATCH] console: add user console daemon
Zbigniew Jędrzejewski-Szmek 62fe94
Zbigniew Jędrzejewski-Szmek 62fe94
This adds a first draft of systemd-consoled. This is still missing a lot
Zbigniew Jędrzejewski-Szmek 62fe94
of features and does some rather primitive rendering. However, it shows
Zbigniew Jędrzejewski-Szmek 62fe94
the direction this code is going and serves as basis for further testing.
Zbigniew Jędrzejewski-Szmek 62fe94
Zbigniew Jędrzejewski-Szmek 62fe94
The systemd-consoled binary should be run as `systemd --user' unit. It
Zbigniew Jędrzejewski-Szmek 62fe94
automatically picks up any session marked as Desktop=SYSTEMD-CONSOLE.
Zbigniew Jędrzejewski-Szmek 62fe94
Therefore, you can use any login-manager you want (ranging from /bin/login
Zbigniew Jędrzejewski-Szmek 62fe94
to gdm) to create sessions for systemd-consoled. However, the sessions
Zbigniew Jędrzejewski-Szmek 62fe94
managers must be prepared to set the Desktop= variable properly.
Zbigniew Jędrzejewski-Szmek 62fe94
Zbigniew Jędrzejewski-Szmek 62fe94
The user-session is called `systemd-console', only the daemon providing
Zbigniew Jędrzejewski-Szmek 62fe94
the terminal environment is called `systemd-consoled' (mind the 'd').
Zbigniew Jędrzejewski-Szmek 62fe94
Zbigniew Jędrzejewski-Szmek 62fe94
So far, only a single terminal session is provided on each opened
Zbigniew Jędrzejewski-Szmek 62fe94
user-session. However, we support multiple user-sessions (even across
Zbigniew Jędrzejewski-Szmek 62fe94
multiple seats) just fine. In the future, the workspace logic will get
Zbigniew Jędrzejewski-Szmek 62fe94
extended so you can have multiple terminal sessions in a single
Zbigniew Jędrzejewski-Szmek 62fe94
user-session for easier access.
Zbigniew Jędrzejewski-Szmek 62fe94
Zbigniew Jędrzejewski-Szmek 62fe94
Note that this is still experimental! Instructions on how to run it will
Zbigniew Jędrzejewski-Szmek 62fe94
follow shortly.
Zbigniew Jędrzejewski-Szmek 62fe94
---
Zbigniew Jędrzejewski-Szmek 62fe94
 .gitignore                       |   1 +
Zbigniew Jędrzejewski-Szmek 62fe94
 Makefile.am                      |  22 +++
Zbigniew Jędrzejewski-Szmek 62fe94
 src/console/Makefile             |   1 +
Zbigniew Jędrzejewski-Szmek 62fe94
 src/console/consoled-display.c   |  82 +++++++++
Zbigniew Jędrzejewski-Szmek 62fe94
 src/console/consoled-manager.c   | 288 +++++++++++++++++++++++++++++++
Zbigniew Jędrzejewski-Szmek 62fe94
 src/console/consoled-session.c   | 283 ++++++++++++++++++++++++++++++
Zbigniew Jędrzejewski-Szmek 62fe94
 src/console/consoled-terminal.c  | 360 +++++++++++++++++++++++++++++++++++++++
Zbigniew Jędrzejewski-Szmek 62fe94
 src/console/consoled-workspace.c | 168 ++++++++++++++++++
Zbigniew Jędrzejewski-Szmek 62fe94
 src/console/consoled.c           |  67 ++++++++
Zbigniew Jędrzejewski-Szmek 62fe94
 src/console/consoled.h           | 170 ++++++++++++++++++
Zbigniew Jędrzejewski-Szmek 62fe94
 10 files changed, 1442 insertions(+)
Zbigniew Jędrzejewski-Szmek 62fe94
 create mode 120000 src/console/Makefile
Zbigniew Jędrzejewski-Szmek 62fe94
 create mode 100644 src/console/consoled-display.c
Zbigniew Jędrzejewski-Szmek 62fe94
 create mode 100644 src/console/consoled-manager.c
Zbigniew Jędrzejewski-Szmek 62fe94
 create mode 100644 src/console/consoled-session.c
Zbigniew Jędrzejewski-Szmek 62fe94
 create mode 100644 src/console/consoled-terminal.c
Zbigniew Jędrzejewski-Szmek 62fe94
 create mode 100644 src/console/consoled-workspace.c
Zbigniew Jędrzejewski-Szmek 62fe94
 create mode 100644 src/console/consoled.c
Zbigniew Jędrzejewski-Szmek 62fe94
 create mode 100644 src/console/consoled.h
Zbigniew Jędrzejewski-Szmek 62fe94
Zbigniew Jędrzejewski-Szmek 62fe94
diff --git a/.gitignore b/.gitignore
Zbigniew Jędrzejewski-Szmek 62fe94
index cb1af8de5b..f119b574c7 100644
Zbigniew Jędrzejewski-Szmek 62fe94
--- a/.gitignore
Zbigniew Jędrzejewski-Szmek 62fe94
+++ b/.gitignore
Zbigniew Jędrzejewski-Szmek 62fe94
@@ -60,6 +60,7 @@
Zbigniew Jędrzejewski-Szmek 62fe94
 /systemd-cgls
Zbigniew Jędrzejewski-Szmek 62fe94
 /systemd-cgroups-agent
Zbigniew Jędrzejewski-Szmek 62fe94
 /systemd-cgtop
Zbigniew Jędrzejewski-Szmek 62fe94
+/systemd-consoled
Zbigniew Jędrzejewski-Szmek 62fe94
 /systemd-coredump
Zbigniew Jędrzejewski-Szmek 62fe94
 /systemd-cryptsetup
Zbigniew Jędrzejewski-Szmek 62fe94
 /systemd-cryptsetup-generator
Zbigniew Jędrzejewski-Szmek 62fe94
diff --git a/Makefile.am b/Makefile.am
Zbigniew Jędrzejewski-Szmek 62fe94
index 503302851b..60011b7d98 100644
Zbigniew Jędrzejewski-Szmek 62fe94
--- a/Makefile.am
Zbigniew Jędrzejewski-Szmek 62fe94
+++ b/Makefile.am
Zbigniew Jędrzejewski-Szmek 62fe94
@@ -3020,6 +3020,9 @@ if ENABLE_TERMINAL
Zbigniew Jędrzejewski-Szmek 62fe94
 noinst_LTLIBRARIES += \
Zbigniew Jędrzejewski-Szmek 62fe94
 	libsystemd-terminal.la
Zbigniew Jędrzejewski-Szmek 62fe94
 
Zbigniew Jędrzejewski-Szmek 62fe94
+bin_PROGRAMS += \
Zbigniew Jędrzejewski-Szmek 62fe94
+	systemd-consoled
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
 noinst_PROGRAMS += \
Zbigniew Jędrzejewski-Szmek 62fe94
 	systemd-evcat \
Zbigniew Jędrzejewski-Szmek 62fe94
 	systemd-modeset \
Zbigniew Jędrzejewski-Szmek 62fe94
@@ -3068,6 +3071,25 @@ libsystemd_terminal_la_LIBADD = \
Zbigniew Jędrzejewski-Szmek 62fe94
 	libsystemd-shared.la \
Zbigniew Jędrzejewski-Szmek 62fe94
 	$(TERMINAL_LIBS)
Zbigniew Jędrzejewski-Szmek 62fe94
 
Zbigniew Jędrzejewski-Szmek 62fe94
+systemd_consoled_CFLAGS = \
Zbigniew Jędrzejewski-Szmek 62fe94
+	$(AM_CFLAGS) \
Zbigniew Jędrzejewski-Szmek 62fe94
+	$(TERMINAL_CFLAGS)
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+systemd_consoled_SOURCES = \
Zbigniew Jędrzejewski-Szmek 62fe94
+	src/console/consoled.h \
Zbigniew Jędrzejewski-Szmek 62fe94
+	src/console/consoled.c \
Zbigniew Jędrzejewski-Szmek 62fe94
+	src/console/consoled-display.c \
Zbigniew Jędrzejewski-Szmek 62fe94
+	src/console/consoled-manager.c \
Zbigniew Jędrzejewski-Szmek 62fe94
+	src/console/consoled-session.c \
Zbigniew Jędrzejewski-Szmek 62fe94
+	src/console/consoled-terminal.c \
Zbigniew Jędrzejewski-Szmek 62fe94
+	src/console/consoled-workspace.c
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+systemd_consoled_LDADD = \
Zbigniew Jędrzejewski-Szmek 62fe94
+	libsystemd-terminal.la \
Zbigniew Jędrzejewski-Szmek 62fe94
+	libsystemd-internal.la \
Zbigniew Jędrzejewski-Szmek 62fe94
+	libsystemd-shared.la \
Zbigniew Jędrzejewski-Szmek 62fe94
+	$(TERMINAL_LIBS)
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
 systemd_evcat_CFLAGS = \
Zbigniew Jędrzejewski-Szmek 62fe94
 	$(AM_CFLAGS) \
Zbigniew Jędrzejewski-Szmek 62fe94
 	$(TERMINAL_CFLAGS)
Zbigniew Jędrzejewski-Szmek 62fe94
diff --git a/src/console/Makefile b/src/console/Makefile
Zbigniew Jędrzejewski-Szmek 62fe94
new file mode 120000
Zbigniew Jędrzejewski-Szmek 62fe94
index 0000000000..d0b0e8e008
Zbigniew Jędrzejewski-Szmek 62fe94
--- /dev/null
Zbigniew Jędrzejewski-Szmek 62fe94
+++ b/src/console/Makefile
Zbigniew Jędrzejewski-Szmek 62fe94
@@ -0,0 +1 @@
Zbigniew Jędrzejewski-Szmek 62fe94
+../Makefile
Zbigniew Jędrzejewski-Szmek 62fe94
\ No newline at end of file
Zbigniew Jędrzejewski-Szmek 62fe94
diff --git a/src/console/consoled-display.c b/src/console/consoled-display.c
Zbigniew Jędrzejewski-Szmek 62fe94
new file mode 100644
Zbigniew Jędrzejewski-Szmek 62fe94
index 0000000000..a30a2f1022
Zbigniew Jędrzejewski-Szmek 62fe94
--- /dev/null
Zbigniew Jędrzejewski-Szmek 62fe94
+++ b/src/console/consoled-display.c
Zbigniew Jędrzejewski-Szmek 62fe94
@@ -0,0 +1,82 @@
Zbigniew Jędrzejewski-Szmek 62fe94
+/*-*- Mode: C; c-basic-offset: 8; indent-tabs-mode: nil -*-*/
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+/***
Zbigniew Jędrzejewski-Szmek 62fe94
+  This file is part of systemd.
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+  Copyright 2014 David Herrmann <dh.herrmann@gmail.com>
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+  systemd is free software; you can redistribute it and/or modify it
Zbigniew Jędrzejewski-Szmek 62fe94
+  under the terms of the GNU Lesser General Public License as published by
Zbigniew Jędrzejewski-Szmek 62fe94
+  the Free Software Foundation; either version 2.1 of the License, or
Zbigniew Jędrzejewski-Szmek 62fe94
+  (at your option) any later version.
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+  systemd is distributed in the hope that it will be useful, but
Zbigniew Jędrzejewski-Szmek 62fe94
+  WITHOUT ANY WARRANTY; without even the implied warranty of
Zbigniew Jędrzejewski-Szmek 62fe94
+  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
Zbigniew Jędrzejewski-Szmek 62fe94
+  Lesser General Public License for more details.
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+  You should have received a copy of the GNU Lesser General Public License
Zbigniew Jędrzejewski-Szmek 62fe94
+  along with systemd; If not, see <http://www.gnu.org/licenses/>.
Zbigniew Jędrzejewski-Szmek 62fe94
+***/
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+#include <errno.h>
Zbigniew Jędrzejewski-Szmek 62fe94
+#include <inttypes.h>
Zbigniew Jędrzejewski-Szmek 62fe94
+#include <stdlib.h>
Zbigniew Jędrzejewski-Szmek 62fe94
+#include "consoled.h"
Zbigniew Jędrzejewski-Szmek 62fe94
+#include "grdev.h"
Zbigniew Jędrzejewski-Szmek 62fe94
+#include "list.h"
Zbigniew Jędrzejewski-Szmek 62fe94
+#include "macro.h"
Zbigniew Jędrzejewski-Szmek 62fe94
+#include "util.h"
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+int display_new(Display **out, Session *s, grdev_display *display) {
Zbigniew Jędrzejewski-Szmek 62fe94
+        _cleanup_(display_freep) Display *d = NULL;
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        assert(out);
Zbigniew Jędrzejewski-Szmek 62fe94
+        assert(s);
Zbigniew Jędrzejewski-Szmek 62fe94
+        assert(display);
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        d = new0(Display, 1);
Zbigniew Jędrzejewski-Szmek 62fe94
+        if (!d)
Zbigniew Jędrzejewski-Szmek 62fe94
+                return -ENOMEM;
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        d->session = s;
Zbigniew Jędrzejewski-Szmek 62fe94
+        d->grdev = display;
Zbigniew Jędrzejewski-Szmek 62fe94
+        d->width = grdev_display_get_width(display);
Zbigniew Jędrzejewski-Szmek 62fe94
+        d->height = grdev_display_get_height(display);
Zbigniew Jędrzejewski-Szmek 62fe94
+        LIST_PREPEND(displays_by_session, d->session->display_list, d);
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        grdev_display_enable(display);
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        *out = d;
Zbigniew Jędrzejewski-Szmek 62fe94
+        d = NULL;
Zbigniew Jędrzejewski-Szmek 62fe94
+        return 0;
Zbigniew Jędrzejewski-Szmek 62fe94
+}
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+Display *display_free(Display *d) {
Zbigniew Jędrzejewski-Szmek 62fe94
+        if (!d)
Zbigniew Jędrzejewski-Szmek 62fe94
+                return NULL;
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        LIST_REMOVE(displays_by_session, d->session->display_list, d);
Zbigniew Jędrzejewski-Szmek 62fe94
+        free(d);
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        return NULL;
Zbigniew Jędrzejewski-Szmek 62fe94
+}
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+void display_refresh(Display *d) {
Zbigniew Jędrzejewski-Szmek 62fe94
+        assert(d);
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        d->width = grdev_display_get_width(d->grdev);
Zbigniew Jędrzejewski-Szmek 62fe94
+        d->height = grdev_display_get_height(d->grdev);
Zbigniew Jędrzejewski-Szmek 62fe94
+}
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+void display_render(Display *d, Workspace *w) {
Zbigniew Jędrzejewski-Szmek 62fe94
+        const grdev_display_target *target;
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        assert(d);
Zbigniew Jędrzejewski-Szmek 62fe94
+        assert(w);
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        GRDEV_DISPLAY_FOREACH_TARGET(d->grdev, target) {
Zbigniew Jędrzejewski-Szmek 62fe94
+                if (workspace_draw(w, target))
Zbigniew Jędrzejewski-Szmek 62fe94
+                        grdev_display_flip_target(d->grdev, target);
Zbigniew Jędrzejewski-Szmek 62fe94
+        }
Zbigniew Jędrzejewski-Szmek 62fe94
+}
Zbigniew Jędrzejewski-Szmek 62fe94
diff --git a/src/console/consoled-manager.c b/src/console/consoled-manager.c
Zbigniew Jędrzejewski-Szmek 62fe94
new file mode 100644
Zbigniew Jędrzejewski-Szmek 62fe94
index 0000000000..8f3823fe46
Zbigniew Jędrzejewski-Szmek 62fe94
--- /dev/null
Zbigniew Jędrzejewski-Szmek 62fe94
+++ b/src/console/consoled-manager.c
Zbigniew Jędrzejewski-Szmek 62fe94
@@ -0,0 +1,288 @@
Zbigniew Jędrzejewski-Szmek 62fe94
+/*-*- Mode: C; c-basic-offset: 8; indent-tabs-mode: nil -*-*/
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+/***
Zbigniew Jędrzejewski-Szmek 62fe94
+  This file is part of systemd.
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+  Copyright 2014 David Herrmann <dh.herrmann@gmail.com>
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+  systemd is free software; you can redistribute it and/or modify it
Zbigniew Jędrzejewski-Szmek 62fe94
+  under the terms of the GNU Lesser General Public License as published by
Zbigniew Jędrzejewski-Szmek 62fe94
+  the Free Software Foundation; either version 2.1 of the License, or
Zbigniew Jędrzejewski-Szmek 62fe94
+  (at your option) any later version.
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+  systemd is distributed in the hope that it will be useful, but
Zbigniew Jędrzejewski-Szmek 62fe94
+  WITHOUT ANY WARRANTY; without even the implied warranty of
Zbigniew Jędrzejewski-Szmek 62fe94
+  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
Zbigniew Jędrzejewski-Szmek 62fe94
+  Lesser General Public License for more details.
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+  You should have received a copy of the GNU Lesser General Public License
Zbigniew Jędrzejewski-Szmek 62fe94
+  along with systemd; If not, see <http://www.gnu.org/licenses/>.
Zbigniew Jędrzejewski-Szmek 62fe94
+***/
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+#include <errno.h>
Zbigniew Jędrzejewski-Szmek 62fe94
+#include <libudev.h>
Zbigniew Jędrzejewski-Szmek 62fe94
+#include <stdlib.h>
Zbigniew Jędrzejewski-Szmek 62fe94
+#include <string.h>
Zbigniew Jędrzejewski-Szmek 62fe94
+#include "consoled.h"
Zbigniew Jędrzejewski-Szmek 62fe94
+#include "grdev.h"
Zbigniew Jędrzejewski-Szmek 62fe94
+#include "idev.h"
Zbigniew Jędrzejewski-Szmek 62fe94
+#include "log.h"
Zbigniew Jędrzejewski-Szmek 62fe94
+#include "sd-bus.h"
Zbigniew Jędrzejewski-Szmek 62fe94
+#include "sd-daemon.h"
Zbigniew Jędrzejewski-Szmek 62fe94
+#include "sd-event.h"
Zbigniew Jędrzejewski-Szmek 62fe94
+#include "sd-login.h"
Zbigniew Jędrzejewski-Szmek 62fe94
+#include "sysview.h"
Zbigniew Jędrzejewski-Szmek 62fe94
+#include "unifont.h"
Zbigniew Jędrzejewski-Szmek 62fe94
+#include "util.h"
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+int manager_new(Manager **out) {
Zbigniew Jędrzejewski-Szmek 62fe94
+        _cleanup_(manager_freep) Manager *m = NULL;
Zbigniew Jędrzejewski-Szmek 62fe94
+        int r;
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        assert(out);
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        m = new0(Manager, 1);
Zbigniew Jędrzejewski-Szmek 62fe94
+        if (!m)
Zbigniew Jędrzejewski-Szmek 62fe94
+                return -ENOMEM;
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        r = sd_event_default(&m->event);
Zbigniew Jędrzejewski-Szmek 62fe94
+        if (r < 0)
Zbigniew Jędrzejewski-Szmek 62fe94
+                return r;
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        r = sd_event_set_watchdog(m->event, true);
Zbigniew Jędrzejewski-Szmek 62fe94
+        if (r < 0)
Zbigniew Jędrzejewski-Szmek 62fe94
+                return r;
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        r = sigprocmask_many(SIG_BLOCK, SIGTERM, SIGQUIT, SIGINT, SIGWINCH, SIGCHLD, -1);
Zbigniew Jędrzejewski-Szmek 62fe94
+        if (r < 0)
Zbigniew Jędrzejewski-Szmek 62fe94
+                return r;
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        r = sd_event_add_signal(m->event, NULL, SIGTERM, NULL, NULL);
Zbigniew Jędrzejewski-Szmek 62fe94
+        if (r < 0)
Zbigniew Jędrzejewski-Szmek 62fe94
+                return r;
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        r = sd_event_add_signal(m->event, NULL, SIGQUIT, NULL, NULL);
Zbigniew Jędrzejewski-Szmek 62fe94
+        if (r < 0)
Zbigniew Jędrzejewski-Szmek 62fe94
+                return r;
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        r = sd_event_add_signal(m->event, NULL, SIGINT, NULL, NULL);
Zbigniew Jędrzejewski-Szmek 62fe94
+        if (r < 0)
Zbigniew Jędrzejewski-Szmek 62fe94
+                return r;
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        r = sd_bus_open_system(&m->sysbus);
Zbigniew Jędrzejewski-Szmek 62fe94
+        if (r < 0)
Zbigniew Jędrzejewski-Szmek 62fe94
+                return r;
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        r = sd_bus_attach_event(m->sysbus, m->event, SD_EVENT_PRIORITY_NORMAL);
Zbigniew Jędrzejewski-Szmek 62fe94
+        if (r < 0)
Zbigniew Jędrzejewski-Szmek 62fe94
+                return r;
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        r = unifont_new(&m->uf);
Zbigniew Jędrzejewski-Szmek 62fe94
+        if (r < 0)
Zbigniew Jędrzejewski-Szmek 62fe94
+                return r;
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        r = sysview_context_new(&m->sysview,
Zbigniew Jędrzejewski-Szmek 62fe94
+                                SYSVIEW_CONTEXT_SCAN_LOGIND |
Zbigniew Jędrzejewski-Szmek 62fe94
+                                SYSVIEW_CONTEXT_SCAN_EVDEV |
Zbigniew Jędrzejewski-Szmek 62fe94
+                                SYSVIEW_CONTEXT_SCAN_DRM,
Zbigniew Jędrzejewski-Szmek 62fe94
+                                m->event,
Zbigniew Jędrzejewski-Szmek 62fe94
+                                m->sysbus,
Zbigniew Jędrzejewski-Szmek 62fe94
+                                NULL);
Zbigniew Jędrzejewski-Szmek 62fe94
+        if (r < 0)
Zbigniew Jędrzejewski-Szmek 62fe94
+                return r;
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        r = grdev_context_new(&m->grdev, m->event, m->sysbus);
Zbigniew Jędrzejewski-Szmek 62fe94
+        if (r < 0)
Zbigniew Jędrzejewski-Szmek 62fe94
+                return r;
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        r = idev_context_new(&m->idev, m->event, m->sysbus);
Zbigniew Jędrzejewski-Szmek 62fe94
+        if (r < 0)
Zbigniew Jędrzejewski-Szmek 62fe94
+                return r;
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        *out = m;
Zbigniew Jędrzejewski-Szmek 62fe94
+        m = NULL;
Zbigniew Jędrzejewski-Szmek 62fe94
+        return 0;
Zbigniew Jędrzejewski-Szmek 62fe94
+}
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+Manager *manager_free(Manager *m) {
Zbigniew Jędrzejewski-Szmek 62fe94
+        if (!m)
Zbigniew Jędrzejewski-Szmek 62fe94
+                return NULL;
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        assert(!m->workspace_list);
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        m->idev = idev_context_unref(m->idev);
Zbigniew Jędrzejewski-Szmek 62fe94
+        m->grdev = grdev_context_unref(m->grdev);
Zbigniew Jędrzejewski-Szmek 62fe94
+        m->sysview = sysview_context_free(m->sysview);
Zbigniew Jędrzejewski-Szmek 62fe94
+        m->uf = unifont_unref(m->uf);
Zbigniew Jędrzejewski-Szmek 62fe94
+        m->sysbus = sd_bus_unref(m->sysbus);
Zbigniew Jędrzejewski-Szmek 62fe94
+        m->event = sd_event_unref(m->event);
Zbigniew Jędrzejewski-Szmek 62fe94
+        free(m);
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        return NULL;
Zbigniew Jędrzejewski-Szmek 62fe94
+}
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+static int manager_sysview_session_filter(Manager *m, sysview_event *event) {
Zbigniew Jędrzejewski-Szmek 62fe94
+        const char *sid = event->session_filter.id;
Zbigniew Jędrzejewski-Szmek 62fe94
+        _cleanup_free_ char *desktop = NULL;
Zbigniew Jędrzejewski-Szmek 62fe94
+        int r;
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        assert(sid);
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        r = sd_session_get_desktop(sid, &desktop);
Zbigniew Jędrzejewski-Szmek 62fe94
+        if (r < 0)
Zbigniew Jędrzejewski-Szmek 62fe94
+                return 0;
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        return streq(desktop, "SYSTEMD-CONSOLE");
Zbigniew Jędrzejewski-Szmek 62fe94
+}
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+static int manager_sysview_session_add(Manager *m, sysview_event *event) {
Zbigniew Jędrzejewski-Szmek 62fe94
+        sysview_session *session = event->session_add.session;
Zbigniew Jędrzejewski-Szmek 62fe94
+        Session *s;
Zbigniew Jędrzejewski-Szmek 62fe94
+        int r;
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        r = sysview_session_take_control(session);
Zbigniew Jędrzejewski-Szmek 62fe94
+        if (r < 0) {
Zbigniew Jędrzejewski-Szmek 62fe94
+                log_error("Cannot request session control on '%s': %s",
Zbigniew Jędrzejewski-Szmek 62fe94
+                          sysview_session_get_name(session), strerror(-r));
Zbigniew Jędrzejewski-Szmek 62fe94
+                return r;
Zbigniew Jędrzejewski-Szmek 62fe94
+        }
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        r = session_new(&s, m, session);
Zbigniew Jędrzejewski-Szmek 62fe94
+        if (r < 0) {
Zbigniew Jędrzejewski-Szmek 62fe94
+                log_error("Cannot create session on '%s': %s",
Zbigniew Jędrzejewski-Szmek 62fe94
+                          sysview_session_get_name(session), strerror(-r));
Zbigniew Jędrzejewski-Szmek 62fe94
+                sysview_session_release_control(session);
Zbigniew Jędrzejewski-Szmek 62fe94
+                return r;
Zbigniew Jędrzejewski-Szmek 62fe94
+        }
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        sysview_session_set_userdata(session, s);
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        return 0;
Zbigniew Jędrzejewski-Szmek 62fe94
+}
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+static int manager_sysview_session_remove(Manager *m, sysview_event *event) {
Zbigniew Jędrzejewski-Szmek 62fe94
+        sysview_session *session = event->session_remove.session;
Zbigniew Jędrzejewski-Szmek 62fe94
+        Session *s;
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        s = sysview_session_get_userdata(session);
Zbigniew Jędrzejewski-Szmek 62fe94
+        if (!s)
Zbigniew Jędrzejewski-Szmek 62fe94
+                return 0;
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        session_free(s);
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        return 0;
Zbigniew Jędrzejewski-Szmek 62fe94
+}
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+static int manager_sysview_session_attach(Manager *m, sysview_event *event) {
Zbigniew Jędrzejewski-Szmek 62fe94
+        sysview_session *session = event->session_attach.session;
Zbigniew Jędrzejewski-Szmek 62fe94
+        sysview_device *device = event->session_attach.device;
Zbigniew Jędrzejewski-Szmek 62fe94
+        Session *s;
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        s = sysview_session_get_userdata(session);
Zbigniew Jędrzejewski-Szmek 62fe94
+        if (!s)
Zbigniew Jędrzejewski-Szmek 62fe94
+                return 0;
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        session_add_device(s, device);
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        return 0;
Zbigniew Jędrzejewski-Szmek 62fe94
+}
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+static int manager_sysview_session_detach(Manager *m, sysview_event *event) {
Zbigniew Jędrzejewski-Szmek 62fe94
+        sysview_session *session = event->session_detach.session;
Zbigniew Jędrzejewski-Szmek 62fe94
+        sysview_device *device = event->session_detach.device;
Zbigniew Jędrzejewski-Szmek 62fe94
+        Session *s;
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        s = sysview_session_get_userdata(session);
Zbigniew Jędrzejewski-Szmek 62fe94
+        if (!s)
Zbigniew Jędrzejewski-Szmek 62fe94
+                return 0;
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        session_remove_device(s, device);
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        return 0;
Zbigniew Jędrzejewski-Szmek 62fe94
+}
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+static int manager_sysview_session_refresh(Manager *m, sysview_event *event) {
Zbigniew Jędrzejewski-Szmek 62fe94
+        sysview_session *session = event->session_refresh.session;
Zbigniew Jędrzejewski-Szmek 62fe94
+        sysview_device *device = event->session_refresh.device;
Zbigniew Jędrzejewski-Szmek 62fe94
+        struct udev_device *ud = event->session_refresh.ud;
Zbigniew Jędrzejewski-Szmek 62fe94
+        Session *s;
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        s = sysview_session_get_userdata(session);
Zbigniew Jędrzejewski-Szmek 62fe94
+        if (!s)
Zbigniew Jędrzejewski-Szmek 62fe94
+                return 0;
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        session_refresh_device(s, device, ud);
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        return 0;
Zbigniew Jędrzejewski-Szmek 62fe94
+}
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+static int manager_sysview_session_control(Manager *m, sysview_event *event) {
Zbigniew Jędrzejewski-Szmek 62fe94
+        sysview_session *session = event->session_control.session;
Zbigniew Jędrzejewski-Szmek 62fe94
+        int error = event->session_control.error;
Zbigniew Jędrzejewski-Szmek 62fe94
+        Session *s;
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        s = sysview_session_get_userdata(session);
Zbigniew Jędrzejewski-Szmek 62fe94
+        if (!s)
Zbigniew Jędrzejewski-Szmek 62fe94
+                return 0;
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        if (error < 0) {
Zbigniew Jędrzejewski-Szmek 62fe94
+                log_error("Cannot take session control on '%s': %s",
Zbigniew Jędrzejewski-Szmek 62fe94
+                          sysview_session_get_name(session), strerror(-error));
Zbigniew Jędrzejewski-Szmek 62fe94
+                session_free(s);
Zbigniew Jędrzejewski-Szmek 62fe94
+                sysview_session_set_userdata(session, NULL);
Zbigniew Jędrzejewski-Szmek 62fe94
+                return -error;
Zbigniew Jędrzejewski-Szmek 62fe94
+        }
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        return 0;
Zbigniew Jędrzejewski-Szmek 62fe94
+}
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+static int manager_sysview_fn(sysview_context *sysview, void *userdata, sysview_event *event) {
Zbigniew Jędrzejewski-Szmek 62fe94
+        Manager *m = userdata;
Zbigniew Jędrzejewski-Szmek 62fe94
+        int r;
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        assert(m);
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        switch (event->type) {
Zbigniew Jędrzejewski-Szmek 62fe94
+        case SYSVIEW_EVENT_SESSION_FILTER:
Zbigniew Jędrzejewski-Szmek 62fe94
+                r = manager_sysview_session_filter(m, event);
Zbigniew Jędrzejewski-Szmek 62fe94
+                break;
Zbigniew Jędrzejewski-Szmek 62fe94
+        case SYSVIEW_EVENT_SESSION_ADD:
Zbigniew Jędrzejewski-Szmek 62fe94
+                r = manager_sysview_session_add(m, event);
Zbigniew Jędrzejewski-Szmek 62fe94
+                break;
Zbigniew Jędrzejewski-Szmek 62fe94
+        case SYSVIEW_EVENT_SESSION_REMOVE:
Zbigniew Jędrzejewski-Szmek 62fe94
+                r = manager_sysview_session_remove(m, event);
Zbigniew Jędrzejewski-Szmek 62fe94
+                break;
Zbigniew Jędrzejewski-Szmek 62fe94
+        case SYSVIEW_EVENT_SESSION_ATTACH:
Zbigniew Jędrzejewski-Szmek 62fe94
+                r = manager_sysview_session_attach(m, event);
Zbigniew Jędrzejewski-Szmek 62fe94
+                break;
Zbigniew Jędrzejewski-Szmek 62fe94
+        case SYSVIEW_EVENT_SESSION_DETACH:
Zbigniew Jędrzejewski-Szmek 62fe94
+                r = manager_sysview_session_detach(m, event);
Zbigniew Jędrzejewski-Szmek 62fe94
+                break;
Zbigniew Jędrzejewski-Szmek 62fe94
+        case SYSVIEW_EVENT_SESSION_REFRESH:
Zbigniew Jędrzejewski-Szmek 62fe94
+                r = manager_sysview_session_refresh(m, event);
Zbigniew Jędrzejewski-Szmek 62fe94
+                break;
Zbigniew Jędrzejewski-Szmek 62fe94
+        case SYSVIEW_EVENT_SESSION_CONTROL:
Zbigniew Jędrzejewski-Szmek 62fe94
+                r = manager_sysview_session_control(m, event);
Zbigniew Jędrzejewski-Szmek 62fe94
+                break;
Zbigniew Jędrzejewski-Szmek 62fe94
+        default:
Zbigniew Jędrzejewski-Szmek 62fe94
+                r = 0;
Zbigniew Jędrzejewski-Szmek 62fe94
+                break;
Zbigniew Jędrzejewski-Szmek 62fe94
+        }
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        return r;
Zbigniew Jędrzejewski-Szmek 62fe94
+}
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+int manager_run(Manager *m) {
Zbigniew Jędrzejewski-Szmek 62fe94
+        int r;
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        assert(m);
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        r = sysview_context_start(m->sysview, manager_sysview_fn, m);
Zbigniew Jędrzejewski-Szmek 62fe94
+        if (r < 0)
Zbigniew Jędrzejewski-Szmek 62fe94
+                return r;
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        r = sd_event_loop(m->event);
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        sysview_context_stop(m->sysview);
Zbigniew Jędrzejewski-Szmek 62fe94
+        return r;
Zbigniew Jędrzejewski-Szmek 62fe94
+}
Zbigniew Jędrzejewski-Szmek 62fe94
diff --git a/src/console/consoled-session.c b/src/console/consoled-session.c
Zbigniew Jędrzejewski-Szmek 62fe94
new file mode 100644
Zbigniew Jędrzejewski-Szmek 62fe94
index 0000000000..8bacacab35
Zbigniew Jędrzejewski-Szmek 62fe94
--- /dev/null
Zbigniew Jędrzejewski-Szmek 62fe94
+++ b/src/console/consoled-session.c
Zbigniew Jędrzejewski-Szmek 62fe94
@@ -0,0 +1,283 @@
Zbigniew Jędrzejewski-Szmek 62fe94
+/*-*- Mode: C; c-basic-offset: 8; indent-tabs-mode: nil -*-*/
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+/***
Zbigniew Jędrzejewski-Szmek 62fe94
+  This file is part of systemd.
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+  Copyright 2014 David Herrmann <dh.herrmann@gmail.com>
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+  systemd is free software; you can redistribute it and/or modify it
Zbigniew Jędrzejewski-Szmek 62fe94
+  under the terms of the GNU Lesser General Public License as published by
Zbigniew Jędrzejewski-Szmek 62fe94
+  the Free Software Foundation; either version 2.1 of the License, or
Zbigniew Jędrzejewski-Szmek 62fe94
+  (at your option) any later version.
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+  systemd is distributed in the hope that it will be useful, but
Zbigniew Jędrzejewski-Szmek 62fe94
+  WITHOUT ANY WARRANTY; without even the implied warranty of
Zbigniew Jędrzejewski-Szmek 62fe94
+  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
Zbigniew Jędrzejewski-Szmek 62fe94
+  Lesser General Public License for more details.
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+  You should have received a copy of the GNU Lesser General Public License
Zbigniew Jędrzejewski-Szmek 62fe94
+  along with systemd; If not, see <http://www.gnu.org/licenses/>.
Zbigniew Jędrzejewski-Szmek 62fe94
+***/
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+#include <errno.h>
Zbigniew Jędrzejewski-Szmek 62fe94
+#include <inttypes.h>
Zbigniew Jędrzejewski-Szmek 62fe94
+#include <libudev.h>
Zbigniew Jędrzejewski-Szmek 62fe94
+#include <stdlib.h>
Zbigniew Jędrzejewski-Szmek 62fe94
+#include "consoled.h"
Zbigniew Jędrzejewski-Szmek 62fe94
+#include "grdev.h"
Zbigniew Jędrzejewski-Szmek 62fe94
+#include "hashmap.h"
Zbigniew Jędrzejewski-Szmek 62fe94
+#include "idev.h"
Zbigniew Jędrzejewski-Szmek 62fe94
+#include "list.h"
Zbigniew Jędrzejewski-Szmek 62fe94
+#include "macro.h"
Zbigniew Jędrzejewski-Szmek 62fe94
+#include "sd-bus.h"
Zbigniew Jędrzejewski-Szmek 62fe94
+#include "sd-event.h"
Zbigniew Jędrzejewski-Szmek 62fe94
+#include "sysview.h"
Zbigniew Jędrzejewski-Szmek 62fe94
+#include "util.h"
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+static bool session_feed_keyboard(Session *s, idev_data *data) {
Zbigniew Jędrzejewski-Szmek 62fe94
+        idev_data_keyboard *kdata = &data->keyboard;
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        if (!data->resync && kdata->value == 1 && kdata->n_syms == 1) {
Zbigniew Jędrzejewski-Szmek 62fe94
+                uint32_t nr;
Zbigniew Jędrzejewski-Szmek 62fe94
+                sysview_seat *seat;
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+                /* handle VT-switch requests */
Zbigniew Jędrzejewski-Szmek 62fe94
+                nr = 0;
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+                switch (kdata->keysyms[0]) {
Zbigniew Jędrzejewski-Szmek 62fe94
+                case XKB_KEY_F1 ... XKB_KEY_F12:
Zbigniew Jędrzejewski-Szmek 62fe94
+                        if (IDEV_KBDMATCH(kdata,
Zbigniew Jędrzejewski-Szmek 62fe94
+                                          IDEV_KBDMOD_CTRL | IDEV_KBDMOD_ALT,
Zbigniew Jędrzejewski-Szmek 62fe94
+                                          kdata->keysyms[0]))
Zbigniew Jędrzejewski-Szmek 62fe94
+                                nr = kdata->keysyms[0] - XKB_KEY_F1 + 1;
Zbigniew Jędrzejewski-Szmek 62fe94
+                        break;
Zbigniew Jędrzejewski-Szmek 62fe94
+                case XKB_KEY_XF86Switch_VT_1 ... XKB_KEY_XF86Switch_VT_12:
Zbigniew Jędrzejewski-Szmek 62fe94
+                        nr = kdata->keysyms[0] - XKB_KEY_XF86Switch_VT_1 + 1;
Zbigniew Jędrzejewski-Szmek 62fe94
+                        break;
Zbigniew Jędrzejewski-Szmek 62fe94
+                }
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+                if (nr != 0) {
Zbigniew Jędrzejewski-Szmek 62fe94
+                        seat = sysview_session_get_seat(s->sysview);
Zbigniew Jędrzejewski-Szmek 62fe94
+                        sysview_seat_switch_to(seat, nr);
Zbigniew Jędrzejewski-Szmek 62fe94
+                        return true;
Zbigniew Jędrzejewski-Szmek 62fe94
+                }
Zbigniew Jędrzejewski-Szmek 62fe94
+        }
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        return false;
Zbigniew Jędrzejewski-Szmek 62fe94
+}
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+static bool session_feed(Session *s, idev_data *data) {
Zbigniew Jędrzejewski-Szmek 62fe94
+        switch (data->type) {
Zbigniew Jędrzejewski-Szmek 62fe94
+        case IDEV_DATA_KEYBOARD:
Zbigniew Jędrzejewski-Szmek 62fe94
+                return session_feed_keyboard(s, data);
Zbigniew Jędrzejewski-Szmek 62fe94
+        default:
Zbigniew Jędrzejewski-Szmek 62fe94
+                return false;
Zbigniew Jędrzejewski-Szmek 62fe94
+        }
Zbigniew Jędrzejewski-Szmek 62fe94
+}
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+static int session_idev_fn(idev_session *idev, void *userdata, idev_event *event) {
Zbigniew Jędrzejewski-Szmek 62fe94
+        Session *s = userdata;
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        switch (event->type) {
Zbigniew Jędrzejewski-Szmek 62fe94
+        case IDEV_EVENT_DEVICE_ADD:
Zbigniew Jędrzejewski-Szmek 62fe94
+                idev_device_enable(event->device_add.device);
Zbigniew Jędrzejewski-Szmek 62fe94
+                break;
Zbigniew Jędrzejewski-Szmek 62fe94
+        case IDEV_EVENT_DEVICE_REMOVE:
Zbigniew Jędrzejewski-Szmek 62fe94
+                idev_device_disable(event->device_remove.device);
Zbigniew Jędrzejewski-Szmek 62fe94
+                break;
Zbigniew Jędrzejewski-Szmek 62fe94
+        case IDEV_EVENT_DEVICE_DATA:
Zbigniew Jędrzejewski-Szmek 62fe94
+                if (!session_feed(s, &event->device_data.data))
Zbigniew Jędrzejewski-Szmek 62fe94
+                        workspace_feed(s->active_ws, &event->device_data.data);
Zbigniew Jędrzejewski-Szmek 62fe94
+                break;
Zbigniew Jędrzejewski-Szmek 62fe94
+        }
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        return 0;
Zbigniew Jędrzejewski-Szmek 62fe94
+}
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+static void session_grdev_fn(grdev_session *grdev, void *userdata, grdev_event *event) {
Zbigniew Jędrzejewski-Szmek 62fe94
+        grdev_display *display;
Zbigniew Jędrzejewski-Szmek 62fe94
+        Session *s = userdata;
Zbigniew Jędrzejewski-Szmek 62fe94
+        Display *d;
Zbigniew Jędrzejewski-Szmek 62fe94
+        int r;
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        switch (event->type) {
Zbigniew Jędrzejewski-Szmek 62fe94
+        case GRDEV_EVENT_DISPLAY_ADD:
Zbigniew Jędrzejewski-Szmek 62fe94
+                display = event->display_add.display;
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+                r = display_new(&d, s, display);
Zbigniew Jędrzejewski-Szmek 62fe94
+                if (r < 0) {
Zbigniew Jędrzejewski-Szmek 62fe94
+                        log_error("Cannot create display '%s' on '%s': %s",
Zbigniew Jędrzejewski-Szmek 62fe94
+                                  grdev_display_get_name(display), sysview_session_get_name(s->sysview), strerror(-r));
Zbigniew Jędrzejewski-Szmek 62fe94
+                        break;
Zbigniew Jędrzejewski-Szmek 62fe94
+                }
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+                grdev_display_set_userdata(display, d);
Zbigniew Jędrzejewski-Szmek 62fe94
+                workspace_refresh(s->active_ws);
Zbigniew Jędrzejewski-Szmek 62fe94
+                break;
Zbigniew Jędrzejewski-Szmek 62fe94
+        case GRDEV_EVENT_DISPLAY_REMOVE:
Zbigniew Jędrzejewski-Szmek 62fe94
+                display = event->display_remove.display;
Zbigniew Jędrzejewski-Szmek 62fe94
+                d = grdev_display_get_userdata(display);
Zbigniew Jędrzejewski-Szmek 62fe94
+                if (!d)
Zbigniew Jędrzejewski-Szmek 62fe94
+                        break;
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+                display_free(d);
Zbigniew Jędrzejewski-Szmek 62fe94
+                workspace_refresh(s->active_ws);
Zbigniew Jędrzejewski-Szmek 62fe94
+                break;
Zbigniew Jędrzejewski-Szmek 62fe94
+        case GRDEV_EVENT_DISPLAY_CHANGE:
Zbigniew Jędrzejewski-Szmek 62fe94
+                display = event->display_remove.display;
Zbigniew Jędrzejewski-Szmek 62fe94
+                d = grdev_display_get_userdata(display);
Zbigniew Jędrzejewski-Szmek 62fe94
+                if (!d)
Zbigniew Jędrzejewski-Szmek 62fe94
+                        break;
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+                display_refresh(d);
Zbigniew Jędrzejewski-Szmek 62fe94
+                workspace_refresh(s->active_ws);
Zbigniew Jędrzejewski-Szmek 62fe94
+                break;
Zbigniew Jędrzejewski-Szmek 62fe94
+        case GRDEV_EVENT_DISPLAY_FRAME:
Zbigniew Jędrzejewski-Szmek 62fe94
+                display = event->display_remove.display;
Zbigniew Jędrzejewski-Szmek 62fe94
+                d = grdev_display_get_userdata(display);
Zbigniew Jędrzejewski-Szmek 62fe94
+                if (!d)
Zbigniew Jędrzejewski-Szmek 62fe94
+                        break;
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+                session_dirty(s);
Zbigniew Jędrzejewski-Szmek 62fe94
+                break;
Zbigniew Jędrzejewski-Szmek 62fe94
+        }
Zbigniew Jędrzejewski-Szmek 62fe94
+}
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+static int session_redraw_fn(sd_event_source *src, void *userdata) {
Zbigniew Jędrzejewski-Szmek 62fe94
+        Session *s = userdata;
Zbigniew Jędrzejewski-Szmek 62fe94
+        Display *d;
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        LIST_FOREACH(displays_by_session, d, s->display_list)
Zbigniew Jędrzejewski-Szmek 62fe94
+                display_render(d, s->active_ws);
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        grdev_session_commit(s->grdev);
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        return 0;
Zbigniew Jędrzejewski-Szmek 62fe94
+}
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+int session_new(Session **out, Manager *m, sysview_session *session) {
Zbigniew Jędrzejewski-Szmek 62fe94
+        _cleanup_(session_freep) Session *s = NULL;
Zbigniew Jędrzejewski-Szmek 62fe94
+        int r;
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        assert(out);
Zbigniew Jędrzejewski-Szmek 62fe94
+        assert(m);
Zbigniew Jędrzejewski-Szmek 62fe94
+        assert(session);
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        s = new0(Session, 1);
Zbigniew Jędrzejewski-Szmek 62fe94
+        if (!s)
Zbigniew Jędrzejewski-Szmek 62fe94
+                return -ENOMEM;
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        s->manager = m;
Zbigniew Jędrzejewski-Szmek 62fe94
+        s->sysview = session;
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        r = grdev_session_new(&s->grdev,
Zbigniew Jędrzejewski-Szmek 62fe94
+                              m->grdev,
Zbigniew Jędrzejewski-Szmek 62fe94
+                              GRDEV_SESSION_MANAGED,
Zbigniew Jędrzejewski-Szmek 62fe94
+                              sysview_session_get_name(session),
Zbigniew Jędrzejewski-Szmek 62fe94
+                              session_grdev_fn,
Zbigniew Jędrzejewski-Szmek 62fe94
+                              s);
Zbigniew Jędrzejewski-Szmek 62fe94
+        if (r < 0)
Zbigniew Jędrzejewski-Szmek 62fe94
+                return r;
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        r = idev_session_new(&s->idev,
Zbigniew Jędrzejewski-Szmek 62fe94
+                             m->idev,
Zbigniew Jędrzejewski-Szmek 62fe94
+                             IDEV_SESSION_MANAGED,
Zbigniew Jędrzejewski-Szmek 62fe94
+                             sysview_session_get_name(session),
Zbigniew Jędrzejewski-Szmek 62fe94
+                             session_idev_fn,
Zbigniew Jędrzejewski-Szmek 62fe94
+                             s);
Zbigniew Jędrzejewski-Szmek 62fe94
+        if (r < 0)
Zbigniew Jędrzejewski-Szmek 62fe94
+                return r;
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        r = workspace_new(&s->my_ws, m);
Zbigniew Jędrzejewski-Szmek 62fe94
+        if (r < 0)
Zbigniew Jędrzejewski-Szmek 62fe94
+                return r;
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        s->active_ws = workspace_attach(s->my_ws, s);
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        r = sd_event_add_defer(m->event, &s->redraw_src, session_redraw_fn, s);
Zbigniew Jędrzejewski-Szmek 62fe94
+        if (r < 0)
Zbigniew Jędrzejewski-Szmek 62fe94
+                return r;
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        grdev_session_enable(s->grdev);
Zbigniew Jędrzejewski-Szmek 62fe94
+        idev_session_enable(s->idev);
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        *out = s;
Zbigniew Jędrzejewski-Szmek 62fe94
+        s = NULL;
Zbigniew Jędrzejewski-Szmek 62fe94
+        return 0;
Zbigniew Jędrzejewski-Szmek 62fe94
+}
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+Session *session_free(Session *s) {
Zbigniew Jędrzejewski-Szmek 62fe94
+        if (!s)
Zbigniew Jędrzejewski-Szmek 62fe94
+                return NULL;
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        assert(!s->display_list);
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        sd_event_source_unref(s->redraw_src);
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        workspace_detach(s->active_ws, s);
Zbigniew Jędrzejewski-Szmek 62fe94
+        workspace_unref(s->my_ws);
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        idev_session_free(s->idev);
Zbigniew Jędrzejewski-Szmek 62fe94
+        grdev_session_free(s->grdev);
Zbigniew Jędrzejewski-Szmek 62fe94
+        free(s);
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        return NULL;
Zbigniew Jędrzejewski-Szmek 62fe94
+}
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+void session_dirty(Session *s) {
Zbigniew Jędrzejewski-Szmek 62fe94
+        int r;
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        assert(s);
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        r = sd_event_source_set_enabled(s->redraw_src, SD_EVENT_ONESHOT);
Zbigniew Jędrzejewski-Szmek 62fe94
+        if (r < 0)
Zbigniew Jędrzejewski-Szmek 62fe94
+                log_error("Cannot enable redraw-source: %s", strerror(-r));
Zbigniew Jędrzejewski-Szmek 62fe94
+}
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+void session_add_device(Session *s, sysview_device *device) {
Zbigniew Jędrzejewski-Szmek 62fe94
+        unsigned int type;
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        assert(s);
Zbigniew Jędrzejewski-Szmek 62fe94
+        assert(device);
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        type = sysview_device_get_type(device);
Zbigniew Jędrzejewski-Szmek 62fe94
+        switch (type) {
Zbigniew Jędrzejewski-Szmek 62fe94
+        case SYSVIEW_DEVICE_DRM:
Zbigniew Jędrzejewski-Szmek 62fe94
+                grdev_session_add_drm(s->grdev, sysview_device_get_ud(device));
Zbigniew Jędrzejewski-Szmek 62fe94
+                break;
Zbigniew Jędrzejewski-Szmek 62fe94
+        case SYSVIEW_DEVICE_EVDEV:
Zbigniew Jędrzejewski-Szmek 62fe94
+                idev_session_add_evdev(s->idev, sysview_device_get_ud(device));
Zbigniew Jędrzejewski-Szmek 62fe94
+                break;
Zbigniew Jędrzejewski-Szmek 62fe94
+        }
Zbigniew Jędrzejewski-Szmek 62fe94
+}
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+void session_remove_device(Session *s, sysview_device *device) {
Zbigniew Jędrzejewski-Szmek 62fe94
+        unsigned int type;
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        assert(s);
Zbigniew Jędrzejewski-Szmek 62fe94
+        assert(device);
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        type = sysview_device_get_type(device);
Zbigniew Jędrzejewski-Szmek 62fe94
+        switch (type) {
Zbigniew Jędrzejewski-Szmek 62fe94
+        case SYSVIEW_DEVICE_DRM:
Zbigniew Jędrzejewski-Szmek 62fe94
+                grdev_session_remove_drm(s->grdev, sysview_device_get_ud(device));
Zbigniew Jędrzejewski-Szmek 62fe94
+                break;
Zbigniew Jędrzejewski-Szmek 62fe94
+        case SYSVIEW_DEVICE_EVDEV:
Zbigniew Jędrzejewski-Szmek 62fe94
+                idev_session_remove_evdev(s->idev, sysview_device_get_ud(device));
Zbigniew Jędrzejewski-Szmek 62fe94
+                break;
Zbigniew Jędrzejewski-Szmek 62fe94
+        }
Zbigniew Jędrzejewski-Szmek 62fe94
+}
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+void session_refresh_device(Session *s, sysview_device *device, struct udev_device *ud) {
Zbigniew Jędrzejewski-Szmek 62fe94
+        unsigned int type;
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        assert(s);
Zbigniew Jędrzejewski-Szmek 62fe94
+        assert(device);
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        type = sysview_device_get_type(device);
Zbigniew Jędrzejewski-Szmek 62fe94
+        switch (type) {
Zbigniew Jędrzejewski-Szmek 62fe94
+        case SYSVIEW_DEVICE_DRM:
Zbigniew Jędrzejewski-Szmek 62fe94
+                grdev_session_hotplug_drm(s->grdev, sysview_device_get_ud(device));
Zbigniew Jędrzejewski-Szmek 62fe94
+                break;
Zbigniew Jędrzejewski-Szmek 62fe94
+        }
Zbigniew Jędrzejewski-Szmek 62fe94
+}
Zbigniew Jędrzejewski-Szmek 62fe94
diff --git a/src/console/consoled-terminal.c b/src/console/consoled-terminal.c
Zbigniew Jędrzejewski-Szmek 62fe94
new file mode 100644
Zbigniew Jędrzejewski-Szmek 62fe94
index 0000000000..d091579aa5
Zbigniew Jędrzejewski-Szmek 62fe94
--- /dev/null
Zbigniew Jędrzejewski-Szmek 62fe94
+++ b/src/console/consoled-terminal.c
Zbigniew Jędrzejewski-Szmek 62fe94
@@ -0,0 +1,360 @@
Zbigniew Jędrzejewski-Szmek 62fe94
+/*-*- Mode: C; c-basic-offset: 8; indent-tabs-mode: nil -*-*/
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+/***
Zbigniew Jędrzejewski-Szmek 62fe94
+  This file is part of systemd.
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+  Copyright 2014 David Herrmann <dh.herrmann@gmail.com>
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+  systemd is free software; you can redistribute it and/or modify it
Zbigniew Jędrzejewski-Szmek 62fe94
+  under the terms of the GNU Lesser General Public License as published by
Zbigniew Jędrzejewski-Szmek 62fe94
+  the Free Software Foundation; either version 2.1 of the License, or
Zbigniew Jędrzejewski-Szmek 62fe94
+  (at your option) any later version.
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+  systemd is distributed in the hope that it will be useful, but
Zbigniew Jędrzejewski-Szmek 62fe94
+  WITHOUT ANY WARRANTY; without even the implied warranty of
Zbigniew Jędrzejewski-Szmek 62fe94
+  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
Zbigniew Jędrzejewski-Szmek 62fe94
+  Lesser General Public License for more details.
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+  You should have received a copy of the GNU Lesser General Public License
Zbigniew Jędrzejewski-Szmek 62fe94
+  along with systemd; If not, see <http://www.gnu.org/licenses/>.
Zbigniew Jędrzejewski-Szmek 62fe94
+***/
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+#include <errno.h>
Zbigniew Jędrzejewski-Szmek 62fe94
+#include <inttypes.h>
Zbigniew Jędrzejewski-Szmek 62fe94
+#include <stdlib.h>
Zbigniew Jędrzejewski-Szmek 62fe94
+#include "consoled.h"
Zbigniew Jędrzejewski-Szmek 62fe94
+#include "list.h"
Zbigniew Jędrzejewski-Szmek 62fe94
+#include "macro.h"
Zbigniew Jędrzejewski-Szmek 62fe94
+#include "util.h"
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+static int terminal_write_fn(term_screen *screen, void *userdata, const void *buf, size_t size) {
Zbigniew Jędrzejewski-Szmek 62fe94
+        Terminal *t = userdata;
Zbigniew Jędrzejewski-Szmek 62fe94
+        int r;
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        if (t->pty) {
Zbigniew Jędrzejewski-Szmek 62fe94
+                r = pty_write(t->pty, buf, size);
Zbigniew Jędrzejewski-Szmek 62fe94
+                if (r < 0)
Zbigniew Jędrzejewski-Szmek 62fe94
+                        return log_oom();
Zbigniew Jędrzejewski-Szmek 62fe94
+        }
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        return 0;
Zbigniew Jędrzejewski-Szmek 62fe94
+}
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+static int terminal_pty_fn(Pty *pty, void *userdata, unsigned int event, const void *ptr, size_t size) {
Zbigniew Jędrzejewski-Szmek 62fe94
+        Terminal *t = userdata;
Zbigniew Jędrzejewski-Szmek 62fe94
+        int r;
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        switch (event) {
Zbigniew Jędrzejewski-Szmek 62fe94
+        case PTY_CHILD:
Zbigniew Jędrzejewski-Szmek 62fe94
+                log_debug("PTY child exited");
Zbigniew Jędrzejewski-Szmek 62fe94
+                t->pty = pty_unref(t->pty);
Zbigniew Jędrzejewski-Szmek 62fe94
+                break;
Zbigniew Jędrzejewski-Szmek 62fe94
+        case PTY_DATA:
Zbigniew Jędrzejewski-Szmek 62fe94
+                r = term_screen_feed_text(t->screen, ptr, size);
Zbigniew Jędrzejewski-Szmek 62fe94
+                if (r < 0)
Zbigniew Jędrzejewski-Szmek 62fe94
+                        log_error("Cannot update screen state: %s", strerror(-r));
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+                workspace_dirty(t->workspace);
Zbigniew Jędrzejewski-Szmek 62fe94
+                break;
Zbigniew Jędrzejewski-Szmek 62fe94
+        }
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        return 0;
Zbigniew Jędrzejewski-Szmek 62fe94
+}
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+int terminal_new(Terminal **out, Workspace *w) {
Zbigniew Jędrzejewski-Szmek 62fe94
+        _cleanup_(terminal_freep) Terminal *t = NULL;
Zbigniew Jędrzejewski-Szmek 62fe94
+        int r;
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        assert(w);
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        t = new0(Terminal, 1);
Zbigniew Jędrzejewski-Szmek 62fe94
+        if (!t)
Zbigniew Jędrzejewski-Szmek 62fe94
+                return -ENOMEM;
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        t->workspace = w;
Zbigniew Jędrzejewski-Szmek 62fe94
+        LIST_PREPEND(terminals_by_workspace, w->terminal_list, t);
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        r = term_parser_new(&t->parser, true);
Zbigniew Jędrzejewski-Szmek 62fe94
+        if (r < 0)
Zbigniew Jędrzejewski-Szmek 62fe94
+                return r;
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        r = term_screen_new(&t->screen, terminal_write_fn, t, NULL, NULL);
Zbigniew Jędrzejewski-Szmek 62fe94
+        if (r < 0)
Zbigniew Jędrzejewski-Szmek 62fe94
+                return r;
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        r = term_screen_set_answerback(t->screen, "systemd-console");
Zbigniew Jędrzejewski-Szmek 62fe94
+        if (r < 0)
Zbigniew Jędrzejewski-Szmek 62fe94
+                return r;
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        if (out)
Zbigniew Jędrzejewski-Szmek 62fe94
+                *out = t;
Zbigniew Jędrzejewski-Szmek 62fe94
+        t = NULL;
Zbigniew Jędrzejewski-Szmek 62fe94
+        return 0;
Zbigniew Jędrzejewski-Szmek 62fe94
+}
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+Terminal *terminal_free(Terminal *t) {
Zbigniew Jędrzejewski-Szmek 62fe94
+        if (!t)
Zbigniew Jędrzejewski-Szmek 62fe94
+                return NULL;
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        assert(t->workspace);
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        if (t->pty) {
Zbigniew Jędrzejewski-Szmek 62fe94
+                (void)pty_signal(t->pty, SIGHUP);
Zbigniew Jędrzejewski-Szmek 62fe94
+                pty_close(t->pty);
Zbigniew Jędrzejewski-Szmek 62fe94
+                pty_unref(t->pty);
Zbigniew Jędrzejewski-Szmek 62fe94
+        }
Zbigniew Jędrzejewski-Szmek 62fe94
+        term_screen_unref(t->screen);
Zbigniew Jędrzejewski-Szmek 62fe94
+        term_parser_free(t->parser);
Zbigniew Jędrzejewski-Szmek 62fe94
+        LIST_REMOVE(terminals_by_workspace, t->workspace->terminal_list, t);
Zbigniew Jędrzejewski-Szmek 62fe94
+        free(t);
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        return NULL;
Zbigniew Jędrzejewski-Szmek 62fe94
+}
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+void terminal_resize(Terminal *t) {
Zbigniew Jędrzejewski-Szmek 62fe94
+        uint32_t width, height, fw, fh;
Zbigniew Jędrzejewski-Szmek 62fe94
+        int r;
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        assert(t);
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        width = t->workspace->width;
Zbigniew Jędrzejewski-Szmek 62fe94
+        height = t->workspace->height;
Zbigniew Jędrzejewski-Szmek 62fe94
+        fw = unifont_get_width(t->workspace->manager->uf);
Zbigniew Jędrzejewski-Szmek 62fe94
+        fh = unifont_get_height(t->workspace->manager->uf);
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        width = (fw > 0) ? width / fw : 0;
Zbigniew Jędrzejewski-Szmek 62fe94
+        height = (fh > 0) ? height / fh : 0;
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        if (t->pty) {
Zbigniew Jędrzejewski-Szmek 62fe94
+                r = pty_resize(t->pty, width, height);
Zbigniew Jędrzejewski-Szmek 62fe94
+                if (r < 0)
Zbigniew Jędrzejewski-Szmek 62fe94
+                        log_error("Cannot resize pty: %s", strerror(-r));
Zbigniew Jędrzejewski-Szmek 62fe94
+        }
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        r = term_screen_resize(t->screen, width, height);
Zbigniew Jędrzejewski-Szmek 62fe94
+        if (r < 0)
Zbigniew Jędrzejewski-Szmek 62fe94
+                log_error("Cannot resize screen: %s", strerror(-r));
Zbigniew Jędrzejewski-Szmek 62fe94
+}
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+void terminal_run(Terminal *t) {
Zbigniew Jędrzejewski-Szmek 62fe94
+        pid_t pid;
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        assert(t);
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        if (t->pty)
Zbigniew Jędrzejewski-Szmek 62fe94
+                return;
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        pid = pty_fork(&t->pty,
Zbigniew Jędrzejewski-Szmek 62fe94
+                       t->workspace->manager->event,
Zbigniew Jędrzejewski-Szmek 62fe94
+                       terminal_pty_fn,
Zbigniew Jędrzejewski-Szmek 62fe94
+                       t,
Zbigniew Jędrzejewski-Szmek 62fe94
+                       term_screen_get_width(t->screen),
Zbigniew Jędrzejewski-Szmek 62fe94
+                       term_screen_get_height(t->screen));
Zbigniew Jędrzejewski-Szmek 62fe94
+        if (pid < 0) {
Zbigniew Jędrzejewski-Szmek 62fe94
+                log_error("Cannot fork PTY: %s", strerror(-pid));
Zbigniew Jędrzejewski-Szmek 62fe94
+                return;
Zbigniew Jędrzejewski-Szmek 62fe94
+        } else if (pid == 0) {
Zbigniew Jędrzejewski-Szmek 62fe94
+                /* child */
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+                char **argv = (char*[]){
Zbigniew Jędrzejewski-Szmek 62fe94
+                        (char*)getenv("SHELL") ? : (char*)_PATH_BSHELL,
Zbigniew Jędrzejewski-Szmek 62fe94
+                        NULL
Zbigniew Jędrzejewski-Szmek 62fe94
+                };
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+                setenv("TERM", "xterm-256color", 1);
Zbigniew Jędrzejewski-Szmek 62fe94
+                setenv("COLORTERM", "systemd-console", 1);
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+                execve(argv[0], argv, environ);
Zbigniew Jędrzejewski-Szmek 62fe94
+                log_error("Cannot exec %s (%d): %m", argv[0], -errno);
Zbigniew Jędrzejewski-Szmek 62fe94
+                _exit(1);
Zbigniew Jędrzejewski-Szmek 62fe94
+        }
Zbigniew Jędrzejewski-Szmek 62fe94
+}
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+static void terminal_feed_keyboard(Terminal *t, idev_data *data) {
Zbigniew Jędrzejewski-Szmek 62fe94
+        idev_data_keyboard *kdata = &data->keyboard;
Zbigniew Jędrzejewski-Szmek 62fe94
+        int r;
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        if (!data->resync && (kdata->value == 1 || kdata->value == 2)) {
Zbigniew Jędrzejewski-Szmek 62fe94
+                assert_cc(TERM_KBDMOD_CNT == (int)IDEV_KBDMOD_CNT);
Zbigniew Jędrzejewski-Szmek 62fe94
+                assert_cc(TERM_KBDMOD_IDX_SHIFT == (int)IDEV_KBDMOD_IDX_SHIFT &&
Zbigniew Jędrzejewski-Szmek 62fe94
+                          TERM_KBDMOD_IDX_CTRL == (int)IDEV_KBDMOD_IDX_CTRL &&
Zbigniew Jędrzejewski-Szmek 62fe94
+                          TERM_KBDMOD_IDX_ALT == (int)IDEV_KBDMOD_IDX_ALT &&
Zbigniew Jędrzejewski-Szmek 62fe94
+                          TERM_KBDMOD_IDX_LINUX == (int)IDEV_KBDMOD_IDX_LINUX &&
Zbigniew Jędrzejewski-Szmek 62fe94
+                          TERM_KBDMOD_IDX_CAPS == (int)IDEV_KBDMOD_IDX_CAPS);
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+                r = term_screen_feed_keyboard(t->screen,
Zbigniew Jędrzejewski-Szmek 62fe94
+                                              kdata->keysyms,
Zbigniew Jędrzejewski-Szmek 62fe94
+                                              kdata->n_syms,
Zbigniew Jędrzejewski-Szmek 62fe94
+                                              kdata->ascii,
Zbigniew Jędrzejewski-Szmek 62fe94
+                                              kdata->codepoints,
Zbigniew Jędrzejewski-Szmek 62fe94
+                                              kdata->mods);
Zbigniew Jędrzejewski-Szmek 62fe94
+                if (r < 0)
Zbigniew Jędrzejewski-Szmek 62fe94
+                        log_error("Cannot feed keyboard data to screen: %s",
Zbigniew Jędrzejewski-Szmek 62fe94
+                                  strerror(-r));
Zbigniew Jędrzejewski-Szmek 62fe94
+        }
Zbigniew Jędrzejewski-Szmek 62fe94
+}
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+void terminal_feed(Terminal *t, idev_data *data) {
Zbigniew Jędrzejewski-Szmek 62fe94
+        switch (data->type) {
Zbigniew Jędrzejewski-Szmek 62fe94
+        case IDEV_DATA_KEYBOARD:
Zbigniew Jędrzejewski-Szmek 62fe94
+                terminal_feed_keyboard(t, data);
Zbigniew Jędrzejewski-Szmek 62fe94
+                break;
Zbigniew Jędrzejewski-Szmek 62fe94
+        }
Zbigniew Jędrzejewski-Szmek 62fe94
+}
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+static void terminal_fill(uint8_t *dst,
Zbigniew Jędrzejewski-Szmek 62fe94
+                          uint32_t width,
Zbigniew Jędrzejewski-Szmek 62fe94
+                          uint32_t height,
Zbigniew Jędrzejewski-Szmek 62fe94
+                          uint32_t stride,
Zbigniew Jędrzejewski-Szmek 62fe94
+                          uint32_t value) {
Zbigniew Jędrzejewski-Szmek 62fe94
+        uint32_t i, j, *px;
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        for (j = 0; j < height; ++j) {
Zbigniew Jędrzejewski-Szmek 62fe94
+                px = (uint32_t*)dst;
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+                for (i = 0; i < width; ++i)
Zbigniew Jędrzejewski-Szmek 62fe94
+                        *px++ = value;
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+                dst += stride;
Zbigniew Jędrzejewski-Szmek 62fe94
+        }
Zbigniew Jędrzejewski-Szmek 62fe94
+}
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+static void terminal_blend(uint8_t *dst,
Zbigniew Jędrzejewski-Szmek 62fe94
+                           uint32_t width,
Zbigniew Jędrzejewski-Szmek 62fe94
+                           uint32_t height,
Zbigniew Jędrzejewski-Szmek 62fe94
+                           uint32_t dst_stride,
Zbigniew Jędrzejewski-Szmek 62fe94
+                           const uint8_t *src,
Zbigniew Jędrzejewski-Szmek 62fe94
+                           uint32_t src_stride,
Zbigniew Jędrzejewski-Szmek 62fe94
+                           uint32_t fg,
Zbigniew Jędrzejewski-Szmek 62fe94
+                           uint32_t bg) {
Zbigniew Jędrzejewski-Szmek 62fe94
+        uint32_t i, j, *px;
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        for (j = 0; j < height; ++j) {
Zbigniew Jędrzejewski-Szmek 62fe94
+                px = (uint32_t*)dst;
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+                for (i = 0; i < width; ++i) {
Zbigniew Jędrzejewski-Szmek 62fe94
+                        if (!src || src[i / 8] & (1 << (7 - i % 8)))
Zbigniew Jędrzejewski-Szmek 62fe94
+                                *px = fg;
Zbigniew Jędrzejewski-Szmek 62fe94
+                        else
Zbigniew Jędrzejewski-Szmek 62fe94
+                                *px = bg;
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+                        ++px;
Zbigniew Jędrzejewski-Szmek 62fe94
+                }
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+                src += src_stride;
Zbigniew Jędrzejewski-Szmek 62fe94
+                dst += dst_stride;
Zbigniew Jędrzejewski-Szmek 62fe94
+        }
Zbigniew Jędrzejewski-Szmek 62fe94
+}
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+typedef struct {
Zbigniew Jędrzejewski-Szmek 62fe94
+        const grdev_display_target *target;
Zbigniew Jędrzejewski-Szmek 62fe94
+        unifont *uf;
Zbigniew Jędrzejewski-Szmek 62fe94
+        uint32_t cell_width;
Zbigniew Jędrzejewski-Szmek 62fe94
+        uint32_t cell_height;
Zbigniew Jędrzejewski-Szmek 62fe94
+        bool dirty;
Zbigniew Jędrzejewski-Szmek 62fe94
+} TerminalDrawContext;
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+static int terminal_draw_cell(term_screen *screen,
Zbigniew Jędrzejewski-Szmek 62fe94
+                              void *userdata,
Zbigniew Jędrzejewski-Szmek 62fe94
+                              unsigned int x,
Zbigniew Jędrzejewski-Szmek 62fe94
+                              unsigned int y,
Zbigniew Jędrzejewski-Szmek 62fe94
+                              const term_attr *attr,
Zbigniew Jędrzejewski-Szmek 62fe94
+                              const uint32_t *ch,
Zbigniew Jędrzejewski-Szmek 62fe94
+                              size_t n_ch,
Zbigniew Jędrzejewski-Szmek 62fe94
+                              unsigned int ch_width) {
Zbigniew Jędrzejewski-Szmek 62fe94
+        TerminalDrawContext *ctx = userdata;
Zbigniew Jędrzejewski-Szmek 62fe94
+        const grdev_display_target *target = ctx->target;
Zbigniew Jędrzejewski-Szmek 62fe94
+        grdev_fb *fb = target->back;
Zbigniew Jędrzejewski-Szmek 62fe94
+        uint32_t xpos, ypos, width, height;
Zbigniew Jędrzejewski-Szmek 62fe94
+        uint32_t fg, bg;
Zbigniew Jędrzejewski-Szmek 62fe94
+        unifont_glyph g;
Zbigniew Jędrzejewski-Szmek 62fe94
+        uint8_t *dst;
Zbigniew Jędrzejewski-Szmek 62fe94
+        int r;
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        if (n_ch > 0) {
Zbigniew Jędrzejewski-Szmek 62fe94
+                r = unifont_lookup(ctx->uf, &g, *ch);
Zbigniew Jędrzejewski-Szmek 62fe94
+                if (r < 0)
Zbigniew Jędrzejewski-Szmek 62fe94
+                        r = unifont_lookup(ctx->uf, &g, 0xfffd);
Zbigniew Jędrzejewski-Szmek 62fe94
+                if (r < 0)
Zbigniew Jędrzejewski-Szmek 62fe94
+                        unifont_fallback(&g);
Zbigniew Jędrzejewski-Szmek 62fe94
+        }
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        xpos = x * ctx->cell_width;
Zbigniew Jędrzejewski-Szmek 62fe94
+        ypos = y * ctx->cell_height;
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        if (xpos >= fb->width || ypos >= fb->height)
Zbigniew Jędrzejewski-Szmek 62fe94
+                return 0;
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        width = MIN(fb->width - xpos, ctx->cell_width * ch_width);
Zbigniew Jędrzejewski-Szmek 62fe94
+        height = MIN(fb->height - ypos, ctx->cell_height);
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        term_attr_to_argb32(attr, &fg, &bg, NULL);
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        ctx->dirty = true;
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        dst = fb->maps[0];
Zbigniew Jędrzejewski-Szmek 62fe94
+        dst += fb->strides[0] * ypos + sizeof(uint32_t) * xpos;
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        if (n_ch < 1) {
Zbigniew Jędrzejewski-Szmek 62fe94
+                terminal_fill(dst,
Zbigniew Jędrzejewski-Szmek 62fe94
+                              width,
Zbigniew Jędrzejewski-Szmek 62fe94
+                              height,
Zbigniew Jędrzejewski-Szmek 62fe94
+                              fb->strides[0],
Zbigniew Jędrzejewski-Szmek 62fe94
+                              bg);
Zbigniew Jędrzejewski-Szmek 62fe94
+        } else {
Zbigniew Jędrzejewski-Szmek 62fe94
+                if (width > g.width)
Zbigniew Jędrzejewski-Szmek 62fe94
+                        terminal_fill(dst + sizeof(uint32_t) * g.width,
Zbigniew Jędrzejewski-Szmek 62fe94
+                                      width - g.width,
Zbigniew Jędrzejewski-Szmek 62fe94
+                                      height,
Zbigniew Jędrzejewski-Szmek 62fe94
+                                      fb->strides[0],
Zbigniew Jędrzejewski-Szmek 62fe94
+                                      bg);
Zbigniew Jędrzejewski-Szmek 62fe94
+                if (height > g.height)
Zbigniew Jędrzejewski-Szmek 62fe94
+                        terminal_fill(dst + fb->strides[0] * g.height,
Zbigniew Jędrzejewski-Szmek 62fe94
+                                      width,
Zbigniew Jędrzejewski-Szmek 62fe94
+                                      height - g.height,
Zbigniew Jędrzejewski-Szmek 62fe94
+                                      fb->strides[0],
Zbigniew Jędrzejewski-Szmek 62fe94
+                                      bg);
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+                terminal_blend(dst,
Zbigniew Jędrzejewski-Szmek 62fe94
+                               width,
Zbigniew Jędrzejewski-Szmek 62fe94
+                               height,
Zbigniew Jędrzejewski-Szmek 62fe94
+                               fb->strides[0],
Zbigniew Jędrzejewski-Szmek 62fe94
+                               g.data,
Zbigniew Jędrzejewski-Szmek 62fe94
+                               g.stride,
Zbigniew Jędrzejewski-Szmek 62fe94
+                               fg,
Zbigniew Jędrzejewski-Szmek 62fe94
+                               bg);
Zbigniew Jędrzejewski-Szmek 62fe94
+        }
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        return 0;
Zbigniew Jędrzejewski-Szmek 62fe94
+}
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+bool terminal_draw(Terminal *t, const grdev_display_target *target) {
Zbigniew Jędrzejewski-Szmek 62fe94
+        TerminalDrawContext ctx = { };
Zbigniew Jędrzejewski-Szmek 62fe94
+        uint64_t age;
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        assert(t);
Zbigniew Jędrzejewski-Szmek 62fe94
+        assert(target);
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        /* start up terminal on first frame */
Zbigniew Jędrzejewski-Szmek 62fe94
+        terminal_run(t);
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        ctx.target = target;
Zbigniew Jędrzejewski-Szmek 62fe94
+        ctx.uf = t->workspace->manager->uf;
Zbigniew Jędrzejewski-Szmek 62fe94
+        ctx.cell_width = unifont_get_width(ctx.uf);
Zbigniew Jędrzejewski-Szmek 62fe94
+        ctx.cell_height = unifont_get_height(ctx.uf);
Zbigniew Jędrzejewski-Szmek 62fe94
+        ctx.dirty = false;
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        if (target->front) {
Zbigniew Jędrzejewski-Szmek 62fe94
+                /* if the frontbuffer is new enough, no reason to redraw */
Zbigniew Jędrzejewski-Szmek 62fe94
+                age = term_screen_get_age(t->screen);
Zbigniew Jędrzejewski-Szmek 62fe94
+                if (age != 0 && age <= target->front->data.u64)
Zbigniew Jędrzejewski-Szmek 62fe94
+                        return false;
Zbigniew Jędrzejewski-Szmek 62fe94
+        } else {
Zbigniew Jędrzejewski-Szmek 62fe94
+                /* force flip if no frontbuffer is set, yet */
Zbigniew Jędrzejewski-Szmek 62fe94
+                ctx.dirty = true;
Zbigniew Jędrzejewski-Szmek 62fe94
+        }
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        term_screen_draw(t->screen, terminal_draw_cell, &ctx, &target->back->data.u64);
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        return ctx.dirty;
Zbigniew Jędrzejewski-Szmek 62fe94
+}
Zbigniew Jędrzejewski-Szmek 62fe94
diff --git a/src/console/consoled-workspace.c b/src/console/consoled-workspace.c
Zbigniew Jędrzejewski-Szmek 62fe94
new file mode 100644
Zbigniew Jędrzejewski-Szmek 62fe94
index 0000000000..56344ef2cf
Zbigniew Jędrzejewski-Szmek 62fe94
--- /dev/null
Zbigniew Jędrzejewski-Szmek 62fe94
+++ b/src/console/consoled-workspace.c
Zbigniew Jędrzejewski-Szmek 62fe94
@@ -0,0 +1,168 @@
Zbigniew Jędrzejewski-Szmek 62fe94
+/*-*- Mode: C; c-basic-offset: 8; indent-tabs-mode: nil -*-*/
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+/***
Zbigniew Jędrzejewski-Szmek 62fe94
+  This file is part of systemd.
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+  Copyright 2014 David Herrmann <dh.herrmann@gmail.com>
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+  systemd is free software; you can redistribute it and/or modify it
Zbigniew Jędrzejewski-Szmek 62fe94
+  under the terms of the GNU Lesser General Public License as published by
Zbigniew Jędrzejewski-Szmek 62fe94
+  the Free Software Foundation; either version 2.1 of the License, or
Zbigniew Jędrzejewski-Szmek 62fe94
+  (at your option) any later version.
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+  systemd is distributed in the hope that it will be useful, but
Zbigniew Jędrzejewski-Szmek 62fe94
+  WITHOUT ANY WARRANTY; without even the implied warranty of
Zbigniew Jędrzejewski-Szmek 62fe94
+  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
Zbigniew Jędrzejewski-Szmek 62fe94
+  Lesser General Public License for more details.
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+  You should have received a copy of the GNU Lesser General Public License
Zbigniew Jędrzejewski-Szmek 62fe94
+  along with systemd; If not, see <http://www.gnu.org/licenses/>.
Zbigniew Jędrzejewski-Szmek 62fe94
+***/
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+#include <errno.h>
Zbigniew Jędrzejewski-Szmek 62fe94
+#include <inttypes.h>
Zbigniew Jędrzejewski-Szmek 62fe94
+#include <stdlib.h>
Zbigniew Jędrzejewski-Szmek 62fe94
+#include "consoled.h"
Zbigniew Jędrzejewski-Szmek 62fe94
+#include "grdev.h"
Zbigniew Jędrzejewski-Szmek 62fe94
+#include "idev.h"
Zbigniew Jędrzejewski-Szmek 62fe94
+#include "list.h"
Zbigniew Jędrzejewski-Szmek 62fe94
+#include "macro.h"
Zbigniew Jędrzejewski-Szmek 62fe94
+#include "util.h"
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+int workspace_new(Workspace **out, Manager *m) {
Zbigniew Jędrzejewski-Szmek 62fe94
+        _cleanup_(workspace_unrefp) Workspace *w = NULL;
Zbigniew Jędrzejewski-Szmek 62fe94
+        int r;
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        assert(out);
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        w = new0(Workspace, 1);
Zbigniew Jędrzejewski-Szmek 62fe94
+        if (!w)
Zbigniew Jędrzejewski-Szmek 62fe94
+                return -ENOMEM;
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        w->ref = 1;
Zbigniew Jędrzejewski-Szmek 62fe94
+        w->manager = m;
Zbigniew Jędrzejewski-Szmek 62fe94
+        LIST_PREPEND(workspaces_by_manager, m->workspace_list, w);
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        r = terminal_new(&w->current, w);
Zbigniew Jędrzejewski-Szmek 62fe94
+        if (r < 0)
Zbigniew Jędrzejewski-Szmek 62fe94
+                return r;
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        *out = w;
Zbigniew Jędrzejewski-Szmek 62fe94
+        w = NULL;
Zbigniew Jędrzejewski-Szmek 62fe94
+        return 0;
Zbigniew Jędrzejewski-Szmek 62fe94
+}
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+static void workspace_cleanup(Workspace *w) {
Zbigniew Jędrzejewski-Szmek 62fe94
+        Terminal *t;
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        assert(w);
Zbigniew Jędrzejewski-Szmek 62fe94
+        assert(w->ref == 0);
Zbigniew Jędrzejewski-Szmek 62fe94
+        assert(w->manager);
Zbigniew Jędrzejewski-Szmek 62fe94
+        assert(!w->session_list);
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        w->current = NULL;
Zbigniew Jędrzejewski-Szmek 62fe94
+        while ((t = w->terminal_list))
Zbigniew Jędrzejewski-Szmek 62fe94
+                terminal_free(t);
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        LIST_REMOVE(workspaces_by_manager, w->manager->workspace_list, w);
Zbigniew Jędrzejewski-Szmek 62fe94
+        free(w);
Zbigniew Jędrzejewski-Szmek 62fe94
+}
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+Workspace *workspace_ref(Workspace *w) {
Zbigniew Jędrzejewski-Szmek 62fe94
+        assert(w);
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        ++w->ref;
Zbigniew Jędrzejewski-Szmek 62fe94
+        return w;
Zbigniew Jędrzejewski-Szmek 62fe94
+}
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+Workspace *workspace_unref(Workspace *w) {
Zbigniew Jędrzejewski-Szmek 62fe94
+        if (!w)
Zbigniew Jędrzejewski-Szmek 62fe94
+                return NULL;
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        assert(w->ref > 0);
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        if (--w->ref == 0)
Zbigniew Jędrzejewski-Szmek 62fe94
+                workspace_cleanup(w);
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        return NULL;
Zbigniew Jędrzejewski-Szmek 62fe94
+}
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+Workspace *workspace_attach(Workspace *w, Session *s) {
Zbigniew Jędrzejewski-Szmek 62fe94
+        assert(w);
Zbigniew Jędrzejewski-Szmek 62fe94
+        assert(s);
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        LIST_PREPEND(sessions_by_workspace, w->session_list, s);
Zbigniew Jędrzejewski-Szmek 62fe94
+        workspace_refresh(w);
Zbigniew Jędrzejewski-Szmek 62fe94
+        return workspace_ref(w);
Zbigniew Jędrzejewski-Szmek 62fe94
+}
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+Workspace *workspace_detach(Workspace *w, Session *s) {
Zbigniew Jędrzejewski-Szmek 62fe94
+        assert(w);
Zbigniew Jędrzejewski-Szmek 62fe94
+        assert(s);
Zbigniew Jędrzejewski-Szmek 62fe94
+        assert(s->active_ws == w);
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        LIST_REMOVE(sessions_by_workspace, w->session_list, s);
Zbigniew Jędrzejewski-Szmek 62fe94
+        workspace_refresh(w);
Zbigniew Jędrzejewski-Szmek 62fe94
+        return workspace_unref(w);
Zbigniew Jędrzejewski-Szmek 62fe94
+}
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+void workspace_refresh(Workspace *w) {
Zbigniew Jędrzejewski-Szmek 62fe94
+        uint32_t width, height;
Zbigniew Jędrzejewski-Szmek 62fe94
+        Terminal *t;
Zbigniew Jędrzejewski-Szmek 62fe94
+        Session *s;
Zbigniew Jędrzejewski-Szmek 62fe94
+        Display *d;
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        assert(w);
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        width = 0;
Zbigniew Jędrzejewski-Szmek 62fe94
+        height = 0;
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        /* find out minimum dimension of all attached displays */
Zbigniew Jędrzejewski-Szmek 62fe94
+        LIST_FOREACH(sessions_by_workspace, s, w->session_list) {
Zbigniew Jędrzejewski-Szmek 62fe94
+                LIST_FOREACH(displays_by_session, d, s->display_list) {
Zbigniew Jędrzejewski-Szmek 62fe94
+                        assert(d->width > 0 && d->height > 0);
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+                        if (width == 0 || d->width < width)
Zbigniew Jędrzejewski-Szmek 62fe94
+                                width = d->width;
Zbigniew Jędrzejewski-Szmek 62fe94
+                        if (height == 0 || d->height < height)
Zbigniew Jędrzejewski-Szmek 62fe94
+                                height = d->height;
Zbigniew Jędrzejewski-Szmek 62fe94
+                }
Zbigniew Jędrzejewski-Szmek 62fe94
+        }
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        /* either both are zero, or none is zero */
Zbigniew Jędrzejewski-Szmek 62fe94
+        assert(!(!width ^ !height));
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        /* update terminal-sizes if dimensions changed */
Zbigniew Jędrzejewski-Szmek 62fe94
+        if (w->width != width || w->height != height) {
Zbigniew Jędrzejewski-Szmek 62fe94
+                w->width = width;
Zbigniew Jędrzejewski-Szmek 62fe94
+                w->height = height;
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+                LIST_FOREACH(terminals_by_workspace, t, w->terminal_list)
Zbigniew Jędrzejewski-Szmek 62fe94
+                        terminal_resize(t);
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+                workspace_dirty(w);
Zbigniew Jędrzejewski-Szmek 62fe94
+        }
Zbigniew Jędrzejewski-Szmek 62fe94
+}
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+void workspace_dirty(Workspace *w) {
Zbigniew Jędrzejewski-Szmek 62fe94
+        Session *s;
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        assert(w);
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        LIST_FOREACH(sessions_by_workspace, s, w->session_list)
Zbigniew Jędrzejewski-Szmek 62fe94
+                session_dirty(s);
Zbigniew Jędrzejewski-Szmek 62fe94
+}
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+void workspace_feed(Workspace *w, idev_data *data) {
Zbigniew Jędrzejewski-Szmek 62fe94
+        assert(w);
Zbigniew Jędrzejewski-Szmek 62fe94
+        assert(data);
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        terminal_feed(w->current, data);
Zbigniew Jędrzejewski-Szmek 62fe94
+}
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+bool workspace_draw(Workspace *w, const grdev_display_target *target) {
Zbigniew Jędrzejewski-Szmek 62fe94
+        assert(w);
Zbigniew Jędrzejewski-Szmek 62fe94
+        assert(target);
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        return terminal_draw(w->current, target);
Zbigniew Jędrzejewski-Szmek 62fe94
+}
Zbigniew Jędrzejewski-Szmek 62fe94
diff --git a/src/console/consoled.c b/src/console/consoled.c
Zbigniew Jędrzejewski-Szmek 62fe94
new file mode 100644
Zbigniew Jędrzejewski-Szmek 62fe94
index 0000000000..b0c9eda1ac
Zbigniew Jędrzejewski-Szmek 62fe94
--- /dev/null
Zbigniew Jędrzejewski-Szmek 62fe94
+++ b/src/console/consoled.c
Zbigniew Jędrzejewski-Szmek 62fe94
@@ -0,0 +1,67 @@
Zbigniew Jędrzejewski-Szmek 62fe94
+/*-*- Mode: C; c-basic-offset: 8; indent-tabs-mode: nil -*-*/
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+/***
Zbigniew Jędrzejewski-Szmek 62fe94
+  This file is part of systemd.
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+  Copyright 2014 David Herrmann <dh.herrmann@gmail.com>
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+  systemd is free software; you can redistribute it and/or modify it
Zbigniew Jędrzejewski-Szmek 62fe94
+  under the terms of the GNU Lesser General Public License as published by
Zbigniew Jędrzejewski-Szmek 62fe94
+  the Free Software Foundation; either version 2.1 of the License, or
Zbigniew Jędrzejewski-Szmek 62fe94
+  (at your option) any later version.
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+  systemd is distributed in the hope that it will be useful, but
Zbigniew Jędrzejewski-Szmek 62fe94
+  WITHOUT ANY WARRANTY; without even the implied warranty of
Zbigniew Jędrzejewski-Szmek 62fe94
+  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
Zbigniew Jędrzejewski-Szmek 62fe94
+  Lesser General Public License for more details.
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+  You should have received a copy of the GNU Lesser General Public License
Zbigniew Jędrzejewski-Szmek 62fe94
+  along with systemd; If not, see <http://www.gnu.org/licenses/>.
Zbigniew Jędrzejewski-Szmek 62fe94
+***/
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+#include <errno.h>
Zbigniew Jędrzejewski-Szmek 62fe94
+#include <stdlib.h>
Zbigniew Jędrzejewski-Szmek 62fe94
+#include <string.h>
Zbigniew Jędrzejewski-Szmek 62fe94
+#include "consoled.h"
Zbigniew Jędrzejewski-Szmek 62fe94
+#include "log.h"
Zbigniew Jędrzejewski-Szmek 62fe94
+#include "sd-daemon.h"
Zbigniew Jędrzejewski-Szmek 62fe94
+#include "util.h"
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+int main(int argc, char *argv[]) {
Zbigniew Jędrzejewski-Szmek 62fe94
+        _cleanup_(manager_freep) Manager *m = NULL;
Zbigniew Jędrzejewski-Szmek 62fe94
+        int r;
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        log_set_target(LOG_TARGET_AUTO);
Zbigniew Jędrzejewski-Szmek 62fe94
+        log_parse_environment();
Zbigniew Jędrzejewski-Szmek 62fe94
+        log_open();
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        umask(0022);
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        if (argc != 1) {
Zbigniew Jędrzejewski-Szmek 62fe94
+                log_error("This program takes no arguments.");
Zbigniew Jędrzejewski-Szmek 62fe94
+                r = -EINVAL;
Zbigniew Jędrzejewski-Szmek 62fe94
+                goto out;
Zbigniew Jędrzejewski-Szmek 62fe94
+        }
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        r = manager_new(&m);
Zbigniew Jędrzejewski-Szmek 62fe94
+        if (r < 0) {
Zbigniew Jędrzejewski-Szmek 62fe94
+                log_error("Could not create manager: %s", strerror(-r));
Zbigniew Jędrzejewski-Szmek 62fe94
+                goto out;
Zbigniew Jędrzejewski-Szmek 62fe94
+        }
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        sd_notify(false,
Zbigniew Jędrzejewski-Szmek 62fe94
+                  "READY=1\n"
Zbigniew Jędrzejewski-Szmek 62fe94
+                  "STATUS=Processing requests...");
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        r = manager_run(m);
Zbigniew Jędrzejewski-Szmek 62fe94
+        if (r < 0) {
Zbigniew Jędrzejewski-Szmek 62fe94
+                log_error("Cannot run manager: %s", strerror(-r));
Zbigniew Jędrzejewski-Szmek 62fe94
+                goto out;
Zbigniew Jędrzejewski-Szmek 62fe94
+        }
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+out:
Zbigniew Jędrzejewski-Szmek 62fe94
+        sd_notify(false,
Zbigniew Jędrzejewski-Szmek 62fe94
+                  "STATUS=Shutting down...");
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        return r < 0 ? EXIT_FAILURE : EXIT_SUCCESS;
Zbigniew Jędrzejewski-Szmek 62fe94
+}
Zbigniew Jędrzejewski-Szmek 62fe94
diff --git a/src/console/consoled.h b/src/console/consoled.h
Zbigniew Jędrzejewski-Szmek 62fe94
new file mode 100644
Zbigniew Jędrzejewski-Szmek 62fe94
index 0000000000..f8a3df4487
Zbigniew Jędrzejewski-Szmek 62fe94
--- /dev/null
Zbigniew Jędrzejewski-Szmek 62fe94
+++ b/src/console/consoled.h
Zbigniew Jędrzejewski-Szmek 62fe94
@@ -0,0 +1,170 @@
Zbigniew Jędrzejewski-Szmek 62fe94
+/*-*- Mode: C; c-basic-offset: 8; indent-tabs-mode: nil -*-*/
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+#pragma once
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+/***
Zbigniew Jędrzejewski-Szmek 62fe94
+  This file is part of systemd.
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+  Copyright 2014 David Herrmann <dh.herrmann@gmail.com>
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+  systemd is free software; you can redistribute it and/or modify it
Zbigniew Jędrzejewski-Szmek 62fe94
+  under the terms of the GNU Lesser General Public License as published by
Zbigniew Jędrzejewski-Szmek 62fe94
+  the Free Software Foundation; either version 2.1 of the License, or
Zbigniew Jędrzejewski-Szmek 62fe94
+  (at your option) any later version.
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+  systemd is distributed in the hope that it will be useful, but
Zbigniew Jędrzejewski-Szmek 62fe94
+  WITHOUT ANY WARRANTY; without even the implied warranty of
Zbigniew Jędrzejewski-Szmek 62fe94
+  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
Zbigniew Jędrzejewski-Szmek 62fe94
+  Lesser General Public License for more details.
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+  You should have received a copy of the GNU Lesser General Public License
Zbigniew Jędrzejewski-Szmek 62fe94
+  along with systemd; If not, see <http://www.gnu.org/licenses/>.
Zbigniew Jędrzejewski-Szmek 62fe94
+***/
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+#include <errno.h>
Zbigniew Jędrzejewski-Szmek 62fe94
+#include <inttypes.h>
Zbigniew Jędrzejewski-Szmek 62fe94
+#include <libudev.h>
Zbigniew Jędrzejewski-Szmek 62fe94
+#include <stdlib.h>
Zbigniew Jędrzejewski-Szmek 62fe94
+#include "grdev.h"
Zbigniew Jędrzejewski-Szmek 62fe94
+#include "hashmap.h"
Zbigniew Jędrzejewski-Szmek 62fe94
+#include "idev.h"
Zbigniew Jędrzejewski-Szmek 62fe94
+#include "list.h"
Zbigniew Jędrzejewski-Szmek 62fe94
+#include "macro.h"
Zbigniew Jędrzejewski-Szmek 62fe94
+#include "pty.h"
Zbigniew Jędrzejewski-Szmek 62fe94
+#include "sd-bus.h"
Zbigniew Jędrzejewski-Szmek 62fe94
+#include "sd-event.h"
Zbigniew Jędrzejewski-Szmek 62fe94
+#include "sysview.h"
Zbigniew Jędrzejewski-Szmek 62fe94
+#include "term.h"
Zbigniew Jędrzejewski-Szmek 62fe94
+#include "unifont.h"
Zbigniew Jędrzejewski-Szmek 62fe94
+#include "util.h"
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+typedef struct Manager Manager;
Zbigniew Jędrzejewski-Szmek 62fe94
+typedef struct Session Session;
Zbigniew Jędrzejewski-Szmek 62fe94
+typedef struct Display Display;
Zbigniew Jędrzejewski-Szmek 62fe94
+typedef struct Workspace Workspace;
Zbigniew Jędrzejewski-Szmek 62fe94
+typedef struct Terminal Terminal;
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+/*
Zbigniew Jędrzejewski-Szmek 62fe94
+ * Terminals
Zbigniew Jędrzejewski-Szmek 62fe94
+ */
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+struct Terminal {
Zbigniew Jędrzejewski-Szmek 62fe94
+        Workspace *workspace;
Zbigniew Jędrzejewski-Szmek 62fe94
+        LIST_FIELDS(Terminal, terminals_by_workspace);
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        term_utf8 utf8;
Zbigniew Jędrzejewski-Szmek 62fe94
+        term_parser *parser;
Zbigniew Jędrzejewski-Szmek 62fe94
+        term_screen *screen;
Zbigniew Jędrzejewski-Szmek 62fe94
+        Pty *pty;
Zbigniew Jędrzejewski-Szmek 62fe94
+};
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+int terminal_new(Terminal **out, Workspace *w);
Zbigniew Jędrzejewski-Szmek 62fe94
+Terminal *terminal_free(Terminal *t);
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+DEFINE_TRIVIAL_CLEANUP_FUNC(Terminal*, terminal_free);
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+void terminal_resize(Terminal *t);
Zbigniew Jędrzejewski-Szmek 62fe94
+void terminal_run(Terminal *t);
Zbigniew Jędrzejewski-Szmek 62fe94
+void terminal_feed(Terminal *t, idev_data *data);
Zbigniew Jędrzejewski-Szmek 62fe94
+bool terminal_draw(Terminal *t, const grdev_display_target *target);
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+/*
Zbigniew Jędrzejewski-Szmek 62fe94
+ * Workspaces
Zbigniew Jędrzejewski-Szmek 62fe94
+ */
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+struct Workspace {
Zbigniew Jędrzejewski-Szmek 62fe94
+        unsigned long ref;
Zbigniew Jędrzejewski-Szmek 62fe94
+        Manager *manager;
Zbigniew Jędrzejewski-Szmek 62fe94
+        LIST_FIELDS(Workspace, workspaces_by_manager);
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        LIST_HEAD(Terminal, terminal_list);
Zbigniew Jędrzejewski-Szmek 62fe94
+        Terminal *current;
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        LIST_HEAD(Session, session_list);
Zbigniew Jędrzejewski-Szmek 62fe94
+        uint32_t width;
Zbigniew Jędrzejewski-Szmek 62fe94
+        uint32_t height;
Zbigniew Jędrzejewski-Szmek 62fe94
+};
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+int workspace_new(Workspace **out, Manager *m);
Zbigniew Jędrzejewski-Szmek 62fe94
+Workspace *workspace_ref(Workspace *w);
Zbigniew Jędrzejewski-Szmek 62fe94
+Workspace *workspace_unref(Workspace *w);
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+DEFINE_TRIVIAL_CLEANUP_FUNC(Workspace*, workspace_unref);
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+Workspace *workspace_attach(Workspace *w, Session *s);
Zbigniew Jędrzejewski-Szmek 62fe94
+Workspace *workspace_detach(Workspace *w, Session *s);
Zbigniew Jędrzejewski-Szmek 62fe94
+void workspace_refresh(Workspace *w);
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+void workspace_dirty(Workspace *w);
Zbigniew Jędrzejewski-Szmek 62fe94
+void workspace_feed(Workspace *w, idev_data *data);
Zbigniew Jędrzejewski-Szmek 62fe94
+bool workspace_draw(Workspace *w, const grdev_display_target *target);
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+/*
Zbigniew Jędrzejewski-Szmek 62fe94
+ * Displays
Zbigniew Jędrzejewski-Szmek 62fe94
+ */
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+struct Display {
Zbigniew Jędrzejewski-Szmek 62fe94
+        Session *session;
Zbigniew Jędrzejewski-Szmek 62fe94
+        LIST_FIELDS(Display, displays_by_session);
Zbigniew Jędrzejewski-Szmek 62fe94
+        grdev_display *grdev;
Zbigniew Jędrzejewski-Szmek 62fe94
+        uint32_t width;
Zbigniew Jędrzejewski-Szmek 62fe94
+        uint32_t height;
Zbigniew Jędrzejewski-Szmek 62fe94
+};
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+int display_new(Display **out, Session *s, grdev_display *grdev);
Zbigniew Jędrzejewski-Szmek 62fe94
+Display *display_free(Display *d);
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+DEFINE_TRIVIAL_CLEANUP_FUNC(Display*, display_free);
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+void display_refresh(Display *d);
Zbigniew Jędrzejewski-Szmek 62fe94
+void display_render(Display *d, Workspace *w);
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+/*
Zbigniew Jędrzejewski-Szmek 62fe94
+ * Sessions
Zbigniew Jędrzejewski-Szmek 62fe94
+ */
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+struct Session {
Zbigniew Jędrzejewski-Szmek 62fe94
+        Manager *manager;
Zbigniew Jędrzejewski-Szmek 62fe94
+        sysview_session *sysview;
Zbigniew Jędrzejewski-Szmek 62fe94
+        grdev_session *grdev;
Zbigniew Jędrzejewski-Szmek 62fe94
+        idev_session *idev;
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        LIST_FIELDS(Session, sessions_by_workspace);
Zbigniew Jędrzejewski-Szmek 62fe94
+        Workspace *my_ws;
Zbigniew Jędrzejewski-Szmek 62fe94
+        Workspace *active_ws;
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        LIST_HEAD(Display, display_list);
Zbigniew Jędrzejewski-Szmek 62fe94
+        sd_event_source *redraw_src;
Zbigniew Jędrzejewski-Szmek 62fe94
+};
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+int session_new(Session **out, Manager *m, sysview_session *session);
Zbigniew Jędrzejewski-Szmek 62fe94
+Session *session_free(Session *s);
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+DEFINE_TRIVIAL_CLEANUP_FUNC(Session*, session_free);
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+void session_dirty(Session *s);
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+void session_add_device(Session *s, sysview_device *device);
Zbigniew Jędrzejewski-Szmek 62fe94
+void session_remove_device(Session *s, sysview_device *device);
Zbigniew Jędrzejewski-Szmek 62fe94
+void session_refresh_device(Session *s, sysview_device *device, struct udev_device *ud);
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+/*
Zbigniew Jędrzejewski-Szmek 62fe94
+ * Managers
Zbigniew Jędrzejewski-Szmek 62fe94
+ */
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+struct Manager {
Zbigniew Jędrzejewski-Szmek 62fe94
+        sd_event *event;
Zbigniew Jędrzejewski-Szmek 62fe94
+        sd_bus *sysbus;
Zbigniew Jędrzejewski-Szmek 62fe94
+        unifont *uf;
Zbigniew Jędrzejewski-Szmek 62fe94
+        sysview_context *sysview;
Zbigniew Jędrzejewski-Szmek 62fe94
+        grdev_context *grdev;
Zbigniew Jędrzejewski-Szmek 62fe94
+        idev_context *idev;
Zbigniew Jędrzejewski-Szmek 62fe94
+        LIST_HEAD(Workspace, workspace_list);
Zbigniew Jędrzejewski-Szmek 62fe94
+};
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+int manager_new(Manager **out);
Zbigniew Jędrzejewski-Szmek 62fe94
+Manager *manager_free(Manager *m);
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+DEFINE_TRIVIAL_CLEANUP_FUNC(Manager*, manager_free);
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+int manager_run(Manager *m);