aboutsummaryrefslogtreecommitdiff
path: root/kernel/drivers/ps2
diff options
context:
space:
mode:
Diffstat (limited to 'kernel/drivers/ps2')
-rw-r--r--kernel/drivers/ps2/keyboard.c83
-rw-r--r--kernel/drivers/ps2/mouse.c196
-rw-r--r--kernel/drivers/ps2/ps2.c229
3 files changed, 508 insertions, 0 deletions
diff --git a/kernel/drivers/ps2/keyboard.c b/kernel/drivers/ps2/keyboard.c
new file mode 100644
index 0000000..9a19b5c
--- /dev/null
+++ b/kernel/drivers/ps2/keyboard.c
@@ -0,0 +1,83 @@
+// MIT License, Copyright (c) 2020 Marvin Borner
+
+#include <cpu.h>
+#include <def.h>
+#include <errno.h>
+#include <fs.h>
+#include <interrupts.h>
+#include <mem.h>
+#include <print.h>
+#include <proc.h>
+#include <ps2.h>
+#include <stack.h>
+#include <str.h>
+#include <sys.h>
+
+PROTECTED static struct stack *queue = NULL;
+PROTECTED static u32 dev_id = 0;
+
+static struct event_keyboard *event = NULL;
+static int state = 0;
+static int merged = 0;
+static void keyboard_handler(struct regs *r)
+{
+ UNUSED(r);
+ u8 scancode = ps2_read_data();
+
+ // TODO: Support more than two-byte scancodes
+ if (scancode == 0xe0) {
+ merged = 0xe0;
+ state = 1;
+ return;
+ } else {
+ merged = scancode << 8 | merged;
+ }
+
+ // TODO: "Merge" scancode to linux keycode?
+ /* printf("%x %x = %x\n", scancode, state ? 0xe0 : 0, merged); */
+
+ event = malloc(sizeof(*event));
+ event->magic = KEYBOARD_MAGIC;
+ event->press = (scancode & 0x80) == 0;
+ event->scancode = event->press ? scancode : scancode & ~0x80;
+ stack_push_bot(queue, event);
+
+ state = 0;
+ merged = 0;
+}
+
+static res keyboard_read(void *buf, u32 offset, u32 count, struct vfs_dev *dev)
+{
+ UNUSED(dev);
+ if (stack_empty(queue))
+ return -EINVAL;
+
+ struct event_keyboard *e = stack_pop(queue);
+ memcpy_user(buf, (u8 *)e + offset, MIN(count, sizeof(*e)));
+ free(e);
+ return MIN(count, sizeof(*e));
+}
+
+static res keyboard_ready(void)
+{
+ return !stack_empty(queue);
+}
+
+CLEAR void ps2_keyboard_reset(void)
+{
+ stack_clear(queue);
+}
+
+CLEAR void ps2_keyboard_install(void)
+{
+ //keyboard_rate(); TODO: Fix keyboard rate?
+ irq_install_handler(1, keyboard_handler);
+
+ queue = stack_new();
+ struct vfs_dev *dev = zalloc(sizeof(*dev));
+ dev->name = strdup("kbd");
+ dev->type = DEV_CHAR;
+ dev->read = keyboard_read;
+ /* device_add(dev); */
+ dev_id = dev->id;
+}
diff --git a/kernel/drivers/ps2/mouse.c b/kernel/drivers/ps2/mouse.c
new file mode 100644
index 0000000..9b3fa2e
--- /dev/null
+++ b/kernel/drivers/ps2/mouse.c
@@ -0,0 +1,196 @@
+// MIT License, Copyright (c) 2020 Marvin Borner
+
+#include <boot.h>
+#include <cpu.h>
+#include <errno.h>
+#include <fs.h>
+#include <interrupts.h>
+#include <mem.h>
+#include <print.h>
+#include <proc.h>
+#include <ps2.h>
+#include <stack.h>
+#include <str.h>
+#include <sys.h>
+
+PROTECTED static struct stack *queue = NULL;
+PROTECTED static u32 dev_id = 0;
+
+static struct event_mouse *event = NULL;
+static char mouse_cycle = 0;
+static char mouse_byte[3] = { 0 };
+static void mouse_handler(struct regs *r)
+{
+ UNUSED(r);
+ switch (mouse_cycle) {
+ case 0:
+ mouse_byte[0] = ps2_read_data();
+ if (((mouse_byte[0] >> 3) & 1) == 1)
+ mouse_cycle++;
+ else
+ mouse_cycle = 0;
+ break;
+ case 1:
+ mouse_byte[1] = ps2_read_data();
+ mouse_cycle++;
+ break;
+ case 2:
+ mouse_byte[2] = ps2_read_data();
+
+ event = malloc(sizeof(*event));
+ event->magic = MOUSE_MAGIC;
+ event->diff_x = mouse_byte[1];
+ event->diff_y = mouse_byte[2];
+ event->but1 = mouse_byte[0] & 1;
+ event->but2 = (mouse_byte[0] >> 1) & 1;
+ event->but3 = (mouse_byte[0] >> 2) & 1;
+ stack_push_bot(queue, event);
+ mouse_cycle = 0;
+ break;
+ default:
+ break;
+ }
+}
+
+#define MOUSE_WAIT_OUT 0
+#define MOUSE_WAIT_IN 1
+CLEAR static void mouse_serial_wait(u8 in)
+{
+ u32 time_out = 100000;
+ if (in) {
+ while (time_out--)
+ if (ps2_read_status().in_full)
+ return;
+ return;
+ } else {
+ while (time_out--)
+ if (ps2_read_status().out_full)
+ return;
+ return;
+ }
+}
+
+CLEAR static void mouse_serial_write(u8 data)
+{
+ mouse_serial_wait(MOUSE_WAIT_IN);
+ ps2_write_command(0xd4);
+ mouse_serial_wait(MOUSE_WAIT_IN);
+ ps2_write_data(data);
+}
+
+CLEAR static u8 mouse_serial_read(void)
+{
+ mouse_serial_wait(MOUSE_WAIT_OUT);
+ return ps2_read_data();
+}
+
+static res mouse_ready(void)
+{
+ return !stack_empty(queue);
+}
+
+static res mouse_read(void *buf, u32 offset, u32 count, struct vfs_dev *dev)
+{
+ (void)dev;
+ if (stack_empty(queue))
+ return -EINVAL;
+
+ struct event_mouse *e = stack_pop(queue);
+ memcpy_user(buf, (u8 *)e + offset, MIN(count, sizeof(*e)));
+ free(e);
+ return MIN(count, sizeof(*e));
+}
+
+CLEAR void ps2_mouse_install(void)
+{
+ u8 status;
+
+ // Enable auxiliary mouse device
+ mouse_serial_wait(MOUSE_WAIT_IN);
+ ps2_write_command(0xa8);
+
+ // Enable interrupts
+ mouse_serial_wait(MOUSE_WAIT_IN);
+ ps2_write_command(0x20);
+ mouse_serial_wait(MOUSE_WAIT_OUT);
+ status = ps2_read_data() | 3;
+ mouse_serial_wait(MOUSE_WAIT_IN);
+ ps2_write_command(0x60);
+ mouse_serial_wait(MOUSE_WAIT_IN);
+ ps2_write_data(status);
+
+ // Use default settings
+ mouse_serial_write(0xf6);
+ mouse_serial_read();
+
+ // Enable mousewheel
+ mouse_serial_write(0xf2);
+ mouse_serial_read();
+ mouse_serial_read();
+ mouse_serial_write(0xf3);
+ mouse_serial_read();
+ mouse_serial_write(200);
+ mouse_serial_read();
+ mouse_serial_write(0xf3);
+ mouse_serial_read();
+ mouse_serial_write(100);
+ mouse_serial_read();
+ mouse_serial_write(0xf3);
+ mouse_serial_read();
+ mouse_serial_write(80);
+ mouse_serial_read();
+ mouse_serial_write(0xf2);
+ mouse_serial_read();
+ status = (u8)mouse_serial_read();
+ if (status == 3) {
+ }
+ /* printf("Scrollwheel support!\n"); */
+
+ // Activate 4th and 5th mouse buttons
+ mouse_serial_write(0xf2);
+ mouse_serial_read();
+ mouse_serial_read();
+ mouse_serial_write(0xf3);
+ mouse_serial_read();
+ mouse_serial_write(200);
+ mouse_serial_read();
+ mouse_serial_write(0xf3);
+ mouse_serial_read();
+ mouse_serial_write(200);
+ mouse_serial_read();
+ mouse_serial_write(0xf3);
+ mouse_serial_read();
+ mouse_serial_write(80);
+ mouse_serial_read();
+ mouse_serial_write(0xf2);
+ mouse_serial_read();
+ status = (u8)mouse_serial_read();
+ if (status == 4) {
+ }
+ /* printf("4th and 5th mouse button support!\n"); */
+
+ /* TODO: Fix mouse laggyness
+ mouse_serial_write(0xE8);
+ mouse_serial_read();
+ mouse_serial_write(0x03);
+ mouse_serial_read();
+ mouse_serial_write(0xF3);
+ mouse_serial_read();
+ mouse_serial_write(200);
+ mouse_serial_read(); */
+
+ // Enable mouse
+ mouse_serial_write(0xf4);
+ mouse_serial_read();
+
+ // Setup the mouse handler
+ irq_install_handler(12, mouse_handler);
+
+ queue = stack_new();
+ struct vfs_dev *dev = zalloc(sizeof(*dev));
+ dev->name = strdup("mouse");
+ dev->type = DEV_CHAR;
+ dev->read = mouse_read;
+ /* device_add(dev); */
+ dev_id = dev->id;
+}
diff --git a/kernel/drivers/ps2/ps2.c b/kernel/drivers/ps2/ps2.c
new file mode 100644
index 0000000..1c73855
--- /dev/null
+++ b/kernel/drivers/ps2/ps2.c
@@ -0,0 +1,229 @@
+// MIT License, Copyright (c) 2021 Marvin Borner
+
+#include <assert.h>
+#include <cpu.h>
+#include <def.h>
+#include <print.h>
+#include <ps2.h>
+
+#define PS2_TIMEOUT 100000
+
+struct ps2_status ps2_read_status(void)
+{
+ u8 byte = inb(0x64);
+ return *(struct ps2_status *)&byte;
+}
+
+static u8 ps2_wait_readable(void)
+{
+ u32 time_out = PS2_TIMEOUT;
+ while (time_out--)
+ if (ps2_read_status().in_full)
+ return 1;
+ /* print("PS/2 readable timeout\n"); */
+ return 0;
+}
+
+static u8 ps2_wait_writable(void)
+{
+ u32 time_out = PS2_TIMEOUT;
+ while (time_out--)
+ if (!ps2_read_status().out_full)
+ return 1;
+ /* print("PS/2 writable timeout\n"); */
+ return 0;
+}
+
+u8 ps2_read_data(void)
+{
+ if (ps2_wait_readable()) {
+ return inb(0x60);
+ } else {
+ return 0;
+ }
+}
+
+u8 ps2_write_data(u8 byte)
+{
+ if (ps2_wait_writable()) {
+ outb(0x60, byte);
+ return 1;
+ } else {
+ return 0;
+ }
+}
+
+u8 ps2_write_command(u8 byte)
+{
+ if (ps2_wait_writable()) {
+ outb(0x64, byte);
+ return 1;
+ } else {
+ return 0;
+ }
+}
+
+CLEAR static struct ps2_config ps2_read_config(void)
+{
+ assert(ps2_write_command(0x20));
+ u8 config = ps2_read_data();
+ return *(struct ps2_config *)&config;
+}
+
+CLEAR static u8 ps2_write_config(struct ps2_config config)
+{
+ // Select first byte
+ if (!ps2_write_command(0x60))
+ return 0;
+
+ return ps2_write_data(*(u8 *)&config);
+}
+
+#define PS2_TYPE_STANDARD_MOUSE 0x0000
+#define PS2_TYPE_WHEEL_MOUSE 0x0003
+#define PS2_TYPE_BUTTON_MOUSE 0x0004
+#define PS2_TYPE_TRANSLATION_KEYBOARD1 0xab41
+#define PS2_TYPE_TRANSLATION_KEYBOARD2 0xabc1
+#define PS2_TYPE_STANDARD_KEYBOARD 0xab83
+
+PROTECTED static struct {
+ u8 detected : 1;
+ struct {
+ u8 exists : 1;
+ u16 type;
+ } first;
+ struct {
+ u8 exists : 1;
+ u16 type;
+ } second;
+} info = { 0 };
+
+CLEAR static u8 ps2_write_device(u8 device, u8 data)
+{
+ u8 resp = PS2_RESEND;
+ for (u8 i = 0; resp == PS2_RESEND && i < 3; i++) {
+ if (device == 1)
+ ps2_write_command(0xd4);
+ ps2_write_data(data);
+ resp = ps2_read_data();
+ }
+
+ if (resp != PS2_ACK)
+ return 0;
+
+ return 1;
+}
+
+CLEAR static u8 ps2_device_keyboard(u16 type)
+{
+ return type == PS2_TYPE_TRANSLATION_KEYBOARD1 || type == PS2_TYPE_TRANSLATION_KEYBOARD2 ||
+ type == PS2_TYPE_STANDARD_KEYBOARD;
+}
+
+CLEAR static u8 ps2_device_mouse(u16 type)
+{
+ return type == PS2_TYPE_STANDARD_MOUSE || type == PS2_TYPE_WHEEL_MOUSE ||
+ type == PS2_TYPE_BUTTON_MOUSE;
+}
+
+CLEAR u8 ps2_keyboard_support(void)
+{
+ if (!info.detected)
+ return 0;
+
+ // Find, reset and self-test
+ if ((info.first.exists && ps2_device_keyboard(info.first.type) &&
+ ps2_write_device(0, 0xff)) ||
+ (info.second.exists && ps2_device_keyboard(info.second.type) &&
+ ps2_write_device(1, 0xff)))
+ return ps2_read_data() == 0xaa;
+
+ return 0;
+}
+
+CLEAR u8 ps2_mouse_support(void)
+{
+ if (!info.detected)
+ return 0;
+
+ // Find, reset and self-test
+ if ((info.first.exists && ps2_device_mouse(info.first.type) && ps2_write_device(0, 0xff)) ||
+ (info.second.exists && ps2_device_mouse(info.second.type) && ps2_write_device(1, 0xff)))
+ return ps2_read_data() == 0xaa;
+
+ return 0;
+}
+
+CLEAR void ps2_detect(void)
+{
+ // TODO: Read ACPI 8042 flag in FADT to verify PS/2 support
+
+ // Disable PS/2 ports
+ ps2_write_command(0xad);
+ ps2_write_command(0xa7);
+
+ // Get config and disable IRQs
+ struct ps2_config config = ps2_read_config();
+ assert(!config.zero1 && !config.zero2);
+ config.first_int = 0;
+ config.second_int = 0;
+ ps2_write_config(config);
+
+ // Test PS/2 controller
+ ps2_write_command(0xaa);
+ if (ps2_read_data() != 0x55)
+ return;
+ ps2_write_config(config);
+
+ // Test first PS/2 port
+ ps2_write_command(0xab);
+ if (ps2_read_data() == 0x0) {
+ // Enable first port
+ ps2_write_command(0xae);
+ config.first_int = 1;
+ config.first_clock_disabled = 0;
+ info.first.exists = 1;
+ }
+
+ // Test and enable second PS/2 port if available
+ if (config.second_clock_disabled) {
+ ps2_write_command(0xa9);
+ if (ps2_read_data() == 0x0) {
+ // Enable second port
+ ps2_write_command(0xa8);
+ config.second_int = 1;
+ config.second_clock_disabled = 0;
+ info.second.exists = 1;
+ }
+ }
+
+ ps2_write_config(config);
+
+ // Detect device type of first port
+ if (info.first.exists) {
+ if (!ps2_write_device(0, 0xf5))
+ return;
+
+ if (!ps2_write_device(0, 0xf2))
+ return;
+
+ u8 first = ps2_read_data();
+ u8 second = ps2_read_data();
+ info.first.type = (first << 8) | second;
+ }
+
+ // Detect device type of second port
+ if (info.second.exists) {
+ if (!ps2_write_device(1, 0xf5))
+ return;
+
+ if (!ps2_write_device(1, 0xf2))
+ return;
+
+ u8 first = ps2_read_data();
+ u8 second = ps2_read_data(); // This shall timeout if it's a mouse
+ info.second.type = (first << 8) | second;
+ }
+
+ info.detected = 1;
+}