/* $NetBSD: stasc.c,v 1.2 2020/07/20 01:06:33 uwe Exp $ */
/*
 * Copyright (c) 2020 Valery Ushakov
 * All rights reserved.
 *
 * Redistribution and use in source and binary forms, with or without
 * modification, are permitted provided that the following conditions
 * are met:
 * 1. Redistributions of source code must retain the above copyright
 *    notice, this list of conditions and the following disclaimer.
 * 2. Redistributions in binary form must reproduce the above copyright
 *    notice, this list of conditions and the following disclaimer in the
 *    documentation and/or other materials provided with the distribution.
 *
 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
 * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
 * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
 * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
 * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
 * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
 * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
 * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
 * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
 */

/*
 * STMicroelectronics ST40 Asynchronous Serial Controller
 */
#include <sys/cdefs.h>
__KERNEL_RCSID(0, "$NetBSD: stasc.c,v 1.2 2020/07/20 01:06:33 uwe Exp $");

#include <sys/param.h>
#include <sys/systm.h>
#include <sys/conf.h>
#include <sys/device.h>
#include <sys/kernel.h>

#include <dev/cons.h>


#define STM_ASC_BASE	0xfd032000

#define ASC_BAUDRATE_OFFSET	0x00
#define ASC_TX_BUFF_OFFSET	0x04
#define ASC_RX_BUFF_OFFSET	0x08
#define ASC_CTRL_OFFSET		0x0C
#define ASC_INT_EN_OFFSET	0x10
#define ASC_INT_STA_OFFSET	0x14
#define ASC_GUARDTIME_OFFSET	0x18
#define ASC_TIMEOUT_OFFSET	0x1C
#define ASC_TX_RST_OFFSET	0x20
#define ASC_RX_RST_OFFSET	0x24
#define ASC_RETRIES_OFFSET	0x28

#define ASC_TX_BUFF	(*(volatile uint32_t *)(STM_ASC_BASE + ASC_TX_BUFF_OFFSET))
#define ASC_RX_BUFF	(*(volatile uint32_t *)(STM_ASC_BASE + ASC_RX_BUFF_OFFSET))
#define ASC_INT_EN	(*(volatile uint32_t *)(STM_ASC_BASE + ASC_INT_EN_OFFSET))
#define ASC_INT_STA	(*(volatile uint32_t *)(STM_ASC_BASE + ASC_INT_STA_OFFSET))


#define ASC_CTRL_NACK_DISABLE	0x2000
#define ASC_CTRL_BAUDMODE	0x1000
#define ASC_CTRL_CTS_EN		0x0800
#define ASC_CTRL_FIFO_EN	0x0400
#define ASC_CTRL_SC_EN		0x0200
#define ASC_CTRL_RX_EN		0x0100
#define ASC_CTRL_RUN		0x0080
#define ASC_CTRL_LOOPBACK	0x0040
#define ASC_CTRL_PARITYODD	0x0020
#define ASC_CTRL_STOPBITS_MASK	0x0018
#define     ASC_CTRL_STOPBITS_0_5	0x0000
#define     ASC_CTRL_STOPBITS_1_0	0x0008
#define     ASC_CTRL_STOPBITS_1_5	0x0010
#define     ASC_CTRL_STOPBITS_2_0	0x0018
#define ASC_CTRL_MODE_MASK	0x0007
#define     ASC_CTRL_MODE_8N		0x0001 /* 8 bit */
#define     ASC_CTRL_MODE_7P		0x0003 /* 7 bit + parity  */
#define     ASC_CTRL_MODE_9N		0x0004 /* 9 bit */
#define     ASC_CTRL_MODE_8W		0x0005 /* 8 bit + wakeup */
#define     ASC_CTRL_MODE_8P		0x0007 /* 8 bit + parity */


