Source-Changes-HG archive

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index][Old Index]

[src/trunk]: src/sys/arch/arm/sunxi sun6i_spi: bring over non-intrusive impro...



details:   https://anonhg.NetBSD.org/src/rev/b394823703c7
branches:  trunk
changeset: 1002516:b394823703c7
user:      tnn <tnn%NetBSD.org@localhost>
date:      Sat Aug 03 14:42:24 2019 +0000

description:
sun6i_spi: bring over non-intrusive improvements from sun4i_spi

- simplify attach error handling
- calculate minfreq & maxfreq instead of hardcoding
- print configured value of SPI_SCLK
- add convenience macro for reading/writing SPI register
- clean up commented out debug printfs

diffstat:

 sys/arch/arm/sunxi/sun6i_spi.c |  175 ++++++++++++++++------------------------
 1 files changed, 70 insertions(+), 105 deletions(-)

diffs (truncated from 360 to 300 lines):

diff -r faf8387b4d0f -r b394823703c7 sys/arch/arm/sunxi/sun6i_spi.c
--- a/sys/arch/arm/sunxi/sun6i_spi.c    Sat Aug 03 14:00:42 2019 +0000
+++ b/sys/arch/arm/sunxi/sun6i_spi.c    Sat Aug 03 14:42:24 2019 +0000
@@ -1,6 +1,7 @@
-/*     $NetBSD: sun6i_spi.c,v 1.2 2018/02/01 14:50:36 jakllsch Exp $   */
+/*     $NetBSD: sun6i_spi.c,v 1.3 2019/08/03 14:42:24 tnn Exp $        */
 
 /*
+ * Copyright (c) 2019 Tobias Nygren
  * Copyright (c) 2018 Jonathan A. Kollasch
  * All rights reserved.
  *
@@ -27,12 +28,11 @@
  */
 
 #include <sys/cdefs.h>
-__KERNEL_RCSID(0, "$NetBSD: sun6i_spi.c,v 1.2 2018/02/01 14:50:36 jakllsch Exp $");
+__KERNEL_RCSID(0, "$NetBSD: sun6i_spi.c,v 1.3 2019/08/03 14:42:24 tnn Exp $");
 
 #include <sys/param.h>
 #include <sys/device.h>
 #include <sys/systm.h>
-#include <sys/mutex.h>
 #include <sys/bus.h>
 #include <sys/intr.h>
 #include <sys/kernel.h>
@@ -64,6 +64,11 @@
        volatile bool           sc_running;
 };
 
+#define SPIREG_READ(sc, reg) \
+    bus_space_read_4((sc)->sc_iot, (sc)->sc_ioh, (reg))
+#define SPIREG_WRITE(sc, reg, val) \
+    bus_space_write_4((sc)->sc_iot, (sc)->sc_ioh, (reg), (val))
+
 static int sun6ispi_match(device_t, cfdata_t, void *);
 static void sun6ispi_attach(device_t, device_t, void *);
 
@@ -97,52 +102,37 @@
        struct sun6ispi_softc * const sc = device_private(self);
        struct fdt_attach_args * const faa = aux;
        struct spibus_attach_args sba;
+       const int phandle = faa->faa_phandle;
+       bus_addr_t addr;
+       bus_size_t size;
        struct fdtbus_reset *rst;
        struct clk *clk, *modclk;
        uint32_t gcr, isr;
        char intrstr[128];
 
-       aprint_naive("\n");
-       aprint_normal(": SPI\n");
-
        sc->sc_dev = self;
        sc->sc_iot = faa->faa_bst;
        SIMPLEQ_INIT(&sc->sc_q);
 
