summaryrefslogtreecommitdiff
path: root/kern
diff options
context:
space:
mode:
authorRichard Braun <rbraun@sceen.net>2017-05-19 19:11:34 +0200
committerRichard Braun <rbraun@sceen.net>2017-05-19 19:11:34 +0200
commit805ba95708f577ea3cfb59f9851ea61ef45abc85 (patch)
tree81e480b1b7d0e730f52ebb5a5668819d15c5e907 /kern
parent3c203f5b93f23927a1e046f120a1a9438c5213bc (diff)
kern/console: new module
Diffstat (limited to 'kern')
-rw-r--r--kern/console.c69
-rw-r--r--kern/console.h71
-rw-r--r--kern/printf.c8
3 files changed, 142 insertions, 6 deletions
diff --git a/kern/console.c b/kern/console.c
new file mode 100644
index 00000000..138a4248
--- /dev/null
+++ b/kern/console.c
@@ -0,0 +1,69 @@
+/*
+ * Copyright (c) 2017 Richard Braun.
+ *
+ * This program is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation, either version 3 of the License, or
+ * (at your option) any later version.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ * GNU General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program. If not, see <http://www.gnu.org/licenses/>.
+ */
+
+#include <stddef.h>
+
+#include <kern/assert.h>
+#include <kern/init.h>
+#include <kern/console.h>
+#include <kern/list.h>
+#include <kern/spinlock.h>
+
+static struct list console_devs;
+static struct spinlock console_lock;
+
+static void
+console_putc(struct console *console, char c)
+{
+ unsigned long flags;
+
+ spinlock_lock_intr_save(&console->lock, &flags);
+ console->putc(console, c);
+ spinlock_unlock_intr_restore(&console->lock, flags);
+}
+
+void __init
+console_setup(void)
+{
+ list_init(&console_devs);
+ spinlock_init(&console_lock);
+}
+
+void __init
+console_register(struct console *console)
+{
+ assert(console->putc != NULL);
+
+ spinlock_lock(&console_lock);
+ list_insert_tail(&console_devs, &console->node);
+ spinlock_unlock(&console_lock);
+}
+
+void
+console_write_char(char c)
+{
+ struct console *console;
+ unsigned long flags;
+
+ spinlock_lock_intr_save(&console_lock, &flags);
+
+ list_for_each_entry(&console_devs, console, node) {
+ console_putc(console, c);
+ }
+
+ spinlock_unlock_intr_restore(&console_lock, flags);
+}
diff --git a/kern/console.h b/kern/console.h
new file mode 100644
index 00000000..8fdef160
--- /dev/null
+++ b/kern/console.h
@@ -0,0 +1,71 @@
+/*
+ * Copyright (c) 2017 Richard Braun.
+ *
+ * This program is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation, either version 3 of the License, or
+ * (at your option) any later version.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ * GNU General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program. If not, see <http://www.gnu.org/licenses/>.
+ *
+ *
+ * Device-independent console interface.
+ */
+
+#ifndef _KERN_CONSOLE_H
+#define _KERN_CONSOLE_H
+
+#include <kern/list.h>
+#include <kern/spinlock.h>
+
+struct console;
+
+/*
+ * Type for character writing functions.
+ */
+typedef void (*console_putc_fn)(struct console *console, char c);
+
+/*
+ * Console device.
+ *
+ * This structure should be embedded in the hardware-specific console
+ * objects. Calls to console operations are all serialized by this module
+ * for each device.
+ */
+struct console {
+ struct spinlock lock;
+ struct list node;
+ console_putc_fn putc;
+};
+
+static inline void
+console_init(struct console *console, console_putc_fn putc)
+{
+ spinlock_init(&console->lock);
+ console->putc = putc;
+}
+
+/*
+ * Initialize the console module.
+ */
+void console_setup(void);
+
+/*
+ * Register a console device.
+ *
+ * The given console must be initialized before calling this function.
+ */
+void console_register(struct console *console);
+
+/*
+ * Write a single character to all registered console devices.
+ */
+void console_write_char(char c);
+
+#endif /* _KERN_CONSOLE_H */
diff --git a/kern/printf.c b/kern/printf.c
index 68ae1037..d7a1d100 100644
--- a/kern/printf.c
+++ b/kern/printf.c
@@ -17,6 +17,7 @@
#include <stdio.h>
+#include <kern/console.h>
#include <kern/spinlock.h>
#include <machine/cpu.h>
@@ -25,11 +26,6 @@
*/
#define PRINTF_BUFSIZE 1024
-/*
- * XXX Must be provided by a console driver.
- */
-extern void console_write_byte(char c);
-
static char printf_buffer[PRINTF_BUFSIZE];
static struct spinlock printf_lock;
@@ -58,7 +54,7 @@ vprintf(const char *format, va_list ap)
length = vsnprintf(printf_buffer, sizeof(printf_buffer), format, ap);
for (ptr = printf_buffer; *ptr != '\0'; ptr++) {
- console_write_byte(*ptr);
+ console_write_char(*ptr);
}
spinlock_unlock_intr_restore(&printf_lock, flags);