summaryrefslogtreecommitdiffstats
path: root/sys
diff options
context:
space:
mode:
authorloos <loos@FreeBSD.org>2014-10-27 12:18:07 +0000
committerloos <loos@FreeBSD.org>2014-10-27 12:18:07 +0000
commita3704ff4d46f150fc2e004a6f0f75c57047b967c (patch)
treec4bc4ec3a3c3e880885481cd0fa6c55e74565204 /sys
parent14fa4d63cd31d3abf0793390e0bbd0f0e5eb9ab0 (diff)
downloadFreeBSD-src-a3704ff4d46f150fc2e004a6f0f75c57047b967c.zip
FreeBSD-src-a3704ff4d46f150fc2e004a6f0f75c57047b967c.tar.gz
MFC: r266336, r270230 and r273263
r266336: Allow us to compile the Ti iic driver for both OMAP4 and AM335x. r270230: Rewrite of ti_i2c based on gonzo's patch, fix the following bugs/problems: . interrupt storm detected on "intr70:"; throttling interrupt source; . Added access serialization on iicbus_transfer(), previously there was no such protection and a new transfer could easily confuse the controller; . Add error checkings (i.e. stop the transfer when a error is detected and do _not_ overwrite the previous error); . On command done interrupt do not assume that the transfer was finished sucessfully as we will receive the command done interrupt even after errors; . Simplify the FIFO handling; . Reset the FIFO between the transfers as the FIFO may contain data from the last (failed) transfer; . Fix the iicbus speed for AM335x, which in turn will make better use of the I2C noise filter (set to one internal clock cycle); . Move the read and write handler to ithread instead of notifying the requesting thread with wakeup(9); . Fix the comments based on OMAP4 TRM. The above changes allows me to read the EDID from my HDMI monitor on BBB with gonzo's patches to support TDA19988 (which does 128 bytes reads) and repeatedly scan the iicbus (with a modified i2c(8)) without lock up the bus. r273263: Fix the chan address for mtx_sleep() on bus wait. Without this fix the threads waiting for the bus would never wake.
Diffstat (limited to 'sys')
-rw-r--r--sys/arm/ti/ti_i2c.c1254
-rw-r--r--sys/arm/ti/ti_i2c.h18
2 files changed, 540 insertions, 732 deletions
diff --git a/sys/arm/ti/ti_i2c.c b/sys/arm/ti/ti_i2c.c
index 19e2a89..e865f91 100644
--- a/sys/arm/ti/ti_i2c.c
+++ b/sys/arm/ti/ti_i2c.c
@@ -1,6 +1,6 @@
/*-
- * Copyright (c) 2011
- * Ben Gray <ben.r.gray@gmail.com>.
+ * Copyright (c) 2011 Ben Gray <ben.r.gray@gmail.com>.
+ * Copyright (c) 2014 Luiz Otavio O Souza <loos@freebsd.org>.
* All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
@@ -58,13 +58,14 @@ __FBSDID("$FreeBSD$");
#include <sys/module.h>
#include <sys/mutex.h>
#include <sys/rman.h>
+#include <sys/sysctl.h>
#include <machine/bus.h>
-#include <dev/fdt/fdt_common.h>
#include <dev/ofw/openfirm.h>
#include <dev/ofw/ofw_bus.h>
#include <dev/ofw/ofw_bus_subr.h>
+#include <arm/ti/ti_cpuid.h>
#include <arm/ti/ti_prcm.h>
#include <arm/ti/ti_i2c.h>
@@ -89,8 +90,13 @@ struct ti_i2c_softc
struct mtx sc_mtx;
- volatile uint16_t sc_stat_flags; /* contains the status flags last IRQ */
+ struct iic_msg* sc_buffer;
+ int sc_bus_inuse;
+ int sc_buffer_pos;
+ int sc_error;
+ int sc_fifo_trsh;
+ uint16_t sc_con_reg;
uint16_t sc_rev;
};
@@ -105,61 +111,54 @@ struct ti_i2c_clock_config
uint8_t hssclh; /* High Speed mode SCL high time */
};
-static struct ti_i2c_clock_config ti_i2c_clock_configs[] = {
+#if defined(SOC_OMAP3)
+#error "Unsupported SoC"
+#endif
#if defined(SOC_OMAP4)
- { IIC_SLOW, 100000, 23, 13, 15, 0, 0},
- { IIC_FAST, 400000, 9, 5, 7, 0, 0},
- { IIC_FASTEST, 3310000, 1, 113, 115, 7, 10},
-#elif defined(SOC_TI_AM335X)
- { IIC_SLOW, 100000, 3, 53, 55, 0, 0},
- { IIC_FAST, 400000, 3, 8, 10, 0, 0},
- { IIC_FASTEST, 400000, 3, 8, 10, 0, 0}, /* This might be higher */
-#else
-#error "TI I2C driver is not supported on this SoC"
-#endif
+static struct ti_i2c_clock_config ti_omap4_i2c_clock_configs[] = {
+ { IIC_UNKNOWN, 100000, 23, 13, 15, 0, 0},
+ { IIC_SLOW, 100000, 23, 13, 15, 0, 0},
+ { IIC_FAST, 400000, 9, 5, 7, 0, 0},
+ { IIC_FASTEST, 1000000, 5, 3, 4, 0, 0},
+ /* { IIC_FASTEST, 3200000, 1, 113, 115, 7, 10}, - HS mode */
{ -1, 0 }
};
+#endif
-
-#define TI_I2C_REV1 0x003C /* OMAP3 */
-#define TI_I2C_REV2 0x000A /* OMAP4 */
+#if defined(SOC_TI_AM335X)
+/*
+ * AM335X doesn't support HS mode. For 100kHz I2C clock set the internal
+ * clock to 12Mhz, for 400kHz I2C clock set the internal clock to 24Mhz.
+ */
+static struct ti_i2c_clock_config ti_am335x_i2c_clock_configs[] = {
+ { IIC_UNKNOWN, 100000, 7, 59, 61, 0, 0},
+ { IIC_SLOW, 100000, 7, 59, 61, 0, 0},
+ { IIC_FAST, 400000, 3, 23, 25, 0, 0},
+ { IIC_FASTEST, 400000, 3, 23, 25, 0, 0},
+ { -1, 0 }
+};
+#endif
/**
* Locking macros used throughout the driver
*/
-#define TI_I2C_LOCK(_sc) mtx_lock(&(_sc)->sc_mtx)
-#define TI_I2C_UNLOCK(_sc) mtx_unlock(&(_sc)->sc_mtx)
-#define TI_I2C_LOCK_INIT(_sc) \
- mtx_init(&_sc->sc_mtx, device_get_nameunit(_sc->sc_dev), \
- "ti_i2c", MTX_DEF)
-#define TI_I2C_LOCK_DESTROY(_sc) mtx_destroy(&_sc->sc_mtx);
-#define TI_I2C_ASSERT_LOCKED(_sc) mtx_assert(&_sc->sc_mtx, MA_OWNED);
-#define TI_I2C_ASSERT_UNLOCKED(_sc) mtx_assert(&_sc->sc_mtx, MA_NOTOWNED);
+#define TI_I2C_LOCK(_sc) mtx_lock(&(_sc)->sc_mtx)
+#define TI_I2C_UNLOCK(_sc) mtx_unlock(&(_sc)->sc_mtx)
+#define TI_I2C_LOCK_INIT(_sc) \
+ mtx_init(&_sc->sc_mtx, device_get_nameunit(_sc->sc_dev), \
+ "ti_i2c", MTX_DEF)
+#define TI_I2C_LOCK_DESTROY(_sc) mtx_destroy(&_sc->sc_mtx)
+#define TI_I2C_ASSERT_LOCKED(_sc) mtx_assert(&_sc->sc_mtx, MA_OWNED)
+#define TI_I2C_ASSERT_UNLOCKED(_sc) mtx_assert(&_sc->sc_mtx, MA_NOTOWNED)
#ifdef DEBUG
-#define ti_i2c_dbg(_sc, fmt, args...) \
- device_printf((_sc)->sc_dev, fmt, ##args)
+#define ti_i2c_dbg(_sc, fmt, args...) \
+ device_printf((_sc)->sc_dev, fmt, ##args)
#else
-#define ti_i2c_dbg(_sc, fmt, args...)
+#define ti_i2c_dbg(_sc, fmt, args...)
#endif
-static devclass_t ti_i2c_devclass;
-
-/* bus entry points */
-
-static int ti_i2c_probe(device_t dev);
-static int ti_i2c_attach(device_t dev);
-static int ti_i2c_detach(device_t dev);
-static void ti_i2c_intr(void *);
-
-/* OFW routine */
-static phandle_t ti_i2c_get_node(device_t bus, device_t dev);
-
-/* helper routines */
-static int ti_i2c_activate(device_t dev);
-static void ti_i2c_deactivate(device_t dev);
-
/**
* ti_i2c_read_2 - reads a 16-bit value from one of the I2C registers
* @sc: I2C device context
@@ -175,7 +174,8 @@ static void ti_i2c_deactivate(device_t dev);
static inline uint16_t
ti_i2c_read_2(struct ti_i2c_softc *sc, bus_size_t off)
{
- return bus_read_2(sc->sc_mem_res, off);
+
+ return (bus_read_2(sc->sc_mem_res, off));
}
/**
@@ -193,129 +193,117 @@ ti_i2c_read_2(struct ti_i2c_softc *sc, bus_size_t off)
static inline void
ti_i2c_write_2(struct ti_i2c_softc *sc, bus_size_t off, uint16_t val)
{
- bus_write_2(sc->sc_mem_res, off, val);
-}
-
-/**
- * ti_i2c_read_reg - reads a 16-bit value from one of the I2C registers
- * take into account revision-dependent register offset
- * @sc: I2C device context
- * @off: the byte offset within the register bank to read from.
- *
- *
- * LOCKING:
- * No locking required
- *
- * RETURNS:
- * 16-bit value read from the register.
- */
-static inline uint16_t
-ti_i2c_read_reg(struct ti_i2c_softc *sc, bus_size_t off)
-{
- /* XXXOMAP3: FIXME add registers mapping here */
- return bus_read_2(sc->sc_mem_res, off);
-}
-/**
- * ti_i2c_write_reg - writes a 16-bit value to one of the I2C registers
- * take into account revision-dependent register offset
- * @sc: I2C device context
- * @off: the byte offset within the register bank to read from.
- * @val: the value to write into the register
- *
- * LOCKING:
- * No locking required
- *
- * RETURNS:
- * 16-bit value read from the register.
- */
-static inline void
-ti_i2c_write_reg(struct ti_i2c_softc *sc, bus_size_t off, uint16_t val)
-{
- /* XXXOMAP3: FIXME add registers mapping here */
bus_write_2(sc->sc_mem_res, off, val);
}
-/**
- * ti_i2c_set_intr_enable - writes the interrupt enable register
- * @sc: I2C device context
- * @ie: bitmask of the interrupts to enable
- *
- * This function is needed as writing the I2C_IE register on the OMAP4 devices
- * doesn't seem to actually enable the interrupt, rather you have to write
- * through the I2C_IRQENABLE_CLR and I2C_IRQENABLE_SET registers.
- *
- * LOCKING:
- * No locking required
- *
- * RETURNS:
- * Nothing.
- */
-static inline void
-ti_i2c_set_intr_enable(struct ti_i2c_softc *sc, uint16_t ie)
-{
- /* XXXOMAP3: FIXME */
- ti_i2c_write_2(sc, I2C_REG_IRQENABLE_CLR, 0xffff);
- if (ie)
- ti_i2c_write_2(sc, I2C_REG_IRQENABLE_SET, ie);
-}
-
-/**
- * ti_i2c_reset - attach function for the driver
- * @dev: i2c device handle
- *
- *
- *
- * LOCKING:
- * Called from timer context
- *
- * RETURNS:
- * EH_HANDLED or EH_NOT_HANDLED
- */
static int
-ti_i2c_reset(device_t dev, u_char speed, u_char addr, u_char *oldaddr)
+ti_i2c_transfer_intr(struct ti_i2c_softc* sc, uint16_t status)
{
- struct ti_i2c_softc *sc = device_get_softc(dev);
- struct ti_i2c_clock_config *clkcfg;
- uint16_t con_reg;
+ int amount, done, i;
+
+ done = 0;
+ amount = 0;
+ /* Check for the error conditions. */
+ if (status & I2C_STAT_NACK) {
+ /* No ACK from slave. */
+ ti_i2c_dbg(sc, "NACK\n");
+ ti_i2c_write_2(sc, I2C_REG_STATUS, I2C_STAT_NACK);
+ sc->sc_error = ENXIO;
+ } else if (status & I2C_STAT_AL) {
+ /* Arbitration lost. */
+ ti_i2c_dbg(sc, "Arbitration lost\n");
+ ti_i2c_write_2(sc, I2C_REG_STATUS, I2C_STAT_AL);
+ sc->sc_error = ENXIO;
+ }
- clkcfg = ti_i2c_clock_configs;
- while (clkcfg->speed != -1) {
- if (clkcfg->speed == speed)
- break;
- /* take slow if speed is unknown */
- if ((speed == IIC_UNKNOWN) && (clkcfg->speed == IIC_SLOW))
- break;
- clkcfg++;
+ /* Check if we have finished. */
+ if (status & I2C_STAT_ARDY) {
+ /* Register access ready - transaction complete basically. */
+ ti_i2c_dbg(sc, "ARDY transaction complete\n");
+ if (sc->sc_error != 0 && sc->sc_buffer->flags & IIC_M_NOSTOP) {
+ ti_i2c_write_2(sc, I2C_REG_CON,
+ sc->sc_con_reg | I2C_CON_STP);
+ }
+ ti_i2c_write_2(sc, I2C_REG_STATUS,
+ I2C_STAT_ARDY | I2C_STAT_RDR | I2C_STAT_RRDY |
+ I2C_STAT_XDR | I2C_STAT_XRDY);
+ return (1);
}
- if (clkcfg->speed == -1)
- return (EINVAL);
- TI_I2C_LOCK(sc);
+ if (sc->sc_buffer->flags & IIC_M_RD) {
+ /* Read some data. */
+ if (status & I2C_STAT_RDR) {
+ /*
+ * Receive draining interrupt - last data received.
+ * The set FIFO threshold wont be reached to trigger
+ * RRDY.
+ */
+ ti_i2c_dbg(sc, "Receive draining interrupt\n");
- /* First disable the controller while changing the clocks */
- con_reg = ti_i2c_read_reg(sc, I2C_REG_CON);
- ti_i2c_write_reg(sc, I2C_REG_CON, 0x0000);
+ /*
+ * Drain the FIFO. Read the pending data in the FIFO.
+ */
+ amount = sc->sc_buffer->len - sc->sc_buffer_pos;
+ } else if (status & I2C_STAT_RRDY) {
+ /*
+ * Receive data ready interrupt - FIFO has reached the
+ * set threshold.
+ */
+ ti_i2c_dbg(sc, "Receive data ready interrupt\n");
- /* Program the prescaler */
- ti_i2c_write_reg(sc, I2C_REG_PSC, clkcfg->psc);
+ amount = min(sc->sc_fifo_trsh,
+ sc->sc_buffer->len - sc->sc_buffer_pos);
+ }
- /* Set the bitrate */
- ti_i2c_write_reg(sc, I2C_REG_SCLL, clkcfg->scll | (clkcfg->hsscll<<8));
- ti_i2c_write_reg(sc, I2C_REG_SCLH, clkcfg->sclh | (clkcfg->hssclh<<8));
+ /* Read the bytes from the fifo. */
+ for (i = 0; i < amount; i++)
+ sc->sc_buffer->buf[sc->sc_buffer_pos++] =
+ (uint8_t)(ti_i2c_read_2(sc, I2C_REG_DATA) & 0xff);
- /* Check if we are dealing with high speed mode */
- if ((clkcfg->hsscll + clkcfg->hssclh) > 0)
- con_reg = I2C_CON_OPMODE_HS;
- else
- con_reg = I2C_CON_OPMODE_STD;
+ if (status & I2C_STAT_RDR)
+ ti_i2c_write_2(sc, I2C_REG_STATUS, I2C_STAT_RDR);
+ if (status & I2C_STAT_RRDY)
+ ti_i2c_write_2(sc, I2C_REG_STATUS, I2C_STAT_RRDY);
- /* Enable the I2C module again */
- ti_i2c_write_reg(sc, I2C_REG_CON, I2C_CON_I2C_EN | con_reg);
+ } else {
+ /* Write some data. */
+ if (status & I2C_STAT_XDR) {
+ /*
+ * Transmit draining interrupt - FIFO level is below
+ * the set threshold and the amount of data still to
+ * be transferred wont reach the set FIFO threshold.
+ */
+ ti_i2c_dbg(sc, "Transmit draining interrupt\n");
- TI_I2C_UNLOCK(sc);
+ /*
+ * Drain the TX data. Write the pending data in the
+ * FIFO.
+ */
+ amount = sc->sc_buffer->len - sc->sc_buffer_pos;
+ } else if (status & I2C_STAT_XRDY) {
+ /*
+ * Transmit data ready interrupt - the FIFO level
+ * is below the set threshold.
+ */
+ ti_i2c_dbg(sc, "Transmit data ready interrupt\n");
- return (IIC_ENOADDR);
+ amount = min(sc->sc_fifo_trsh,
+ sc->sc_buffer->len - sc->sc_buffer_pos);
+ }
+
+ /* Write the bytes from the fifo. */
+ for (i = 0; i < amount; i++)
+ ti_i2c_write_2(sc, I2C_REG_DATA,
+ sc->sc_buffer->buf[sc->sc_buffer_pos++]);
+
+ if (status & I2C_STAT_XDR)
+ ti_i2c_write_2(sc, I2C_REG_STATUS, I2C_STAT_XDR);
+ if (status & I2C_STAT_XRDY)
+ ti_i2c_write_2(sc, I2C_REG_STATUS, I2C_STAT_XRDY);
+ }
+
+ return (done);
}
/**
@@ -333,439 +321,163 @@ ti_i2c_reset(device_t dev, u_char speed, u_char addr, u_char *oldaddr)
static void
ti_i2c_intr(void *arg)
{
- struct ti_i2c_softc *sc = (struct ti_i2c_softc*) arg;
- uint16_t status;
+ int done;
+ struct ti_i2c_softc *sc;
+ uint16_t events, status;
- status = ti_i2c_read_reg(sc, I2C_REG_STAT);
- if (status == 0)
- return;
+ sc = (struct ti_i2c_softc *)arg;
TI_I2C_LOCK(sc);
- /* save the flags */
- sc->sc_stat_flags |= status;
-
- /* clear the status flags */
- ti_i2c_write_reg(sc, I2C_REG_STAT, status);
-
- /* wakeup the process the started the transaction */
- wakeup(sc);
-
- TI_I2C_UNLOCK(sc);
-
- return;
-}
-
-/**
- * ti_i2c_wait - waits for the specific event to occur
- * @sc: i2c driver context
- * @flags: the event(s) to wait on, this is a bitmask of the I2C_STAT_??? flags
- * @statp: if not null will contain the status flags upon return
- * @timo: the number of ticks to wait
- *
- *
- *
- * LOCKING:
- * The driver context must be locked before calling this function. Internally
- * the function sleeps, releasing the lock as it does so, however the lock is
- * always retaken before this function returns.
- *
- * RETURNS:
- * 0 if the event(s) were tripped within timeout period
- * EBUSY if timedout waiting for the events
- * ENXIO if a NACK event was received
- */
-static int
-ti_i2c_wait(struct ti_i2c_softc *sc, uint16_t flags, uint16_t *statp, int timo)
-{
- int waittime = timo;
- int start_ticks = ticks;
- int rc;
-
- TI_I2C_ASSERT_LOCKED(sc);
-
- /* check if the condition has already occured, the interrupt routine will
- * clear the status flags.
- */
- if ((sc->sc_stat_flags & flags) == 0) {
-
- /* condition(s) haven't occured so sleep on the IRQ */
- while (waittime > 0) {
-
- rc = mtx_sleep(sc, &sc->sc_mtx, 0, "I2Cwait", waittime);
- if (rc == EWOULDBLOCK) {
- /* timed-out, simply break out of the loop */
- break;
- } else {
- /* IRQ has been tripped, but need to sanity check we have the
- * right events in the status flag.
- */
- if ((sc->sc_stat_flags & flags) != 0)
- break;
-
- /* event hasn't been tripped so wait some more */
- waittime -= (ticks - start_ticks);
- start_ticks = ticks;
- }
- }
- }
-
- /* copy the actual status bits */
- if (statp != NULL)
- *statp = sc->sc_stat_flags;
-
- /* return the status found */
- if ((sc->sc_stat_flags & flags) != 0)
- rc = 0;
- else
- rc = EBUSY;
-
- /* clear the flags set by the interrupt handler */
- sc->sc_stat_flags = 0;
-
- return (rc);
-}
-
-/**
- * ti_i2c_wait_for_free_bus - waits for the bus to become free
- * @sc: i2c driver context
- * @timo: the time to wait for the bus to become free
- *
- *
- *
- * LOCKING:
- * The driver context must be locked before calling this function. Internally
- * the function sleeps, releasing the lock as it does so, however the lock is
- * always taken before this function returns.
- *
- * RETURNS:
- * 0 if the event(s) were tripped within timeout period
- * EBUSY if timedout waiting for the events
- * ENXIO if a NACK event was received
- */
-static int
-ti_i2c_wait_for_free_bus(struct ti_i2c_softc *sc, int timo)
-{
- /* check if the bus is free, BB bit = 0 */
- if ((ti_i2c_read_reg(sc, I2C_REG_STAT) & I2C_STAT_BB) == 0)
- return 0;
-
- /* enable bus free interrupts */
- ti_i2c_set_intr_enable(sc, I2C_IE_BF);
-
- /* wait for the bus free interrupt to be tripped */
- return ti_i2c_wait(sc, I2C_STAT_BF, NULL, timo);
-}
-
-/**
- * ti_i2c_read_bytes - attempts to perform a read operation
- * @sc: i2c driver context
- * @buf: buffer to hold the received bytes
- * @len: the number of bytes to read
- *
- * This function assumes the slave address is already set
- *
- * LOCKING:
- * The context lock should be held before calling this function
- *
- * RETURNS:
- * 0 on function succeeded
- * EINVAL if invalid message is passed as an arg
- */
-static int
-ti_i2c_read_bytes(struct ti_i2c_softc *sc, uint8_t *buf, uint16_t len)
-{
- int timo = (hz / 4);
- int err = 0;
- uint16_t con_reg;
- uint16_t events;
- uint16_t status;
- uint32_t amount = 0;
- uint32_t sofar = 0;
- uint32_t i;
-
- /* wait for the bus to become free */
- err = ti_i2c_wait_for_free_bus(sc, timo);
- if (err != 0) {
- device_printf(sc->sc_dev, "bus never freed\n");
- return (err);
+ status = ti_i2c_read_2(sc, I2C_REG_STATUS);
+ if (status == 0) {
+ TI_I2C_UNLOCK(sc);
+ return;
}
- /* set the events to wait for */
- events = I2C_IE_RDR | /* Receive draining interrupt */
- I2C_IE_RRDY | /* Receive Data Ready interrupt */
- I2C_IE_ARDY | /* Register Access Ready interrupt */
- I2C_IE_NACK | /* No Acknowledgment interrupt */
- I2C_IE_AL;
-
- /* enable interrupts for the events we want */
- ti_i2c_set_intr_enable(sc, events);
-
- /* write the number of bytes to read */
- ti_i2c_write_reg(sc, I2C_REG_CNT, len);
-
- /* clear the write bit and initiate the read transaction. Setting the STT
- * (start) bit initiates the transfer.
- */
- con_reg = ti_i2c_read_reg(sc, I2C_REG_CON);
- con_reg &= ~I2C_CON_TRX;
- con_reg |= I2C_CON_MST | I2C_CON_STT | I2C_CON_STP;
- ti_i2c_write_reg(sc, I2C_REG_CON, con_reg);
-
- /* reading loop */
- while (1) {
-
- /* wait for an event */
- err = ti_i2c_wait(sc, events, &status, timo);
- if (err != 0) {
- break;
- }
-
- /* check for the error conditions */
- if (status & I2C_STAT_NACK) {
- /* no ACK from slave */
- ti_i2c_dbg(sc, "NACK\n");
- err = ENXIO;
- break;
- }
- if (status & I2C_STAT_AL) {
- /* arbitration lost */
- ti_i2c_dbg(sc, "Arbitration lost\n");
- err = ENXIO;
- break;
- }
-
- /* check if we have finished */
- if (status & I2C_STAT_ARDY) {
- /* register access ready - transaction complete basically */
- ti_i2c_dbg(sc, "ARDY transaction complete\n");
- err = 0;
- break;
- }
+ /* Save enabled interrupts. */
+ events = ti_i2c_read_2(sc, I2C_REG_IRQENABLE_SET);
- /* read some data */
- if (status & I2C_STAT_RDR) {
- /* Receive draining interrupt - last data received */
- ti_i2c_dbg(sc, "Receive draining interrupt\n");
-
- /* get the number of bytes in the FIFO */
- amount = ti_i2c_read_reg(sc, I2C_REG_BUFSTAT);
- amount >>= 8;
- amount &= 0x3f;
- }
- else if (status & I2C_STAT_RRDY) {
- /* Receive data ready interrupt - enough data received */
- ti_i2c_dbg(sc, "Receive data ready interrupt\n");
+ /* We only care about enabled interrupts. */
+ status &= events;
- /* get the number of bytes in the FIFO */
- amount = ti_i2c_read_reg(sc, I2C_REG_BUF);
- amount >>= 8;
- amount &= 0x3f;
- amount += 1;
- }
+ done = 0;
- /* sanity check we haven't overwritten the array */
- if ((sofar + amount) > len) {
- ti_i2c_dbg(sc, "to many bytes to read\n");
- amount = (len - sofar);
- }
-
- /* read the bytes from the fifo */
- for (i = 0; i < amount; i++) {
- buf[sofar++] = (uint8_t)(ti_i2c_read_reg(sc, I2C_REG_DATA) & 0xff);
- }
-
- /* attempt to clear the receive ready bits */
- ti_i2c_write_reg(sc, I2C_REG_STAT, I2C_STAT_RDR | I2C_STAT_RRDY);
+ if (sc->sc_buffer != NULL)
+ done = ti_i2c_transfer_intr(sc, status);
+ else {
+ ti_i2c_dbg(sc, "Transfer interrupt without buffer\n");
+ sc->sc_error = EINVAL;
+ done = 1;
}
- /* reset the registers regardless if there was an error or not */
- ti_i2c_set_intr_enable(sc, 0x0000);
- ti_i2c_write_reg(sc, I2C_REG_CON, I2C_CON_I2C_EN | I2C_CON_MST | I2C_CON_STP);
+ if (done)
+ /* Wakeup the process that started the transaction. */
+ wakeup(sc);
- return (err);
+ TI_I2C_UNLOCK(sc);
}
/**
- * ti_i2c_write_bytes - attempts to perform a read operation
- * @sc: i2c driver context
- * @buf: buffer containing the bytes to write
- * @len: the number of bytes to write
+ * ti_i2c_transfer - called to perform the transfer
+ * @dev: i2c device handle
+ * @msgs: the messages to send/receive
+ * @nmsgs: the number of messages in the msgs array
*
- * This function assumes the slave address is already set
*
* LOCKING:
- * The context lock should be held before calling this function
+ * Internally locked
*
* RETURNS:
* 0 on function succeeded
* EINVAL if invalid message is passed as an arg
*/
static int
-ti_i2c_write_bytes(struct ti_i2c_softc *sc, const uint8_t *buf, uint16_t len)
+ti_i2c_transfer(device_t dev, struct iic_msg *msgs, uint32_t nmsgs)
{
- int timo = (hz / 4);
- int err = 0;
- uint16_t con_reg;
- uint16_t events;
- uint16_t status;
- uint32_t amount = 0;
- uint32_t sofar = 0;
- uint32_t i;
-
- /* wait for the bus to become free */
- err = ti_i2c_wait_for_free_bus(sc, timo);
- if (err != 0)
- return (err);
+ int err, i, repstart, timeout;
+ struct ti_i2c_softc *sc;
+ uint16_t reg;
- /* set the events to wait for */
- events = I2C_IE_XDR | /* Transmit draining interrupt */
- I2C_IE_XRDY | /* Transmit Data Ready interrupt */
- I2C_IE_ARDY | /* Register Access Ready interrupt */
- I2C_IE_NACK | /* No Acknowledgment interrupt */
- I2C_IE_AL;
+ sc = device_get_softc(dev);
+ TI_I2C_LOCK(sc);
- /* enable interrupts for the events we want*/
- ti_i2c_set_intr_enable(sc, events);
+ /* If the controller is busy wait until it is available. */
+ while (sc->sc_bus_inuse == 1)
+ mtx_sleep(sc, &sc->sc_mtx, 0, "i2cbuswait", 0);
- /* write the number of bytes to write */
- ti_i2c_write_reg(sc, I2C_REG_CNT, len);
+ /* Now we have control over the I2C controller. */
+ sc->sc_bus_inuse = 1;
- /* set the write bit and initiate the write transaction. Setting the STT
- * (start) bit initiates the transfer.
- */
- con_reg = ti_i2c_read_reg(sc, I2C_REG_CON);
- con_reg |= I2C_CON_TRX | I2C_CON_MST | I2C_CON_STT | I2C_CON_STP;
- ti_i2c_write_reg(sc, I2C_REG_CON, con_reg);
+ err = 0;
+ repstart = 0;
+ for (i = 0; i < nmsgs; i++) {
- /* writing loop */
- while (1) {
+ sc->sc_buffer = &msgs[i];
+ sc->sc_buffer_pos = 0;
+ sc->sc_error = 0;
- /* wait for an event */
- err = ti_i2c_wait(sc, events, &status, timo);
- if (err != 0) {
+ /* Zero byte transfers aren't allowed. */
+ if (sc->sc_buffer == NULL || sc->sc_buffer->buf == NULL ||
+ sc->sc_buffer->len == 0) {
+ err = EINVAL;
break;
}
- /* check for the error conditions */
- if (status & I2C_STAT_NACK) {
- /* no ACK from slave */
- ti_i2c_dbg(sc, "NACK\n");
- err = ENXIO;
- break;
- }
- if (status & I2C_STAT_AL) {
- /* arbitration lost */
- ti_i2c_dbg(sc, "Arbitration lost\n");
- err = ENXIO;
- break;
- }
+ /* Check if the i2c bus is free. */
+ if (repstart == 0) {
+ /*
+ * On repeated start we send the START condition while
+ * the bus _is_ busy.
+ */
+ timeout = 0;
+ while (ti_i2c_read_2(sc, I2C_REG_STATUS_RAW) & I2C_STAT_BB) {
+ if (timeout++ > 100) {
+ err = EBUSY;
+ goto out;
+ }
+ DELAY(1000);
+ }
+ timeout = 0;
+ } else
+ repstart = 0;
- /* check if we have finished */
- if (status & I2C_STAT_ARDY) {
- /* register access ready - transaction complete basically */
- ti_i2c_dbg(sc, "ARDY transaction complete\n");
- err = 0;
- break;
- }
+ if (sc->sc_buffer->flags & IIC_M_NOSTOP)
+ repstart = 1;
- /* read some data */
- if (status & I2C_STAT_XDR) {
- /* Receive draining interrupt - last data received */
- ti_i2c_dbg(sc, "Transmit draining interrupt\n");
+ /* Set the slave address. */
+ ti_i2c_write_2(sc, I2C_REG_SA, msgs[i].slave >> 1);
- /* get the number of bytes in the FIFO */
- amount = ti_i2c_read_reg(sc, I2C_REG_BUFSTAT);
- amount &= 0x3f;
- }
- else if (status & I2C_STAT_XRDY) {
- /* Receive data ready interrupt - enough data received */
- ti_i2c_dbg(sc, "Transmit data ready interrupt\n");
+ /* Write the data length. */
+ ti_i2c_write_2(sc, I2C_REG_CNT, sc->sc_buffer->len);
- /* get the number of bytes in the FIFO */
- amount = ti_i2c_read_reg(sc, I2C_REG_BUF);
- amount &= 0x3f;
- amount += 1;
- }
+ /* Clear the RX and the TX FIFO. */
+ reg = ti_i2c_read_2(sc, I2C_REG_BUF);
+ reg |= I2C_BUF_RXFIFO_CLR | I2C_BUF_TXFIFO_CLR;
+ ti_i2c_write_2(sc, I2C_REG_BUF, reg);
- /* sanity check we haven't overwritten the array */
- if ((sofar + amount) > len) {
- ti_i2c_dbg(sc, "to many bytes to write\n");
- amount = (len - sofar);
- }
+ reg = sc->sc_con_reg | I2C_CON_STT;
+ if (repstart == 0)
+ reg |= I2C_CON_STP;
+ if ((sc->sc_buffer->flags & IIC_M_RD) == 0)
+ reg |= I2C_CON_TRX;
+ ti_i2c_write_2(sc, I2C_REG_CON, reg);
- /* write the bytes from the fifo */
- for (i = 0; i < amount; i++) {
- ti_i2c_write_reg(sc, I2C_REG_DATA, buf[sofar++]);
- }
+ /* Wait for an event. */
+ err = mtx_sleep(sc, &sc->sc_mtx, 0, "i2ciowait", hz);
+ if (err == 0)
+ err = sc->sc_error;
- /* attempt to clear the transmit ready bits */
- ti_i2c_write_reg(sc, I2C_REG_STAT, I2C_STAT_XDR | I2C_STAT_XRDY);
+ if (err)
+ break;
}
- /* reset the registers regardless if there was an error or not */
- ti_i2c_set_intr_enable(sc, 0x0000);
- ti_i2c_write_reg(sc, I2C_REG_CON, I2C_CON_I2C_EN | I2C_CON_MST | I2C_CON_STP);
-
- return (err);
-}
-
-/**
- * ti_i2c_transfer - called to perform the transfer
- * @dev: i2c device handle
- * @msgs: the messages to send/receive
- * @nmsgs: the number of messages in the msgs array
- *
- *
- * LOCKING:
- * Internally locked
- *
- * RETURNS:
- * 0 on function succeeded
- * EINVAL if invalid message is passed as an arg
- */
-static int
-ti_i2c_transfer(device_t dev, struct iic_msg *msgs, uint32_t nmsgs)
-{
- struct ti_i2c_softc *sc = device_get_softc(dev);
- int err = 0;
- uint32_t i;
- uint16_t len;
- uint8_t *buf;
-
- TI_I2C_LOCK(sc);
-
- for (i = 0; i < nmsgs; i++) {
-
- len = msgs[i].len;
- buf = msgs[i].buf;
-
- /* zero byte transfers aren't allowed */
- if (len == 0 || buf == NULL) {
- err = EINVAL;
- goto out;
+out:
+ if (timeout == 0) {
+ while (ti_i2c_read_2(sc, I2C_REG_STATUS_RAW) & I2C_STAT_BB) {
+ if (timeout++ > 100)
+ break;
+ DELAY(1000);
}
+ }
+ /* Put the controller in master mode again. */
+ if ((ti_i2c_read_2(sc, I2C_REG_CON) & I2C_CON_MST) == 0)
+ ti_i2c_write_2(sc, I2C_REG_CON, sc->sc_con_reg);
- /* set the slave address */
- ti_i2c_write_reg(sc, I2C_REG_SA, msgs[i].slave >> 1);
-
- /* perform the read or write */
- if (msgs[i].flags & IIC_M_RD) {
- err = ti_i2c_read_bytes(sc, buf, len);
- } else {
- err = ti_i2c_write_bytes(sc, buf, len);
- }
+ sc->sc_buffer = NULL;
+ sc->sc_bus_inuse = 0;
- }
+ /* Wake up the processes that are waiting for the bus. */
+ wakeup(sc);
-out:
TI_I2C_UNLOCK(sc);
return (err);
}
/**
- * ti_i2c_callback - not sure about this one
+ * ti_i2c_callback - as we only provide iicbus_transfer() interface
+ * we don't need to implement the serialization here.
* @dev: i2c device handle
*
*
@@ -795,158 +507,231 @@ ti_i2c_callback(device_t dev, int index, caddr_t data)
return (error);
}
-/**
- * ti_i2c_activate - initialises and activates an I2C bus
- * @dev: i2c device handle
- * @num: the number of the I2C controller to activate; 1, 2 or 3
- *
- *
- * LOCKING:
- * Assumed called in an atomic context.
- *
- * RETURNS:
- * nothing
- */
static int
-ti_i2c_activate(device_t dev)
+ti_i2c_reset(struct ti_i2c_softc *sc, u_char speed)
{
- struct ti_i2c_softc *sc = (struct ti_i2c_softc*) device_get_softc(dev);
- unsigned int timeout = 0;
- uint16_t con_reg;
- int err;
- clk_ident_t clk;
+ int timeout;
+ struct ti_i2c_clock_config *clkcfg;
+ uint16_t fifo_trsh, reg, scll, sclh;
- /*
- * The following sequence is taken from the OMAP3530 technical reference
- *
- * 1. Enable the functional and interface clocks (see Section 18.3.1.1.1).
- */
- clk = I2C0_CLK + sc->device_id;
- err = ti_prcm_clk_enable(clk);
- if (err)
- return (err);
+ switch (ti_chip()) {
+#ifdef SOC_OMAP4
+ case CHIP_OMAP_4:
+ clkcfg = ti_omap4_i2c_clock_configs;
+ break;
+#endif
+#ifdef SOC_TI_AM335X
+ case CHIP_AM335X:
+ clkcfg = ti_am335x_i2c_clock_configs;
+ break;
+#endif
+ default:
+ panic("Unknown Ti SoC, unable to reset the i2c");
+ }
+ while (clkcfg->speed != -1) {
+ if (clkcfg->speed == speed)
+ break;
+ clkcfg++;
+ }
+ if (clkcfg->speed == -1)
+ return (EINVAL);
- /* There seems to be a bug in the I2C reset mechanism, for some reason you
- * need to disable the I2C module before issuing the reset and then enable
- * it again after to detect the reset done.
+ /*
+ * 23.1.4.3 - HS I2C Software Reset
+ * From OMAP4 TRM at page 4068.
*
- * I found this out by looking at the Linux driver implementation, thanks
- * linux guys!
+ * 1. Ensure that the module is disabled.
*/
+ sc->sc_con_reg = 0;
+ ti_i2c_write_2(sc, I2C_REG_CON, sc->sc_con_reg);
- /* Disable the I2C controller */
- ti_i2c_write_reg(sc, I2C_REG_CON, 0x0000);
+ /* 2. Issue a softreset to the controller. */
+ bus_write_2(sc->sc_mem_res, I2C_REG_SYSC, I2C_REG_SYSC_SRST);
- /* Issue a softreset to the controller */
- /* XXXOMAP3: FIXME */
- bus_write_2(sc->sc_mem_res, I2C_REG_SYSC, 0x0002);
-
- /* Re-enable the module and then check for the reset done */
- ti_i2c_write_reg(sc, I2C_REG_CON, I2C_CON_I2C_EN);
+ /*
+ * 3. Enable the module.
+ * The I2Ci.I2C_SYSS[0] RDONE bit is asserted only after the module
+ * is enabled by setting the I2Ci.I2C_CON[15] I2C_EN bit to 1.
+ */
+ ti_i2c_write_2(sc, I2C_REG_CON, I2C_CON_I2C_EN);
- while ((ti_i2c_read_reg(sc, I2C_REG_SYSS) & 0x01) == 0x00) {
- if (timeout++ > 100) {
+ /* 4. Wait for the software reset to complete. */
+ timeout = 0;
+ while ((ti_i2c_read_2(sc, I2C_REG_SYSS) & I2C_SYSS_RDONE) == 0) {
+ if (timeout++ > 100)
return (EBUSY);
- }
DELAY(100);
}
- /* Disable the I2C controller once again, now that the reset has finished */
- ti_i2c_write_reg(sc, I2C_REG_CON, 0x0000);
-
- /* 2. Program the prescaler to obtain an approximately 12-MHz internal
- * sampling clock (I2Ci_INTERNAL_CLK) by programming the corresponding
- * value in the I2Ci.I2C_PSC[3:0] PSC field.
- * This value depends on the frequency of the functional clock (I2Ci_FCLK).
- * Because this frequency is 96MHz, the I2Ci.I2C_PSC[7:0] PSC field value
- * is 0x7.
+ /*
+ * Disable the I2C controller once again, now that the reset has
+ * finished.
*/
+ ti_i2c_write_2(sc, I2C_REG_CON, sc->sc_con_reg);
- /* Program the prescaler to obtain an approximately 12-MHz internal
- * sampling clock.
+ /*
+ * The following sequence is taken from the OMAP4 TRM at page 4077.
+ *
+ * 1. Enable the functional and interface clocks (see Section
+ * 23.1.5.1.1.1.1). Done at ti_i2c_activate().
+ *
+ * 2. Program the prescaler to obtain an approximately 12MHz internal
+ * sampling clock (I2Ci_INTERNAL_CLK) by programming the
+ * corresponding value in the I2Ci.I2C_PSC[3:0] PSC field.
+ * This value depends on the frequency of the functional clock
+ * (I2Ci_FCLK). Because this frequency is 96MHz, the
+ * I2Ci.I2C_PSC[7:0] PSC field value is 0x7.
*/
- ti_i2c_write_reg(sc, I2C_REG_PSC, 0x0017);
+ ti_i2c_write_2(sc, I2C_REG_PSC, clkcfg->psc);
- /* 3. Program the I2Ci.I2C_SCLL[7:0] SCLL and I2Ci.I2C_SCLH[7:0] SCLH fields
- * to obtain a bit rate of 100K bps or 400K bps. These values depend on
- * the internal sampling clock frequency (see Table 18-12).
+ /*
+ * 3. Program the I2Ci.I2C_SCLL[7:0] SCLL and I2Ci.I2C_SCLH[7:0] SCLH
+ * bit fields to obtain a bit rate of 100 Kbps, 400 Kbps or 1Mbps.
+ * These values depend on the internal sampling clock frequency
+ * (see Table 23-8).
*/
+ scll = clkcfg->scll & I2C_SCLL_MASK;
+ sclh = clkcfg->sclh & I2C_SCLH_MASK;
- /* Set the bitrate to 100kbps */
- ti_i2c_write_reg(sc, I2C_REG_SCLL, 0x000d);
- ti_i2c_write_reg(sc, I2C_REG_SCLH, 0x000f);
-
- /* 4. (Optional) Program the I2Ci.I2C_SCLL[15:8] HSSCLL and
- * I2Ci.I2C_SCLH[15:8] HSSCLH fields to obtain a bit rate of 400K bps or
- * 3.4M bps (for the second phase of HS mode). These values depend on the
- * internal sampling clock frequency (see Table 18-12).
+ /*
+ * 4. (Optional) Program the I2Ci.I2C_SCLL[15:8] HSSCLL and
+ * I2Ci.I2C_SCLH[15:8] HSSCLH fields to obtain a bit rate of
+ * 400K bps or 3.4M bps (for the second phase of HS mode). These
+ * values depend on the internal sampling clock frequency (see
+ * Table 23-8).
*
* 5. (Optional) If a bit rate of 3.4M bps is used and the bus line
- * capacitance exceeds 45 pF, program the CONTROL.CONTROL_DEVCONF1[12]
- * I2C1HSMASTER bit for I2C1, the CONTROL.CONTROL_DEVCONF1[13]
- * I2C2HSMASTER bit for I2C2, or the CONTROL.CONTROL_DEVCONF1[14]
- * I2C3HSMASTER bit for I2C3.
- */
-
- /* 6. Configure the Own Address of the I2C controller by storing it in the
- * I2Ci.I2C_OA0 register. Up to four Own Addresses can be programmed in
- * the I2Ci.I2C_OAi registers (with I = 0, 1, 2, 3) for each I2C
- * controller.
- *
- * Note: For a 10-bit address, set the corresponding expand Own Address bit
- * in the I2Ci.I2C_CON register.
+ * capacitance exceeds 45 pF, (see Section 18.4.8, PAD Functional
+ * Multiplexing and Configuration).
*/
+ switch (ti_chip()) {
+#ifdef SOC_OMAP4
+ case CHIP_OMAP_4:
+ if ((clkcfg->hsscll + clkcfg->hssclh) > 0) {
+ scll |= clkcfg->hsscll << I2C_HSSCLL_SHIFT;
+ sclh |= clkcfg->hssclh << I2C_HSSCLH_SHIFT;
+ sc->sc_con_reg |= I2C_CON_OPMODE_HS;
+ }
+ break;
+#endif
+ }
- /* Driver currently always in single master mode so ignore this step */
-
- /* 7. Set the TX threshold (in transmitter mode) and the RX threshold (in
- * receiver mode) by setting the I2Ci.I2C_BUF[5:0]XTRSH field to (TX
- * threshold - 1) and the I2Ci.I2C_BUF[13:8]RTRSH field to (RX threshold
- * - 1), where the TX and RX thresholds are greater than or equal to 1.
- */
+ /* Write the selected bit rate. */
+ ti_i2c_write_2(sc, I2C_REG_SCLL, scll);
+ ti_i2c_write_2(sc, I2C_REG_SCLH, sclh);
- /* Set the FIFO buffer threshold, note I2C1 & I2C2 have 8 byte FIFO, whereas
- * I2C3 has 64 bytes. Threshold set to 5 for now.
+ /*
+ * 6. Configure the Own Address of the I2C controller by storing it in
+ * the I2Ci.I2C_OA0 register. Up to four Own Addresses can be
+ * programmed in the I2Ci.I2C_OAi registers (where i = 0, 1, 2, 3)
+ * for each I2C controller.
+ *
+ * Note: For a 10-bit address, set the corresponding expand Own Address
+ * bit in the I2Ci.I2C_CON register.
+ *
+ * Driver currently always in single master mode so ignore this step.
*/
- ti_i2c_write_reg(sc, I2C_REG_BUF, 0x0404);
/*
- * 8. Take the I2C controller out of reset by setting the I2Ci.I2C_CON[15]
- * I2C_EN bit to 1.
+ * 7. Set the TX threshold (in transmitter mode) and the RX threshold
+ * (in receiver mode) by setting the I2Ci.I2C_BUF[5:0]XTRSH field to
+ * (TX threshold - 1) and the I2Ci.I2C_BUF[13:8]RTRSH field to (RX
+ * threshold - 1), where the TX and RX thresholds are greater than
+ * or equal to 1.
+ *
+ * The threshold is set to 5 for now.
*/
- ti_i2c_write_reg(sc, I2C_REG_CON, I2C_CON_I2C_EN | I2C_CON_OPMODE_STD);
+ fifo_trsh = (sc->sc_fifo_trsh - 1) & I2C_BUF_TRSH_MASK;
+ reg = fifo_trsh | (fifo_trsh << I2C_BUF_RXTRSH_SHIFT);
+ ti_i2c_write_2(sc, I2C_REG_BUF, reg);
/*
+ * 8. Take the I2C controller out of reset by setting the
+ * I2Ci.I2C_CON[15] I2C_EN bit to 1.
+ *
+ * 23.1.5.1.1.1.2 - Initialize the I2C Controller
+ *
* To initialize the I2C controller, perform the following steps:
*
* 1. Configure the I2Ci.I2C_CON register:
- * · For master or slave mode, set the I2Ci.I2C_CON[10] MST bit (0: slave,
- * 1: master).
- * · For transmitter or receiver mode, set the I2Ci.I2C_CON[9] TRX bit
- * (0: receiver, 1: transmitter).
+ * . For master or slave mode, set the I2Ci.I2C_CON[10] MST bit
+ * (0: slave, 1: master).
+ * . For transmitter or receiver mode, set the I2Ci.I2C_CON[9] TRX
+ * bit (0: receiver, 1: transmitter).
*/
- con_reg = ti_i2c_read_reg(sc, I2C_REG_CON);
- con_reg |= I2C_CON_MST;
- ti_i2c_write_reg(sc, I2C_REG_CON, con_reg);
- /* 2. If using an interrupt to transmit/receive data, set to 1 the
+ /* Enable the I2C controller in master mode. */
+ sc->sc_con_reg |= I2C_CON_I2C_EN | I2C_CON_MST;
+ ti_i2c_write_2(sc, I2C_REG_CON, sc->sc_con_reg);
+
+ /*
+ * 2. If using an interrupt to transmit/receive data, set the
* corresponding bit in the I2Ci.I2C_IE register (the I2Ci.I2C_IE[4]
- * XRDY_IE bit for the transmit interrupt, the I2Ci.I2C_IE[3] RRDY bit
- * for the receive interrupt).
+ * XRDY_IE bit for the transmit interrupt, the I2Ci.I2C_IE[3] RRDY
+ * bit for the receive interrupt).
*/
- ti_i2c_set_intr_enable(sc, I2C_IE_XRDY | I2C_IE_RRDY);
- /* 3. If using DMA to receive/transmit data, set to 1 the corresponding bit
- * in the I2Ci.I2C_BUF register (the I2Ci.I2C_BUF[15] RDMA_EN bit for the
- * receive DMA channel, the I2Ci.I2C_BUF[7] XDMA_EN bit for the transmit
- * DMA channel).
- */
+ /* Set the interrupts we want to be notified. */
+ reg = I2C_IE_XDR | /* Transmit draining interrupt. */
+ I2C_IE_XRDY | /* Transmit Data Ready interrupt. */
+ I2C_IE_RDR | /* Receive draining interrupt. */
+ I2C_IE_RRDY | /* Receive Data Ready interrupt. */
+ I2C_IE_ARDY | /* Register Access Ready interrupt. */
+ I2C_IE_NACK | /* No Acknowledgment interrupt. */
+ I2C_IE_AL; /* Arbitration lost interrupt. */
- /* not using DMA for now, so ignore this */
+ /* Enable the interrupts. */
+ ti_i2c_write_2(sc, I2C_REG_IRQENABLE_SET, reg);
+
+ /*
+ * 3. If using DMA to receive/transmit data, set to 1 the corresponding
+ * bit in the I2Ci.I2C_BUF register (the I2Ci.I2C_BUF[15] RDMA_EN
+ * bit for the receive DMA channel, the I2Ci.I2C_BUF[7] XDMA_EN bit
+ * for the transmit DMA channel).
+ *
+ * Not using DMA for now, so ignore this.
+ */
return (0);
}
+static int
+ti_i2c_iicbus_reset(device_t dev, u_char speed, u_char addr, u_char *oldaddr)
+{
+ struct ti_i2c_softc *sc;
+ int err;
+
+ sc = device_get_softc(dev);
+ TI_I2C_LOCK(sc);
+ err = ti_i2c_reset(sc, speed);
+ TI_I2C_UNLOCK(sc);
+ if (err)
+ return (err);
+
+ return (IIC_ENOADDR);
+}
+
+static int
+ti_i2c_activate(device_t dev)
+{
+ clk_ident_t clk;
+ int err;
+ struct ti_i2c_softc *sc;
+
+ sc = (struct ti_i2c_softc*)device_get_softc(dev);
+
+ /*
+ * 1. Enable the functional and interface clocks (see Section
+ * 23.1.5.1.1.1.1).
+ */
+ clk = I2C0_CLK + sc->device_id;
+ err = ti_prcm_clk_enable(clk);
+ if (err)
+ return (err);
+
+ return (ti_i2c_reset(sc, IIC_UNKNOWN));
+}
+
/**
* ti_i2c_deactivate - deactivates the controller and releases resources
* @dev: i2c device handle
@@ -965,136 +750,154 @@ ti_i2c_deactivate(device_t dev)
struct ti_i2c_softc *sc = device_get_softc(dev);
clk_ident_t clk;
- /* Disable the controller - cancel all transactions */
- ti_i2c_write_reg(sc, I2C_REG_CON, 0x0000);
+ /* Disable the controller - cancel all transactions. */
+ ti_i2c_write_2(sc, I2C_REG_IRQENABLE_CLR, 0xffff);
+ ti_i2c_write_2(sc, I2C_REG_STATUS, 0xffff);
+ ti_i2c_write_2(sc, I2C_REG_CON, 0);
- /* Release the interrupt handler */
- if (sc->sc_irq_h) {
+ /* Release the interrupt handler. */
+ if (sc->sc_irq_h != NULL) {
bus_teardown_intr(dev, sc->sc_irq_res, sc->sc_irq_h);
- sc->sc_irq_h = 0;
+ sc->sc_irq_h = NULL;
}
bus_generic_detach(sc->sc_dev);
- /* Unmap the I2C controller registers */
- if (sc->sc_mem_res != 0) {
- bus_release_resource(dev, SYS_RES_MEMORY, rman_get_rid(sc->sc_irq_res),
- sc->sc_mem_res);
+ /* Unmap the I2C controller registers. */
+ if (sc->sc_mem_res != NULL) {
+ bus_release_resource(dev, SYS_RES_MEMORY, 0, sc->sc_mem_res);
sc->sc_mem_res = NULL;
}
- /* Release the IRQ resource */
+ /* Release the IRQ resource. */
if (sc->sc_irq_res != NULL) {
- bus_release_resource(dev, SYS_RES_IRQ, rman_get_rid(sc->sc_irq_res),
- sc->sc_irq_res);
+ bus_release_resource(dev, SYS_RES_IRQ, 0, sc->sc_irq_res);
sc->sc_irq_res = NULL;
}
- /* Finally disable the functional and interface clocks */
+ /* Finally disable the functional and interface clocks. */
clk = I2C0_CLK + sc->device_id;
ti_prcm_clk_disable(clk);
+}
+
+static int
+ti_i2c_sysctl_clk(SYSCTL_HANDLER_ARGS)
+{
+ device_t dev;
+ int clk, psc, sclh, scll;
+ struct ti_i2c_softc *sc;
+
+ dev = (device_t)arg1;
+ sc = device_get_softc(dev);
- return;
+ TI_I2C_LOCK(sc);
+ /* Get the system prescaler value. */
+ psc = (int)ti_i2c_read_2(sc, I2C_REG_PSC) + 1;
+
+ /* Get the bitrate. */
+ scll = (int)ti_i2c_read_2(sc, I2C_REG_SCLL) & I2C_SCLL_MASK;
+ sclh = (int)ti_i2c_read_2(sc, I2C_REG_SCLH) & I2C_SCLH_MASK;
+
+ clk = I2C_CLK / psc / (scll + 7 + sclh + 5);
+ TI_I2C_UNLOCK(sc);
+
+ return (sysctl_handle_int(oidp, &clk, 0, req));
}
-/**
- * ti_i2c_probe - probe function for the driver
- * @dev: i2c device handle
- *
- *
- *
- * LOCKING:
- *
- *
- * RETURNS:
- * Always returns 0
- */
static int
ti_i2c_probe(device_t dev)
{
if (!ofw_bus_status_okay(dev))
return (ENXIO);
-
if (!ofw_bus_is_compatible(dev, "ti,i2c"))
return (ENXIO);
-
device_set_desc(dev, "TI I2C Controller");
+
return (0);
}
-/**
- * ti_i2c_attach - attach function for the driver
- * @dev: i2c device handle
- *
- * Initialised driver data structures and activates the I2C controller.
- *
- * LOCKING:
- *
- *
- * RETURNS:
- *
- */
static int
ti_i2c_attach(device_t dev)
{
- struct ti_i2c_softc *sc = device_get_softc(dev);
+ int err, rid;
phandle_t node;
- pcell_t did;
- int err;
- int rid;
+ struct ti_i2c_softc *sc;
+ struct sysctl_ctx_list *ctx;
+ struct sysctl_oid_list *tree;
+ uint16_t fifosz;
+ sc = device_get_softc(dev);
sc->sc_dev = dev;
- /* Get the i2c device id from FDT */
+ /* Get the i2c device id from FDT. */
node = ofw_bus_get_node(dev);
- if ((OF_getprop(node, "i2c-device-id", &did, sizeof(did))) <= 0) {
+ if ((OF_getencprop(node, "i2c-device-id", &sc->device_id,
+ sizeof(sc->device_id))) <= 0) {
device_printf(dev, "missing i2c-device-id attribute in FDT\n");
return (ENXIO);
}
- sc->device_id = fdt32_to_cpu(did);
- TI_I2C_LOCK_INIT(sc);
-
- /* Get the memory resource for the register mapping */
+ /* Get the memory resource for the register mapping. */
rid = 0;
sc->sc_mem_res = bus_alloc_resource_any(dev, SYS_RES_MEMORY, &rid,
- RF_ACTIVE);
- if (sc->sc_mem_res == NULL)
- panic("%s: Cannot map registers", device_get_name(dev));
+ RF_ACTIVE);
+ if (sc->sc_mem_res == NULL) {
+ device_printf(dev, "Cannot map registers.\n");
+ return (ENXIO);
+ }
- /* Allocate an IRQ resource for the MMC controller */
+ /* Allocate our IRQ resource. */
rid = 0;
sc->sc_irq_res = bus_alloc_resource_any(dev, SYS_RES_IRQ, &rid,
- RF_ACTIVE | RF_SHAREABLE);
+ RF_ACTIVE | RF_SHAREABLE);
if (sc->sc_irq_res == NULL) {
- err = ENOMEM;
- goto out;
+ bus_release_resource(dev, SYS_RES_MEMORY, 0, sc->sc_mem_res);
+ device_printf(dev, "Cannot allocate interrupt.\n");
+ return (ENXIO);
}
- /* First we _must_ activate the H/W */
+ TI_I2C_LOCK_INIT(sc);
+
+ /* First of all, we _must_ activate the H/W. */
err = ti_i2c_activate(dev);
if (err) {
device_printf(dev, "ti_i2c_activate failed\n");
goto out;
}
- /* XXXOMAP3: FIXME get proper revision here */
/* Read the version number of the I2C module */
sc->sc_rev = ti_i2c_read_2(sc, I2C_REG_REVNB_HI) & 0xff;
- device_printf(dev, "I2C revision %d.%d\n", sc->sc_rev >> 4,
- sc->sc_rev & 0xf);
+ /* Get the fifo size. */
+ fifosz = ti_i2c_read_2(sc, I2C_REG_BUFSTAT);
+ fifosz >>= I2C_BUFSTAT_FIFODEPTH_SHIFT;
+ fifosz &= I2C_BUFSTAT_FIFODEPTH_MASK;
+
+ device_printf(dev, "I2C revision %d.%d FIFO size: %d bytes\n",
+ sc->sc_rev >> 4, sc->sc_rev & 0xf, 8 << fifosz);
- /* activate the interrupt */
+ /* Set the FIFO threshold to 5 for now. */
+ sc->sc_fifo_trsh = 5;
+
+ ctx = device_get_sysctl_ctx(dev);
+ tree = SYSCTL_CHILDREN(device_get_sysctl_tree(dev));
+ SYSCTL_ADD_PROC(ctx, tree, OID_AUTO, "i2c_clock",
+ CTLFLAG_RD | CTLTYPE_UINT | CTLFLAG_MPSAFE, dev, 0,
+ ti_i2c_sysctl_clk, "IU", "I2C bus clock");
+
+ /* Activate the interrupt. */
err = bus_setup_intr(dev, sc->sc_irq_res, INTR_TYPE_MISC | INTR_MPSAFE,
- NULL, ti_i2c_intr, sc, &sc->sc_irq_h);
+ NULL, ti_i2c_intr, sc, &sc->sc_irq_h);
if (err)
goto out;
- /* Attach to the iicbus */
- if ((sc->sc_iicbus = device_add_child(dev, "iicbus", -1)) == NULL)
+ /* Attach the iicbus. */
+ if ((sc->sc_iicbus = device_add_child(dev, "iicbus", -1)) == NULL) {
device_printf(dev, "could not allocate iicbus instance\n");
+ err = ENXIO;
+ goto out;
+ }
/* Probe and attach the iicbus */
bus_generic_attach(dev);
@@ -1108,42 +911,28 @@ out:
return (err);
}
-/**
- * ti_i2c_detach - detach function for the driver
- * @dev: i2c device handle
- *
- *
- *
- * LOCKING:
- *
- *
- * RETURNS:
- * Always returns 0
- */
static int
ti_i2c_detach(device_t dev)
{
- struct ti_i2c_softc *sc = device_get_softc(dev);
+ struct ti_i2c_softc *sc;
int rv;
+ sc = device_get_softc(dev);
ti_i2c_deactivate(dev);
-
- if (sc->sc_iicbus && (rv = device_delete_child(dev, sc->sc_iicbus)) != 0)
- return (rv);
-
TI_I2C_LOCK_DESTROY(sc);
+ if (sc->sc_iicbus &&
+ (rv = device_delete_child(dev, sc->sc_iicbus)) != 0)
+ return (rv);
return (0);
}
-
static phandle_t
ti_i2c_get_node(device_t bus, device_t dev)
{
- /*
- * Share controller node with iibus device
- */
- return ofw_bus_get_node(bus);
+
+ /* Share controller node with iibus device. */
+ return (ofw_bus_get_node(bus));
}
static device_method_t ti_i2c_methods[] = {
@@ -1157,9 +946,10 @@ static device_method_t ti_i2c_methods[] = {
/* iicbus interface */
DEVMETHOD(iicbus_callback, ti_i2c_callback),
- DEVMETHOD(iicbus_reset, ti_i2c_reset),
+ DEVMETHOD(iicbus_reset, ti_i2c_iicbus_reset),
DEVMETHOD(iicbus_transfer, ti_i2c_transfer),
- { 0, 0 }
+
+ DEVMETHOD_END
};
static driver_t ti_i2c_driver = {
@@ -1168,6 +958,8 @@ static driver_t ti_i2c_driver = {
sizeof(struct ti_i2c_softc),
};
+static devclass_t ti_i2c_devclass;
+
DRIVER_MODULE(ti_iic, simplebus, ti_i2c_driver, ti_i2c_devclass, 0, 0);
DRIVER_MODULE(iicbus, ti_iic, iicbus_driver, iicbus_devclass, 0, 0);
diff --git a/sys/arm/ti/ti_i2c.h b/sys/arm/ti/ti_i2c.h
index f569f87..d8dac60 100644
--- a/sys/arm/ti/ti_i2c.h
+++ b/sys/arm/ti/ti_i2c.h
@@ -69,7 +69,12 @@
#define I2C_STAT_NACK (1UL << 1)
#define I2C_STAT_AL (1UL << 0)
#define I2C_REG_SYSS 0x90
+#define I2C_SYSS_RDONE (1UL << 0)
#define I2C_REG_BUF 0x94
+#define I2C_BUF_RXFIFO_CLR (1UL << 14)
+#define I2C_BUF_TXFIFO_CLR (1UL << 6)
+#define I2C_BUF_RXTRSH_SHIFT 8
+#define I2C_BUF_TRSH_MASK 0x3f
#define I2C_REG_CNT 0x98
#define I2C_REG_DATA 0x9c
#define I2C_REG_CON 0xa4
@@ -91,10 +96,17 @@
#define I2C_REG_OA0 0xa8
#define I2C_REG_SA 0xac
#define I2C_REG_PSC 0xb0
+#define I2C_PSC_MASK 0xff
#define I2C_REG_SCLL 0xb4
+#define I2C_SCLL_MASK 0xff
+#define I2C_HSSCLL_SHIFT 8
#define I2C_REG_SCLH 0xb8
+#define I2C_SCLH_MASK 0xff
+#define I2C_HSSCLH_SHIFT 8
#define I2C_REG_SYSTEST 0xbc
#define I2C_REG_BUFSTAT 0xc0
+#define I2C_BUFSTAT_FIFODEPTH_MASK 0x3
+#define I2C_BUFSTAT_FIFODEPTH_SHIFT 14
#define I2C_REG_OA1 0xc4
#define I2C_REG_OA2 0xc8
#define I2C_REG_OA3 0xcc
@@ -107,9 +119,13 @@
#define I2C_REG_REVNB_LO 0x00
#define I2C_REG_REVNB_HI 0x04
#define I2C_REG_SYSC 0x10
+#define I2C_REG_SYSC_SRST (1UL << 1)
+#define I2C_REG_STATUS_RAW 0x24
+#define I2C_REG_STATUS 0x28
#define I2C_REG_IRQENABLE_SET 0x2C
#define I2C_REG_IRQENABLE_CLR 0x30
-
+#define I2C_CLK 96000000UL /* 96MHz */
+#define I2C_ICLK 12000000UL /* 12MHz */
#endif /* _TI_I2C_H_ */
OpenPOWER on IntegriCloud