tty: Add handler for local tty mirroring
This change adds a simple handler for mirroring data to local tty
device, like a hardware UART.
This is specified through a new config parameter 'local-tty'.
Signed-off-by: Jeremy Kerr <jk@ozlabs.org>
diff --git a/Makefile.am b/Makefile.am
index 5f0a1bb..2df24d8 100644
--- a/Makefile.am
+++ b/Makefile.am
@@ -34,6 +34,7 @@
config.c \
log-handler.c \
socket-handler.c \
+ tty-handler.c \
console-socket.c
obmc_console_client_SOURCES = \
diff --git a/obmc-console.conf.sample b/obmc-console.conf.sample
index a5b8808..d473af2 100644
--- a/obmc-console.conf.sample
+++ b/obmc-console.conf.sample
@@ -4,3 +4,6 @@
# For VUART devices, we can specify the LPC address and SIRQ parameters
lpc-address = 0x3f8
sirq = 4
+
+# To mirror to a local tty device (typically a hardware UART), set local-tty
+# local-tty = ttyS0
diff --git a/tty-handler.c b/tty-handler.c
new file mode 100644
index 0000000..00ada7a
--- /dev/null
+++ b/tty-handler.c
@@ -0,0 +1,123 @@
+/**
+ * Copyright © 2016 IBM Corporation
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+#define _GNU_SOURCE
+
+#include <assert.h>
+#include <err.h>
+#include <fcntl.h>
+#include <stdio.h>
+#include <stdlib.h>
+#include <unistd.h>
+
+#include "console-server.h"
+
+struct tty_handler {
+ struct handler handler;
+ struct console *console;
+ struct poller *poller;
+ int fd;
+};
+
+static struct tty_handler *to_tty_handler(struct handler *handler)
+{
+ return container_of(handler, struct tty_handler, handler);
+}
+
+static enum poller_ret tty_poll(struct handler *handler,
+ int events, void __attribute__((unused)) *data)
+{
+ struct tty_handler *th = to_tty_handler(handler);
+ uint8_t buf[4096];
+ ssize_t len;
+
+ if (!(events & POLLIN))
+ return POLLER_OK;
+
+ len = read(th->fd, buf, sizeof(buf));
+ if (len <= 0) {
+ th->poller = NULL;
+ close(th->fd);
+ return POLLER_REMOVE;
+ }
+
+ console_data_out(th->console, buf, len);
+
+ return POLLER_OK;
+}
+
+static int tty_init(struct handler *handler, struct console *console,
+ struct config *config __attribute__((unused)))
+{
+ struct tty_handler *th = to_tty_handler(handler);
+ const char *tty_name;
+ char *tty_path;
+ int rc, flags;
+
+ tty_name = config_get_value(config, "local-tty");
+ if (!tty_name)
+ return -1;
+
+ rc = asprintf(&tty_path, "/dev/%s", tty_name);
+ if (!rc)
+ return -1;
+
+ th->fd = open(tty_path, O_RDWR);
+ if (th->fd < 0) {
+ warn("Can't open %s; disabling local tty", tty_name);
+ free(tty_path);
+ return -1;
+ }
+
+ free(tty_path);
+
+ /* initial tty setup */
+ flags = fcntl(th->fd, F_GETFL, 0);
+ flags |= FNDELAY;
+ fcntl(th->fd, F_SETFL, flags);
+
+ th->poller = console_register_poller(console, handler, tty_poll,
+ th->fd, POLLIN, NULL);
+ th->console = console;
+
+ return 0;
+}
+
+static int tty_data(struct handler *handler, uint8_t *buf, size_t len)
+{
+ struct tty_handler *th = to_tty_handler(handler);
+ return write_buf_to_fd(th->fd, buf, len);
+}
+
+static void tty_fini(struct handler *handler)
+{
+ struct tty_handler *th = to_tty_handler(handler);
+ if (th->poller)
+ console_unregister_poller(th->console, th->poller);
+ close(th->fd);
+}
+
+static struct tty_handler tty_handler = {
+ .handler = {
+ .name = "tty",
+ .init = tty_init,
+ .data_in = tty_data,
+ .fini = tty_fini,
+ },
+};
+
+console_register_handler(&tty_handler.handler);
+