shithub: riscv

Download patch

ref: 9a6bf7f0787d050429da8b22a82c89ab5d869943
parent: 91f79ce9fab7c9920b6f331ca074d58c17755fc9
author: Sigrid Solveig Haflínudóttir <[email protected]>
date: Wed Aug 17 19:24:42 EDT 2022

imx8: add a semi-working SAI2 audio driver

--- a/sys/src/9/imx8/io.h
+++ b/sys/src/9/imx8/io.h
@@ -30,6 +30,8 @@
 
 	IRQpci2		= SPI+74,
 
+	IRQsai2		= SPI+96,
+
 	IRQenet1	= SPI+118,
 
 	IRQpci1		= SPI+122,
--- a/sys/src/9/imx8/reform
+++ b/sys/src/9/imx8/reform
@@ -22,6 +22,7 @@
 	i2c
 	rtc	devi2c
 	sd
+	audio
 
 link
 	usbxhciimx
@@ -31,6 +32,7 @@
 	netdevmedium
 	i2cimx		devi2c
 	pciimx		pci
+	sai
 
 ip
 	tcp
--- /dev/null
+++ b/sys/src/9/imx8/sai.c
@@ -1,0 +1,393 @@
+#include "u.h"
+#include "../port/lib.h"
+#include "mem.h"
+#include "dat.h"
+#include "fns.h"
+#include "io.h"
+#include "../port/audioif.h"
+#include "../port/error.h"
+
+typedef struct Ctlr Ctlr;
+typedef struct Ring Ring;
+
+enum {
+	Byteps = 4,
+
+	TCSR = 0x08/4,
+		TCSR_TE = 1<<31,
+		TCSR_FR = 1<<25,
+		TCSR_SR = 1<<24,
+		TCSR_FEF = 1<<18, /* w1c fifo error */
+		TCSR_FWF = 1<<17,
+		TCSR_FRF = 1<<16, /* watermark hit */
+		TCSR_FEIE = 1<<10,
+		TCSR_FWIE = 1<<9,
+		TCSR_FRIE = 1<<8,
+	TCR1 = 0x0c/4,
+	TCR2 = 0x10/4,
+		TCR2_MSEL_MCLK1 = 1<<26,
+		TCR2_BCP = 1<<25,
+		TCR2_BCD_MASTER = 1<<24,
+	TCR3 = 0x14/4,
+		TCR3_TCE_SHIFT = 16,
+	TCR4 = 0x18/4,
+		TCR4_FPACK_16BIT = 3<<24,
+		TCR4_FRSZ_SHIFT = 16,
+		TCR4_SYWD_SHIFT = 8,
+		TCR4_CHMOD = 1<<5,
+		TCR4_MSB_FIRST = 1<<4,
+		TCR4_FSE = 1<<3,
+		TCR4_FSP_LOW = 1<<1,
+		TCR4_FSD_MASTER = 1<<0,
+	TCR5 = 0x1c/4,
+		TCR5_WNW_SHIFT = 24,
+		TCR5_W0W_SHIFT = 16,
+		TCR5_FBT_SHIFT = 8,
+	TDR0 = 0x20/4,
+	TFR0 = 0x40/4,
+		TFRx_WFP_SHIFT = 16,
+		TFRx_RFP_SHIFT = 0,
+	TMR = 0x60/4,
+};
+
+struct Ring {
+	Rendez r;
+
+	uchar *buf;
+	ulong nbuf;
+
+	ulong ri;
+	ulong wi;
+};
+
+struct Ctlr {
+	u32int *reg;
+	Audio *adev;
+
+	Ring w;
+	int wactive;
+
+	Lock;
+};
+
+#define wr(a, v) ctlr->reg[a] = v
+#define rd(a) ctlr->reg[a]
+
+static long
+buffered(Ring *r)
+{
+	ulong ri, wi;
+
+	ri = r->ri;
+	wi = r->wi;
+	if(wi >= ri)
+		return wi - ri;
+	else
+		return r->nbuf - (ri - wi);
+}
+
+static long
+available(Ring *r)
+{
+	long m;
+
+	m = (r->nbuf - Byteps) - buffered(r);
+	if(m < 0)
+		m = 0;
+	return m;
+}
+
+static long
+writering(Ring *r, uchar *p, long n)
+{
+	long n0, m;
+
+	n0 = n;
+	while(n > 0){
+		if((m = available(r)) <= 0)
+			break;
+		if(m > n)
+			m = n;
+		if(p){
+			if(r->wi + m > r->nbuf)
+				m = r->nbuf - r->wi;
+			memmove(r->buf + r->wi, p, m);
+			p += m;
+		}
+		r->wi = (r->wi + m) % r->nbuf;
+		n -= m;
+	}
+	return n0 - n;
+}
+
+static int
+outavail(void *arg)
+{
+	Ring *r = arg;
+
+	return available(r) > 0;
+}
+
+static int
+outrate(void *arg)
+{
+	Ctlr *ctlr = arg;
+	int delay = ctlr->adev->delay*Byteps;
+
+	return delay <= 0 || buffered(&ctlr->w) <= delay || ctlr->wactive == 0;
+}
+
+static void
+saikick(Ctlr *ctlr)
+{
+	int delay;
+
+	delay = ctlr->adev->delay*Byteps;
+	if(buffered(&ctlr->w) >= delay){
+		ctlr->wactive = 1;
+		/* activate channel 1 */
+		wr(TCR3, 1<<TCR3_TCE_SHIFT);
+		wr(TCSR, TCSR_TE | TCSR_FEF | TCSR_FRIE | TCSR_FWIE | TCSR_FEIE);
+	}
+}
+
+static void
+saistop(Ctlr *ctlr)
+{
+	if(!ctlr->wactive)
+		return;
+	ctlr->wactive = 0;
+	wr(TCSR, TCSR_FR | TCSR_SR);
+}
+
+static long
+saiwrite(Audio *adev, void *a, long n, vlong)
+{
+	Ctlr *ctlr = adev->ctlr;
+	uchar *p, *e;
+	Ring *r;
+
+	p = a;
+	e = p + n;
+	r = &ctlr->w;
+	while(p < e){
+		if((n = writering(r, p, e - p)) <= 0){
+			saikick(ctlr);
+			sleep(&r->r, outavail, r);
+			continue;
+		}
+		p += n;
+	}
+	saikick(ctlr);
+	while(outrate(ctlr) == 0)
+		sleep(&r->r, outrate, ctlr);
+	return p - (uchar*)a;
+}
+
+static void
+saiclose(Audio *adev, int mode)
+{
+	Ctlr *ctlr = adev->ctlr;
+
+	if(mode == OWRITE || mode == ORDWR)
+		saistop(ctlr);
+}
+
+static void
+saireset(Ctlr *ctlr)
+{
+	/* fifo+software reset */
+	wr(TCSR, TCSR_FR | TCSR_SR);
+	delay(1);
+	wr(TCSR, 0);
+	delay(1);
+
+	/* watermark - hit early enough */
+	wr(TCR1, 32);
+	/* derive from mclk1; generate bitclock (active low), f = mclk1/(8+1)*2 = mclk1/18 */
+	wr(TCR2, TCR2_MSEL_MCLK1 | TCR2_BCD_MASTER | TCR2_BCP | 8);
+	/* activate channel 1 */
+	wr(TCR3, 1<<TCR3_TCE_SHIFT);
+	/* set up for i²s */
+	wr(TCR4,
+		TCR4_CHMOD | /* output mode, no TDM */
+		TCR4_MSB_FIRST |
+		TCR4_FPACK_16BIT | /* 16-bit packed words */
+		1<<TCR4_FRSZ_SHIFT | /* two words per frame */
+		15<<TCR4_SYWD_SHIFT | /* frame sync per word */
+		/* frame sync */
+		TCR4_FSE | /* one bit earlier */
+		TCR4_FSP_LOW | /* active high */
+		TCR4_FSD_MASTER /* generate internally */
+	);
+	/* 16-bit words, MSB first */
+	wr(TCR5, 15<<TCR5_WNW_SHIFT | 15<<TCR5_W0W_SHIFT | 15<<TCR5_FBT_SHIFT);
+	/* mask all but first two words */
+	wr(TMR, ~3UL);
+}
+
+static long
+saictl(Audio *adev, void *a, long n, vlong)
+{
+	char *p, *e, *x, *tok[4];
+	Ctlr *ctlr = adev->ctlr;
+	int ntok;
+	u32int v;
+
+	p = a;
+	e = p + n;
+	for(; p < e; p = x){
+		if(x = strchr(p, '\n'))
+			*x++ = 0;
+		else
+			x = e;
+		ntok = tokenize(p, tok, 4);
+		if(ntok <= 0)
+			continue;
+
+		if(cistrcmp(tok[0], "div") == 0 && ntok >= 2){
+			v = strtoul(tok[1], nil, 0);
+			wr(TCR2, (rd(TCR2) & ~0xff) | (v & 0xff));
+		}else if(cistrcmp(tok[0], "msel") == 0 && ntok >= 2){
+			v = strtoul(tok[1], nil, 0);
+			wr(TCR2, (rd(TCR2) & ~(3<<26)) | (v & 3)<<26);
+		}else if(cistrcmp(tok[0], "reset") == 0){
+			saireset(ctlr);
+		}else
+			error(Ebadctl);
+	}
+	return n;
+}
+
+static long
+fifo(Ctlr *ctlr, long n)
+{
+	long n0, m;
+	u32int *p;
+	Ring *r;
+
+	n0 = n;
+	r = &ctlr->w;
+	while(n > 0){
+		if((m = buffered(r)) <= 0 || m < 4)
+			break;
+		if(m > n)
+			m = n;
+
+		if(r->ri + m > r->nbuf)
+			m = r->nbuf - r->ri;
+		m &= ~(Byteps-1);
+		for(p = (u32int*)(r->buf + r->ri); p < (u32int*)(r->buf + r->ri + m); p++)
+			wr(TDR0, *p);
+
+		r->ri = (r->ri + m) % r->nbuf;
+		n -= m;
+	}
+	return n0 - n;
+}
+
+static void
+saiinterrupt(Ureg *, void *arg)
+{
+	Ctlr *ctlr = arg;
+	u32int v;
+	Ring *r;
+
+	ilock(ctlr);
+	v = rd(TCSR);
+	if(v & (TCSR_FEF | TCSR_FRF | TCSR_FWF)){
+		r = &ctlr->w;
+		if(ctlr->wactive){
+			if(buffered(r) < 128*Byteps) /* having less than fifo buffered */
+				saistop(ctlr);
+			else if(fifo(ctlr, (128-32)*Byteps) > 0)
+				v |= TCSR_TE;
+		}
+		wakeup(&r->r);
+	}
+	wr(TCSR, v | TCSR_FEF);
+	iunlock(ctlr);
+}
+
+static long
+saistatus(Audio *adev, void *a, long n, vlong)
+{
+	Ctlr *ctlr = adev->ctlr;
+	u32int v, p;
+	char *s, *e;
+
+	s = a;
+	e = s + n;
+	v = rd(TCSR);
+	p = rd(TFR0);
+	s = seprint(s, e, "transmit wfp %d rfp %d delay %d buf %ld avail %ld active %d %s%s%s%s\n",
+		(p>>TFRx_WFP_SHIFT) & 0xff,
+		(p>>TFRx_RFP_SHIFT) & 0xff,
+		adev->delay,
+		buffered(&ctlr->w),
+		available(&ctlr->w),
+		ctlr->wactive,
+		(v & TCSR_TE) ? " enabled" : "",
+		(v & TCSR_FEF) ? " fifo_error" : "",
+		(v & TCSR_FWF) ? " fifo_warn" : "",
+		(v & TCSR_FRF) ? " fifo_req" : ""
+	);
+
+	return s - (char*)a;
+}
+
+static long
+saibuffered(Audio *adev)
+{
+	Ctlr *ctlr = adev->ctlr;
+
+	return buffered(&ctlr->w);
+}
+
+static int
+saiprobe(Audio *adev)
+{
+	Ctlr *ctlr;
+
+	if(adev->ctlrno > 0)
+		return -1;
+
+	ctlr = mallocz(sizeof(Ctlr), 1);
+	if(ctlr == nil)
+		return -1;
+
+	ctlr->w.buf = malloc(ctlr->w.nbuf = 44100*Byteps*2);
+	ctlr->reg = (u32int*)(VIRTIO + 0x8b0000);
+	ctlr->adev = adev;
+
+	adev->delay = 1024;
+	adev->ctlr = ctlr;
+	adev->write = saiwrite;
+	adev->close = saiclose;
+	adev->buffered = saibuffered;
+	adev->status = saistatus;
+	adev->ctl = saictl;
+
+	saireset(ctlr);
+
+	intrenable(IRQsai2, saiinterrupt, ctlr, BUSUNKNOWN, "sai2");
+
+	return 0;
+}
+
+void
+sailink(void)
+{
+
+	iomuxpad("pad_sai2_rxfs", "sai2_rx_sync", "SION ~LVTTL HYS PUE ~ODE FAST 45_OHM VSEL_0");
+	iomuxpad("pad_sai2_rxc", "sai2_rx_bclk", "SION ~LVTTL HYS PUE ~ODE FAST 45_OHM VSEL_0");
+	iomuxpad("pad_sai2_rxd0", "sai2_rx_data0", "SION ~LVTTL HYS PUE ~ODE FAST 45_OHM VSEL_0");
+	iomuxpad("pad_sai2_txfs", "sai2_tx_sync", "SION ~LVTTL HYS PUE ~ODE FAST 45_OHM VSEL_0");
+	iomuxpad("pad_sai2_txc", "sai2_tx_bclk", "SION ~LVTTL HYS PUE ~ODE FAST 45_OHM VSEL_0");
+	iomuxpad("pad_sai2_txd0", "sai2_tx_data0", "SION ~LVTTL HYS PUE ~ODE FAST 45_OHM VSEL_0");
+	iomuxpad("pad_sai2_mclk", "sai2_mclk", "SION ~LVTTL HYS PUE ~ODE FAST 45_OHM VSEL_0");
+
+	setclkgate("sai2.ipg_clk", 1);
+
+	addaudiocard("sai", saiprobe);
+}