xref: /netbsd-src/sys/arch/arm/sunxi/sun6i_spi.c (revision c7fb772b85b2b5d4cfb282f868f454b4701534fd)
1*c7fb772bSthorpej /*	$NetBSD: sun6i_spi.c,v 1.10 2021/08/07 16:18:45 thorpej Exp $	*/
2ad63a6a0Sjakllsch 
3ad63a6a0Sjakllsch /*
44a1c6b68Stnn  * Copyright (c) 2019 Tobias Nygren
5ad63a6a0Sjakllsch  * Copyright (c) 2018 Jonathan A. Kollasch
6ad63a6a0Sjakllsch  * All rights reserved.
7ad63a6a0Sjakllsch  *
8ad63a6a0Sjakllsch  * Redistribution and use in source and binary forms, with or without
9ad63a6a0Sjakllsch  * modification, are permitted provided that the following conditions
10ad63a6a0Sjakllsch  * are met:
11ad63a6a0Sjakllsch  * 1. Redistributions of source code must retain the above copyright
12ad63a6a0Sjakllsch  *    notice, this list of conditions and the following disclaimer.
13ad63a6a0Sjakllsch  * 2. Redistributions in binary form must reproduce the above copyright
14ad63a6a0Sjakllsch  *    notice, this list of conditions and the following disclaimer in the
15ad63a6a0Sjakllsch  *    documentation and/or other materials provided with the distribution.
16ad63a6a0Sjakllsch  *
17ad63a6a0Sjakllsch  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
18ad63a6a0Sjakllsch  * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
19ad63a6a0Sjakllsch  * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
20ad63a6a0Sjakllsch  * PURPOSE ARE DISCLAIMED.  IN NO EVENT SHALL THE COPYRIGHT HOLDER OR
21ad63a6a0Sjakllsch  * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
22ad63a6a0Sjakllsch  * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
23ad63a6a0Sjakllsch  * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS;
24ad63a6a0Sjakllsch  * OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
25ad63a6a0Sjakllsch  * WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR
26ad63a6a0Sjakllsch  * OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF
27ad63a6a0Sjakllsch  * ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
28ad63a6a0Sjakllsch  */
29ad63a6a0Sjakllsch 
30ad63a6a0Sjakllsch #include <sys/cdefs.h>
31*c7fb772bSthorpej __KERNEL_RCSID(0, "$NetBSD: sun6i_spi.c,v 1.10 2021/08/07 16:18:45 thorpej Exp $");
32ad63a6a0Sjakllsch 
33ad63a6a0Sjakllsch #include <sys/param.h>
34ad63a6a0Sjakllsch #include <sys/device.h>
35ad63a6a0Sjakllsch #include <sys/systm.h>
36ad63a6a0Sjakllsch #include <sys/bus.h>
37ad63a6a0Sjakllsch #include <sys/intr.h>
38ad63a6a0Sjakllsch #include <sys/kernel.h>
39ad63a6a0Sjakllsch 
40ad63a6a0Sjakllsch #include <sys/bitops.h>
41ad63a6a0Sjakllsch #include <dev/spi/spivar.h>
42ad63a6a0Sjakllsch 
43ad63a6a0Sjakllsch #include <arm/sunxi/sun6i_spireg.h>
44ad63a6a0Sjakllsch 
45ad63a6a0Sjakllsch #include <dev/fdt/fdtvar.h>
46ad63a6a0Sjakllsch 
47ad63a6a0Sjakllsch #include <arm/fdt/arm_fdtvar.h>
48ad63a6a0Sjakllsch 
49ad63a6a0Sjakllsch #define SPI_IER_DEFAULT (SPI_IER_TC_INT_EN | SPI_IER_TF_UDR_INT_EN | \
50ad63a6a0Sjakllsch 	SPI_IER_TF_OVF_INT_EN | SPI_IER_RF_UDR_INT_EN | SPI_IER_RF_OVF_INT_EN)
51ad63a6a0Sjakllsch 
52ad63a6a0Sjakllsch struct sun6ispi_softc {
53ad63a6a0Sjakllsch 	device_t		sc_dev;
54ad63a6a0Sjakllsch 	bus_space_tag_t		sc_iot;
55ad63a6a0Sjakllsch 	bus_space_handle_t	sc_ioh;
56ad63a6a0Sjakllsch 	void			*sc_intrh;
57ad63a6a0Sjakllsch 	struct spi_controller	sc_spi;
58ad63a6a0Sjakllsch 	SIMPLEQ_HEAD(,spi_transfer) sc_q;
59ad63a6a0Sjakllsch 	struct spi_transfer	*sc_transfer;
60ad63a6a0Sjakllsch 	struct spi_chunk	*sc_wchunk;
61ad63a6a0Sjakllsch 	struct spi_chunk	*sc_rchunk;
62ad63a6a0Sjakllsch 	uint32_t		sc_TCR;
63ad63a6a0Sjakllsch 	u_int			sc_modclkrate;
64ad63a6a0Sjakllsch 	volatile bool		sc_running;
65ad63a6a0Sjakllsch };
66ad63a6a0Sjakllsch 
674a1c6b68Stnn #define SPIREG_READ(sc, reg) \
684a1c6b68Stnn     bus_space_read_4((sc)->sc_iot, (sc)->sc_ioh, (reg))
694a1c6b68Stnn #define SPIREG_WRITE(sc, reg, val) \
704a1c6b68Stnn     bus_space_write_4((sc)->sc_iot, (sc)->sc_ioh, (reg), (val))
714a1c6b68Stnn 
72ad63a6a0Sjakllsch static int sun6ispi_match(device_t, cfdata_t, void *);
73ad63a6a0Sjakllsch static void sun6ispi_attach(device_t, device_t, void *);
74ad63a6a0Sjakllsch 
75ad63a6a0Sjakllsch static int sun6ispi_configure(void *, int, int, int);
76ad63a6a0Sjakllsch static int sun6ispi_transfer(void *, struct spi_transfer *);
77ad63a6a0Sjakllsch 
78ad63a6a0Sjakllsch static void sun6ispi_start(struct sun6ispi_softc * const);
79ad63a6a0Sjakllsch static int sun6ispi_intr(void *);
80ad63a6a0Sjakllsch 
81ad63a6a0Sjakllsch static void sun6ispi_send(struct sun6ispi_softc * const);
82ad63a6a0Sjakllsch static void sun6ispi_recv(struct sun6ispi_softc * const);
83ad63a6a0Sjakllsch 
84ad63a6a0Sjakllsch CFATTACH_DECL_NEW(sun6i_spi, sizeof(struct sun6ispi_softc),
85ad63a6a0Sjakllsch     sun6ispi_match, sun6ispi_attach, NULL, NULL);
86ad63a6a0Sjakllsch 
876e54367aSthorpej static const struct device_compatible_entry compat_data[] = {
886e54367aSthorpej 	{ .compat = "allwinner,sun8i-h3-spi" },
896e54367aSthorpej 	DEVICE_COMPAT_EOL
906e54367aSthorpej };
916e54367aSthorpej 
92ad63a6a0Sjakllsch static int
sun6ispi_match(device_t parent,cfdata_t cf,void * aux)93ad63a6a0Sjakllsch sun6ispi_match(device_t parent, cfdata_t cf, void *aux)
94ad63a6a0Sjakllsch {
95ad63a6a0Sjakllsch 	struct fdt_attach_args * const faa = aux;
96ad63a6a0Sjakllsch 
976e54367aSthorpej 	return of_compatible_match(faa->faa_phandle, compat_data);
98ad63a6a0Sjakllsch }
99ad63a6a0Sjakllsch 
100ad63a6a0Sjakllsch static void
sun6ispi_attach(device_t parent,device_t self,void * aux)101ad63a6a0Sjakllsch sun6ispi_attach(device_t parent, device_t self, void *aux)
102ad63a6a0Sjakllsch {
103ad63a6a0Sjakllsch 	struct sun6ispi_softc * const sc = device_private(self);
104ad63a6a0Sjakllsch 	struct fdt_attach_args * const faa = aux;
105ad63a6a0Sjakllsch 	struct spibus_attach_args sba;
1064a1c6b68Stnn 	const int phandle = faa->faa_phandle;
1074a1c6b68Stnn 	bus_addr_t addr;
1084a1c6b68Stnn 	bus_size_t size;
109ad63a6a0Sjakllsch 	struct fdtbus_reset *rst;
110ad63a6a0Sjakllsch 	struct clk *clk, *modclk;
111ad63a6a0Sjakllsch 	uint32_t gcr, isr;
112ad63a6a0Sjakllsch 	char intrstr[128];
113ad63a6a0Sjakllsch 
114ad63a6a0Sjakllsch 	sc->sc_dev = self;
115ad63a6a0Sjakllsch 	sc->sc_iot = faa->faa_bst;
116ad63a6a0Sjakllsch 	SIMPLEQ_INIT(&sc->sc_q);
117ad63a6a0Sjakllsch 
1184a1c6b68Stnn 	if ((clk = fdtbus_clock_get_index(phandle, 0)) == NULL
1194a1c6b68Stnn 	    || clk_enable(clk) != 0) {
1204a1c6b68Stnn 		aprint_error(": couldn't enable clock\n");
121ad63a6a0Sjakllsch 		return;
122ad63a6a0Sjakllsch 	}
123ad63a6a0Sjakllsch 
124ad63a6a0Sjakllsch 	/* 200MHz max on H3,H5 */
1254a1c6b68Stnn 	if ((modclk = fdtbus_clock_get(phandle, "mod")) == NULL
1264a1c6b68Stnn 	    || clk_set_rate(modclk, 200000000) != 0
1274a1c6b68Stnn 	    || clk_enable(modclk) != 0) {
1284a1c6b68Stnn 		aprint_error(": couldn't enable module clock\n");
129ad63a6a0Sjakllsch 		return;
130ad63a6a0Sjakllsch 	}
131ad63a6a0Sjakllsch 	sc->sc_modclkrate = clk_get_rate(modclk);
1324a1c6b68Stnn 
1334a1c6b68Stnn 	if (fdtbus_get_reg(phandle, 0, &addr, &size) != 0
1344a1c6b68Stnn 	    || bus_space_map(sc->sc_iot, addr, size, 0, &sc->sc_ioh) != 0) {
1354a1c6b68Stnn 		aprint_error(": couldn't map registers\n");
1364a1c6b68Stnn 		return;
137ad63a6a0Sjakllsch 	}
138ad63a6a0Sjakllsch 
139ad63a6a0Sjakllsch 	if ((rst = fdtbus_reset_get_index(phandle, 0)) != NULL)
140ad63a6a0Sjakllsch 		if (fdtbus_reset_deassert(rst) != 0) {
141ad63a6a0Sjakllsch 			aprint_error(": couldn't de-assert reset\n");
142ad63a6a0Sjakllsch 			return;
143ad63a6a0Sjakllsch 		}
144ad63a6a0Sjakllsch 
1454a1c6b68Stnn 	isr = SPIREG_READ(sc, SPI_INT_STA);
1464a1c6b68Stnn 	SPIREG_WRITE(sc, SPI_INT_STA, isr);
147ad63a6a0Sjakllsch 
148ad63a6a0Sjakllsch 	if (!fdtbus_intr_str(phandle, 0, intrstr, sizeof(intrstr))) {
149ad63a6a0Sjakllsch 		aprint_error(": failed to decode interrupt\n");
150ad63a6a0Sjakllsch 		return;
151ad63a6a0Sjakllsch 	}
152ad63a6a0Sjakllsch 
1538027ed92Sjmcneill 	sc->sc_intrh = fdtbus_intr_establish_xname(phandle, 0, IPL_VM, 0,
1548027ed92Sjmcneill 	    sun6ispi_intr, sc, device_xname(self));
155ad63a6a0Sjakllsch 	if (sc->sc_intrh == NULL) {
1564a1c6b68Stnn 		aprint_error(": unable to establish interrupt\n");
157ad63a6a0Sjakllsch 		return;
158ad63a6a0Sjakllsch 	}
1594a1c6b68Stnn 
1604a1c6b68Stnn 	aprint_naive("\n");
1614a1c6b68Stnn 	aprint_normal(": SPI\n");
1624a1c6b68Stnn 
163ad63a6a0Sjakllsch 	aprint_normal_dev(self, "interrupting on %s\n", intrstr);
164ad63a6a0Sjakllsch 
165ad63a6a0Sjakllsch 	gcr = SPI_GCR_SRST;
1664a1c6b68Stnn 	SPIREG_WRITE(sc, SPI_GCR, gcr);
167fccc36fcSjakllsch 	for (u_int i = 0; ; i++) {
168fccc36fcSjakllsch 		if (i >= 1000000) {
169fccc36fcSjakllsch 			aprint_error_dev(self, "reset timeout\n");
170fccc36fcSjakllsch 			return;
171fccc36fcSjakllsch 		}
172fccc36fcSjakllsch 		gcr = bus_space_read_4(sc->sc_iot, sc->sc_ioh, SPI_GCR);
173fccc36fcSjakllsch 		if ((gcr & SPI_GCR_SRST) == 0)
174fccc36fcSjakllsch 			break;
175fccc36fcSjakllsch 		else
176fccc36fcSjakllsch 			DELAY(1);
177fccc36fcSjakllsch 	}
178ad63a6a0Sjakllsch 	gcr = SPI_GCR_TP_EN | SPI_GCR_MODE | SPI_GCR_EN;
1794a1c6b68Stnn 	SPIREG_WRITE(sc, SPI_GCR, gcr);
180ad63a6a0Sjakllsch 
1814a1c6b68Stnn 	SPIREG_WRITE(sc, SPI_IER, SPI_IER_DEFAULT);
182ad63a6a0Sjakllsch 
183ad63a6a0Sjakllsch 	sc->sc_spi.sct_cookie = sc;
184ad63a6a0Sjakllsch 	sc->sc_spi.sct_configure = sun6ispi_configure;
185ad63a6a0Sjakllsch 	sc->sc_spi.sct_transfer = sun6ispi_transfer;
186ad63a6a0Sjakllsch 	sc->sc_spi.sct_nslaves = 4;
187ad63a6a0Sjakllsch 
1889555f417Stnn 	memset(&sba, 0, sizeof(sba));
189ad63a6a0Sjakllsch 	sba.sba_controller = &sc->sc_spi;
190ad63a6a0Sjakllsch 
191*c7fb772bSthorpej 	(void) config_found(self, &sba, spibus_print, CFARGS_NONE);
192ad63a6a0Sjakllsch }
193ad63a6a0Sjakllsch 
194ad63a6a0Sjakllsch static int
sun6ispi_configure(void * cookie,int slave,int mode,int speed)195ad63a6a0Sjakllsch sun6ispi_configure(void *cookie, int slave, int mode, int speed)
196ad63a6a0Sjakllsch {
197ad63a6a0Sjakllsch 	struct sun6ispi_softc * const sc = cookie;
198ad63a6a0Sjakllsch 	uint32_t tcr, cctl;
1994a1c6b68Stnn 	uint32_t minfreq, maxfreq;
200ad63a6a0Sjakllsch 
2014a1c6b68Stnn 	minfreq = sc->sc_modclkrate >> 16;
2024a1c6b68Stnn 	maxfreq = sc->sc_modclkrate >> 1;
203ad63a6a0Sjakllsch 
2044a1c6b68Stnn 	if (speed <= 0 || speed < minfreq || speed > maxfreq)
2054a1c6b68Stnn 		return EINVAL;
2064a1c6b68Stnn 
2074a1c6b68Stnn 	if (slave >= sc->sc_spi.sct_nslaves)
2084a1c6b68Stnn 		return EINVAL;
209ad63a6a0Sjakllsch 
210ad63a6a0Sjakllsch 	tcr = SPI_TCR_SS_LEVEL | SPI_TCR_SPOL;
211ad63a6a0Sjakllsch 
212ad63a6a0Sjakllsch 	switch (mode) {
213ad63a6a0Sjakllsch 	case SPI_MODE_0:
214ad63a6a0Sjakllsch 		tcr |= 0;
215ad63a6a0Sjakllsch 		break;
216ad63a6a0Sjakllsch 	case SPI_MODE_1:
217ad63a6a0Sjakllsch 		tcr |= SPI_TCR_CPHA;
218ad63a6a0Sjakllsch 		break;
219ad63a6a0Sjakllsch 	case SPI_MODE_2:
220ad63a6a0Sjakllsch 		tcr |= SPI_TCR_CPOL;
221ad63a6a0Sjakllsch 		break;
222ad63a6a0Sjakllsch 	case SPI_MODE_3:
223ad63a6a0Sjakllsch 		tcr |= SPI_TCR_CPHA|SPI_TCR_CPOL;
224ad63a6a0Sjakllsch 		break;
225ad63a6a0Sjakllsch 	default:
226ad63a6a0Sjakllsch 		return EINVAL;
227ad63a6a0Sjakllsch 	}
228ad63a6a0Sjakllsch 
229ad63a6a0Sjakllsch 	sc->sc_TCR = tcr;
230ad63a6a0Sjakllsch 
2314a1c6b68Stnn 	if (speed < sc->sc_modclkrate / 512) {
232ad63a6a0Sjakllsch 		for (cctl = 0; cctl <= __SHIFTOUT_MASK(SPI_CCTL_CDR1); cctl++) {
233ad63a6a0Sjakllsch 			if ((sc->sc_modclkrate / (1<<cctl)) <= speed)
234ad63a6a0Sjakllsch 				goto cdr1_found;
235ad63a6a0Sjakllsch 		}
236ad63a6a0Sjakllsch 		return EINVAL;
237ad63a6a0Sjakllsch cdr1_found:
238ad63a6a0Sjakllsch 		cctl = __SHIFTIN(cctl, SPI_CCTL_CDR1);
239ad63a6a0Sjakllsch 	} else {
240ad63a6a0Sjakllsch 		cctl = howmany(sc->sc_modclkrate, 2 * speed) - 1;
241ad63a6a0Sjakllsch 		cctl = SPI_CCTL_DRS|__SHIFTIN(cctl, SPI_CCTL_CDR2);
242ad63a6a0Sjakllsch 	}
243ad63a6a0Sjakllsch 
2444a1c6b68Stnn 	device_printf(sc->sc_dev, "tcr 0x%x, cctl 0x%x, CLK %uHz, SCLK %uHz\n",
2454a1c6b68Stnn 	    tcr, cctl, sc->sc_modclkrate,
2464a1c6b68Stnn 	    (cctl & SPI_CCTL_DRS)
2474a1c6b68Stnn 	    ? (sc->sc_modclkrate / (u_int)(2 * (__SHIFTOUT(cctl, SPI_CCTL_CDR2) + 1)))
2484a1c6b68Stnn 	    : (sc->sc_modclkrate >> (__SHIFTOUT(cctl, SPI_CCTL_CDR1) + 1))
2494a1c6b68Stnn 	);
2504a1c6b68Stnn 
2514a1c6b68Stnn 	SPIREG_WRITE(sc, SPI_CCTL, cctl);
252ad63a6a0Sjakllsch 
253ad63a6a0Sjakllsch 	return 0;
254ad63a6a0Sjakllsch }
255ad63a6a0Sjakllsch 
256ad63a6a0Sjakllsch static int
sun6ispi_transfer(void * cookie,struct spi_transfer * st)257ad63a6a0Sjakllsch sun6ispi_transfer(void *cookie, struct spi_transfer *st)
258ad63a6a0Sjakllsch {
259ad63a6a0Sjakllsch 	struct sun6ispi_softc * const sc = cookie;
260ad63a6a0Sjakllsch 	int s;
261ad63a6a0Sjakllsch 
262ad63a6a0Sjakllsch 	s = splbio();
263ad63a6a0Sjakllsch 	spi_transq_enqueue(&sc->sc_q, st);
264ad63a6a0Sjakllsch 	if (sc->sc_running == false) {
265ad63a6a0Sjakllsch 		sun6ispi_start(sc);
266ad63a6a0Sjakllsch 	}
267ad63a6a0Sjakllsch 	splx(s);
268ad63a6a0Sjakllsch 	return 0;
269ad63a6a0Sjakllsch }
270ad63a6a0Sjakllsch 
271ad63a6a0Sjakllsch static void
sun6ispi_start(struct sun6ispi_softc * const sc)272ad63a6a0Sjakllsch sun6ispi_start(struct sun6ispi_softc * const sc)
273ad63a6a0Sjakllsch {
274ad63a6a0Sjakllsch 	struct spi_transfer *st;
275ad63a6a0Sjakllsch 	uint32_t isr, tcr;
2764a1c6b68Stnn 	struct spi_chunk *chunk;
2774a1c6b68Stnn 	size_t burstcount;
278ad63a6a0Sjakllsch 
279ad63a6a0Sjakllsch 	while ((st = spi_transq_first(&sc->sc_q)) != NULL) {
280ad63a6a0Sjakllsch 
281ad63a6a0Sjakllsch 		spi_transq_dequeue(&sc->sc_q);
282ad63a6a0Sjakllsch 
283ad63a6a0Sjakllsch 		KASSERT(sc->sc_transfer == NULL);
284ad63a6a0Sjakllsch 		sc->sc_transfer = st;
285ad63a6a0Sjakllsch 		sc->sc_rchunk = sc->sc_wchunk = st->st_chunks;
286ad63a6a0Sjakllsch 		sc->sc_running = true;
287ad63a6a0Sjakllsch 
2884a1c6b68Stnn 		isr = SPIREG_READ(sc, SPI_INT_STA);
2894a1c6b68Stnn 		SPIREG_WRITE(sc, SPI_INT_STA, isr);
290ad63a6a0Sjakllsch 
2914a1c6b68Stnn 		burstcount = 0;
2924a1c6b68Stnn 		for (chunk = st->st_chunks; chunk; chunk = chunk->chunk_next) {
2934a1c6b68Stnn 			burstcount += chunk->chunk_count;
2944a1c6b68Stnn 		}
2954a1c6b68Stnn 		KASSERT(burstcount <= SPI_BC_MBC);
2964a1c6b68Stnn 		SPIREG_WRITE(sc, SPI_BC, __SHIFTIN(burstcount, SPI_BC_MBC));
2974a1c6b68Stnn 		SPIREG_WRITE(sc, SPI_TC, __SHIFTIN(burstcount, SPI_TC_MWTC));
2984a1c6b68Stnn 		SPIREG_WRITE(sc, SPI_BCC, __SHIFTIN(burstcount, SPI_BCC_STC));
299ad63a6a0Sjakllsch 
300ad63a6a0Sjakllsch 		KASSERT(st->st_slave <= 3);
301ad63a6a0Sjakllsch 		tcr = sc->sc_TCR | __SHIFTIN(st->st_slave, SPI_TCR_SS_SEL);
302ad63a6a0Sjakllsch 
303ad63a6a0Sjakllsch 		sun6ispi_send(sc);
304ad63a6a0Sjakllsch 
305ad63a6a0Sjakllsch 		const uint32_t ier = SPI_IER_DEFAULT | SPI_IER_RF_RDY_INT_EN | SPI_IER_TX_ERQ_INT_EN;
3064a1c6b68Stnn 		SPIREG_WRITE(sc, SPI_IER, ier);
307ad63a6a0Sjakllsch 
3084a1c6b68Stnn 		SPIREG_WRITE(sc, SPI_TCR, tcr|SPI_TCR_XCH);
309ad63a6a0Sjakllsch 
310ad63a6a0Sjakllsch 		if (!cold)
311ad63a6a0Sjakllsch 			return;
312ad63a6a0Sjakllsch 
313ad63a6a0Sjakllsch 		for (;;) {
314ad63a6a0Sjakllsch 			sun6ispi_intr(sc);
315ad63a6a0Sjakllsch 			if (ISSET(st->st_flags, SPI_F_DONE))
316ad63a6a0Sjakllsch 				break;
317ad63a6a0Sjakllsch 		}
318ad63a6a0Sjakllsch 	}
319ad63a6a0Sjakllsch 
320ad63a6a0Sjakllsch 	sc->sc_running = false;
321ad63a6a0Sjakllsch }
322ad63a6a0Sjakllsch 
323ad63a6a0Sjakllsch static void
sun6ispi_send(struct sun6ispi_softc * const sc)324ad63a6a0Sjakllsch sun6ispi_send(struct sun6ispi_softc * const sc)
325ad63a6a0Sjakllsch {
326ad63a6a0Sjakllsch 	uint8_t fd;
327ad63a6a0Sjakllsch 	uint32_t fsr;
328ad63a6a0Sjakllsch 	struct spi_chunk *chunk;
329ad63a6a0Sjakllsch 
330ad63a6a0Sjakllsch 	while ((chunk = sc->sc_wchunk) != NULL) {
331ad63a6a0Sjakllsch 		while (chunk->chunk_wresid) {
3324a1c6b68Stnn 			fsr = SPIREG_READ(sc, SPI_FSR);
333ad63a6a0Sjakllsch 			if (__SHIFTOUT(fsr, SPI_FSR_TF_CNT) >= 64) {
334ad63a6a0Sjakllsch 				return;
335ad63a6a0Sjakllsch 			}
336ad63a6a0Sjakllsch 			if (chunk->chunk_wptr) {
337ad63a6a0Sjakllsch 				fd = *chunk->chunk_wptr++;
338ad63a6a0Sjakllsch 			} else {
339ad63a6a0Sjakllsch 				fd = '\0';
340ad63a6a0Sjakllsch 			}
341ad63a6a0Sjakllsch 			bus_space_write_1(sc->sc_iot, sc->sc_ioh, SPI_TXD, fd);
342ad63a6a0Sjakllsch 			chunk->chunk_wresid--;
343ad63a6a0Sjakllsch 		}
344ad63a6a0Sjakllsch 		sc->sc_wchunk = sc->sc_wchunk->chunk_next;
345ad63a6a0Sjakllsch 	}
346ad63a6a0Sjakllsch }
347ad63a6a0Sjakllsch 
348ad63a6a0Sjakllsch static void
sun6ispi_recv(struct sun6ispi_softc * const sc)349ad63a6a0Sjakllsch sun6ispi_recv(struct sun6ispi_softc * const sc)
350ad63a6a0Sjakllsch {
351ad63a6a0Sjakllsch 	uint8_t fd;
352ad63a6a0Sjakllsch 	uint32_t fsr;
353ad63a6a0Sjakllsch 	struct spi_chunk *chunk;
354ad63a6a0Sjakllsch 
355ad63a6a0Sjakllsch 	while ((chunk = sc->sc_rchunk) != NULL) {
356ad63a6a0Sjakllsch 		while (chunk->chunk_rresid) {
3574a1c6b68Stnn 			fsr = SPIREG_READ(sc, SPI_FSR);
358ad63a6a0Sjakllsch 			if (__SHIFTOUT(fsr, SPI_FSR_RF_CNT) == 0) {
359ad63a6a0Sjakllsch 				return;
360ad63a6a0Sjakllsch 			}
361ad63a6a0Sjakllsch 			fd = bus_space_read_1(sc->sc_iot, sc->sc_ioh, SPI_RXD);
362ad63a6a0Sjakllsch 			if (chunk->chunk_rptr) {
363ad63a6a0Sjakllsch 				*chunk->chunk_rptr++ = fd;
364ad63a6a0Sjakllsch 			}
365ad63a6a0Sjakllsch 			chunk->chunk_rresid--;
366ad63a6a0Sjakllsch 		}
367ad63a6a0Sjakllsch 		sc->sc_rchunk = sc->sc_rchunk->chunk_next;
368ad63a6a0Sjakllsch 	}
369ad63a6a0Sjakllsch }
370ad63a6a0Sjakllsch 
371ad63a6a0Sjakllsch static int
sun6ispi_intr(void * cookie)372ad63a6a0Sjakllsch sun6ispi_intr(void *cookie)
373ad63a6a0Sjakllsch {
374ad63a6a0Sjakllsch 	struct sun6ispi_softc * const sc = cookie;
375ad63a6a0Sjakllsch 	struct spi_transfer *st;
376ad63a6a0Sjakllsch 	uint32_t isr;
377ad63a6a0Sjakllsch 
3784a1c6b68Stnn 	isr = SPIREG_READ(sc, SPI_INT_STA);
3794a1c6b68Stnn 	SPIREG_WRITE(sc, SPI_INT_STA, isr);
380ad63a6a0Sjakllsch 
381ad63a6a0Sjakllsch 	if (ISSET(isr, SPI_ISR_RX_RDY)) {
382ad63a6a0Sjakllsch 		sun6ispi_recv(sc);
383ad63a6a0Sjakllsch 		sun6ispi_send(sc);
384ad63a6a0Sjakllsch 	}
385ad63a6a0Sjakllsch 
386ad63a6a0Sjakllsch 	if (ISSET(isr, SPI_ISR_TC)) {
3874a1c6b68Stnn 		SPIREG_WRITE(sc, SPI_IER, SPI_IER_DEFAULT);
388ad63a6a0Sjakllsch 
389ad63a6a0Sjakllsch 		sc->sc_rchunk = sc->sc_wchunk = NULL;
390ad63a6a0Sjakllsch 		st = sc->sc_transfer;
391ad63a6a0Sjakllsch 		sc->sc_transfer = NULL;
392ad63a6a0Sjakllsch 		KASSERT(st != NULL);
393ad63a6a0Sjakllsch 		spi_done(st, 0);
394ad63a6a0Sjakllsch 		sc->sc_running = false;
395ad63a6a0Sjakllsch 	}
396ad63a6a0Sjakllsch 
397ad63a6a0Sjakllsch 	return isr;
398ad63a6a0Sjakllsch }
399