2024-11-16 14:51:55 -06:00
|
|
|
.{memory, log} := @use("../../../libraries/stn/src/lib.hb");
|
2024-11-17 10:57:06 -06:00
|
|
|
.{bit0, bit1, bit5, bit6, bit7} := @use("bits.hb");
|
2024-11-17 14:30:58 -06:00
|
|
|
.{Port, PORT_AT_STARTUP} := @use("port.hb")
|
2024-11-16 15:56:00 -06:00
|
|
|
|
2024-11-17 15:38:07 -06:00
|
|
|
port1 := @as(Port, PORT_AT_STARTUP)
|
|
|
|
port2 := @as(Port, PORT_AT_STARTUP)
|
2024-11-17 12:11:13 -06:00
|
|
|
|
2024-12-01 06:10:41 -06:00
|
|
|
//wiki.osdev.org/"8042"_PS/2_Controller#PS/2_Controller_IO_Ports
|
|
|
|
$CONTROLLER_PORT := 0x64
|
|
|
|
$DATA_PORT := 0x60n
|
|
|
|
|
|
|
|
$disable_port1 := fn(): void memory.outb(CONTROLLER_PORT, 0xAD)
|
|
|
|
$enable_port1 := fn(): void memory.outb(CONTROLLER_PORT, 0xAE)
|
|
|
|
$disable_port2 := fn(): void memory.outb(CONTROLLER_PORT, 0xA7)
|
|
|
|
$enable_port2 := fn(): void memory.outb(CONTROLLER_PORT, 0xA8)
|
2024-11-16 15:56:00 -06:00
|
|
|
|
2024-11-17 04:17:32 -06:00
|
|
|
test_port1 := fn(): bool {
|
2024-12-01 06:10:41 -06:00
|
|
|
memory.outb(CONTROLLER_PORT, 0xAB)
|
2024-11-17 04:17:32 -06:00
|
|
|
loop if has_input(get_info()) break
|
|
|
|
input := get_input()
|
|
|
|
return input == 0x0
|
|
|
|
}
|
2024-11-16 15:56:00 -06:00
|
|
|
|
2024-11-17 04:17:32 -06:00
|
|
|
test_port2 := fn(): bool {
|
2024-12-01 06:10:41 -06:00
|
|
|
memory.outb(CONTROLLER_PORT, 0xA9)
|
2024-11-17 04:17:32 -06:00
|
|
|
loop if has_input(get_info()) break
|
|
|
|
input := get_input()
|
|
|
|
return input == 0x0
|
|
|
|
}
|
2024-11-16 15:56:00 -06:00
|
|
|
|
|
|
|
get_config_byte := fn(): u8 {
|
2024-12-01 06:10:41 -06:00
|
|
|
memory.outb(CONTROLLER_PORT, 0x20)
|
2024-11-16 15:56:00 -06:00
|
|
|
loop if has_input(get_info()) break
|
2024-11-17 04:17:32 -06:00
|
|
|
return get_input()
|
2024-11-16 15:56:00 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
Info := struct {d: u8}
|
|
|
|
|
2024-12-01 06:10:41 -06:00
|
|
|
$get_info := fn(): Info return .(memory.inb(CONTROLLER_PORT))
|
2024-11-17 10:57:06 -06:00
|
|
|
//inline when can
|
|
|
|
has_input := fn(info: Info): bool return bit0(info.d)
|
|
|
|
can_send := fn(info: Info): bool return bit1(info.d) == false
|
|
|
|
timed_out := fn(info: Info): bool return bit6(info.d)
|
|
|
|
check_parity := fn(info: Info): bool return bit7(info.d)
|
2024-11-17 04:17:32 -06:00
|
|
|
get_port := fn(info: Info): ^Port {
|
|
|
|
if bit5(info.d) {
|
|
|
|
return &port2
|
|
|
|
} else {
|
|
|
|
return &port1
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-12-01 06:10:41 -06:00
|
|
|
//T
|
|
|
|
port2_ptr := &port2
|
2024-11-17 10:57:06 -06:00
|
|
|
send_byte := fn(port: ^Port, byte: u8): void {
|
2024-12-01 06:10:41 -06:00
|
|
|
if port == port2_ptr {
|
|
|
|
memory.outb(CONTROLLER_PORT, 0xD4)
|
2024-11-17 10:57:06 -06:00
|
|
|
}
|
|
|
|
loop if can_send(get_info()) break
|
2024-12-01 06:10:41 -06:00
|
|
|
memory.outb(DATA_PORT, byte)
|
2024-11-17 10:57:06 -06:00
|
|
|
}
|
|
|
|
|
2024-12-01 06:10:41 -06:00
|
|
|
$get_input := fn(): u8 return memory.inb(DATA_PORT)
|
|
|
|
$write_out := fn(data: u8): void memory.outb(DATA_PORT, data)
|
2024-11-16 15:56:00 -06:00
|
|
|
|
|
|
|
flush_input := fn(): void {
|
2024-11-17 04:17:32 -06:00
|
|
|
loop if has_input(get_info()) == false break else get_info()
|
2024-11-16 15:56:00 -06:00
|
|
|
}
|
2024-11-16 14:51:55 -06:00
|
|
|
|
|
|
|
init := fn(): void {
|
2024-11-16 15:56:00 -06:00
|
|
|
disable_port1()
|
|
|
|
disable_port2()
|
2024-11-16 14:51:55 -06:00
|
|
|
//Disables ports to make sure that they won't interfere with the setup process.
|
|
|
|
|
2024-11-16 15:56:00 -06:00
|
|
|
flush_input()
|
2024-11-16 14:51:55 -06:00
|
|
|
|
2024-11-16 15:56:00 -06:00
|
|
|
enable_port2()
|
2024-11-17 04:17:32 -06:00
|
|
|
port2.exists = bit5(@inline(get_config_byte)) == false
|
2024-11-16 15:56:00 -06:00
|
|
|
disable_port2()
|
2024-11-16 14:51:55 -06:00
|
|
|
|
2024-11-16 15:56:00 -06:00
|
|
|
flush_input()
|
2024-11-16 14:51:55 -06:00
|
|
|
|
2024-11-17 04:17:32 -06:00
|
|
|
port1.exists = test_port1()
|
2024-11-16 14:51:55 -06:00
|
|
|
|
2024-11-17 04:17:32 -06:00
|
|
|
if port2.exists {
|
|
|
|
port2.exists = test_port2()
|
2024-11-16 14:51:55 -06:00
|
|
|
}
|
|
|
|
|
2024-11-17 04:17:32 -06:00
|
|
|
if (port1.exists | port2.exists) == false {
|
2024-11-16 14:51:55 -06:00
|
|
|
log.error("No ports detected! No input will be processed! Cannot handle this!\0")
|
|
|
|
}
|
|
|
|
|
2024-11-17 04:17:32 -06:00
|
|
|
if port1.exists {
|
2024-11-16 14:51:55 -06:00
|
|
|
log.info("Port 1 exists.\0")
|
2024-11-16 15:56:00 -06:00
|
|
|
enable_port1()
|
2024-11-16 14:51:55 -06:00
|
|
|
}
|
2024-11-17 04:17:32 -06:00
|
|
|
if port2.exists {
|
2024-11-16 14:51:55 -06:00
|
|
|
log.info("Port 2 exists.\0")
|
2024-11-16 15:56:00 -06:00
|
|
|
enable_port2()
|
2024-11-16 14:51:55 -06:00
|
|
|
}
|
|
|
|
}
|