-       const int phandle = faa->faa_phandle;
-       bus_addr_t addr;
-       bus_size_t size;
-
-       if (fdtbus_get_reg(phandle, 0, &addr, &size) != 0) {
-               aprint_error_dev(sc->sc_dev, "missing 'reg' property\n");
-               return;
-       }
-
-       if (bus_space_map(sc->sc_iot, addr, size, 0, &sc->sc_ioh) != 0) {
-               aprint_error_dev(sc->sc_dev, "unable to map device\n");
+       if ((clk = fdtbus_clock_get_index(phandle, 0)) == NULL
+           || clk_enable(clk) != 0) {
+               aprint_error(": couldn't enable clock\n");
                return;
        }
 
-       if ((clk = fdtbus_clock_get_index(phandle, 0)) != NULL) {
-               if (clk_enable(clk) != 0) {
-                       aprint_error_dev(sc->sc_dev, "couldn't enable clock\n");
-                       return;
-               }
-               device_printf(self, "%s ahb @ %uHz\n", __func__, clk_get_rate(clk));
+       /* 200MHz max on H3,H5 */
+       if ((modclk = fdtbus_clock_get(phandle, "mod")) == NULL
+           || clk_set_rate(modclk, 200000000) != 0
+           || clk_enable(modclk) != 0) {
+               aprint_error(": couldn't enable module clock\n");
+               return;
        }
+       sc->sc_modclkrate = clk_get_rate(modclk);
 
-       if ((modclk = fdtbus_clock_get(phandle, "mod")) != NULL) {
-               /* 200MHz max on H3,H5 */
-               if (clk_set_rate(modclk, 200000000) != 0) {
-                       aprint_error_dev(sc->sc_dev, "couldn't configure module clock\n");
-                       return;
-               }
-               if (clk_enable(modclk) != 0) {
-                       aprint_error_dev(sc->sc_dev, "couldn't enable module clock\n");
-                       return;
-               }
-               sc->sc_modclkrate = clk_get_rate(modclk);
-               device_printf(self, "%s %uHz\n", __func__, sc->sc_modclkrate);
+       if (fdtbus_get_reg(phandle, 0, &addr, &size) != 0
+           || bus_space_map(sc->sc_iot, addr, size, 0, &sc->sc_ioh) != 0) {
+               aprint_error(": couldn't map registers\n");
+               return;
        }
 
        if ((rst = fdtbus_reset_get_index(phandle, 0)) != NULL)
@@ -151,8 +141,8 @@
                        return;
                }
 
-       isr = bus_space_read_4(sc->sc_iot, sc->sc_ioh, SPI_INT_STA);
-       bus_space_write_4(sc->sc_iot, sc->sc_ioh, SPI_INT_STA, isr);
+       isr = SPIREG_READ(sc, SPI_INT_STA);
+       SPIREG_WRITE(sc, SPI_INT_STA, isr);
 
        if (!fdtbus_intr_str(phandle, 0, intrstr, sizeof(intrstr))) {
                aprint_error(": failed to decode interrupt\n");
@@ -162,13 +152,17 @@
        sc->sc_intrh = fdtbus_intr_establish(phandle, 0, IPL_VM, 0,
            sun6ispi_intr, sc);
        if (sc->sc_intrh == NULL) {
-               aprint_error_dev(sc->sc_dev, "unable to establish interrupt\n");
+               aprint_error(": unable to establish interrupt\n");
                return;
        }
+
+       aprint_naive("\n");
+       aprint_normal(": SPI\n");
+       
        aprint_normal_dev(self, "interrupting on %s\n", intrstr);
 
        gcr = SPI_GCR_SRST;
-       bus_space_write_4(sc->sc_iot, sc->sc_ioh, SPI_GCR, gcr);
+       SPIREG_WRITE(sc, SPI_GCR, gcr);
        for (u_int i = 0; ; i++) {
                if (i >= 1000000) {
                        aprint_error_dev(self, "reset timeout\n");
@@ -181,9 +175,9 @@
                        DELAY(1);
        }
        gcr = SPI_GCR_TP_EN | SPI_GCR_MODE | SPI_GCR_EN;
-       bus_space_write_4(sc->sc_iot, sc->sc_ioh, SPI_GCR, gcr);
+       SPIREG_WRITE(sc, SPI_GCR, gcr);
 
-       bus_space_write_4(sc->sc_iot, sc->sc_ioh, SPI_IER, SPI_IER_DEFAULT);
+       SPIREG_WRITE(sc, SPI_IER, SPI_IER_DEFAULT);
 
        sc->sc_spi.sct_cookie = sc;
        sc->sc_spi.sct_configure = sun6ispi_configure;
@@ -200,21 +194,18 @@
 {
        struct sun6ispi_softc * const sc = cookie;
        uint32_t tcr, cctl;
-
-#if 0
-       device_printf(sc->sc_dev, "%s slave %d mode %d speed %d\n", __func__, slave, mode, speed);
+       uint32_t minfreq, maxfreq;
 
-       if (speed > 200000)
-               speed = 200000;
-#endif
+       minfreq = sc->sc_modclkrate >> 16;
+       maxfreq = sc->sc_modclkrate >> 1;
 
-       tcr = SPI_TCR_SS_LEVEL | SPI_TCR_SPOL;
-
-       if (slave > 3)
+       if (speed <= 0 || speed < minfreq || speed > maxfreq)
                return EINVAL;
 
-       if (speed <= 0)
+       if (slave >= sc->sc_spi.sct_nslaves)
                return EINVAL;
+       
+       tcr = SPI_TCR_SS_LEVEL | SPI_TCR_SPOL;
 
        switch (mode) {
        case SPI_MODE_0:
@@ -235,10 +226,7 @@
 
        sc->sc_TCR = tcr;
 
-       if (speed < 3000 || speed > 200000000)
-               return EINVAL;
-
-       if (speed > sc->sc_modclkrate / 2 || speed < sc->sc_modclkrate / 512) {
+       if (speed < sc->sc_modclkrate / 512) {
                for (cctl = 0; cctl <= __SHIFTOUT_MASK(SPI_CCTL_CDR1); cctl++) {
                        if ((sc->sc_modclkrate / (1<<cctl)) <= speed)
                                goto cdr1_found;
@@ -251,8 +239,14 @@
                cctl = SPI_CCTL_DRS|__SHIFTIN(cctl, SPI_CCTL_CDR2);
        }
 
-       device_printf(sc->sc_dev, "%s CCTL 0x%08x\n", __func__, cctl);
-       bus_space_write_4(sc->sc_iot, sc->sc_ioh, SPI_CCTL, cctl);
+       device_printf(sc->sc_dev, "tcr 0x%x, cctl 0x%x, CLK %uHz, SCLK %uHz\n",
+           tcr, cctl, sc->sc_modclkrate,
+           (cctl & SPI_CCTL_DRS)
+           ? (sc->sc_modclkrate / (u_int)(2 * (__SHIFTOUT(cctl, SPI_CCTL_CDR2) + 1)))
+           : (sc->sc_modclkrate >> (__SHIFTOUT(cctl, SPI_CCTL_CDR1) + 1))
+       );
+
+       SPIREG_WRITE(sc, SPI_CCTL, cctl);
 
        return 0;
 }
@@ -272,27 +266,14 @@
        return 0;
 }
 
-static size_t
-chunks_total_count(struct spi_transfer * st)
-{
-       struct spi_chunk *chunk = st->st_chunks;
-       size_t len = 0;
-
-       do {
-               len += chunk->chunk_count;
-       } while ((chunk = chunk->chunk_next) != NULL);
-
-       return len;
-}
-
 static void
 sun6ispi_start(struct sun6ispi_softc * const sc)
 {
        struct spi_transfer *st;
        uint32_t isr, tcr;
-       size_t ctc;
+       struct spi_chunk *chunk;
+       size_t burstcount;
 
-       //device_printf(sc->sc_dev, "%s\n", __func__);
        while ((st = spi_transq_first(&sc->sc_q)) != NULL) {
 
                spi_transq_dequeue(&sc->sc_q);
@@ -302,37 +283,31 @@
                sc->sc_rchunk = sc->sc_wchunk = st->st_chunks;
                sc->sc_running = true;
 
-               isr = bus_space_read_4(sc->sc_iot, sc->sc_ioh, SPI_INT_STA);
-               bus_space_write_4(sc->sc_iot, sc->sc_ioh, SPI_INT_STA, isr);
-
-               //printf("%s fcr 0x%08x\n", __func__, bus_space_read_4(sc->sc_iot, sc->sc_ioh, SPI_FCR));
+               isr = SPIREG_READ(sc, SPI_INT_STA);
+               SPIREG_WRITE(sc, SPI_INT_STA, isr);
 
-               ctc = chunks_total_count(st);
-               //printf("%s total count %zu\n", __func__, ctc);
-
-               bus_space_write_4(sc->sc_iot, sc->sc_ioh, SPI_BC, __SHIFTIN(ctc, SPI_BC_MBC));
-               bus_space_write_4(sc->sc_iot, sc->sc_ioh, SPI_TC, __SHIFTIN(ctc, SPI_TC_MWTC));
-               bus_space_write_4(sc->sc_iot, sc->sc_ioh, SPI_BCC, __SHIFTIN(ctc, SPI_BCC_STC));
+               burstcount = 0;
+               for (chunk = st->st_chunks; chunk; chunk = chunk->chunk_next) {
+                       burstcount += chunk->chunk_count;
+               }
+               KASSERT(burstcount <= SPI_BC_MBC);
+               SPIREG_WRITE(sc, SPI_BC, __SHIFTIN(burstcount, SPI_BC_MBC));
+               SPIREG_WRITE(sc, SPI_TC, __SHIFTIN(burstcount, SPI_TC_MWTC));
+               SPIREG_WRITE(sc, SPI_BCC, __SHIFTIN(burstcount, SPI_BCC_STC));
 
                KASSERT(st->st_slave <= 3);
                tcr = sc->sc_TCR | __SHIFTIN(st->st_slave, SPI_TCR_SS_SEL);
 
-               //device_printf(sc->sc_dev, "%s before TCR write\n", __func__);
-               //bus_space_write_4(sc->sc_iot, sc->sc_ioh, SPI_TCR, tcr);
-
                sun6ispi_send(sc);
 
                const uint32_t ier = SPI_IER_DEFAULT | SPI_IER_RF_RDY_INT_EN | SPI_IER_TX_ERQ_INT_EN;
-               bus_space_write_4(sc->sc_iot, sc->sc_ioh, SPI_IER, ier);
+               SPIREG_WRITE(sc, SPI_IER, ier);
 
-               bus_space_write_4(sc->sc_iot, sc->sc_ioh, SPI_TCR, tcr|SPI_TCR_XCH);
-               //device_printf(sc->sc_dev, "%s after TCR write\n", __func__);
+               SPIREG_WRITE(sc, SPI_TCR, tcr|SPI_TCR_XCH);
 
                if (!cold)
                        return;
 
-               //device_printf(sc->sc_dev, "%s cold\n", __func__);
-
                int s = splbio();
                for (;;) {
                        sun6ispi_intr(sc);
@@ -343,7 +318,6 @@
        }
 
        sc->sc_running = false;
-//     device_printf(sc->sc_dev, "%s finishes\n", __func__);
 }
 
 static void
@@ -353,13 +327,10 @@
        uint32_t fsr;
        struct spi_chunk *chunk;



Home | Main Index | Thread Index | Old Index