#include <stdbool.h>
#include <stdlib.h>
#include <stdio.h>
#include <console.h>
#include <sys/io.h>
#include <sys/cpu.h>
#include <syslinux/config.h>

#include "serial.h"

enum {
    THR = 0,
    RBR = 0,
    DLL = 0,
    DLM = 1,
    IER = 1,
    IIR = 2,
    FCR = 2,
    LCR = 3,
    MCR = 4,
    LSR = 5,
    MSR = 6,
    SCR = 7,
};


int serial_init(struct serial_if *sif, const char *argv[])
{
    const struct syslinux_serial_console_info *sci
	= syslinux_serial_console_info();
    uint16_t port;
    unsigned int divisor;
    uint8_t dll, dlm, lcr;

    if (!argv[0]) {
	if (sci->iobase) {
	    port = sci->iobase;
	} else {
	    printf("No port number specified and not using serial console!\n");
	    return -1;
	}
    } else {
	port = strtoul(argv[0], NULL, 0);
	if (port <= 3) {
	    uint16_t addr = ((uint16_t *)0x400)[port];
	    if (!addr) {
		printf("No serial port address found!\n");
	    return -1;
	    }
	    printf("Serial port %u is at 0x%04x\n", port, addr);
	    port = addr;
	}
    }

    sif->port = port;
    sif->console = false;

    divisor = 1;		/* Default speed = 115200 bps */

    /* Check to see if this is the same as the serial console */
    if (port == sci->iobase) {
	/* Overlaying the console... */
	sif->console = true;

	/* Default to already configured speed */
	divisor = sci->divisor;

	/* Shut down I/O to the console for the time being */
	openconsole(&dev_null_r, &dev_null_w);
    }

    if (argv[0] && argv[1])
	divisor = 115200/strtoul(argv[1], NULL, 0);

    cli();			/* Just in case... */

    /* Save old register settings */
    sif->old.lcr = inb(port + LCR);
    sif->old.mcr = inb(port + MCR);
    sif->old.iir = inb(port + IIR);

    /* Set speed */
    outb(0x83, port + LCR);	/* Enable divisor access */
    sif->old.dll = inb(port + DLL);
    sif->old.dlm = inb(port + DLM);
    outb(divisor, port + DLL);
    outb(divisor >> 8, port + DLM);
    (void)inb(port + IER);	/* Synchronize */

    dll = inb(port + DLL);
    dlm = inb(port + DLM);
    lcr = inb(port + LCR);
    outb(0x03, port + LCR);	/* Enable data access, n81 */
    (void)inb(port + IER);	/* Synchronize */
    sif->old.ier = inb(port + IER);

    /* Disable interrupts */
    outb(0, port + IER);

    sti();

    if (dll != (uint8_t)divisor ||
	dlm != (uint8_t)(divisor >> 8) ||
	lcr != 0x83) {
	serial_cleanup(sif);
	printf("No serial port detected!\n");
	return -1;		/* This doesn't look like a serial port */
    }

    /* Enable 16550A FIFOs if available */
    outb(0x01, port + FCR);	/* Enable FIFO */
    (void)inb(port + IER);	/* Synchronize */
    if (inb(port + IIR) < 0xc0)
	outb(0x00, port + FCR);	/* Disable FIFOs if non-functional */
    (void)inb(port + IER);	/* Synchronize */

    return 0;
}

void serial_write(struct serial_if *sif, const void *data, size_t n)
{
    uint16_t port = sif->port;
    const char *p = data;
    uint8_t lsr;

    while (n--) {
	do {
	    lsr = inb(port + LSR);
	} while (!(lsr & 0x20));

	outb(*p++, port + THR);
    }
}

void serial_read(struct serial_if *sif, void *data, size_t n)
{
    uint16_t port = sif->port;
    char *p = data;
    uint8_t lsr;

    while (n--) {
	do {
	    lsr = inb(port + LSR);
	} while (!(lsr & 0x01));

	*p++ = inb(port + RBR);
    }
}

void serial_cleanup(struct serial_if *sif)
{
    uint16_t port = sif->port;

    outb(0x83, port + LCR);
    (void)inb(port + IER);
    outb(sif->old.dll, port + DLL);
    outb(sif->old.dlm, port + DLM);
    (void)inb(port + IER);
    outb(sif->old.lcr & 0x7f, port + LCR);
    (void)inb(port + IER);
    outb(sif->old.mcr, port + MCR);
    outb(sif->old.ier, port + IER);
    if (sif->old.iir < 0xc0)
	outb(0x00, port + FCR);	/* Disable FIFOs */

    /* Re-enable console messages, if we shut them down */
    if (sif->console)
	openconsole(&dev_null_r, &dev_stdcon_w);
}