84 lines
1.7 KiB
C
84 lines
1.7 KiB
C
/* vim: tabstop=4 shiftwidth=4 noexpandtab
|
|
*
|
|
* Serial Port Driver
|
|
*/
|
|
#include <system.h>
|
|
#include <logging.h>
|
|
|
|
#define SERIAL_PORT_A 0x3F8
|
|
#define SERIAL_PORT_B 0x2F8
|
|
#define SERIAL_PORT_C 0x3E8
|
|
#define SERIAL_PORT_D 0x2E8
|
|
|
|
#define SERIAL_IRQ 4
|
|
|
|
void
|
|
serial_handler(
|
|
struct regs *r
|
|
) {
|
|
char serial = serial_recv();
|
|
irq_ack(SERIAL_IRQ);
|
|
/*
|
|
* Fix the serial input assuming it is ascii
|
|
*/
|
|
switch (serial) {
|
|
case 127:
|
|
serial = 0x08;
|
|
break;
|
|
case 13:
|
|
serial = '\n';
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
if (keyboard_buffer_handler) {
|
|
keyboard_buffer_handler(serial);
|
|
}
|
|
}
|
|
|
|
void
|
|
serial_install() {
|
|
blog("Installing serial communication driver...");
|
|
LOG(INFO, "Installing serial communication driver");
|
|
/* We will initialize the first serial port */
|
|
outportb(SERIAL_PORT_A + 1, 0x00);
|
|
outportb(SERIAL_PORT_A + 3, 0x80); /* Enable divisor mode */
|
|
outportb(SERIAL_PORT_A + 0, 0x03); /* Div Low: 03 Set the port to 38400 bps */
|
|
outportb(SERIAL_PORT_A + 1, 0x00); /* Div High: 00 */
|
|
outportb(SERIAL_PORT_A + 3, 0x03);
|
|
outportb(SERIAL_PORT_A + 2, 0xC7);
|
|
outportb(SERIAL_PORT_A + 4, 0x0B);
|
|
irq_install_handler(SERIAL_IRQ, serial_handler); /* Install the serial input handler */
|
|
outportb(SERIAL_PORT_A + 1, 0x01); /* Enable interrupts on receive */
|
|
bfinish(0);
|
|
}
|
|
|
|
int
|
|
serial_rcvd() {
|
|
return inportb(SERIAL_PORT_A + 5) & 1;
|
|
}
|
|
|
|
char
|
|
serial_recv() {
|
|
while (serial_rcvd() == 0);
|
|
return inportb(SERIAL_PORT_A);
|
|
}
|
|
|
|
int
|
|
serial_transmit_empty() {
|
|
return inportb(SERIAL_PORT_A + 5) & 0x20;
|
|
}
|
|
|
|
void
|
|
serial_send(char out) {
|
|
while (serial_transmit_empty() == 0);
|
|
outportb(SERIAL_PORT_A, out);
|
|
}
|
|
|
|
void
|
|
serial_string(char * out) {
|
|
for (uint32_t i = 0; i < strlen(out); ++i) {
|
|
serial_send(out[i]);
|
|
}
|
|
}
|