/* $NetBSD: bcm2835_bsc.c,v 1.15 2020/03/31 12:23:17 jmcneill Exp $ */ /* * Copyright (c) 2019 Jason R. Thorpe * Copyright (c) 2012 Jonathan A. Kollasch * 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 COPYRIGHT HOLDERS AND CONTRIBUTORS * "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 COPYRIGHT HOLDER OR * CONTRIBUTORS 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. */ #include __KERNEL_RCSID(0, "$NetBSD: bcm2835_bsc.c,v 1.15 2020/03/31 12:23:17 jmcneill Exp $"); #include #include #include #include #include #include #include #include #include #include #include static void bsciic_exec_func_idle(struct bsciic_softc * const); static void bsciic_exec_func_send_addr(struct bsciic_softc * const); static void bsciic_exec_func_send_cmd(struct bsciic_softc * const); static void bsciic_exec_func_send_data(struct bsciic_softc * const); static void bsciic_exec_func_recv_data(struct bsciic_softc * const); static void bsciic_exec_func_done(struct bsciic_softc * const); static void bsciic_exec_func_error(struct bsciic_softc * const); const struct { void (*func)(struct bsciic_softc * const); uint32_t c_bits; uint32_t s_bits; } bsciic_exec_state_data[] = { [BSC_EXEC_STATE_IDLE] = { .func = bsciic_exec_func_idle, }, [BSC_EXEC_STATE_SEND_ADDR] = { .func = bsciic_exec_func_send_addr, }, [BSC_EXEC_STATE_SEND_CMD] = { .func = bsciic_exec_func_send_cmd, .c_bits = BSC_C_INTT, .s_bits = BSC_S_TXW, }, [BSC_EXEC_STATE_SEND_DATA] = { .func = bsciic_exec_func_send_data, .c_bits = BSC_C_INTT, .s_bits = BSC_S_TXW, }, [BSC_EXEC_STATE_RECV_DATA] = { .func = bsciic_exec_func_recv_data, .c_bits = BSC_C_READ | BSC_C_INTR, .s_bits = BSC_S_RXR, }, [BSC_EXEC_STATE_DONE] = { .func = bsciic_exec_func_done, }, [BSC_EXEC_STATE_ERROR] = { .func = bsciic_exec_func_error, }, }; int bsciic_debug = 0; void bsciic_attach(struct bsciic_softc *sc) { mutex_init(&sc->sc_intr_lock, MUTEX_DEFAULT, IPL_VM); cv_init(&sc->sc_intr_wait, device_xname(sc->sc_dev)); /* clear FIFO, disable controller */ bus_space_write_4(sc->sc_iot, sc->sc_ioh, BSC_C, BSC_C_CLEAR_CLEAR); bus_space_write_4(sc->sc_iot, sc->sc_ioh, BSC_S, BSC_S_CLKT | BSC_S_ERR | BSC_S_DONE); bus_space_write_4(sc->sc_iot, sc->sc_ioh, BSC_DLEN, 0); u_int divider = howmany(sc->sc_frequency, sc->sc_clkrate); bus_space_write_4(sc->sc_iot, sc->sc_ioh, BSC_DIV, __SHIFTIN(divider, BSC_DIV_CDIV)); } int bsciic_acquire_bus(void *v, int flags) { struct bsciic_softc * const sc = v; uint32_t s __diagused; bus_space_write_4(sc->sc_iot, sc->sc_ioh, BSC_S, BSC_S_CLKT | BSC_S_ERR | BSC_S_DONE); bus_space_write_4(sc->sc_iot, sc->sc_ioh, BSC_C, BSC_C_I2CEN | BSC_C_CLEAR_CLEAR); s = bus_space_read_4(sc->sc_iot, sc->sc_ioh, BSC_S); KASSERT((s & BSC_S_TA) == 0); return 0; } void bsciic_release_bus(void *v, int flags) { struct bsciic_softc * const sc = v; bus_space_write_4(sc->sc_iot, sc->sc_ioh, BSC_C, BSC_C_CLEAR_CLEAR); } static void bsciic_exec_lock(struct bsciic_softc * const sc) { if ((sc->sc_exec.flags & I2C_F_POLL) == 0) { mutex_enter(&sc->sc_intr_lock); } } static void bsciic_exec_unlock(struct bsciic_softc * const sc) { if ((sc->sc_exec.flags & I2C_F_POLL) == 0) { mutex_exit(&sc->sc_intr_lock); } } static void bsciic_txfill(struct bsciic_softc * const sc) { uint32_t s; while (sc->sc_bufpos != sc->sc_buflen) { s = bus_space_read_4(sc->sc_iot, sc->sc_ioh, BSC_S); if ((s & BSC_S_TXD) == 0) break; bus_space_write_4(sc->sc_iot, sc->sc_ioh, BSC_FIFO, sc->sc_buf[sc->sc_bufpos++]); } } static void bsciic_rxdrain(struct bsciic_softc * const sc) { uint32_t s; while (sc->sc_bufpos != sc->sc_buflen) { s = bus_space_read_4(sc->sc_iot, sc->sc_ioh, BSC_S); if ((s & BSC_S_RXD) == 0) break; sc->sc_buf[sc->sc_bufpos++] = (uint8_t)bus_space_read_4(sc->sc_iot, sc->sc_ioh, BSC_FIFO); } } static bsc_exec_state_t bsciic_next_state(struct bsciic_softc * const sc) { switch (sc->sc_exec_state) { case BSC_EXEC_STATE_IDLE: if (sc->sc_exec.addr > 0x7f) { return BSC_EXEC_STATE_SEND_ADDR; } /* FALLTHROUGH */ case BSC_EXEC_STATE_SEND_ADDR: if (sc->sc_exec.cmdlen) { return BSC_EXEC_STATE_SEND_CMD; } /* FALLTHROUGH */ case BSC_EXEC_STATE_SEND_CMD: if (sc->sc_exec.datalen == 0) { return BSC_EXEC_STATE_DONE; } if (I2C_OP_READ_P(sc->sc_exec.op)) { return BSC_EXEC_STATE_RECV_DATA; } return BSC_EXEC_STATE_SEND_DATA; case BSC_EXEC_STATE_SEND_DATA: case BSC_EXEC_STATE_RECV_DATA: return BSC_EXEC_STATE_DONE; case BSC_EXEC_STATE_DONE: case BSC_EXEC_STATE_ERROR: return sc->sc_exec_state; } panic("bsciic_next_state: invalid state: %d", sc->sc_exec_state); } #define BSC_EXEC_PHASE_COMPLETE(sc) \ ((sc)->sc_exec_state == BSC_EXEC_STATE_ERROR || \ (sc)->sc_bufpos == (sc)->sc_buflen) static void bsciic_signal(struct bsciic_softc * const sc) { if ((sc->sc_exec.flags & I2C_F_POLL) == 0) { cv_signal(&sc->sc_intr_wait); } } static void bsciic_phase_done(struct bsciic_softc * const sc) { sc->sc_exec_state = bsciic_next_state(sc); (*bsciic_exec_state_data[sc->sc_exec_state].func)(sc); } static void bsciic_abort(struct bsciic_softc * const sc) { sc->sc_exec_state = BSC_EXEC_STATE_ERROR; bsciic_phase_done(sc); } int bsciic_intr(void *v) { struct bsciic_softc * const sc = v; uint32_t s; bsciic_exec_lock(sc); if ((sc->sc_exec.flags & I2C_F_POLL) == 0 && sc->sc_expecting_interrupt == false) { bsciic_exec_unlock(sc); return 0; } s = bus_space_read_4(sc->sc_iot, sc->sc_ioh, BSC_S); bus_space_write_4(sc->sc_iot, sc->sc_ioh, BSC_S, BSC_S_CLKT | BSC_S_ERR | BSC_S_DONE); if (s & (BSC_S_CLKT | BSC_S_ERR)) { /* * ERR might be a normal "probing for device" sort * of thing, so don't complain about that one. * Do complain about CLKT, though. */ if ((s & BSC_S_CLKT) || (bsciic_debug && (s & BSC_S_ERR))) { device_printf(sc->sc_dev, "error s=0x%08x, aborting transfer\n", s); } bsciic_abort(sc); goto out; } if (BSC_EXEC_STATE_SENDING(sc)) { /* * When transmitting, we need to wait for one final * interrupt after pushing out the last of our data. * Catch that case here and go to the next state. */ if (BSC_EXEC_PHASE_COMPLETE(sc)) { bsciic_phase_done(sc); } else { bsciic_txfill(sc); } } else if (BSC_EXEC_STATE_RECEIVING(sc)) { bsciic_rxdrain(sc); /* * If we've received all of the data, go to the next * state now; we might not get another interrupt. */ if (BSC_EXEC_PHASE_COMPLETE(sc)) { bsciic_phase_done(sc); } } else { device_printf(sc->sc_dev, "unexpected interrupt: state=%d s=0x%08x\n", sc->sc_exec_state, s); bsciic_abort(sc); } out: bsciic_exec_unlock(sc); return (1); } static void bsciic_wait(struct bsciic_softc * const sc, const uint32_t events) { if ((sc->sc_exec.flags & I2C_F_POLL) == 0) { return; } const uint32_t s_bits = events | BSC_S_CLKT | BSC_S_ERR | BSC_S_DONE; uint32_t s; /* sc_intr_lock is not held in this case. */ for (;;) { s = bus_space_read_4(sc->sc_iot, sc->sc_ioh, BSC_S); if (s & s_bits) { (void) bsciic_intr(sc); } if (BSC_EXEC_PHASE_COMPLETE(sc)) { bsciic_phase_done(sc); } if (sc->sc_exec_state >= BSC_EXEC_STATE_DONE) { return; } delay(1); } } static void bsciic_start(struct bsciic_softc * const sc) { sc->sc_c_bits = BSC_C_I2CEN | BSC_C_INTD | bsciic_exec_state_data[sc->sc_exec_state].c_bits; /* Clear the interrupt-enable bits if we're polling. */ if (sc->sc_exec.flags & I2C_F_POLL) { sc->sc_c_bits &= ~(BSC_C_INTD | BSC_C_INTT | BSC_C_INTR); } sc->sc_expecting_interrupt = (sc->sc_c_bits & (BSC_C_INTD | BSC_C_INTT | BSC_C_INTR)) ? true : false; bus_space_write_4(sc->sc_iot, sc->sc_ioh, BSC_C, sc->sc_c_bits | BSC_C_ST); bsciic_wait(sc, bsciic_exec_state_data[sc->sc_exec_state].s_bits); } static void bsciic_exec_func_idle(struct bsciic_softc * const sc) { /* We kick off a transfer by setting the slave address register. */ bus_space_write_4(sc->sc_iot, sc->sc_ioh, BSC_A, sc->sc_exec.addr); /* Immediately transition to the next state. */ bsciic_phase_done(sc); } static void bsciic_exec_func_send_addr(struct bsciic_softc * const sc) { /* XXX For 10-bit addressing; not implemented yet. */ panic("bsciic_exec_func_send_addr is not supposed to be called"); } static void bsciic_exec_func_send_cmd(struct bsciic_softc * const sc) { sc->sc_buf = __UNCONST(sc->sc_exec.cmdbuf); sc->sc_bufpos = 0; sc->sc_buflen = sc->sc_exec.cmdlen; uint32_t dlen = sc->sc_exec.cmdlen; if (! I2C_OP_READ_P(sc->sc_exec.op)) { dlen += sc->sc_exec.datalen; } bus_space_write_4(sc->sc_iot, sc->sc_ioh, BSC_DLEN, dlen); bsciic_start(sc); } static void bsciic_exec_func_send_data(struct bsciic_softc * const sc) { sc->sc_buf = sc->sc_exec.databuf; sc->sc_bufpos = 0; sc->sc_buflen = sc->sc_exec.datalen; if (sc->sc_exec.cmdlen) { /* * Output has already been started in this case; we just * needed to switch buffers. */ bsciic_wait(sc, BSC_S_TXW); } else { uint32_t dlen = sc->sc_exec.datalen; bus_space_write_4(sc->sc_iot, sc->sc_ioh, BSC_DLEN, dlen); bsciic_start(sc); } } static void bsciic_exec_func_recv_data(struct bsciic_softc * const sc) { sc->sc_buf = sc->sc_exec.databuf; sc->sc_bufpos = 0; sc->sc_buflen = sc->sc_exec.datalen; uint32_t dlen = sc->sc_exec.datalen; bus_space_write_4(sc->sc_iot, sc->sc_ioh, BSC_DLEN, dlen); bsciic_start(sc); } static void bsciic_exec_func_done(struct bsciic_softc * const sc) { /* We're done! Disable interrupts. */ bus_space_write_4(sc->sc_iot, sc->sc_ioh, BSC_C, BSC_C_I2CEN); sc->sc_expecting_interrupt = false; bsciic_signal(sc); } static void bsciic_exec_func_error(struct bsciic_softc * const sc) { /* Clear the FIFO and disable interrupts. */ bus_space_write_4(sc->sc_iot, sc->sc_ioh, BSC_C, BSC_C_I2CEN | BSC_C_CLEAR_CLEAR); sc->sc_expecting_interrupt = false; bsciic_signal(sc); } int bsciic_exec(void *v, i2c_op_t op, i2c_addr_t addr, const void *cmdbuf, size_t cmdlen, void *databuf, size_t datalen, int flags) { struct bsciic_softc * const sc = v; /* XXX We don't do 10-bit addressing correctly yet. */ if (addr > 0x7f) return (ENOTSUP); /* * The I2C middle layer has ensured that the client device has * exclusive access to the controller. Copy the parameters * and start the state machine that runs through the necessary * phases of the request. */ KASSERT(sc->sc_exec_state == BSC_EXEC_STATE_IDLE); sc->sc_exec.op = op; sc->sc_exec.addr = addr; sc->sc_exec.cmdbuf = cmdbuf; sc->sc_exec.cmdlen = cmdlen; sc->sc_exec.databuf = databuf; sc->sc_exec.datalen = datalen; sc->sc_exec.flags = flags; bsciic_exec_lock(sc); (*bsciic_exec_state_data[sc->sc_exec_state].func)(sc); while (sc->sc_exec_state < BSC_EXEC_STATE_DONE) { KASSERT((flags & I2C_F_POLL) == 0); cv_wait(&sc->sc_intr_wait, &sc->sc_intr_lock); } int error = sc->sc_exec_state == BSC_EXEC_STATE_ERROR ? EIO : 0; uint32_t s; do { s = bus_space_read_4(sc->sc_iot, sc->sc_ioh, BSC_S); } while ((s & BSC_S_TA) != 0); if (s & (BSC_S_CLKT | BSC_S_ERR)) { error = EIO; } bus_space_write_4(sc->sc_iot, sc->sc_ioh, BSC_C, BSC_C_I2CEN | BSC_C_CLEAR_CLEAR); bus_space_write_4(sc->sc_iot, sc->sc_ioh, BSC_S, BSC_S_CLKT | BSC_S_ERR | BSC_S_DONE); bus_space_write_4(sc->sc_iot, sc->sc_ioh, BSC_DLEN, 0); bsciic_exec_unlock(sc); sc->sc_exec.flags = 0; sc->sc_exec_state = BSC_EXEC_STATE_IDLE; memset(&sc->sc_exec, 0, sizeof(sc->sc_exec)); return error; }