#define ASC_INT_EN_RHF		0x0100 /* ASC_INT_STA_RHF */
#define ASC_INT_EN_TOE		0x0080 /* ASC_INT_STA_TOE */
#define ASC_INT_EN_TNE		0x0040 /* ASC_INT_STA_TNE */
#define ASC_INT_EN_OE		0x0020 /* ASC_INT_STA_OE */
#define ASC_INT_EN_FE		0x0010 /* ASC_INT_STA_FE */
#define ASC_INT_EN_PE		0x0008 /* ASC_INT_STA_PE */
#define	ASC_INT_EN_THE		0x0004 /* ASC_INT_STA_THE */
#define ASC_INT_EN_TE		0x0002 /* ASC_INT_STA_TE */
#define ASC_INT_EN_RBF		0x0001 /* ASC_INT_STA_RBF */

#define ASC_INT_STA_NKD		0x0400 /* Tx: NACK Data */
#define ASC_INT_STA_TF		0x0200 /* Tx: Transmitter Full */
#define ASC_INT_STA_RHF		0x0100 /* Rx: Receiver FIFO Half Full */
#define ASC_INT_STA_TOE		0x0080 /* Rx: Timeout Or Empty */
#define ASC_INT_STA_TNE		0x0040 /* Rx: Timeout Or Not Empty */
#define ASC_INT_STA_OE		0x0020 /* Rx: Overrun Error */
#define ASC_INT_STA_FE		0x0010 /* Rx: Frame Error */
#define ASC_INT_STA_PE		0x0008 /* Rx: Parity Error */
#define	ASC_INT_STA_THE		0x0004 /* Tx: Transmitter FIFO Half Empty */
#define ASC_INT_STA_TE		0x0002 /* Tx: Transmitter Empty */
#define ASC_INT_STA_RBF		0x0001 /* Rx: Reciever Buffer Full */



struct stasc_softc {
	device_t sc_dev;
};


static int stasc_match(device_t, cfdata_t, void *);
static void stasc_attach(device_t, device_t, void *);

CFATTACH_DECL_NEW(stasc, sizeof(struct stasc_softc),
    stasc_match, stasc_attach, NULL, NULL);


/* console */
cons_decl(stasc_)

/* assign to cn_tab after cleaning bss to get printf early for the cpu setup */
struct consdev stasc_earlycons = cons_init(stasc_);

extern struct cfdriver stasc_cd;

const struct cdevsw stasc_cdevsw = {
	.d_open = noopen,
	.d_close = noclose,
	.d_read = noread,
	.d_write = nowrite,
	.d_ioctl = noioctl,
	.d_stop = nostop,
	.d_tty = notty,
	.d_poll = nopoll,
	.d_mmap = nommap,
	.d_kqfilter = nokqfilter,
	.d_discard = nodiscard,
	.d_flag = D_TTY
};



static int
stasc_match(device_t parent, cfdata_t cfp, void *aux)
{

	if (strcmp(cfp->cf_name, "stasc") != 0)
		return 0;

	return 0;		/* just stub it out for now */
}


static void
stasc_attach(device_t parent, device_t self, void *aux)
{
	struct stasc_softc *sc;

	sc = device_private(self);
	sc->sc_dev = self;

	aprint_normal("\n");
}


void
stasc_cnprobe(struct consdev *cp)
{

	cp->cn_pri = CN_NORMAL;
}


void
stasc_cninit(struct consdev *cp)
{

	return;
}


int
stasc_cngetc(dev_t dev)
{
	int s = splserial();
	uint32_t status;
	int c;

	/* don't block if Rx buffer is empty */
	status = ASC_INT_STA;
	if (!ISSET(status, ASC_INT_STA_RBF)) {
		splx(s);
		return -1;
	}

	/* can read the character now */
	c = ASC_RX_BUFF;
	splx(s);
	return (unsigned char)c;
}


void
stasc_cnputc(dev_t dev, int c)
{
	int s = splserial();
	uint32_t timo, status;

	/* wait for Tx Full to become clear */
	timo = 150000;
	do {
		status = ASC_INT_STA;
	} while (ISSET(status, ASC_INT_STA_TF) && --timo);

	/* can write the character now */
	ASC_TX_BUFF = c;
	splx(s);
}


void
stasc_cnpollc(dev_t dev, int on)
{

	return;
}