* [PATCH][resend] iMX/MXC support for I2C
@ 2008-12-02 14:00 Darius
2008-12-02 14:04 ` Darius
` (2 more replies)
0 siblings, 3 replies; 21+ messages in thread
From: Darius @ 2008-12-02 14:00 UTC (permalink / raw)
To: linux-i2c-u79uwXL29TY76Z2rM5mHXA
Cc: linux-arm-kernel-xIg/pKzrS19vn6HldHNs0ANdhmdF6hFW
[-- Attachment #1: Type: text/plain, Size: 0 bytes --]
[-- Attachment #2: patch-i2c-imx --]
[-- Type: text/plain, Size: 21545 bytes --]
From: Darius Augulis <augulis.darius-Re5JQEeQqe8AvxtiuMwx3w@public.gmane.org>
Implementation of I2C Adapter/Algorithm Driver for I2C Bus integrated
in Freescale's i.MX/MXC processors. Tested only on MX1, but should work on MX2
and MX3 too, because all three processors have the same I2C controller.
Signed-off-by: Darius Augulis <augulis.darius-Re5JQEeQqe8AvxtiuMwx3w@public.gmane.org>
Index: linux-2.6.28-rc6/drivers/i2c/busses/i2c-imx.c
===================================================================
--- /dev/null
+++ linux-2.6.28-rc6/drivers/i2c/busses/i2c-imx.c
@@ -0,0 +1,656 @@
+/*
+ * Copyright (C) 2002 Motorola GSG-China
+ *
+ * This program is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU General Public License
+ * as published by the Free Software Foundation; either version 2
+ * of the License, or (at your option) any later version.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ * GNU General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program; if not, write to the Free Software
+ * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307,
+ * USA.
+ *
+ * Author:
+ * Darius Augulis, Teltonika Inc.
+ *
+ * Desc.:
+ * Implementation of I2C Adapter/Algorithm Driver
+ * for I2C Bus integrated in Freescale i.MX/MXC processors
+ *
+ * module parameters:
+ * - clkfreq:
+ * Sets the desired clock rate
+ * The default value is 100000
+ * Max value is 400000
+ *
+ * Derived from Motorola GSG China I2C example driver
+ *
+ * Copyright (C) 2005 Torsten Koschorrek <koschorrek at synertronixx.de
+ * Copyright (C) 2005 Matthias Blaschke <blaschke at synertronixx.de
+ * Copyright (C) 2007 RightHand Technologies, Inc.
+ * Copyright (C) 2008 Darius Augulis <darius.augulis at teltonika.lt>
+ *
+ */
+
+/** Includes *******************************************************************
+*******************************************************************************/
+
+#include <linux/init.h>
+#include <linux/kernel.h>
+#include <linux/module.h>
+#include <linux/errno.h>
+#include <linux/err.h>
+#include <linux/interrupt.h>
+#include <linux/delay.h>
+#include <linux/i2c.h>
+#include <linux/io.h>
+#include <linux/sched.h>
+#include <linux/platform_device.h>
+#include <linux/clk.h>
+
+#include <mach/irqs.h>
+#include <mach/hardware.h>
+#include <mach/i2c.h>
+
+/** Defines ********************************************************************
+*******************************************************************************/
+
+/* This will be the driver name the kernel reports */
+#define DRIVER_NAME "imx-i2c"
+
+/* Default values of module parameters */
+#define IMX_I2C_BIT_RATE 100000 /* 100kHz */
+
+/* Timeouts */
+#define I2C_IMX_TIME_BUSY 2000 /* loop count */
+#define I2C_IMX_TIME_ACK 2000 /* loop count */
+#define I2C_IMX_TIME_TRX 5 /* seconds */
+
+/* IMX I2C registers */
+#define IMX_I2C_IADR 0x00 /* i2c slave address */
+#define IMX_I2C_IFDR 0x04 /* i2c frequency divider */
+#define IMX_I2C_I2CR 0x08 /* i2c control */
+#define IMX_I2C_I2SR 0x0C /* i2c status */
+#define IMX_I2C_I2DR 0x10 /* i2c transfer data */
+
+/* Bits of IMX I2C registers */
+#define I2SR_RXAK 0x01
+#define I2SR_IIF 0x02
+#define I2SR_SRW 0x04
+#define I2SR_IAL 0x10
+#define I2SR_IBB 0x20
+#define I2SR_IAAS 0x40
+#define I2SR_ICF 0x80
+#define I2CR_RSTA 0x04
+#define I2CR_TXAK 0x08
+#define I2CR_MTX 0x10
+#define I2CR_MSTA 0x20
+#define I2CR_IIEN 0x40
+#define I2CR_IEN 0x80
+
+/** Variables ******************************************************************
+*******************************************************************************/
+
+static unsigned int clkfreq = IMX_I2C_BIT_RATE;
+static unsigned int disable_delay; /* Dummy delay */
+
+/*
+ * sorted list of clock divider, register value pairs
+ * taken from table 26-5, p.26-9, Freescale i.MX
+ * Integrated Portable System Processor Reference Manual
+ * Document Number: MC9328MXLRM, Rev. 5.1, 06/2007
+ *
+ * Duplicated divider values removed from list
+ */
+
+static u16 __initdata i2c_clk_div[50][2] = {
+ { 22, 0x20 }, { 24, 0x21 }, { 26, 0x22 }, { 28, 0x23 },
+ { 30, 0x00 }, { 32, 0x24 }, { 36, 0x25 }, { 40, 0x26 },
+ { 42, 0x03 }, { 44, 0x27 }, { 48, 0x28 }, { 52, 0x05 },
+ { 56, 0x29 }, { 60, 0x06 }, { 64, 0x2A }, { 72, 0x2B },
+ { 80, 0x2C }, { 88, 0x09 }, { 96, 0x2D }, { 104, 0x0A },
+ { 112, 0x2E }, { 128, 0x2F }, { 144, 0x0C }, { 160, 0x30 },
+ { 192, 0x31 }, { 224, 0x32 }, { 240, 0x0F }, { 256, 0x33 },
+ { 288, 0x10 }, { 320, 0x34 }, { 384, 0x35 }, { 448, 0x36 },
+ { 480, 0x13 }, { 512, 0x37 }, { 576, 0x14 }, { 640, 0x38 },
+ { 768, 0x39 }, { 896, 0x3A }, { 960, 0x17 }, { 1024, 0x3B },
+ { 1152, 0x18 }, { 1280, 0x3C }, { 1536, 0x3D }, { 1792, 0x3E },
+ { 1920, 0x1B }, { 2048, 0x3F }, { 2304, 0x1C }, { 2560, 0x1D },
+ { 3072, 0x1E }, { 3840, 0x1F }
+};
+
+struct imx_i2c_struct {
+ struct i2c_adapter adapter;
+ struct resource *res;
+ struct clk *clk;
+ void __iomem *base;
+ int irq;
+ wait_queue_head_t queue;
+ unsigned long i2csr;
+};
+
+/** Functions for IMX I2C adapter driver ***************************************
+*******************************************************************************/
+
+static int i2c_imx_bus_busy(struct imx_i2c_struct *i2c_imx)
+{
+ unsigned long orig_jiffies = jiffies;
+
+ dev_dbg(&i2c_imx->adapter.dev, "<%s>\n", __func__);
+
+ /* wait for bus not busy */
+ while (readb(i2c_imx->base + IMX_I2C_I2SR) & I2SR_IBB) {
+ if (signal_pending(current)) {
+ dev_dbg(&i2c_imx->adapter.dev,
+ "<%s> I2C Interrupted\n", __func__);
+ return -EINTR;
+ }
+ if (time_after(jiffies, orig_jiffies + HZ)) {
+ dev_dbg(&i2c_imx->adapter.dev,
+ "<%s> I2C bus is busy\n", __func__);
+ return -EIO;
+ }
+ schedule();
+ }
+
+ return 0;
+}
+
+static int i2c_imx_trx_complete(struct imx_i2c_struct *i2c_imx)
+{
+ int result;
+
+ result = wait_event_interruptible_timeout(i2c_imx->queue,
+ i2c_imx->i2csr & I2SR_IIF, I2C_IMX_TIME_TRX * HZ);
+
+ if (unlikely(result < 0)) {
+ dev_dbg(&i2c_imx->adapter.dev, "<%s> result < 0\n", __func__);
+ return result;
+ } else if (unlikely(!(i2c_imx->i2csr & I2SR_IIF))) {
+ dev_dbg(&i2c_imx->adapter.dev, "<%s> Timeout\n", __func__);
+ return -ETIMEDOUT;
+ }
+ dev_dbg(&i2c_imx->adapter.dev, "<%s> TRX complete\n", __func__);
+ i2c_imx->i2csr = 0;
+ return 0;
+}
+
+static int i2c_imx_acked(struct imx_i2c_struct *i2c_imx)
+{
+ unsigned long orig_jiffies = jiffies;
+
+ /* Wait for ack */
+ while (readb(i2c_imx->base + IMX_I2C_I2SR) & I2SR_RXAK) {
+ if (signal_pending(current)) {
+ dev_dbg(&i2c_imx->adapter.dev,
+ "<%s> I2C Interrupted\n", __func__);
+ return -EINTR;
+ }
+ if (time_after(jiffies, orig_jiffies + HZ)) {
+ dev_dbg(&i2c_imx->adapter.dev,
+ "<%s> No ACK\n", __func__);
+ return -EIO;
+ }
+ schedule();
+ }
+
+ dev_dbg(&i2c_imx->adapter.dev, "<%s> ACK received\n", __func__);
+ return 0; /* No ACK */
+}
+
+static void i2c_imx_start(struct imx_i2c_struct *i2c_imx)
+{
+ unsigned int temp = 0;
+
+ dev_dbg(&i2c_imx->adapter.dev, "<%s>\n", __func__);
+
+ /* Enable I2C controller */
+ writeb(I2CR_IEN, i2c_imx->base + IMX_I2C_I2CR);
+ /* Start I2C transaction */
+ temp = readb(i2c_imx->base + IMX_I2C_I2CR);
+ temp |= I2CR_MSTA;
+ writeb(temp, i2c_imx->base + IMX_I2C_I2CR);
+ temp |= I2CR_IIEN | I2CR_MTX | I2CR_TXAK;
+ writeb(temp, i2c_imx->base + IMX_I2C_I2CR);
+}
+
+static void i2c_imx_stop(struct imx_i2c_struct *i2c_imx)
+{
+ unsigned int temp = 0;
+
+ /* Stop I2C transaction */
+ dev_dbg(&i2c_imx->adapter.dev, "<%s>\n", __func__);
+ temp = readb(i2c_imx->base + IMX_I2C_I2CR);
+ temp &= ~I2CR_MSTA;
+ writeb(temp, i2c_imx->base + IMX_I2C_I2CR);
+ /* setup chip registers to defaults */
+ writeb(I2CR_IEN, i2c_imx->base + IMX_I2C_I2CR);
+ writeb(0, i2c_imx->base + IMX_I2C_I2SR);
+ /*
+ * This delay caused by an i.MXL hardware bug.
+ * If no (or too short) delay, no "STOP" bit will be generated.
+ */
+ udelay(disable_delay);
+ /* Disable I2C controller */
+ writeb(0, i2c_imx->base + IMX_I2C_I2CR);
+}
+
+static void __init i2c_imx_set_clk(struct imx_i2c_struct *i2c_imx,
+ unsigned int rate)
+{
+ unsigned int i2c_clk_rate;
+ unsigned int div;
+ int i;
+
+ /* Divider value calculation */
+ i2c_clk_rate = clk_get_rate(i2c_imx->clk);
+ div = (i2c_clk_rate + rate - 1) / rate;
+ if (div < i2c_clk_div[0][0])
+ i = 0;
+ else if (div > i2c_clk_div[ARRAY_SIZE(i2c_clk_div) - 1][0])
+ i = ARRAY_SIZE(i2c_clk_div) - 1;
+ else
+ for (i = 0; i2c_clk_div[i][0] < div; i++);
+
+ /* Write divider value to register */
+ writeb(i2c_clk_div[i][1], i2c_imx->base + IMX_I2C_IFDR);
+
+ /*
+ * There dummy delay is calculated.
+ * It should be about one I2C clock period long.
+ * This delay is used in I2C bus disable function
+ * to fix chip hardware bug.
+ */
+ disable_delay = (500000U * i2c_clk_div[i][0]
+ + (i2c_clk_rate / 2) - 1) / (i2c_clk_rate / 2);
+
+ /* dev_dbg() can't be used, because adapter is not yet registered */
+#ifdef CONFIG_I2C_DEBUG_BUS
+ printk(KERN_DEBUG "I2C: <%s> I2C_CLK=%d, REQ DIV=%d\n",
+ __func__, i2c_clk_rate, div);
+ printk(KERN_DEBUG "I2C: <%s> IFDR[IC]=0x%x, REAL DIV=%d\n",
+ __func__, i2c_clk_div[i][1], i2c_clk_div[i][0]);
+#endif
+}
+
+static irqreturn_t i2c_imx_isr(int irq, void *dev_id)
+{
+ struct imx_i2c_struct *i2c_imx = dev_id;
+ unsigned int temp;
+
+ temp = readb(i2c_imx->base + IMX_I2C_I2SR);
+ if (temp & I2SR_IIF) {
+ /* save status register */
+ i2c_imx->i2csr = temp;
+ temp &= ~I2SR_IIF;
+ writeb(temp, i2c_imx->base + IMX_I2C_I2SR);
+ wake_up_interruptible(&i2c_imx->queue);
+ }
+
+ return IRQ_HANDLED;
+}
+
+static int i2c_imx_write(struct imx_i2c_struct *i2c_imx, struct i2c_msg *msgs)
+{
+ int i, result;
+
+ dev_dbg(&i2c_imx->adapter.dev, "<%s> write slave address: addr=0x%x\n",
+ __func__, msgs->addr);
+
+ /* write slave address */
+ writeb(msgs->addr, i2c_imx->base + IMX_I2C_I2DR);
+ result = i2c_imx_trx_complete(i2c_imx);
+ if (result)
+ return result;
+ result = i2c_imx_acked(i2c_imx);
+ if (result)
+ return result;
+ dev_dbg(&i2c_imx->adapter.dev, "<%s> write data\n", __func__);
+
+ /* write data */
+ for (i = 0; i < msgs->len; i++) {
+ dev_dbg(&i2c_imx->adapter.dev,
+ "<%s> write byte: B%d=0x%X\n",
+ __func__, i, msgs->buf[i]);
+ writeb(msgs->buf[i], i2c_imx->base + IMX_I2C_I2DR);
+ result = i2c_imx_trx_complete(i2c_imx);
+ if (result)
+ return result;
+ result = i2c_imx_acked(i2c_imx);
+ if (result)
+ return result;
+ }
+ return 0;
+}
+
+static int i2c_imx_read(struct imx_i2c_struct *i2c_imx, struct i2c_msg *msgs)
+{
+ int i, result;
+ unsigned int temp;
+
+ dev_dbg(&i2c_imx->adapter.dev,
+ "<%s> write slave address: addr=0x%x\n",
+ __func__, msgs->addr | 0x01);
+
+ /* write slave address */
+ writeb(msgs->addr | 0x01, i2c_imx->base + IMX_I2C_I2DR);
+ result = i2c_imx_trx_complete(i2c_imx);
+ if (result)
+ return result;
+ result = i2c_imx_acked(i2c_imx);
+ if (result)
+ return result;
+
+ dev_dbg(&i2c_imx->adapter.dev, "<%s> setup bus\n", __func__);
+
+ /* setup bus to read data */
+ temp = readb(i2c_imx->base + IMX_I2C_I2CR);
+ temp &= ~I2CR_MTX;
+ if (msgs->len - 1)
+ temp &= ~I2CR_TXAK;
+ writeb(temp, i2c_imx->base + IMX_I2C_I2CR);
+ readb(i2c_imx->base + IMX_I2C_I2DR); /* dummy read */
+
+ dev_dbg(&i2c_imx->adapter.dev, "<%s> read data\n", __func__);
+
+ /* read data */
+ for (i = 0; i < msgs->len; i++) {
+ result = i2c_imx_trx_complete(i2c_imx);
+ if (result)
+ return result;
+ if (i == (msgs->len - 1)) {
+ dev_dbg(&i2c_imx->adapter.dev,
+ "<%s> clear MSTA\n", __func__);
+ temp = readb(i2c_imx->base + IMX_I2C_I2CR);
+ temp &= ~I2CR_MSTA;
+ writeb(temp, i2c_imx->base + IMX_I2C_I2CR);
+ } else if (i == (msgs->len - 2)) {
+ dev_dbg(&i2c_imx->adapter.dev,
+ "<%s> set TXAK\n", __func__);
+ temp = readb(i2c_imx->base + IMX_I2C_I2CR);
+ temp |= I2CR_TXAK;
+ writeb(temp, i2c_imx->base + IMX_I2C_I2CR);
+ }
+ msgs->buf[i] = readb(i2c_imx->base + IMX_I2C_I2DR);
+ dev_dbg(&i2c_imx->adapter.dev,
+ "<%s> read byte: B%d=0x%X\n",
+ __func__, i, msgs->buf[i]);
+ }
+ return 0;
+}
+
+static int i2c_imx_xfer(struct i2c_adapter *adapter,
+ struct i2c_msg *msgs, int num)
+{
+ unsigned int i, temp;
+ int result;
+ struct imx_i2c_struct *i2c_imx = i2c_get_adapdata(adapter);
+
+ dev_dbg(&i2c_imx->adapter.dev, "<%s>\n", __func__);
+
+ /* Check or i2c bus is not busy */
+ result = i2c_imx_bus_busy(i2c_imx);
+ if (result)
+ goto fail0;
+
+ /* Start I2C transfer */
+ i2c_imx_start(i2c_imx);
+
+ /* read/write data */
+ for (i = 0; i < num; i++) {
+ if (i) {
+ dev_dbg(&i2c_imx->adapter.dev,
+ "<%s> repeated start\n", __func__);
+ temp = readb(i2c_imx->base + IMX_I2C_I2CR);
+ temp |= I2CR_RSTA;
+ writeb(temp, i2c_imx->base + IMX_I2C_I2CR);
+ }
+ dev_dbg(&i2c_imx->adapter.dev,
+ "<%s> transfer message: %d\n", __func__, i);
+ /* write/read data */
+#ifdef CONFIG_I2C_DEBUG_BUS
+ temp = readb(i2c_imx->base + IMX_I2C_I2CR);
+ dev_dbg(&i2c_imx->adapter.dev, "<%s> CONTROL: IEN=%d, IIEN=%d, "
+ "MSTA=%d, MTX=%d, TXAK=%d, RSTA=%d\n", __func__,
+ (temp & I2CR_IEN ? 1 : 0), (temp & I2CR_IIEN ? 1 : 0),
+ (temp & I2CR_MSTA ? 1 : 0), (temp & I2CR_MTX ? 1 : 0),
+ (temp & I2CR_TXAK ? 1 : 0), (temp & I2CR_RSTA ? 1 : 0));
+ temp = readb(i2c_imx->base + IMX_I2C_I2SR);
+ dev_dbg(&i2c_imx->adapter.dev,
+ "<%s> STATUS: ICF=%d, IAAS=%d, IBB=%d, "
+ "IAL=%d, SRW=%d, IIF=%d, RXAK=%d\n", __func__,
+ (temp & I2SR_ICF ? 1 : 0), (temp & I2SR_IAAS ? 1 : 0),
+ (temp & I2SR_IBB ? 1 : 0), (temp & I2SR_IAL ? 1 : 0),
+ (temp & I2SR_SRW ? 1 : 0), (temp & I2SR_IIF ? 1 : 0),
+ (temp & I2SR_RXAK ? 1 : 0));
+#endif
+ if (msgs[i].flags & I2C_M_RD)
+ result = i2c_imx_read(i2c_imx, &msgs[i]);
+ else
+ result = i2c_imx_write(i2c_imx, &msgs[i]);
+ }
+
+fail0:
+ /* Stop I2C transfer */
+ i2c_imx_stop(i2c_imx);
+
+ dev_dbg(&i2c_imx->adapter.dev, "<%s> exit with: %s: %d\n", __func__,
+ (result < 0) ? "error" : "success msg",
+ (result < 0) ? result : num);
+ return (result < 0) ? result : num;
+}
+
+static u32 i2c_imx_func(struct i2c_adapter *adapter)
+{
+ return I2C_FUNC_I2C | I2C_FUNC_SMBUS_EMUL;
+}
+
+static struct i2c_algorithm i2c_imx_algo = {
+ .master_xfer = i2c_imx_xfer,
+ .functionality = i2c_imx_func,
+};
+
+static int __init i2c_imx_probe(struct platform_device *pdev)
+{
+ struct imx_i2c_struct *i2c_imx;
+ struct resource *res;
+ struct imxi2c_platform_data *pdata;
+ void __iomem *base;
+ int irq;
+ int res_size;
+ int ret;
+
+ dev_dbg(&pdev->dev, "<%s>\n", __func__);
+
+ res = platform_get_resource(pdev, IORESOURCE_MEM, 0);
+ if (!res) {
+ dev_err(&pdev->dev, "can't get device resources\n");
+ return -ENODEV;
+ }
+ irq = platform_get_irq(pdev, 0);
+ if (irq < 0) {
+ dev_err(&pdev->dev, "can't get irq number\n");
+ return -ENODEV;
+ }
+
+ pdata = pdev->dev.platform_data;
+ if (!pdata) {
+ dev_err(&pdev->dev, "I2C driver needs platform data\n");
+ return -ENODEV;
+ }
+
+ res_size = res->end - res->start + 1;
+ if (!request_mem_region(res->start, res_size, res->name)) {
+ dev_err(&pdev->dev, "can't allocate %d bytes at %d address\n",
+ res_size, res->start);
+ return -ENOMEM;
+ }
+
+ if (pdata->init) {
+ ret = pdata->init(&pdev->dev);
+ if (ret)
+ goto fail0;
+ }
+
+ base = ioremap(res->start, res_size);
+ if (!base) {
+ dev_err(&pdev->dev, "ioremap failed\n");
+ ret = -EIO;
+ goto fail1;
+ }
+
+ i2c_imx = kzalloc(sizeof(struct imx_i2c_struct), GFP_KERNEL);
+ if (!i2c_imx) {
+ dev_err(&pdev->dev, "can't allocate interface\n");
+ ret = -ENOMEM;
+ goto fail2;
+ }
+
+ /* Setup i2c_imx driver structure */
+ strcpy(i2c_imx->adapter.name, pdev->name);
+ i2c_imx->adapter.owner = THIS_MODULE;
+ i2c_imx->adapter.algo = &i2c_imx_algo;
+ i2c_imx->adapter.dev.parent = &pdev->dev;
+ i2c_imx->adapter.nr = pdev->id;
+ i2c_imx->irq = irq;
+ i2c_imx->base = base;
+ i2c_imx->res = res;
+
+ /* Get I2C clock */
+ i2c_imx->clk = clk_get(NULL, "i2c_clk");
+ if (IS_ERR(i2c_imx->clk)) {
+ ret = PTR_ERR(i2c_imx->clk);
+ dev_err(&pdev->dev, "can't get I2C clock\n");
+ goto fail3;
+ }
+ clk_enable(i2c_imx->clk);
+
+ /* Request IRQ */
+ ret = request_irq(i2c_imx->irq, i2c_imx_isr,
+ IRQF_DISABLED, pdev->name, i2c_imx);
+ if (ret) {
+ dev_err(&pdev->dev, "can't claim irq %d\n", i2c_imx->irq);
+ goto fail4;
+ }
+
+ /* Init queue */
+ init_waitqueue_head(&i2c_imx->queue);
+
+ /* Set up adapter data */
+ i2c_set_adapdata(&i2c_imx->adapter, i2c_imx);
+
+ /* Set up clock divider */
+ i2c_imx_set_clk(i2c_imx, clkfreq);
+
+ /* Set up chip registers to defaults */
+ writeb(0, i2c_imx->base + IMX_I2C_I2CR);
+ writeb(0, i2c_imx->base + IMX_I2C_I2SR);
+
+ /* Add I2C adapter */
+ ret = i2c_add_numbered_adapter(&i2c_imx->adapter);
+ if (ret < 0) {
+ dev_err(&pdev->dev, "registration failed\n");
+ goto fail5;
+ }
+
+ /* Set up platform driver data */
+ platform_set_drvdata(pdev, i2c_imx);
+
+ dev_dbg(&i2c_imx->adapter.dev, "claimed irq %d\n", i2c_imx->irq);
+ dev_dbg(&i2c_imx->adapter.dev, "device resources from 0x%x to 0x%x\n",
+ i2c_imx->res->start, i2c_imx->res->end);
+ dev_dbg(&i2c_imx->adapter.dev, "allocated %d bytes at 0x%x \n",
+ res_size, i2c_imx->res->start);
+ dev_dbg(&i2c_imx->adapter.dev, "adapter name: \"%s\"\n",
+ i2c_imx->adapter.name);
+ dev_dbg(&i2c_imx->adapter.dev, "IMX I2C adapter registered\n");
+
+ return 0; /* Return OK */
+
+fail5:
+ free_irq(i2c_imx->irq, i2c_imx);
+fail4:
+ clk_disable(i2c_imx->clk);
+ clk_put(i2c_imx->clk);
+fail3:
+ kfree(i2c_imx);
+fail2:
+ iounmap(base);
+fail1:
+ if (pdata->exit)
+ pdata->exit(&pdev->dev);
+fail0:
+ release_mem_region(res->start, res_size);
+ return ret; /* Return error number */
+}
+
+static int __exit i2c_imx_remove(struct platform_device *pdev)
+{
+ struct imx_i2c_struct *i2c_imx = platform_get_drvdata(pdev);
+ struct imxi2c_platform_data *pdata = pdev->dev.platform_data;
+
+ /* remove adapter */
+ dev_dbg(&i2c_imx->adapter.dev, "adapter removed\n");
+ i2c_del_adapter(&i2c_imx->adapter);
+ platform_set_drvdata(pdev, NULL);
+
+ /* free interrupt */
+ free_irq(i2c_imx->irq, i2c_imx);
+
+ /* setup chip registers to defaults */
+ writeb(0, i2c_imx->base + IMX_I2C_IADR);
+ writeb(0, i2c_imx->base + IMX_I2C_IFDR);
+ writeb(0, i2c_imx->base + IMX_I2C_I2CR);
+ writeb(0, i2c_imx->base + IMX_I2C_I2SR);
+
+ /* Shut down hardware */
+ if (pdata->exit)
+ pdata->exit(&pdev->dev);
+
+ /* Disable I2C clock */
+ clk_disable(i2c_imx->clk);
+ clk_put(i2c_imx->clk);
+
+ /* Release memory */
+ release_mem_region(i2c_imx->res->start,
+ i2c_imx->res->end - i2c_imx->res->start + 1);
+ iounmap(i2c_imx->base);
+ kfree(i2c_imx);
+ return 0;
+}
+
+static struct platform_driver i2c_imx_driver = {
+ .probe = i2c_imx_probe,
+ .remove = __exit_p(i2c_imx_remove),
+ .driver = {
+ .name = DRIVER_NAME,
+ .owner = THIS_MODULE,
+ }
+};
+
+static int __init i2c_adap_imx_init(void)
+{
+ return platform_driver_probe(&i2c_imx_driver, i2c_imx_probe);
+}
+
+static void __exit i2c_adap_imx_exit(void)
+{
+ platform_driver_unregister(&i2c_imx_driver);
+}
+
+module_init(i2c_adap_imx_init);
+module_exit(i2c_adap_imx_exit);
+
+module_param(clkfreq, uint, S_IRUGO);
+MODULE_PARM_DESC(clkfreq, "desired IMX I2C Clock Rate in Hz");
+
+MODULE_LICENSE("GPL");
+MODULE_AUTHOR("Darius Augulis");
+MODULE_DESCRIPTION("I2C adapter driver for IMX I2C bus");
+MODULE_ALIAS("platform:" DRIVER_NAME);
Index: linux-2.6.28-rc6/drivers/i2c/busses/Kconfig
===================================================================
--- linux-2.6.28-rc6.orig/drivers/i2c/busses/Kconfig
+++ linux-2.6.28-rc6/drivers/i2c/busses/Kconfig
@@ -355,6 +355,16 @@ config I2C_IBM_IIC
This driver can also be built as a module. If so, the module
will be called i2c-ibm_iic.
+config I2C_IMX
+ tristate "IMX I2C interface"
+ depends on ARCH_MXC
+ help
+ Say Y here if you want to use the IIC bus controller on
+ the Freescale i.MX/MXC processors.
+
+ This driver can also be built as a module. If so, the module
+ will be called i2c-imx.
+
config I2C_IOP3XX
tristate "Intel IOPx3xx and IXP4xx on-chip I2C interface"
depends on ARCH_IOP32X || ARCH_IOP33X || ARCH_IXP4XX || ARCH_IOP13XX
Index: linux-2.6.28-rc6/drivers/i2c/busses/Makefile
===================================================================
--- linux-2.6.28-rc6.orig/drivers/i2c/busses/Makefile
+++ linux-2.6.28-rc6/drivers/i2c/busses/Makefile
@@ -33,6 +33,7 @@ obj-$(CONFIG_I2C_DAVINCI) += i2c-davinci
obj-$(CONFIG_I2C_GPIO) += i2c-gpio.o
obj-$(CONFIG_I2C_HIGHLANDER) += i2c-highlander.o
obj-$(CONFIG_I2C_IBM_IIC) += i2c-ibm_iic.o
+obj-$(CONFIG_I2C_IMX) += i2c-imx.o
obj-$(CONFIG_I2C_IOP3XX) += i2c-iop3xx.o
obj-$(CONFIG_I2C_IXP2000) += i2c-ixp2000.o
obj-$(CONFIG_I2C_MPC) += i2c-mpc.o
Index: linux-2.6.28-rc6/arch/arm/plat-mxc/include/mach/i2c.h
===================================================================
--- /dev/null
+++ linux-2.6.28-rc6/arch/arm/plat-mxc/include/mach/i2c.h
@@ -0,0 +1,17 @@
+/*
+ * i2c.h - i.MX I2C driver header file
+ *
+ * Copyright (c) 2008, Darius Augulis <augulis.darius-Re5JQEeQqe8AvxtiuMwx3w@public.gmane.org>
+ *
+ * This file is released under the GPLv2
+ */
+
+#ifndef __ASM_ARCH_I2C_H_
+#define __ASM_ARCH_I2C_H_
+
+struct imxi2c_platform_data {
+ int (*init)(struct device *dev);
+ int (*exit)(struct device *dev);
+};
+
+#endif /* __ASM_ARCH_I2C_H_ */
^ permalink raw reply [flat|nested] 21+ messages in thread* Re: [PATCH][resend] iMX/MXC support for I2C 2008-12-02 14:00 [PATCH][resend] iMX/MXC support for I2C Darius @ 2008-12-02 14:04 ` Darius 2008-12-03 8:28 ` Jean Delvare 2008-12-03 11:17 ` Guennadi Liakhovetski 2008-12-03 12:56 ` Guennadi Liakhovetski 2 siblings, 1 reply; 21+ messages in thread From: Darius @ 2008-12-02 14:04 UTC (permalink / raw) To: linux-i2c-u79uwXL29TY76Z2rM5mHXA Cc: linux-arm-kernel-xIg/pKzrS19vn6HldHNs0ANdhmdF6hFW Please review and add this for 2.6.29 if it's ok. ^ permalink raw reply [flat|nested] 21+ messages in thread
* Re: [PATCH][resend] iMX/MXC support for I2C 2008-12-02 14:04 ` Darius @ 2008-12-03 8:28 ` Jean Delvare 0 siblings, 0 replies; 21+ messages in thread From: Jean Delvare @ 2008-12-03 8:28 UTC (permalink / raw) To: Ben Dooks Cc: Darius, linux-i2c-u79uwXL29TY76Z2rM5mHXA, linux-arm-kernel-xIg/pKzrS19vn6HldHNs0ANdhmdF6hFW On Tue, 02 Dec 2008 16:04:55 +0200, Darius wrote: > Please review and add this for 2.6.29 if it's ok. Ben, can you please take care of this? Thanks, -- Jean Delvare ^ permalink raw reply [flat|nested] 21+ messages in thread
* Re: [PATCH][resend] iMX/MXC support for I2C 2008-12-02 14:00 [PATCH][resend] iMX/MXC support for I2C Darius 2008-12-02 14:04 ` Darius @ 2008-12-03 11:17 ` Guennadi Liakhovetski [not found] ` <Pine.LNX.4.64.0812030917440.4717-0199iw4Nj15frtckUFj5Ag@public.gmane.org> 2008-12-03 12:56 ` Guennadi Liakhovetski 2 siblings, 1 reply; 21+ messages in thread From: Guennadi Liakhovetski @ 2008-12-03 11:17 UTC (permalink / raw) To: Darius Cc: linux-i2c-u79uwXL29TY76Z2rM5mHXA, linux-arm-kernel-xIg/pKzrS19vn6HldHNs0ANdhmdF6hFW The good news is - I was able to get this driver running on i.MX31 - somewhat... Now the comments: On Tue, 2 Dec 2008, Darius wrote: Commenting would have been easier, if you had inlined your patch... > From augulis.darius-Re5JQEeQqe8AvxtiuMwx3w@public.gmane.org Tue Dec 2 15:15:21 2008 > Date: Tue, 02 Dec 2008 16:00:20 +0200 > From: Darius <augulis.darius-Re5JQEeQqe8AvxtiuMwx3w@public.gmane.org> > To: linux-i2c-u79uwXL29TY76Z2rM5mHXA@public.gmane.org > Cc: linux-arm-kernel-xIg/pKzrS19vn6HldHNs0ANdhmdF6hFW@public.gmane.org > Subject: [PATCH][resend] iMX/MXC support for I2C > > > [ Empty or malformed message. Use "H" to see raw text. ] > > > [ Part 2: "Attached Text" ] > > From: Darius Augulis <augulis.darius-Re5JQEeQqe8AvxtiuMwx3w@public.gmane.org> > > Implementation of I2C Adapter/Algorithm Driver for I2C Bus integrated > in Freescale's i.MX/MXC processors. Tested only on MX1, but should work on MX2 > and MX3 too, because all three processors have the same I2C controller. > > Signed-off-by: Darius Augulis <augulis.darius-Re5JQEeQqe8AvxtiuMwx3w@public.gmane.org> > > Index: linux-2.6.28-rc6/drivers/i2c/busses/i2c-imx.c > =================================================================== > --- /dev/null > +++ linux-2.6.28-rc6/drivers/i2c/busses/i2c-imx.c > @@ -0,0 +1,656 @@ > +/* > + * Copyright (C) 2002 Motorola GSG-China > + * > + * This program is free software; you can redistribute it and/or > + * modify it under the terms of the GNU General Public License > + * as published by the Free Software Foundation; either version 2 > + * of the License, or (at your option) any later version. > + * > + * This program is distributed in the hope that it will be useful, > + * but WITHOUT ANY WARRANTY; without even the implied warranty of > + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the > + * GNU General Public License for more details. > + * > + * You should have received a copy of the GNU General Public License > + * along with this program; if not, write to the Free Software > + * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, > + * USA. > + * > + * Author: > + * Darius Augulis, Teltonika Inc. > + * > + * Desc.: > + * Implementation of I2C Adapter/Algorithm Driver > + * for I2C Bus integrated in Freescale i.MX/MXC processors > + * > + * module parameters: > + * - clkfreq: > + * Sets the desired clock rate > + * The default value is 100000 > + * Max value is 400000 > + * > + * Derived from Motorola GSG China I2C example driver > + * > + * Copyright (C) 2005 Torsten Koschorrek <koschorrek at synertronixx.de > + * Copyright (C) 2005 Matthias Blaschke <blaschke at synertronixx.de > + * Copyright (C) 2007 RightHand Technologies, Inc. > + * Copyright (C) 2008 Darius Augulis <darius.augulis at teltonika.lt> > + * > + */ > + > +/** Includes ******************************************************************* > +*******************************************************************************/ > + > +#include <linux/init.h> > +#include <linux/kernel.h> > +#include <linux/module.h> > +#include <linux/errno.h> > +#include <linux/err.h> > +#include <linux/interrupt.h> > +#include <linux/delay.h> > +#include <linux/i2c.h> > +#include <linux/io.h> > +#include <linux/sched.h> > +#include <linux/platform_device.h> > +#include <linux/clk.h> > + > +#include <mach/irqs.h> > +#include <mach/hardware.h> > +#include <mach/i2c.h> > + > +/** Defines ******************************************************************** > +*******************************************************************************/ > + > +/* This will be the driver name the kernel reports */ > +#define DRIVER_NAME "imx-i2c" > + > +/* Default values of module parameters */ > +#define IMX_I2C_BIT_RATE 100000 /* 100kHz */ > + > +/* Timeouts */ > +#define I2C_IMX_TIME_BUSY 2000 /* loop count */ > +#define I2C_IMX_TIME_ACK 2000 /* loop count */ > +#define I2C_IMX_TIME_TRX 5 /* seconds */ > + > +/* IMX I2C registers */ > +#define IMX_I2C_IADR 0x00 /* i2c slave address */ > +#define IMX_I2C_IFDR 0x04 /* i2c frequency divider */ > +#define IMX_I2C_I2CR 0x08 /* i2c control */ > +#define IMX_I2C_I2SR 0x0C /* i2c status */ > +#define IMX_I2C_I2DR 0x10 /* i2c transfer data */ > + > +/* Bits of IMX I2C registers */ > +#define I2SR_RXAK 0x01 > +#define I2SR_IIF 0x02 > +#define I2SR_SRW 0x04 > +#define I2SR_IAL 0x10 > +#define I2SR_IBB 0x20 > +#define I2SR_IAAS 0x40 > +#define I2SR_ICF 0x80 > +#define I2CR_RSTA 0x04 > +#define I2CR_TXAK 0x08 > +#define I2CR_MTX 0x10 > +#define I2CR_MSTA 0x20 > +#define I2CR_IIEN 0x40 > +#define I2CR_IEN 0x80 > + > +/** Variables ****************************************************************** > +*******************************************************************************/ > + > +static unsigned int clkfreq = IMX_I2C_BIT_RATE; > +static unsigned int disable_delay; /* Dummy delay */ > + > +/* > + * sorted list of clock divider, register value pairs > + * taken from table 26-5, p.26-9, Freescale i.MX > + * Integrated Portable System Processor Reference Manual > + * Document Number: MC9328MXLRM, Rev. 5.1, 06/2007 > + * > + * Duplicated divider values removed from list > + */ > + > +static u16 __initdata i2c_clk_div[50][2] = { > + { 22, 0x20 }, { 24, 0x21 }, { 26, 0x22 }, { 28, 0x23 }, > + { 30, 0x00 }, { 32, 0x24 }, { 36, 0x25 }, { 40, 0x26 }, > + { 42, 0x03 }, { 44, 0x27 }, { 48, 0x28 }, { 52, 0x05 }, > + { 56, 0x29 }, { 60, 0x06 }, { 64, 0x2A }, { 72, 0x2B }, > + { 80, 0x2C }, { 88, 0x09 }, { 96, 0x2D }, { 104, 0x0A }, > + { 112, 0x2E }, { 128, 0x2F }, { 144, 0x0C }, { 160, 0x30 }, > + { 192, 0x31 }, { 224, 0x32 }, { 240, 0x0F }, { 256, 0x33 }, > + { 288, 0x10 }, { 320, 0x34 }, { 384, 0x35 }, { 448, 0x36 }, > + { 480, 0x13 }, { 512, 0x37 }, { 576, 0x14 }, { 640, 0x38 }, > + { 768, 0x39 }, { 896, 0x3A }, { 960, 0x17 }, { 1024, 0x3B }, > + { 1152, 0x18 }, { 1280, 0x3C }, { 1536, 0x3D }, { 1792, 0x3E }, > + { 1920, 0x1B }, { 2048, 0x3F }, { 2304, 0x1C }, { 2560, 0x1D }, > + { 3072, 0x1E }, { 3840, 0x1F } > +}; > + > +struct imx_i2c_struct { > + struct i2c_adapter adapter; > + struct resource *res; > + struct clk *clk; > + void __iomem *base; > + int irq; > + wait_queue_head_t queue; > + unsigned long i2csr; > +}; > + > +/** Functions for IMX I2C adapter driver *************************************** > +*******************************************************************************/ > + > +static int i2c_imx_bus_busy(struct imx_i2c_struct *i2c_imx) > +{ > + unsigned long orig_jiffies = jiffies; > + > + dev_dbg(&i2c_imx->adapter.dev, "<%s>\n", __func__); > + > + /* wait for bus not busy */ > + while (readb(i2c_imx->base + IMX_I2C_I2SR) & I2SR_IBB) { The i2c-mxc driver I ported over from Freescale sources uses 16-bit access, and the datasheet defines registers to be 16-bit wide, although only low 8 bits are used. Whereas an i.MX datasheet (MC9328MX1) I have here defines i2c registers to be 32-bit wide. > + if (signal_pending(current)) { > + dev_dbg(&i2c_imx->adapter.dev, > + "<%s> I2C Interrupted\n", __func__); > + return -EINTR; > + } > + if (time_after(jiffies, orig_jiffies + HZ)) { > + dev_dbg(&i2c_imx->adapter.dev, > + "<%s> I2C bus is busy\n", __func__); > + return -EIO; > + } > + schedule(); For comparison: the MXC driver only waits for bus busy to clear for 600us at most. > + } > + > + return 0; > +} > + > +static int i2c_imx_trx_complete(struct imx_i2c_struct *i2c_imx) > +{ > + int result; > + > + result = wait_event_interruptible_timeout(i2c_imx->queue, > + i2c_imx->i2csr & I2SR_IIF, I2C_IMX_TIME_TRX * HZ); 5s is much too long! > + > + if (unlikely(result < 0)) { > + dev_dbg(&i2c_imx->adapter.dev, "<%s> result < 0\n", __func__); > + return result; > + } else if (unlikely(!(i2c_imx->i2csr & I2SR_IIF))) { > + dev_dbg(&i2c_imx->adapter.dev, "<%s> Timeout\n", __func__); > + return -ETIMEDOUT; > + } > + dev_dbg(&i2c_imx->adapter.dev, "<%s> TRX complete\n", __func__); > + i2c_imx->i2csr = 0; > + return 0; > +} > + > +static int i2c_imx_acked(struct imx_i2c_struct *i2c_imx) > +{ > + unsigned long orig_jiffies = jiffies; > + > + /* Wait for ack */ > + while (readb(i2c_imx->base + IMX_I2C_I2SR) & I2SR_RXAK) { > + if (signal_pending(current)) { > + dev_dbg(&i2c_imx->adapter.dev, > + "<%s> I2C Interrupted\n", __func__); > + return -EINTR; > + } > + if (time_after(jiffies, orig_jiffies + HZ)) { > + dev_dbg(&i2c_imx->adapter.dev, > + "<%s> No ACK\n", __func__); > + return -EIO; > + } > + schedule(); > + } 1s is way too long here! i2cdetect takes ages to complete. The mxc driver doesn't wait here at all - as soon as a tx is complete, it checks the ack status. Could you verify if this would work on your MX1 too? > + > + dev_dbg(&i2c_imx->adapter.dev, "<%s> ACK received\n", __func__); > + return 0; /* No ACK */ > +} > + > +static void i2c_imx_start(struct imx_i2c_struct *i2c_imx) > +{ > + unsigned int temp = 0; > + > + dev_dbg(&i2c_imx->adapter.dev, "<%s>\n", __func__); > + > + /* Enable I2C controller */ > + writeb(I2CR_IEN, i2c_imx->base + IMX_I2C_I2CR); > + /* Start I2C transaction */ > + temp = readb(i2c_imx->base + IMX_I2C_I2CR); > + temp |= I2CR_MSTA; > + writeb(temp, i2c_imx->base + IMX_I2C_I2CR); > + temp |= I2CR_IIEN | I2CR_MTX | I2CR_TXAK; > + writeb(temp, i2c_imx->base + IMX_I2C_I2CR); > +} > + > +static void i2c_imx_stop(struct imx_i2c_struct *i2c_imx) > +{ > + unsigned int temp = 0; > + > + /* Stop I2C transaction */ > + dev_dbg(&i2c_imx->adapter.dev, "<%s>\n", __func__); > + temp = readb(i2c_imx->base + IMX_I2C_I2CR); > + temp &= ~I2CR_MSTA; > + writeb(temp, i2c_imx->base + IMX_I2C_I2CR); > + /* setup chip registers to defaults */ > + writeb(I2CR_IEN, i2c_imx->base + IMX_I2C_I2CR); > + writeb(0, i2c_imx->base + IMX_I2C_I2SR); > + /* > + * This delay caused by an i.MXL hardware bug. > + * If no (or too short) delay, no "STOP" bit will be generated. > + */ > + udelay(disable_delay); > + /* Disable I2C controller */ > + writeb(0, i2c_imx->base + IMX_I2C_I2CR); > +} > + > +static void __init i2c_imx_set_clk(struct imx_i2c_struct *i2c_imx, > + unsigned int rate) > +{ > + unsigned int i2c_clk_rate; > + unsigned int div; > + int i; > + > + /* Divider value calculation */ > + i2c_clk_rate = clk_get_rate(i2c_imx->clk); > + div = (i2c_clk_rate + rate - 1) / rate; > + if (div < i2c_clk_div[0][0]) > + i = 0; > + else if (div > i2c_clk_div[ARRAY_SIZE(i2c_clk_div) - 1][0]) > + i = ARRAY_SIZE(i2c_clk_div) - 1; > + else > + for (i = 0; i2c_clk_div[i][0] < div; i++); > + > + /* Write divider value to register */ > + writeb(i2c_clk_div[i][1], i2c_imx->base + IMX_I2C_IFDR); > + > + /* > + * There dummy delay is calculated. > + * It should be about one I2C clock period long. > + * This delay is used in I2C bus disable function > + * to fix chip hardware bug. > + */ > + disable_delay = (500000U * i2c_clk_div[i][0] > + + (i2c_clk_rate / 2) - 1) / (i2c_clk_rate / 2); > + > + /* dev_dbg() can't be used, because adapter is not yet registered */ > +#ifdef CONFIG_I2C_DEBUG_BUS > + printk(KERN_DEBUG "I2C: <%s> I2C_CLK=%d, REQ DIV=%d\n", > + __func__, i2c_clk_rate, div); > + printk(KERN_DEBUG "I2C: <%s> IFDR[IC]=0x%x, REAL DIV=%d\n", > + __func__, i2c_clk_div[i][1], i2c_clk_div[i][0]); > +#endif > +} > + > +static irqreturn_t i2c_imx_isr(int irq, void *dev_id) > +{ > + struct imx_i2c_struct *i2c_imx = dev_id; > + unsigned int temp; > + > + temp = readb(i2c_imx->base + IMX_I2C_I2SR); > + if (temp & I2SR_IIF) { > + /* save status register */ > + i2c_imx->i2csr = temp; > + temp &= ~I2SR_IIF; > + writeb(temp, i2c_imx->base + IMX_I2C_I2SR); > + wake_up_interruptible(&i2c_imx->queue); > + } > + > + return IRQ_HANDLED; > +} > + > +static int i2c_imx_write(struct imx_i2c_struct *i2c_imx, struct i2c_msg *msgs) > +{ > + int i, result; > + > + dev_dbg(&i2c_imx->adapter.dev, "<%s> write slave address: addr=0x%x\n", > + __func__, msgs->addr); > + > + /* write slave address */ > + writeb(msgs->addr, i2c_imx->base + IMX_I2C_I2DR); This is wrong! You have to double the address before writing to the register. > + result = i2c_imx_trx_complete(i2c_imx); > + if (result) > + return result; > + result = i2c_imx_acked(i2c_imx); > + if (result) > + return result; > + dev_dbg(&i2c_imx->adapter.dev, "<%s> write data\n", __func__); > + > + /* write data */ > + for (i = 0; i < msgs->len; i++) { > + dev_dbg(&i2c_imx->adapter.dev, > + "<%s> write byte: B%d=0x%X\n", > + __func__, i, msgs->buf[i]); > + writeb(msgs->buf[i], i2c_imx->base + IMX_I2C_I2DR); > + result = i2c_imx_trx_complete(i2c_imx); > + if (result) > + return result; > + result = i2c_imx_acked(i2c_imx); > + if (result) > + return result; > + } > + return 0; > +} > + > +static int i2c_imx_read(struct imx_i2c_struct *i2c_imx, struct i2c_msg *msgs) > +{ > + int i, result; > + unsigned int temp; > + > + dev_dbg(&i2c_imx->adapter.dev, > + "<%s> write slave address: addr=0x%x\n", > + __func__, msgs->addr | 0x01); > + > + /* write slave address */ > + writeb(msgs->addr | 0x01, i2c_imx->base + IMX_I2C_I2DR); Same here - (msgs->addr << 1) | 0x01. > + result = i2c_imx_trx_complete(i2c_imx); > + if (result) > + return result; > + result = i2c_imx_acked(i2c_imx); > + if (result) > + return result; > + > + dev_dbg(&i2c_imx->adapter.dev, "<%s> setup bus\n", __func__); > + > + /* setup bus to read data */ > + temp = readb(i2c_imx->base + IMX_I2C_I2CR); > + temp &= ~I2CR_MTX; > + if (msgs->len - 1) > + temp &= ~I2CR_TXAK; > + writeb(temp, i2c_imx->base + IMX_I2C_I2CR); > + readb(i2c_imx->base + IMX_I2C_I2DR); /* dummy read */ > + > + dev_dbg(&i2c_imx->adapter.dev, "<%s> read data\n", __func__); > + > + /* read data */ > + for (i = 0; i < msgs->len; i++) { > + result = i2c_imx_trx_complete(i2c_imx); > + if (result) > + return result; > + if (i == (msgs->len - 1)) { > + dev_dbg(&i2c_imx->adapter.dev, > + "<%s> clear MSTA\n", __func__); > + temp = readb(i2c_imx->base + IMX_I2C_I2CR); > + temp &= ~I2CR_MSTA; > + writeb(temp, i2c_imx->base + IMX_I2C_I2CR); > + } else if (i == (msgs->len - 2)) { > + dev_dbg(&i2c_imx->adapter.dev, > + "<%s> set TXAK\n", __func__); > + temp = readb(i2c_imx->base + IMX_I2C_I2CR); > + temp |= I2CR_TXAK; > + writeb(temp, i2c_imx->base + IMX_I2C_I2CR); > + } > + msgs->buf[i] = readb(i2c_imx->base + IMX_I2C_I2DR); > + dev_dbg(&i2c_imx->adapter.dev, > + "<%s> read byte: B%d=0x%X\n", > + __func__, i, msgs->buf[i]); > + } > + return 0; > +} > + > +static int i2c_imx_xfer(struct i2c_adapter *adapter, > + struct i2c_msg *msgs, int num) > +{ > + unsigned int i, temp; > + int result; > + struct imx_i2c_struct *i2c_imx = i2c_get_adapdata(adapter); > + > + dev_dbg(&i2c_imx->adapter.dev, "<%s>\n", __func__); > + > + /* Check or i2c bus is not busy */ > + result = i2c_imx_bus_busy(i2c_imx); > + if (result) > + goto fail0; > + > + /* Start I2C transfer */ > + i2c_imx_start(i2c_imx); > + > + /* read/write data */ > + for (i = 0; i < num; i++) { > + if (i) { > + dev_dbg(&i2c_imx->adapter.dev, > + "<%s> repeated start\n", __func__); > + temp = readb(i2c_imx->base + IMX_I2C_I2CR); > + temp |= I2CR_RSTA; > + writeb(temp, i2c_imx->base + IMX_I2C_I2CR); > + } > + dev_dbg(&i2c_imx->adapter.dev, > + "<%s> transfer message: %d\n", __func__, i); > + /* write/read data */ > +#ifdef CONFIG_I2C_DEBUG_BUS > + temp = readb(i2c_imx->base + IMX_I2C_I2CR); > + dev_dbg(&i2c_imx->adapter.dev, "<%s> CONTROL: IEN=%d, IIEN=%d, " > + "MSTA=%d, MTX=%d, TXAK=%d, RSTA=%d\n", __func__, > + (temp & I2CR_IEN ? 1 : 0), (temp & I2CR_IIEN ? 1 : 0), > + (temp & I2CR_MSTA ? 1 : 0), (temp & I2CR_MTX ? 1 : 0), > + (temp & I2CR_TXAK ? 1 : 0), (temp & I2CR_RSTA ? 1 : 0)); > + temp = readb(i2c_imx->base + IMX_I2C_I2SR); > + dev_dbg(&i2c_imx->adapter.dev, > + "<%s> STATUS: ICF=%d, IAAS=%d, IBB=%d, " > + "IAL=%d, SRW=%d, IIF=%d, RXAK=%d\n", __func__, > + (temp & I2SR_ICF ? 1 : 0), (temp & I2SR_IAAS ? 1 : 0), > + (temp & I2SR_IBB ? 1 : 0), (temp & I2SR_IAL ? 1 : 0), > + (temp & I2SR_SRW ? 1 : 0), (temp & I2SR_IIF ? 1 : 0), > + (temp & I2SR_RXAK ? 1 : 0)); > +#endif > + if (msgs[i].flags & I2C_M_RD) > + result = i2c_imx_read(i2c_imx, &msgs[i]); > + else > + result = i2c_imx_write(i2c_imx, &msgs[i]); > + } > + > +fail0: > + /* Stop I2C transfer */ > + i2c_imx_stop(i2c_imx); > + > + dev_dbg(&i2c_imx->adapter.dev, "<%s> exit with: %s: %d\n", __func__, > + (result < 0) ? "error" : "success msg", > + (result < 0) ? result : num); > + return (result < 0) ? result : num; > +} > + > +static u32 i2c_imx_func(struct i2c_adapter *adapter) > +{ > + return I2C_FUNC_I2C | I2C_FUNC_SMBUS_EMUL; > +} > + > +static struct i2c_algorithm i2c_imx_algo = { > + .master_xfer = i2c_imx_xfer, > + .functionality = i2c_imx_func, > +}; > + > +static int __init i2c_imx_probe(struct platform_device *pdev) > +{ > + struct imx_i2c_struct *i2c_imx; > + struct resource *res; > + struct imxi2c_platform_data *pdata; > + void __iomem *base; > + int irq; > + int res_size; > + int ret; > + > + dev_dbg(&pdev->dev, "<%s>\n", __func__); > + > + res = platform_get_resource(pdev, IORESOURCE_MEM, 0); > + if (!res) { > + dev_err(&pdev->dev, "can't get device resources\n"); > + return -ENODEV; > + } > + irq = platform_get_irq(pdev, 0); > + if (irq < 0) { > + dev_err(&pdev->dev, "can't get irq number\n"); > + return -ENODEV; > + } > + > + pdata = pdev->dev.platform_data; > + if (!pdata) { > + dev_err(&pdev->dev, "I2C driver needs platform data\n"); > + return -ENODEV; > + } > + > + res_size = res->end - res->start + 1; You can use resource_size() > + if (!request_mem_region(res->start, res_size, res->name)) { > + dev_err(&pdev->dev, "can't allocate %d bytes at %d address\n", > + res_size, res->start); > + return -ENOMEM; > + } I was once told, one doesn't need request_mem_region() on regions from platform data resources - this is already done implicitly. > + > + if (pdata->init) { > + ret = pdata->init(&pdev->dev); > + if (ret) > + goto fail0; > + } > + > + base = ioremap(res->start, res_size); > + if (!base) { > + dev_err(&pdev->dev, "ioremap failed\n"); > + ret = -EIO; > + goto fail1; > + } > + > + i2c_imx = kzalloc(sizeof(struct imx_i2c_struct), GFP_KERNEL); > + if (!i2c_imx) { > + dev_err(&pdev->dev, "can't allocate interface\n"); > + ret = -ENOMEM; > + goto fail2; > + } > + > + /* Setup i2c_imx driver structure */ > + strcpy(i2c_imx->adapter.name, pdev->name); > + i2c_imx->adapter.owner = THIS_MODULE; > + i2c_imx->adapter.algo = &i2c_imx_algo; > + i2c_imx->adapter.dev.parent = &pdev->dev; > + i2c_imx->adapter.nr = pdev->id; > + i2c_imx->irq = irq; > + i2c_imx->base = base; > + i2c_imx->res = res; > + > + /* Get I2C clock */ > + i2c_imx->clk = clk_get(NULL, "i2c_clk"); There can be several i2c busses on the system, so you want: + i2c_imx->clk = clk_get(&pdev->dev, "i2c_clk"); > + if (IS_ERR(i2c_imx->clk)) { > + ret = PTR_ERR(i2c_imx->clk); > + dev_err(&pdev->dev, "can't get I2C clock\n"); > + goto fail3; > + } > + clk_enable(i2c_imx->clk); > + > + /* Request IRQ */ > + ret = request_irq(i2c_imx->irq, i2c_imx_isr, > + IRQF_DISABLED, pdev->name, i2c_imx); > + if (ret) { > + dev_err(&pdev->dev, "can't claim irq %d\n", i2c_imx->irq); > + goto fail4; > + } > + > + /* Init queue */ > + init_waitqueue_head(&i2c_imx->queue); > + > + /* Set up adapter data */ > + i2c_set_adapdata(&i2c_imx->adapter, i2c_imx); > + > + /* Set up clock divider */ > + i2c_imx_set_clk(i2c_imx, clkfreq); > + > + /* Set up chip registers to defaults */ > + writeb(0, i2c_imx->base + IMX_I2C_I2CR); > + writeb(0, i2c_imx->base + IMX_I2C_I2SR); > + > + /* Add I2C adapter */ > + ret = i2c_add_numbered_adapter(&i2c_imx->adapter); > + if (ret < 0) { > + dev_err(&pdev->dev, "registration failed\n"); > + goto fail5; > + } > + > + /* Set up platform driver data */ > + platform_set_drvdata(pdev, i2c_imx); > + > + dev_dbg(&i2c_imx->adapter.dev, "claimed irq %d\n", i2c_imx->irq); > + dev_dbg(&i2c_imx->adapter.dev, "device resources from 0x%x to 0x%x\n", > + i2c_imx->res->start, i2c_imx->res->end); > + dev_dbg(&i2c_imx->adapter.dev, "allocated %d bytes at 0x%x \n", > + res_size, i2c_imx->res->start); > + dev_dbg(&i2c_imx->adapter.dev, "adapter name: \"%s\"\n", > + i2c_imx->adapter.name); > + dev_dbg(&i2c_imx->adapter.dev, "IMX I2C adapter registered\n"); > + > + return 0; /* Return OK */ > + > +fail5: > + free_irq(i2c_imx->irq, i2c_imx); > +fail4: > + clk_disable(i2c_imx->clk); > + clk_put(i2c_imx->clk); > +fail3: > + kfree(i2c_imx); > +fail2: > + iounmap(base); > +fail1: > + if (pdata->exit) > + pdata->exit(&pdev->dev); > +fail0: > + release_mem_region(res->start, res_size); > + return ret; /* Return error number */ > +} > + > +static int __exit i2c_imx_remove(struct platform_device *pdev) > +{ > + struct imx_i2c_struct *i2c_imx = platform_get_drvdata(pdev); > + struct imxi2c_platform_data *pdata = pdev->dev.platform_data; > + > + /* remove adapter */ > + dev_dbg(&i2c_imx->adapter.dev, "adapter removed\n"); > + i2c_del_adapter(&i2c_imx->adapter); > + platform_set_drvdata(pdev, NULL); > + > + /* free interrupt */ > + free_irq(i2c_imx->irq, i2c_imx); > + > + /* setup chip registers to defaults */ > + writeb(0, i2c_imx->base + IMX_I2C_IADR); > + writeb(0, i2c_imx->base + IMX_I2C_IFDR); > + writeb(0, i2c_imx->base + IMX_I2C_I2CR); > + writeb(0, i2c_imx->base + IMX_I2C_I2SR); > + > + /* Shut down hardware */ > + if (pdata->exit) > + pdata->exit(&pdev->dev); > + > + /* Disable I2C clock */ > + clk_disable(i2c_imx->clk); > + clk_put(i2c_imx->clk); > + > + /* Release memory */ > + release_mem_region(i2c_imx->res->start, > + i2c_imx->res->end - i2c_imx->res->start + 1); > + iounmap(i2c_imx->base); > + kfree(i2c_imx); > + return 0; > +} > + > +static struct platform_driver i2c_imx_driver = { > + .probe = i2c_imx_probe, > + .remove = __exit_p(i2c_imx_remove), > + .driver = { > + .name = DRIVER_NAME, > + .owner = THIS_MODULE, > + } > +}; > + > +static int __init i2c_adap_imx_init(void) > +{ > + return platform_driver_probe(&i2c_imx_driver, i2c_imx_probe); > +} > + > +static void __exit i2c_adap_imx_exit(void) > +{ > + platform_driver_unregister(&i2c_imx_driver); > +} > + > +module_init(i2c_adap_imx_init); > +module_exit(i2c_adap_imx_exit); > + > +module_param(clkfreq, uint, S_IRUGO); > +MODULE_PARM_DESC(clkfreq, "desired IMX I2C Clock Rate in Hz"); Making clkfreq a module parameter you force the same frequency on all i2c busses. On my i.MX31 system 2 busses are internal and one goes to a connector, to which a camera is connected. This third bus can only handle a lower frequency, which, however, doesn't mean we also have to throttle the other two busses. Can we put this into platform data? > + > +MODULE_LICENSE("GPL"); > +MODULE_AUTHOR("Darius Augulis"); > +MODULE_DESCRIPTION("I2C adapter driver for IMX I2C bus"); > +MODULE_ALIAS("platform:" DRIVER_NAME); > Index: linux-2.6.28-rc6/drivers/i2c/busses/Kconfig > =================================================================== > --- linux-2.6.28-rc6.orig/drivers/i2c/busses/Kconfig > +++ linux-2.6.28-rc6/drivers/i2c/busses/Kconfig > @@ -355,6 +355,16 @@ config I2C_IBM_IIC > This driver can also be built as a module. If so, the module > will be called i2c-ibm_iic. > > +config I2C_IMX > + tristate "IMX I2C interface" > + depends on ARCH_MXC > + help > + Say Y here if you want to use the IIC bus controller on > + the Freescale i.MX/MXC processors. > + > + This driver can also be built as a module. If so, the module > + will be called i2c-imx. > + > config I2C_IOP3XX > tristate "Intel IOPx3xx and IXP4xx on-chip I2C interface" > depends on ARCH_IOP32X || ARCH_IOP33X || ARCH_IXP4XX || ARCH_IOP13XX > Index: linux-2.6.28-rc6/drivers/i2c/busses/Makefile > =================================================================== > --- linux-2.6.28-rc6.orig/drivers/i2c/busses/Makefile > +++ linux-2.6.28-rc6/drivers/i2c/busses/Makefile > @@ -33,6 +33,7 @@ obj-$(CONFIG_I2C_DAVINCI) += i2c-davinci > obj-$(CONFIG_I2C_GPIO) += i2c-gpio.o > obj-$(CONFIG_I2C_HIGHLANDER) += i2c-highlander.o > obj-$(CONFIG_I2C_IBM_IIC) += i2c-ibm_iic.o > +obj-$(CONFIG_I2C_IMX) += i2c-imx.o > obj-$(CONFIG_I2C_IOP3XX) += i2c-iop3xx.o > obj-$(CONFIG_I2C_IXP2000) += i2c-ixp2000.o > obj-$(CONFIG_I2C_MPC) += i2c-mpc.o > Index: linux-2.6.28-rc6/arch/arm/plat-mxc/include/mach/i2c.h > =================================================================== > --- /dev/null > +++ linux-2.6.28-rc6/arch/arm/plat-mxc/include/mach/i2c.h > @@ -0,0 +1,17 @@ > +/* > + * i2c.h - i.MX I2C driver header file > + * > + * Copyright (c) 2008, Darius Augulis <augulis.darius-Re5JQEeQqe8AvxtiuMwx3w@public.gmane.org> > + * > + * This file is released under the GPLv2 > + */ > + > +#ifndef __ASM_ARCH_I2C_H_ > +#define __ASM_ARCH_I2C_H_ > + > +struct imxi2c_platform_data { > + int (*init)(struct device *dev); > + int (*exit)(struct device *dev); What are you going to use .exit() for? Is it really needed? Even if it is, it can easily return void I guess? > +}; > + > +#endif /* __ASM_ARCH_I2C_H_ */ Thanks Guennadi --- Guennadi Liakhovetski, Ph.D. Freelance Open-Source Software Developer ^ permalink raw reply [flat|nested] 21+ messages in thread
[parent not found: <Pine.LNX.4.64.0812030917440.4717-0199iw4Nj15frtckUFj5Ag@public.gmane.org>]
* Re: [PATCH][resend] iMX/MXC support for I2C [not found] ` <Pine.LNX.4.64.0812030917440.4717-0199iw4Nj15frtckUFj5Ag@public.gmane.org> @ 2008-12-03 12:52 ` Darius 2008-12-03 15:19 ` Mark Brown 2008-12-03 21:19 ` Guennadi Liakhovetski 2008-12-03 23:43 ` Ben Dooks 1 sibling, 2 replies; 21+ messages in thread From: Darius @ 2008-12-03 12:52 UTC (permalink / raw) To: linux-i2c-u79uwXL29TY76Z2rM5mHXA Cc: linux-arm-kernel-xIg/pKzrS19vn6HldHNs0ANdhmdF6hFW Guennadi Liakhovetski wrote: > The good news is - I was able to get this driver running on i.MX31 - > somewhat... Now the comments: Thank you for review and test on MX3! > > On Tue, 2 Dec 2008, Darius wrote: > > Commenting would have been easier, if you had inlined your patch... > >> From augulis.darius-Re5JQEeQqe8AvxtiuMwx3w@public.gmane.org Tue Dec 2 15:15:21 2008 >> Date: Tue, 02 Dec 2008 16:00:20 +0200 >> From: Darius <augulis.darius-Re5JQEeQqe8AvxtiuMwx3w@public.gmane.org> >> To: linux-i2c-u79uwXL29TY76Z2rM5mHXA@public.gmane.org >> Cc: linux-arm-kernel-xIg/pKzrS19vn6HldHNs0ANdhmdF6hFW@public.gmane.org >> Subject: [PATCH][resend] iMX/MXC support for I2C >> >> >> [ Empty or malformed message. Use "H" to see raw text. ] >> >> >> [ Part 2: "Attached Text" ] >> >> From: Darius Augulis <augulis.darius-Re5JQEeQqe8AvxtiuMwx3w@public.gmane.org> >> >> Implementation of I2C Adapter/Algorithm Driver for I2C Bus integrated >> in Freescale's i.MX/MXC processors. Tested only on MX1, but should work on MX2 >> and MX3 too, because all three processors have the same I2C controller. >> >> Signed-off-by: Darius Augulis <augulis.darius-Re5JQEeQqe8AvxtiuMwx3w@public.gmane.org> >> >> Index: linux-2.6.28-rc6/drivers/i2c/busses/i2c-imx.c >> =================================================================== >> --- /dev/null >> +++ linux-2.6.28-rc6/drivers/i2c/busses/i2c-imx.c >> @@ -0,0 +1,656 @@ >> +/* >> + * Copyright (C) 2002 Motorola GSG-China >> + * >> + * This program is free software; you can redistribute it and/or >> + * modify it under the terms of the GNU General Public License >> + * as published by the Free Software Foundation; either version 2 >> + * of the License, or (at your option) any later version. >> + * >> + * This program is distributed in the hope that it will be useful, >> + * but WITHOUT ANY WARRANTY; without even the implied warranty of >> + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the >> + * GNU General Public License for more details. >> + * >> + * You should have received a copy of the GNU General Public License >> + * along with this program; if not, write to the Free Software >> + * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, >> + * USA. >> + * >> + * Author: >> + * Darius Augulis, Teltonika Inc. >> + * >> + * Desc.: >> + * Implementation of I2C Adapter/Algorithm Driver >> + * for I2C Bus integrated in Freescale i.MX/MXC processors >> + * >> + * module parameters: >> + * - clkfreq: >> + * Sets the desired clock rate >> + * The default value is 100000 >> + * Max value is 400000 >> + * >> + * Derived from Motorola GSG China I2C example driver >> + * >> + * Copyright (C) 2005 Torsten Koschorrek <koschorrek at synertronixx.de >> + * Copyright (C) 2005 Matthias Blaschke <blaschke at synertronixx.de >> + * Copyright (C) 2007 RightHand Technologies, Inc. >> + * Copyright (C) 2008 Darius Augulis <darius.augulis at teltonika.lt> >> + * >> + */ >> + >> +/** Includes ******************************************************************* >> +*******************************************************************************/ >> + >> +#include <linux/init.h> >> +#include <linux/kernel.h> >> +#include <linux/module.h> >> +#include <linux/errno.h> >> +#include <linux/err.h> >> +#include <linux/interrupt.h> >> +#include <linux/delay.h> >> +#include <linux/i2c.h> >> +#include <linux/io.h> >> +#include <linux/sched.h> >> +#include <linux/platform_device.h> >> +#include <linux/clk.h> >> + >> +#include <mach/irqs.h> >> +#include <mach/hardware.h> >> +#include <mach/i2c.h> >> + >> +/** Defines ******************************************************************** >> +*******************************************************************************/ >> + >> +/* This will be the driver name the kernel reports */ >> +#define DRIVER_NAME "imx-i2c" >> + >> +/* Default values of module parameters */ >> +#define IMX_I2C_BIT_RATE 100000 /* 100kHz */ >> + >> +/* Timeouts */ >> +#define I2C_IMX_TIME_BUSY 2000 /* loop count */ >> +#define I2C_IMX_TIME_ACK 2000 /* loop count */ >> +#define I2C_IMX_TIME_TRX 5 /* seconds */ >> + >> +/* IMX I2C registers */ >> +#define IMX_I2C_IADR 0x00 /* i2c slave address */ >> +#define IMX_I2C_IFDR 0x04 /* i2c frequency divider */ >> +#define IMX_I2C_I2CR 0x08 /* i2c control */ >> +#define IMX_I2C_I2SR 0x0C /* i2c status */ >> +#define IMX_I2C_I2DR 0x10 /* i2c transfer data */ >> + >> +/* Bits of IMX I2C registers */ >> +#define I2SR_RXAK 0x01 >> +#define I2SR_IIF 0x02 >> +#define I2SR_SRW 0x04 >> +#define I2SR_IAL 0x10 >> +#define I2SR_IBB 0x20 >> +#define I2SR_IAAS 0x40 >> +#define I2SR_ICF 0x80 >> +#define I2CR_RSTA 0x04 >> +#define I2CR_TXAK 0x08 >> +#define I2CR_MTX 0x10 >> +#define I2CR_MSTA 0x20 >> +#define I2CR_IIEN 0x40 >> +#define I2CR_IEN 0x80 >> + >> +/** Variables ****************************************************************** >> +*******************************************************************************/ >> + >> +static unsigned int clkfreq = IMX_I2C_BIT_RATE; >> +static unsigned int disable_delay; /* Dummy delay */ >> + >> +/* >> + * sorted list of clock divider, register value pairs >> + * taken from table 26-5, p.26-9, Freescale i.MX >> + * Integrated Portable System Processor Reference Manual >> + * Document Number: MC9328MXLRM, Rev. 5.1, 06/2007 >> + * >> + * Duplicated divider values removed from list >> + */ >> + >> +static u16 __initdata i2c_clk_div[50][2] = { >> + { 22, 0x20 }, { 24, 0x21 }, { 26, 0x22 }, { 28, 0x23 }, >> + { 30, 0x00 }, { 32, 0x24 }, { 36, 0x25 }, { 40, 0x26 }, >> + { 42, 0x03 }, { 44, 0x27 }, { 48, 0x28 }, { 52, 0x05 }, >> + { 56, 0x29 }, { 60, 0x06 }, { 64, 0x2A }, { 72, 0x2B }, >> + { 80, 0x2C }, { 88, 0x09 }, { 96, 0x2D }, { 104, 0x0A }, >> + { 112, 0x2E }, { 128, 0x2F }, { 144, 0x0C }, { 160, 0x30 }, >> + { 192, 0x31 }, { 224, 0x32 }, { 240, 0x0F }, { 256, 0x33 }, >> + { 288, 0x10 }, { 320, 0x34 }, { 384, 0x35 }, { 448, 0x36 }, >> + { 480, 0x13 }, { 512, 0x37 }, { 576, 0x14 }, { 640, 0x38 }, >> + { 768, 0x39 }, { 896, 0x3A }, { 960, 0x17 }, { 1024, 0x3B }, >> + { 1152, 0x18 }, { 1280, 0x3C }, { 1536, 0x3D }, { 1792, 0x3E }, >> + { 1920, 0x1B }, { 2048, 0x3F }, { 2304, 0x1C }, { 2560, 0x1D }, >> + { 3072, 0x1E }, { 3840, 0x1F } >> +}; >> + >> +struct imx_i2c_struct { >> + struct i2c_adapter adapter; >> + struct resource *res; >> + struct clk *clk; >> + void __iomem *base; >> + int irq; >> + wait_queue_head_t queue; >> + unsigned long i2csr; >> +}; >> + >> +/** Functions for IMX I2C adapter driver *************************************** >> +*******************************************************************************/ >> + >> +static int i2c_imx_bus_busy(struct imx_i2c_struct *i2c_imx) >> +{ >> + unsigned long orig_jiffies = jiffies; >> + >> + dev_dbg(&i2c_imx->adapter.dev, "<%s>\n", __func__); >> + >> + /* wait for bus not busy */ >> + while (readb(i2c_imx->base + IMX_I2C_I2SR) & I2SR_IBB) { > > The i2c-mxc driver I ported over from Freescale sources uses 16-bit > access, and the datasheet defines registers to be 16-bit wide, although > only low 8 bits are used. Whereas an i.MX datasheet (MC9328MX1) I have > here defines i2c registers to be 32-bit wide. yes, it's no difference how to access registers - in 8bit or 16bit wide, because only 8 lower bits are used. > >> + if (signal_pending(current)) { >> + dev_dbg(&i2c_imx->adapter.dev, >> + "<%s> I2C Interrupted\n", __func__); >> + return -EINTR; >> + } >> + if (time_after(jiffies, orig_jiffies + HZ)) { >> + dev_dbg(&i2c_imx->adapter.dev, >> + "<%s> I2C bus is busy\n", __func__); >> + return -EIO; >> + } >> + schedule(); > > For comparison: the MXC driver only waits for bus busy to clear for 600us > at most. but there is shedule() used, no impact to performace. but I will decrease timeouts. > >> + } >> + >> + return 0; >> +} >> + >> +static int i2c_imx_trx_complete(struct imx_i2c_struct *i2c_imx) >> +{ >> + int result; >> + >> + result = wait_event_interruptible_timeout(i2c_imx->queue, >> + i2c_imx->i2csr & I2SR_IIF, I2C_IMX_TIME_TRX * HZ); > > 5s is much too long! how much? 600us? > >> + >> + if (unlikely(result < 0)) { >> + dev_dbg(&i2c_imx->adapter.dev, "<%s> result < 0\n", __func__); >> + return result; >> + } else if (unlikely(!(i2c_imx->i2csr & I2SR_IIF))) { >> + dev_dbg(&i2c_imx->adapter.dev, "<%s> Timeout\n", __func__); >> + return -ETIMEDOUT; >> + } >> + dev_dbg(&i2c_imx->adapter.dev, "<%s> TRX complete\n", __func__); >> + i2c_imx->i2csr = 0; >> + return 0; >> +} >> + >> +static int i2c_imx_acked(struct imx_i2c_struct *i2c_imx) >> +{ >> + unsigned long orig_jiffies = jiffies; >> + >> + /* Wait for ack */ >> + while (readb(i2c_imx->base + IMX_I2C_I2SR) & I2SR_RXAK) { >> + if (signal_pending(current)) { >> + dev_dbg(&i2c_imx->adapter.dev, >> + "<%s> I2C Interrupted\n", __func__); >> + return -EINTR; >> + } >> + if (time_after(jiffies, orig_jiffies + HZ)) { >> + dev_dbg(&i2c_imx->adapter.dev, >> + "<%s> No ACK\n", __func__); >> + return -EIO; >> + } >> + schedule(); >> + } > > 1s is way too long here! i2cdetect takes ages to complete. The mxc driver > doesn't wait here at all - as soon as a tx is complete, it checks the ack > status. Could you verify if this would work on your MX1 too? I will try. > >> + >> + dev_dbg(&i2c_imx->adapter.dev, "<%s> ACK received\n", __func__); >> + return 0; /* No ACK */ >> +} >> + >> +static void i2c_imx_start(struct imx_i2c_struct *i2c_imx) >> +{ >> + unsigned int temp = 0; >> + >> + dev_dbg(&i2c_imx->adapter.dev, "<%s>\n", __func__); >> + >> + /* Enable I2C controller */ >> + writeb(I2CR_IEN, i2c_imx->base + IMX_I2C_I2CR); >> + /* Start I2C transaction */ >> + temp = readb(i2c_imx->base + IMX_I2C_I2CR); >> + temp |= I2CR_MSTA; >> + writeb(temp, i2c_imx->base + IMX_I2C_I2CR); >> + temp |= I2CR_IIEN | I2CR_MTX | I2CR_TXAK; >> + writeb(temp, i2c_imx->base + IMX_I2C_I2CR); >> +} >> + >> +static void i2c_imx_stop(struct imx_i2c_struct *i2c_imx) >> +{ >> + unsigned int temp = 0; >> + >> + /* Stop I2C transaction */ >> + dev_dbg(&i2c_imx->adapter.dev, "<%s>\n", __func__); >> + temp = readb(i2c_imx->base + IMX_I2C_I2CR); >> + temp &= ~I2CR_MSTA; >> + writeb(temp, i2c_imx->base + IMX_I2C_I2CR); >> + /* setup chip registers to defaults */ >> + writeb(I2CR_IEN, i2c_imx->base + IMX_I2C_I2CR); >> + writeb(0, i2c_imx->base + IMX_I2C_I2SR); >> + /* >> + * This delay caused by an i.MXL hardware bug. >> + * If no (or too short) delay, no "STOP" bit will be generated. >> + */ >> + udelay(disable_delay); >> + /* Disable I2C controller */ >> + writeb(0, i2c_imx->base + IMX_I2C_I2CR); >> +} >> + >> +static void __init i2c_imx_set_clk(struct imx_i2c_struct *i2c_imx, >> + unsigned int rate) >> +{ >> + unsigned int i2c_clk_rate; >> + unsigned int div; >> + int i; >> + >> + /* Divider value calculation */ >> + i2c_clk_rate = clk_get_rate(i2c_imx->clk); >> + div = (i2c_clk_rate + rate - 1) / rate; >> + if (div < i2c_clk_div[0][0]) >> + i = 0; >> + else if (div > i2c_clk_div[ARRAY_SIZE(i2c_clk_div) - 1][0]) >> + i = ARRAY_SIZE(i2c_clk_div) - 1; >> + else >> + for (i = 0; i2c_clk_div[i][0] < div; i++); >> + >> + /* Write divider value to register */ >> + writeb(i2c_clk_div[i][1], i2c_imx->base + IMX_I2C_IFDR); >> + >> + /* >> + * There dummy delay is calculated. >> + * It should be about one I2C clock period long. >> + * This delay is used in I2C bus disable function >> + * to fix chip hardware bug. >> + */ >> + disable_delay = (500000U * i2c_clk_div[i][0] >> + + (i2c_clk_rate / 2) - 1) / (i2c_clk_rate / 2); >> + >> + /* dev_dbg() can't be used, because adapter is not yet registered */ >> +#ifdef CONFIG_I2C_DEBUG_BUS >> + printk(KERN_DEBUG "I2C: <%s> I2C_CLK=%d, REQ DIV=%d\n", >> + __func__, i2c_clk_rate, div); >> + printk(KERN_DEBUG "I2C: <%s> IFDR[IC]=0x%x, REAL DIV=%d\n", >> + __func__, i2c_clk_div[i][1], i2c_clk_div[i][0]); >> +#endif >> +} >> + >> +static irqreturn_t i2c_imx_isr(int irq, void *dev_id) >> +{ >> + struct imx_i2c_struct *i2c_imx = dev_id; >> + unsigned int temp; >> + >> + temp = readb(i2c_imx->base + IMX_I2C_I2SR); >> + if (temp & I2SR_IIF) { >> + /* save status register */ >> + i2c_imx->i2csr = temp; >> + temp &= ~I2SR_IIF; >> + writeb(temp, i2c_imx->base + IMX_I2C_I2SR); >> + wake_up_interruptible(&i2c_imx->queue); >> + } >> + >> + return IRQ_HANDLED; >> +} >> + >> +static int i2c_imx_write(struct imx_i2c_struct *i2c_imx, struct i2c_msg *msgs) >> +{ >> + int i, result; >> + >> + dev_dbg(&i2c_imx->adapter.dev, "<%s> write slave address: addr=0x%x\n", >> + __func__, msgs->addr); >> + >> + /* write slave address */ >> + writeb(msgs->addr, i2c_imx->base + IMX_I2C_I2DR); > > This is wrong! You have to double the address before writing to the > register. strange! there are I2c board data in my MXLADS code: struct i2c_board_info __initdata mx1ads_i2c_devices[] = { { I2C_BOARD_INFO("ov7xxx", 0x42), .platform_data = &iclink[0], }, { I2C_BOARD_INFO("mt9v111", 0x90), .platform_data = &iclink[0], } } slave addresses are exactly 0x42 and 0x90 (from datasheets). my driver works with these devices with address not doubled. I saw this in other I2C drivers, but If I double address in my driver, it works wrong. I tested this with oscilloscope - now it works ok, with all devices I have tryed. > >> + result = i2c_imx_trx_complete(i2c_imx); >> + if (result) >> + return result; >> + result = i2c_imx_acked(i2c_imx); >> + if (result) >> + return result; >> + dev_dbg(&i2c_imx->adapter.dev, "<%s> write data\n", __func__); >> + >> + /* write data */ >> + for (i = 0; i < msgs->len; i++) { >> + dev_dbg(&i2c_imx->adapter.dev, >> + "<%s> write byte: B%d=0x%X\n", >> + __func__, i, msgs->buf[i]); >> + writeb(msgs->buf[i], i2c_imx->base + IMX_I2C_I2DR); >> + result = i2c_imx_trx_complete(i2c_imx); >> + if (result) >> + return result; >> + result = i2c_imx_acked(i2c_imx); >> + if (result) >> + return result; >> + } >> + return 0; >> +} >> + >> +static int i2c_imx_read(struct imx_i2c_struct *i2c_imx, struct i2c_msg *msgs) >> +{ >> + int i, result; >> + unsigned int temp; >> + >> + dev_dbg(&i2c_imx->adapter.dev, >> + "<%s> write slave address: addr=0x%x\n", >> + __func__, msgs->addr | 0x01); >> + >> + /* write slave address */ >> + writeb(msgs->addr | 0x01, i2c_imx->base + IMX_I2C_I2DR); > > Same here - (msgs->addr << 1) | 0x01. > >> + result = i2c_imx_trx_complete(i2c_imx); >> + if (result) >> + return result; >> + result = i2c_imx_acked(i2c_imx); >> + if (result) >> + return result; >> + >> + dev_dbg(&i2c_imx->adapter.dev, "<%s> setup bus\n", __func__); >> + >> + /* setup bus to read data */ >> + temp = readb(i2c_imx->base + IMX_I2C_I2CR); >> + temp &= ~I2CR_MTX; >> + if (msgs->len - 1) >> + temp &= ~I2CR_TXAK; >> + writeb(temp, i2c_imx->base + IMX_I2C_I2CR); >> + readb(i2c_imx->base + IMX_I2C_I2DR); /* dummy read */ >> + >> + dev_dbg(&i2c_imx->adapter.dev, "<%s> read data\n", __func__); >> + >> + /* read data */ >> + for (i = 0; i < msgs->len; i++) { >> + result = i2c_imx_trx_complete(i2c_imx); >> + if (result) >> + return result; >> + if (i == (msgs->len - 1)) { >> + dev_dbg(&i2c_imx->adapter.dev, >> + "<%s> clear MSTA\n", __func__); >> + temp = readb(i2c_imx->base + IMX_I2C_I2CR); >> + temp &= ~I2CR_MSTA; >> + writeb(temp, i2c_imx->base + IMX_I2C_I2CR); >> + } else if (i == (msgs->len - 2)) { >> + dev_dbg(&i2c_imx->adapter.dev, >> + "<%s> set TXAK\n", __func__); >> + temp = readb(i2c_imx->base + IMX_I2C_I2CR); >> + temp |= I2CR_TXAK; >> + writeb(temp, i2c_imx->base + IMX_I2C_I2CR); >> + } >> + msgs->buf[i] = readb(i2c_imx->base + IMX_I2C_I2DR); >> + dev_dbg(&i2c_imx->adapter.dev, >> + "<%s> read byte: B%d=0x%X\n", >> + __func__, i, msgs->buf[i]); >> + } >> + return 0; >> +} >> + >> +static int i2c_imx_xfer(struct i2c_adapter *adapter, >> + struct i2c_msg *msgs, int num) >> +{ >> + unsigned int i, temp; >> + int result; >> + struct imx_i2c_struct *i2c_imx = i2c_get_adapdata(adapter); >> + >> + dev_dbg(&i2c_imx->adapter.dev, "<%s>\n", __func__); >> + >> + /* Check or i2c bus is not busy */ >> + result = i2c_imx_bus_busy(i2c_imx); >> + if (result) >> + goto fail0; >> + >> + /* Start I2C transfer */ >> + i2c_imx_start(i2c_imx); >> + >> + /* read/write data */ >> + for (i = 0; i < num; i++) { >> + if (i) { >> + dev_dbg(&i2c_imx->adapter.dev, >> + "<%s> repeated start\n", __func__); >> + temp = readb(i2c_imx->base + IMX_I2C_I2CR); >> + temp |= I2CR_RSTA; >> + writeb(temp, i2c_imx->base + IMX_I2C_I2CR); >> + } >> + dev_dbg(&i2c_imx->adapter.dev, >> + "<%s> transfer message: %d\n", __func__, i); >> + /* write/read data */ >> +#ifdef CONFIG_I2C_DEBUG_BUS >> + temp = readb(i2c_imx->base + IMX_I2C_I2CR); >> + dev_dbg(&i2c_imx->adapter.dev, "<%s> CONTROL: IEN=%d, IIEN=%d, " >> + "MSTA=%d, MTX=%d, TXAK=%d, RSTA=%d\n", __func__, >> + (temp & I2CR_IEN ? 1 : 0), (temp & I2CR_IIEN ? 1 : 0), >> + (temp & I2CR_MSTA ? 1 : 0), (temp & I2CR_MTX ? 1 : 0), >> + (temp & I2CR_TXAK ? 1 : 0), (temp & I2CR_RSTA ? 1 : 0)); >> + temp = readb(i2c_imx->base + IMX_I2C_I2SR); >> + dev_dbg(&i2c_imx->adapter.dev, >> + "<%s> STATUS: ICF=%d, IAAS=%d, IBB=%d, " >> + "IAL=%d, SRW=%d, IIF=%d, RXAK=%d\n", __func__, >> + (temp & I2SR_ICF ? 1 : 0), (temp & I2SR_IAAS ? 1 : 0), >> + (temp & I2SR_IBB ? 1 : 0), (temp & I2SR_IAL ? 1 : 0), >> + (temp & I2SR_SRW ? 1 : 0), (temp & I2SR_IIF ? 1 : 0), >> + (temp & I2SR_RXAK ? 1 : 0)); >> +#endif >> + if (msgs[i].flags & I2C_M_RD) >> + result = i2c_imx_read(i2c_imx, &msgs[i]); >> + else >> + result = i2c_imx_write(i2c_imx, &msgs[i]); >> + } >> + >> +fail0: >> + /* Stop I2C transfer */ >> + i2c_imx_stop(i2c_imx); >> + >> + dev_dbg(&i2c_imx->adapter.dev, "<%s> exit with: %s: %d\n", __func__, >> + (result < 0) ? "error" : "success msg", >> + (result < 0) ? result : num); >> + return (result < 0) ? result : num; >> +} >> + >> +static u32 i2c_imx_func(struct i2c_adapter *adapter) >> +{ >> + return I2C_FUNC_I2C | I2C_FUNC_SMBUS_EMUL; >> +} >> + >> +static struct i2c_algorithm i2c_imx_algo = { >> + .master_xfer = i2c_imx_xfer, >> + .functionality = i2c_imx_func, >> +}; >> + >> +static int __init i2c_imx_probe(struct platform_device *pdev) >> +{ >> + struct imx_i2c_struct *i2c_imx; >> + struct resource *res; >> + struct imxi2c_platform_data *pdata; >> + void __iomem *base; >> + int irq; >> + int res_size; >> + int ret; >> + >> + dev_dbg(&pdev->dev, "<%s>\n", __func__); >> + >> + res = platform_get_resource(pdev, IORESOURCE_MEM, 0); >> + if (!res) { >> + dev_err(&pdev->dev, "can't get device resources\n"); >> + return -ENODEV; >> + } >> + irq = platform_get_irq(pdev, 0); >> + if (irq < 0) { >> + dev_err(&pdev->dev, "can't get irq number\n"); >> + return -ENODEV; >> + } >> + >> + pdata = pdev->dev.platform_data; >> + if (!pdata) { >> + dev_err(&pdev->dev, "I2C driver needs platform data\n"); >> + return -ENODEV; >> + } >> + >> + res_size = res->end - res->start + 1; > > You can use resource_size() > >> + if (!request_mem_region(res->start, res_size, res->name)) { >> + dev_err(&pdev->dev, "can't allocate %d bytes at %d address\n", >> + res_size, res->start); >> + return -ENOMEM; >> + } > > I was once told, one doesn't need request_mem_region() on regions from > platform data resources - this is already done implicitly. ok, i will fix it. > >> + >> + if (pdata->init) { >> + ret = pdata->init(&pdev->dev); >> + if (ret) >> + goto fail0; >> + } >> + >> + base = ioremap(res->start, res_size); >> + if (!base) { >> + dev_err(&pdev->dev, "ioremap failed\n"); >> + ret = -EIO; >> + goto fail1; >> + } >> + >> + i2c_imx = kzalloc(sizeof(struct imx_i2c_struct), GFP_KERNEL); >> + if (!i2c_imx) { >> + dev_err(&pdev->dev, "can't allocate interface\n"); >> + ret = -ENOMEM; >> + goto fail2; >> + } >> + >> + /* Setup i2c_imx driver structure */ >> + strcpy(i2c_imx->adapter.name, pdev->name); >> + i2c_imx->adapter.owner = THIS_MODULE; >> + i2c_imx->adapter.algo = &i2c_imx_algo; >> + i2c_imx->adapter.dev.parent = &pdev->dev; >> + i2c_imx->adapter.nr = pdev->id; >> + i2c_imx->irq = irq; >> + i2c_imx->base = base; >> + i2c_imx->res = res; >> + >> + /* Get I2C clock */ >> + i2c_imx->clk = clk_get(NULL, "i2c_clk"); > > There can be several i2c busses on the system, so you want: > > + i2c_imx->clk = clk_get(&pdev->dev, "i2c_clk"); ok, fixed. > >> + if (IS_ERR(i2c_imx->clk)) { >> + ret = PTR_ERR(i2c_imx->clk); >> + dev_err(&pdev->dev, "can't get I2C clock\n"); >> + goto fail3; >> + } >> + clk_enable(i2c_imx->clk); >> + >> + /* Request IRQ */ >> + ret = request_irq(i2c_imx->irq, i2c_imx_isr, >> + IRQF_DISABLED, pdev->name, i2c_imx); >> + if (ret) { >> + dev_err(&pdev->dev, "can't claim irq %d\n", i2c_imx->irq); >> + goto fail4; >> + } >> + >> + /* Init queue */ >> + init_waitqueue_head(&i2c_imx->queue); >> + >> + /* Set up adapter data */ >> + i2c_set_adapdata(&i2c_imx->adapter, i2c_imx); >> + >> + /* Set up clock divider */ >> + i2c_imx_set_clk(i2c_imx, clkfreq); >> + >> + /* Set up chip registers to defaults */ >> + writeb(0, i2c_imx->base + IMX_I2C_I2CR); >> + writeb(0, i2c_imx->base + IMX_I2C_I2SR); >> + >> + /* Add I2C adapter */ >> + ret = i2c_add_numbered_adapter(&i2c_imx->adapter); >> + if (ret < 0) { >> + dev_err(&pdev->dev, "registration failed\n"); >> + goto fail5; >> + } >> + >> + /* Set up platform driver data */ >> + platform_set_drvdata(pdev, i2c_imx); >> + >> + dev_dbg(&i2c_imx->adapter.dev, "claimed irq %d\n", i2c_imx->irq); >> + dev_dbg(&i2c_imx->adapter.dev, "device resources from 0x%x to 0x%x\n", >> + i2c_imx->res->start, i2c_imx->res->end); >> + dev_dbg(&i2c_imx->adapter.dev, "allocated %d bytes at 0x%x \n", >> + res_size, i2c_imx->res->start); >> + dev_dbg(&i2c_imx->adapter.dev, "adapter name: \"%s\"\n", >> + i2c_imx->adapter.name); >> + dev_dbg(&i2c_imx->adapter.dev, "IMX I2C adapter registered\n"); >> + >> + return 0; /* Return OK */ >> + >> +fail5: >> + free_irq(i2c_imx->irq, i2c_imx); >> +fail4: >> + clk_disable(i2c_imx->clk); >> + clk_put(i2c_imx->clk); >> +fail3: >> + kfree(i2c_imx); >> +fail2: >> + iounmap(base); >> +fail1: >> + if (pdata->exit) >> + pdata->exit(&pdev->dev); >> +fail0: >> + release_mem_region(res->start, res_size); >> + return ret; /* Return error number */ >> +} >> + >> +static int __exit i2c_imx_remove(struct platform_device *pdev) >> +{ >> + struct imx_i2c_struct *i2c_imx = platform_get_drvdata(pdev); >> + struct imxi2c_platform_data *pdata = pdev->dev.platform_data; >> + >> + /* remove adapter */ >> + dev_dbg(&i2c_imx->adapter.dev, "adapter removed\n"); >> + i2c_del_adapter(&i2c_imx->adapter); >> + platform_set_drvdata(pdev, NULL); >> + >> + /* free interrupt */ >> + free_irq(i2c_imx->irq, i2c_imx); >> + >> + /* setup chip registers to defaults */ >> + writeb(0, i2c_imx->base + IMX_I2C_IADR); >> + writeb(0, i2c_imx->base + IMX_I2C_IFDR); >> + writeb(0, i2c_imx->base + IMX_I2C_I2CR); >> + writeb(0, i2c_imx->base + IMX_I2C_I2SR); >> + >> + /* Shut down hardware */ >> + if (pdata->exit) >> + pdata->exit(&pdev->dev); >> + >> + /* Disable I2C clock */ >> + clk_disable(i2c_imx->clk); >> + clk_put(i2c_imx->clk); >> + >> + /* Release memory */ >> + release_mem_region(i2c_imx->res->start, >> + i2c_imx->res->end - i2c_imx->res->start + 1); >> + iounmap(i2c_imx->base); >> + kfree(i2c_imx); >> + return 0; >> +} >> + >> +static struct platform_driver i2c_imx_driver = { >> + .probe = i2c_imx_probe, >> + .remove = __exit_p(i2c_imx_remove), >> + .driver = { >> + .name = DRIVER_NAME, >> + .owner = THIS_MODULE, >> + } >> +}; >> + >> +static int __init i2c_adap_imx_init(void) >> +{ >> + return platform_driver_probe(&i2c_imx_driver, i2c_imx_probe); >> +} >> + >> +static void __exit i2c_adap_imx_exit(void) >> +{ >> + platform_driver_unregister(&i2c_imx_driver); >> +} >> + >> +module_init(i2c_adap_imx_init); >> +module_exit(i2c_adap_imx_exit); >> + >> +module_param(clkfreq, uint, S_IRUGO); >> +MODULE_PARM_DESC(clkfreq, "desired IMX I2C Clock Rate in Hz"); > > Making clkfreq a module parameter you force the same frequency on all i2c > busses. On my i.MX31 system 2 busses are internal and one goes to a > connector, to which a camera is connected. This third bus can only handle > a lower frequency, which, however, doesn't mean we also have to throttle > the other two busses. Can we put this into platform data? We can do that, but now there is possibility to change bitrate when re-loading module. What is better? > >> + >> +MODULE_LICENSE("GPL"); >> +MODULE_AUTHOR("Darius Augulis"); >> +MODULE_DESCRIPTION("I2C adapter driver for IMX I2C bus"); >> +MODULE_ALIAS("platform:" DRIVER_NAME); >> Index: linux-2.6.28-rc6/drivers/i2c/busses/Kconfig >> =================================================================== >> --- linux-2.6.28-rc6.orig/drivers/i2c/busses/Kconfig >> +++ linux-2.6.28-rc6/drivers/i2c/busses/Kconfig >> @@ -355,6 +355,16 @@ config I2C_IBM_IIC >> This driver can also be built as a module. If so, the module >> will be called i2c-ibm_iic. >> >> +config I2C_IMX >> + tristate "IMX I2C interface" >> + depends on ARCH_MXC >> + help >> + Say Y here if you want to use the IIC bus controller on >> + the Freescale i.MX/MXC processors. >> + >> + This driver can also be built as a module. If so, the module >> + will be called i2c-imx. >> + >> config I2C_IOP3XX >> tristate "Intel IOPx3xx and IXP4xx on-chip I2C interface" >> depends on ARCH_IOP32X || ARCH_IOP33X || ARCH_IXP4XX || ARCH_IOP13XX >> Index: linux-2.6.28-rc6/drivers/i2c/busses/Makefile >> =================================================================== >> --- linux-2.6.28-rc6.orig/drivers/i2c/busses/Makefile >> +++ linux-2.6.28-rc6/drivers/i2c/busses/Makefile >> @@ -33,6 +33,7 @@ obj-$(CONFIG_I2C_DAVINCI) += i2c-davinci >> obj-$(CONFIG_I2C_GPIO) += i2c-gpio.o >> obj-$(CONFIG_I2C_HIGHLANDER) += i2c-highlander.o >> obj-$(CONFIG_I2C_IBM_IIC) += i2c-ibm_iic.o >> +obj-$(CONFIG_I2C_IMX) += i2c-imx.o >> obj-$(CONFIG_I2C_IOP3XX) += i2c-iop3xx.o >> obj-$(CONFIG_I2C_IXP2000) += i2c-ixp2000.o >> obj-$(CONFIG_I2C_MPC) += i2c-mpc.o >> Index: linux-2.6.28-rc6/arch/arm/plat-mxc/include/mach/i2c.h >> =================================================================== >> --- /dev/null >> +++ linux-2.6.28-rc6/arch/arm/plat-mxc/include/mach/i2c.h >> @@ -0,0 +1,17 @@ >> +/* >> + * i2c.h - i.MX I2C driver header file >> + * >> + * Copyright (c) 2008, Darius Augulis <augulis.darius-Re5JQEeQqe8AvxtiuMwx3w@public.gmane.org> >> + * >> + * This file is released under the GPLv2 >> + */ >> + >> +#ifndef __ASM_ARCH_I2C_H_ >> +#define __ASM_ARCH_I2C_H_ >> + >> +struct imxi2c_platform_data { >> + int (*init)(struct device *dev); >> + int (*exit)(struct device *dev); > > What are you going to use .exit() for? Is it really needed? Even if it is, > it can easily return void I guess? .init is used to request and setup gpio pins, .exit used to free gpio. yes, .exit can return void - I will fix it. > >> +}; >> + >> +#endif /* __ASM_ARCH_I2C_H_ */ > > Thanks > Guennadi > --- > Guennadi Liakhovetski, Ph.D. > Freelance Open-Source Software Developer ^ permalink raw reply [flat|nested] 21+ messages in thread
* Re: [PATCH][resend] iMX/MXC support for I2C 2008-12-03 12:52 ` Darius @ 2008-12-03 15:19 ` Mark Brown 2008-12-03 21:19 ` Guennadi Liakhovetski 1 sibling, 0 replies; 21+ messages in thread From: Mark Brown @ 2008-12-03 15:19 UTC (permalink / raw) To: Darius Cc: linux-i2c-u79uwXL29TY76Z2rM5mHXA, linux-arm-kernel-xIg/pKzrS19vn6HldHNs0ANdhmdF6hFW On Wed, Dec 03, 2008 at 02:52:56PM +0200, Darius wrote: > Guennadi Liakhovetski wrote: > >>+ /* write slave address */ > >>+ writeb(msgs->addr, i2c_imx->base + IMX_I2C_I2DR); > >This is wrong! You have to double the address before writing to the > >register. > strange! there are I2c board data in my MXLADS code: ... > slave addresses are exactly 0x42 and 0x90 (from datasheets). > my driver works with these devices with address not doubled. > I saw this in other I2C drivers, but If I double address in my driver, it > works wrong. Many datasheets quote the I2C address in wire format (ie, after the doubling, including the read/write bit as LSB) that the hardware is looking for here. The Linux I2C framework doesn't do this, it doesn't include the read/write bit in the address. > >Making clkfreq a module parameter you force the same frequency on all i2c > >busses. On my i.MX31 system 2 busses are internal and one goes to a > >connector, to which a camera is connected. This third bus can only handle > >a lower frequency, which, however, doesn't mean we also have to throttle > >the other two busses. Can we put this into platform data? > We can do that, but now there is possibility to change bitrate when > re-loading module. > What is better? Platform data is better but you could also include a default for use when no platform data is specified and then have a module parameter to override that. ^ permalink raw reply [flat|nested] 21+ messages in thread
* Re: [PATCH][resend] iMX/MXC support for I2C 2008-12-03 12:52 ` Darius 2008-12-03 15:19 ` Mark Brown @ 2008-12-03 21:19 ` Guennadi Liakhovetski [not found] ` <Pine.LNX.4.64.0812032158150.7961-0199iw4Nj15frtckUFj5Ag@public.gmane.org> 1 sibling, 1 reply; 21+ messages in thread From: Guennadi Liakhovetski @ 2008-12-03 21:19 UTC (permalink / raw) To: Darius Cc: linux-i2c-u79uwXL29TY76Z2rM5mHXA, linux-arm-kernel-xIg/pKzrS19vn6HldHNs0ANdhmdF6hFW On Wed, 3 Dec 2008, Darius wrote: > Guennadi Liakhovetski wrote: > > > + > > > +static int i2c_imx_trx_complete(struct imx_i2c_struct *i2c_imx) > > > +{ > > > + int result; > > > + > > > + result = wait_event_interruptible_timeout(i2c_imx->queue, > > > + i2c_imx->i2csr & I2SR_IIF, I2C_IMX_TIME_TRX * HZ); > > > > 5s is much too long! > > how much? 600us? mxc uses 1 jiffie. > > > + /* write slave address */ > > > + writeb(msgs->addr, i2c_imx->base + IMX_I2C_I2DR); > > > > This is wrong! You have to double the address before writing to the > > register. > > strange! there are I2c board data in my MXLADS code: > > struct i2c_board_info __initdata mx1ads_i2c_devices[] = { > { > I2C_BOARD_INFO("ov7xxx", 0x42), > .platform_data = &iclink[0], > }, { > I2C_BOARD_INFO("mt9v111", 0x90), > .platform_data = &iclink[0], > } > } > > slave addresses are exactly 0x42 and 0x90 (from datasheets). > my driver works with these devices with address not doubled. > I saw this in other I2C drivers, but If I double address in my driver, it > works wrong. > I tested this with oscilloscope - now it works ok, with all devices I have > tryed. As Mark explained - Linux uses i2c addresses without the read/write bit, i.e., shifted one bit right. > > > +module_param(clkfreq, uint, S_IRUGO); > > > +MODULE_PARM_DESC(clkfreq, "desired IMX I2C Clock Rate in Hz"); > > > > Making clkfreq a module parameter you force the same frequency on all i2c > > busses. On my i.MX31 system 2 busses are internal and one goes to a > > connector, to which a camera is connected. This third bus can only handle a > > lower frequency, which, however, doesn't mean we also have to throttle the > > other two busses. Can we put this into platform data? > > We can do that, but now there is possibility to change bitrate when re-loading > module. > What is better? But is it really necessary to be able to override this at load-time? At least not as one single parameter. If you absolutely need this possibility, maybe an array of frequencies? But then you don't know how many busses you are going to have. Having an array of 8 ints will probably be enough for a while:-) > > > +struct imxi2c_platform_data { > > > + int (*init)(struct device *dev); > > > + int (*exit)(struct device *dev); > > > > What are you going to use .exit() for? Is it really needed? Even if it is, > > it can easily return void I guess? > > .init is used to request and setup gpio pins, .exit used to free gpio. > yes, .exit can return void - I will fix it. You mean in your .init() you not only configure iomux pins for i2c, you also gpio_request(pin_to_gpio(), "i2c")? Now that I think about this, maybe this is indeed correct, and then you gpio_free() in .exit()... Is this what you mean? Thanks Guennadi --- Guennadi Liakhovetski, Ph.D. Freelance Open-Source Software Developer ^ permalink raw reply [flat|nested] 21+ messages in thread
[parent not found: <Pine.LNX.4.64.0812032158150.7961-0199iw4Nj15frtckUFj5Ag@public.gmane.org>]
* Re: [PATCH][resend] iMX/MXC support for I2C [not found] ` <Pine.LNX.4.64.0812032158150.7961-0199iw4Nj15frtckUFj5Ag@public.gmane.org> @ 2008-12-03 23:45 ` Ben Dooks 2008-12-04 7:47 ` Darius 1 sibling, 0 replies; 21+ messages in thread From: Ben Dooks @ 2008-12-03 23:45 UTC (permalink / raw) To: Guennadi Liakhovetski Cc: Darius, linux-i2c-u79uwXL29TY76Z2rM5mHXA, linux-arm-kernel-xIg/pKzrS19vn6HldHNs0ANdhmdF6hFW On Wed, Dec 03, 2008 at 10:19:32PM +0100, Guennadi Liakhovetski wrote: > On Wed, 3 Dec 2008, Darius wrote: > > > Guennadi Liakhovetski wrote: > > > > + > > > > +static int i2c_imx_trx_complete(struct imx_i2c_struct *i2c_imx) > > > > +{ > > > > + int result; > > > > + > > > > + result = wait_event_interruptible_timeout(i2c_imx->queue, > > > > + i2c_imx->i2csr & I2SR_IIF, I2C_IMX_TIME_TRX * HZ); > > > > > > 5s is much too long! > > > > how much? 600us? > > mxc uses 1 jiffie. > > > > > + /* write slave address */ > > > > + writeb(msgs->addr, i2c_imx->base + IMX_I2C_I2DR); > > > > > > This is wrong! You have to double the address before writing to the > > > register. > > > > strange! there are I2c board data in my MXLADS code: > > > > struct i2c_board_info __initdata mx1ads_i2c_devices[] = { > > { > > I2C_BOARD_INFO("ov7xxx", 0x42), > > .platform_data = &iclink[0], > > }, { > > I2C_BOARD_INFO("mt9v111", 0x90), > > .platform_data = &iclink[0], > > } > > } > > > > slave addresses are exactly 0x42 and 0x90 (from datasheets). > > my driver works with these devices with address not doubled. > > I saw this in other I2C drivers, but If I double address in my driver, it > > works wrong. > > I tested this with oscilloscope - now it works ok, with all devices I have > > tryed. > > As Mark explained - Linux uses i2c addresses without the read/write bit, > i.e., shifted one bit right. > > > > > +module_param(clkfreq, uint, S_IRUGO); > > > > +MODULE_PARM_DESC(clkfreq, "desired IMX I2C Clock Rate in Hz"); > > > > > > Making clkfreq a module parameter you force the same frequency on all i2c > > > busses. On my i.MX31 system 2 busses are internal and one goes to a > > > connector, to which a camera is connected. This third bus can only handle a > > > lower frequency, which, however, doesn't mean we also have to throttle the > > > other two busses. Can we put this into platform data? > > > > We can do that, but now there is possibility to change bitrate when re-loading > > module. > > What is better? > > But is it really necessary to be able to override this at load-time? At > least not as one single parameter. If you absolutely need this > possibility, maybe an array of frequencies? But then you don't know how > many busses you are going to have. Having an array of 8 ints will probably > be enough for a while:-) Platform data seems a good way of getting a board to declare a bus rate. Otherwise a standard i2c command line option for all busses to use may be the best thing to do. > > > > +struct imxi2c_platform_data { > > > > + int (*init)(struct device *dev); > > > > + int (*exit)(struct device *dev); > > > > > > What are you going to use .exit() for? Is it really needed? Even if it is, > > > it can easily return void I guess? > > > > .init is used to request and setup gpio pins, .exit used to free gpio. > > yes, .exit can return void - I will fix it. > > You mean in your .init() you not only configure iomux pins for i2c, you > also gpio_request(pin_to_gpio(), "i2c")? Now that I think about this, > maybe this is indeed correct, and then you gpio_free() in .exit()... Is > this what you mean? What does David Brownell think of that? -- Ben (ben-elnMNo+KYs3YtjvyW6yDsg@public.gmane.org, http://www.fluff.org/) 'a smiley only costs 4 bytes' ^ permalink raw reply [flat|nested] 21+ messages in thread
* Re: [PATCH][resend] iMX/MXC support for I2C [not found] ` <Pine.LNX.4.64.0812032158150.7961-0199iw4Nj15frtckUFj5Ag@public.gmane.org> 2008-12-03 23:45 ` Ben Dooks @ 2008-12-04 7:47 ` Darius [not found] ` <49378B1A.7030909-Re5JQEeQqe8AvxtiuMwx3w@public.gmane.org> 1 sibling, 1 reply; 21+ messages in thread From: Darius @ 2008-12-04 7:47 UTC (permalink / raw) To: Guennadi Liakhovetski Cc: linux-i2c-u79uwXL29TY76Z2rM5mHXA, linux-arm-kernel-xIg/pKzrS19vn6HldHNs0ANdhmdF6hFW Guennadi Liakhovetski wrote: > On Wed, 3 Dec 2008, Darius wrote: > > >> Guennadi Liakhovetski wrote: >> >>>> + >>>> +static int i2c_imx_trx_complete(struct imx_i2c_struct *i2c_imx) >>>> +{ >>>> + int result; >>>> + >>>> + result = wait_event_interruptible_timeout(i2c_imx->queue, >>>> + i2c_imx->i2csr & I2SR_IIF, I2C_IMX_TIME_TRX * HZ); >>>> >>> 5s is much too long! >>> >> how much? 600us? >> > > mxc uses 1 jiffie. > > >>>> + /* write slave address */ >>>> + writeb(msgs->addr, i2c_imx->base + IMX_I2C_I2DR); >>>> >>> This is wrong! You have to double the address before writing to the >>> register. >>> >> strange! there are I2c board data in my MXLADS code: >> >> struct i2c_board_info __initdata mx1ads_i2c_devices[] = { >> { >> I2C_BOARD_INFO("ov7xxx", 0x42), >> .platform_data = &iclink[0], >> }, { >> I2C_BOARD_INFO("mt9v111", 0x90), >> .platform_data = &iclink[0], >> } >> } >> >> slave addresses are exactly 0x42 and 0x90 (from datasheets). >> my driver works with these devices with address not doubled. >> I saw this in other I2C drivers, but If I double address in my driver, it >> works wrong. >> I tested this with oscilloscope - now it works ok, with all devices I have >> tryed. >> > > As Mark explained - Linux uses i2c addresses without the read/write bit, > i.e., shifted one bit right. > This means I should use shifted address in platform data? For example: OV7670 image sensor has slave address 0x42 for write, and 0x43 for read operations. This is hardware slave address, to which device responds. But I should use 0x21 in platform data, because it must be doubled in adapter driver? If I use original 0x42 address, after doubling it will be 0x108, to which device does not respond... > >>>> +module_param(clkfreq, uint, S_IRUGO); >>>> +MODULE_PARM_DESC(clkfreq, "desired IMX I2C Clock Rate in Hz"); >>>> >>> Making clkfreq a module parameter you force the same frequency on all i2c >>> busses. On my i.MX31 system 2 busses are internal and one goes to a >>> connector, to which a camera is connected. This third bus can only handle a >>> lower frequency, which, however, doesn't mean we also have to throttle the >>> other two busses. Can we put this into platform data? >>> >> We can do that, but now there is possibility to change bitrate when re-loading >> module. >> What is better? >> > > But is it really necessary to be able to override this at load-time? At > least not as one single parameter. If you absolutely need this > possibility, maybe an array of frequencies? But then you don't know how > many busses you are going to have. Having an array of 8 ints will probably > be enough for a while:-) > No it's not necessary. I will leave this setting only in platform data and default value if platform data is not provided. > >>>> +struct imxi2c_platform_data { >>>> + int (*init)(struct device *dev); >>>> + int (*exit)(struct device *dev); >>>> >>> What are you going to use .exit() for? Is it really needed? Even if it is, >>> it can easily return void I guess? >>> >> .init is used to request and setup gpio pins, .exit used to free gpio. >> yes, .exit can return void - I will fix it. >> > > You mean in your .init() you not only configure iomux pins for i2c, you > also gpio_request(pin_to_gpio(), "i2c")? Now that I think about this, > maybe this is indeed correct, and then you gpio_free() in .exit()... Is > this what you mean? > yes, you're right. > Thanks > Guennadi > --- > Guennadi Liakhovetski, Ph.D. > Freelance Open-Source Software Developer > > ^ permalink raw reply [flat|nested] 21+ messages in thread
[parent not found: <49378B1A.7030909-Re5JQEeQqe8AvxtiuMwx3w@public.gmane.org>]
* Re: [PATCH][resend] iMX/MXC support for I2C [not found] ` <49378B1A.7030909-Re5JQEeQqe8AvxtiuMwx3w@public.gmane.org> @ 2008-12-04 13:30 ` Mark Brown 0 siblings, 0 replies; 21+ messages in thread From: Mark Brown @ 2008-12-04 13:30 UTC (permalink / raw) To: Darius Cc: Guennadi Liakhovetski, linux-i2c-u79uwXL29TY76Z2rM5mHXA, linux-arm-kernel-xIg/pKzrS19vn6HldHNs0ANdhmdF6hFW On Thu, Dec 04, 2008 at 09:47:38AM +0200, Darius wrote: > This means I should use shifted address in platform data? > For example: OV7670 image sensor has slave address 0x42 for write, and > 0x43 for read operations. > This is hardware slave address, to which device responds. > But I should use 0x21 in platform data, because it must be doubled in > adapter driver? Yes, exactly. Linux does not treat the read/write bit as part of the address. ^ permalink raw reply [flat|nested] 21+ messages in thread
* Re: [PATCH][resend] iMX/MXC support for I2C [not found] ` <Pine.LNX.4.64.0812030917440.4717-0199iw4Nj15frtckUFj5Ag@public.gmane.org> 2008-12-03 12:52 ` Darius @ 2008-12-03 23:43 ` Ben Dooks [not found] ` <20081203234317.GA4256-elnMNo+KYs3pIgCt6eIbzw@public.gmane.org> 1 sibling, 1 reply; 21+ messages in thread From: Ben Dooks @ 2008-12-03 23:43 UTC (permalink / raw) To: Guennadi Liakhovetski Cc: Darius, linux-i2c-u79uwXL29TY76Z2rM5mHXA, linux-arm-kernel-xIg/pKzrS19vn6HldHNs0ANdhmdF6hFW On Wed, Dec 03, 2008 at 12:17:58PM +0100, Guennadi Liakhovetski wrote: > The good news is - I was able to get this driver running on i.MX31 - > somewhat... Now the comments: > > On Tue, 2 Dec 2008, Darius wrote: > > Commenting would have been easier, if you had inlined your patch... > > > From augulis.darius-Re5JQEeQqe8AvxtiuMwx3w@public.gmane.org Tue Dec 2 15:15:21 2008 > > Date: Tue, 02 Dec 2008 16:00:20 +0200 > > From: Darius <augulis.darius-Re5JQEeQqe8AvxtiuMwx3w@public.gmane.org> > > To: linux-i2c-u79uwXL29TY76Z2rM5mHXA@public.gmane.org > > Cc: linux-arm-kernel-xIg/pKzrS19vn6HldHNs0ANdhmdF6hFW@public.gmane.org > > Subject: [PATCH][resend] iMX/MXC support for I2C > > > > > > [ Empty or malformed message. Use "H" to see raw text. ] > > > > > > [ Part 2: "Attached Text" ] > > > > From: Darius Augulis <augulis.darius-Re5JQEeQqe8AvxtiuMwx3w@public.gmane.org> > > > > Implementation of I2C Adapter/Algorithm Driver for I2C Bus integrated > > in Freescale's i.MX/MXC processors. Tested only on MX1, but should work on MX2 > > and MX3 too, because all three processors have the same I2C controller. > > > > Signed-off-by: Darius Augulis <augulis.darius-Re5JQEeQqe8AvxtiuMwx3w@public.gmane.org> > > > > Index: linux-2.6.28-rc6/drivers/i2c/busses/i2c-imx.c > > =================================================================== > > --- /dev/null > > +++ linux-2.6.28-rc6/drivers/i2c/busses/i2c-imx.c > > @@ -0,0 +1,656 @@ > > +/* > > + * Copyright (C) 2002 Motorola GSG-China > > + * > > + * This program is free software; you can redistribute it and/or > > + * modify it under the terms of the GNU General Public License > > + * as published by the Free Software Foundation; either version 2 > > + * of the License, or (at your option) any later version. > > + * > > + * This program is distributed in the hope that it will be useful, > > + * but WITHOUT ANY WARRANTY; without even the implied warranty of > > + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the > > + * GNU General Public License for more details. > > + * > > + * You should have received a copy of the GNU General Public License > > + * along with this program; if not, write to the Free Software > > + * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, > > + * USA. > > + * > > + * Author: > > + * Darius Augulis, Teltonika Inc. > > + * > > + * Desc.: > > + * Implementation of I2C Adapter/Algorithm Driver > > + * for I2C Bus integrated in Freescale i.MX/MXC processors > > + * > > + * module parameters: > > + * - clkfreq: > > + * Sets the desired clock rate > > + * The default value is 100000 > > + * Max value is 400000 > > + * > > + * Derived from Motorola GSG China I2C example driver > > + * > > + * Copyright (C) 2005 Torsten Koschorrek <koschorrek at synertronixx.de > > + * Copyright (C) 2005 Matthias Blaschke <blaschke at synertronixx.de > > + * Copyright (C) 2007 RightHand Technologies, Inc. > > + * Copyright (C) 2008 Darius Augulis <darius.augulis at teltonika.lt> > > + * > > + */ > > + > > +/** Includes ******************************************************************* > > +*******************************************************************************/ > > + > > +#include <linux/init.h> > > +#include <linux/kernel.h> > > +#include <linux/module.h> > > +#include <linux/errno.h> > > +#include <linux/err.h> > > +#include <linux/interrupt.h> > > +#include <linux/delay.h> > > +#include <linux/i2c.h> > > +#include <linux/io.h> > > +#include <linux/sched.h> > > +#include <linux/platform_device.h> > > +#include <linux/clk.h> > > + > > +#include <mach/irqs.h> > > +#include <mach/hardware.h> > > +#include <mach/i2c.h> > > + > > +/** Defines ******************************************************************** > > +*******************************************************************************/ > > + > > +/* This will be the driver name the kernel reports */ > > +#define DRIVER_NAME "imx-i2c" > > + > > +/* Default values of module parameters */ > > +#define IMX_I2C_BIT_RATE 100000 /* 100kHz */ > > + > > +/* Timeouts */ > > +#define I2C_IMX_TIME_BUSY 2000 /* loop count */ > > +#define I2C_IMX_TIME_ACK 2000 /* loop count */ > > +#define I2C_IMX_TIME_TRX 5 /* seconds */ > > + > > +/* IMX I2C registers */ > > +#define IMX_I2C_IADR 0x00 /* i2c slave address */ > > +#define IMX_I2C_IFDR 0x04 /* i2c frequency divider */ > > +#define IMX_I2C_I2CR 0x08 /* i2c control */ > > +#define IMX_I2C_I2SR 0x0C /* i2c status */ > > +#define IMX_I2C_I2DR 0x10 /* i2c transfer data */ > > + > > +/* Bits of IMX I2C registers */ > > +#define I2SR_RXAK 0x01 > > +#define I2SR_IIF 0x02 > > +#define I2SR_SRW 0x04 > > +#define I2SR_IAL 0x10 > > +#define I2SR_IBB 0x20 > > +#define I2SR_IAAS 0x40 > > +#define I2SR_ICF 0x80 > > +#define I2CR_RSTA 0x04 > > +#define I2CR_TXAK 0x08 > > +#define I2CR_MTX 0x10 > > +#define I2CR_MSTA 0x20 > > +#define I2CR_IIEN 0x40 > > +#define I2CR_IEN 0x80 > > + > > +/** Variables ****************************************************************** > > +*******************************************************************************/ > > + > > +static unsigned int clkfreq = IMX_I2C_BIT_RATE; > > +static unsigned int disable_delay; /* Dummy delay */ > > + > > +/* > > + * sorted list of clock divider, register value pairs > > + * taken from table 26-5, p.26-9, Freescale i.MX > > + * Integrated Portable System Processor Reference Manual > > + * Document Number: MC9328MXLRM, Rev. 5.1, 06/2007 > > + * > > + * Duplicated divider values removed from list > > + */ > > + > > +static u16 __initdata i2c_clk_div[50][2] = { > > + { 22, 0x20 }, { 24, 0x21 }, { 26, 0x22 }, { 28, 0x23 }, > > + { 30, 0x00 }, { 32, 0x24 }, { 36, 0x25 }, { 40, 0x26 }, > > + { 42, 0x03 }, { 44, 0x27 }, { 48, 0x28 }, { 52, 0x05 }, > > + { 56, 0x29 }, { 60, 0x06 }, { 64, 0x2A }, { 72, 0x2B }, > > + { 80, 0x2C }, { 88, 0x09 }, { 96, 0x2D }, { 104, 0x0A }, > > + { 112, 0x2E }, { 128, 0x2F }, { 144, 0x0C }, { 160, 0x30 }, > > + { 192, 0x31 }, { 224, 0x32 }, { 240, 0x0F }, { 256, 0x33 }, > > + { 288, 0x10 }, { 320, 0x34 }, { 384, 0x35 }, { 448, 0x36 }, > > + { 480, 0x13 }, { 512, 0x37 }, { 576, 0x14 }, { 640, 0x38 }, > > + { 768, 0x39 }, { 896, 0x3A }, { 960, 0x17 }, { 1024, 0x3B }, > > + { 1152, 0x18 }, { 1280, 0x3C }, { 1536, 0x3D }, { 1792, 0x3E }, > > + { 1920, 0x1B }, { 2048, 0x3F }, { 2304, 0x1C }, { 2560, 0x1D }, > > + { 3072, 0x1E }, { 3840, 0x1F } > > +}; > > + > > +struct imx_i2c_struct { > > + struct i2c_adapter adapter; > > + struct resource *res; > > + struct clk *clk; > > + void __iomem *base; > > + int irq; > > + wait_queue_head_t queue; > > + unsigned long i2csr; > > +}; > > + > > +/** Functions for IMX I2C adapter driver *************************************** > > +*******************************************************************************/ > > + > > +static int i2c_imx_bus_busy(struct imx_i2c_struct *i2c_imx) > > +{ > > + unsigned long orig_jiffies = jiffies; > > + > > + dev_dbg(&i2c_imx->adapter.dev, "<%s>\n", __func__); > > + > > + /* wait for bus not busy */ > > + while (readb(i2c_imx->base + IMX_I2C_I2SR) & I2SR_IBB) { > > The i2c-mxc driver I ported over from Freescale sources uses 16-bit > access, and the datasheet defines registers to be 16-bit wide, although > only low 8 bits are used. Whereas an i.MX datasheet (MC9328MX1) I have > here defines i2c registers to be 32-bit wide. > > > + if (signal_pending(current)) { > > + dev_dbg(&i2c_imx->adapter.dev, > > + "<%s> I2C Interrupted\n", __func__); > > + return -EINTR; > > + } > > + if (time_after(jiffies, orig_jiffies + HZ)) { > > + dev_dbg(&i2c_imx->adapter.dev, > > + "<%s> I2C bus is busy\n", __func__); > > + return -EIO; > > + } > > + schedule(); is there a function that would do the equiv of wait_event_interruptible? > For comparison: the MXC driver only waits for bus busy to clear for 600us > at most. > > > + } > > + > > + return 0; > > +} > > + > > +static int i2c_imx_trx_complete(struct imx_i2c_struct *i2c_imx) > > +{ > > + int result; > > + > > + result = wait_event_interruptible_timeout(i2c_imx->queue, > > + i2c_imx->i2csr & I2SR_IIF, I2C_IMX_TIME_TRX * HZ); > > 5s is much too long! Hmm, I would tend to agree, a half-second would probably be better unless you know you are really going to pushing a lot of data over a low speed i2c bus. > > + > > + if (unlikely(result < 0)) { > > + dev_dbg(&i2c_imx->adapter.dev, "<%s> result < 0\n", __func__); > > + return result; > > + } else if (unlikely(!(i2c_imx->i2csr & I2SR_IIF))) { > > + dev_dbg(&i2c_imx->adapter.dev, "<%s> Timeout\n", __func__); > > + return -ETIMEDOUT; > > + } > > + dev_dbg(&i2c_imx->adapter.dev, "<%s> TRX complete\n", __func__); > > + i2c_imx->i2csr = 0; > > + return 0; > > +} > > + > > +static int i2c_imx_acked(struct imx_i2c_struct *i2c_imx) > > +{ > > + unsigned long orig_jiffies = jiffies; > > + > > + /* Wait for ack */ > > + while (readb(i2c_imx->base + IMX_I2C_I2SR) & I2SR_RXAK) { > > + if (signal_pending(current)) { > > + dev_dbg(&i2c_imx->adapter.dev, > > + "<%s> I2C Interrupted\n", __func__); > > + return -EINTR; > > + } > > + if (time_after(jiffies, orig_jiffies + HZ)) { > > + dev_dbg(&i2c_imx->adapter.dev, > > + "<%s> No ACK\n", __func__); > > + return -EIO; > > + } > > + schedule(); > > + } > > 1s is way too long here! i2cdetect takes ages to complete. The mxc driver > doesn't wait here at all - as soon as a tx is complete, it checks the ack > status. Could you verify if this would work on your MX1 too? That does seem a problem, if we can detect a NAK immediately then all the better. > > + > > + dev_dbg(&i2c_imx->adapter.dev, "<%s> ACK received\n", __func__); > > + return 0; /* No ACK */ > > +} > > + > > +static void i2c_imx_start(struct imx_i2c_struct *i2c_imx) > > +{ > > + unsigned int temp = 0; > > + > > + dev_dbg(&i2c_imx->adapter.dev, "<%s>\n", __func__); > > + > > + /* Enable I2C controller */ > > + writeb(I2CR_IEN, i2c_imx->base + IMX_I2C_I2CR); > > + /* Start I2C transaction */ > > + temp = readb(i2c_imx->base + IMX_I2C_I2CR); > > + temp |= I2CR_MSTA; > > + writeb(temp, i2c_imx->base + IMX_I2C_I2CR); > > + temp |= I2CR_IIEN | I2CR_MTX | I2CR_TXAK; > > + writeb(temp, i2c_imx->base + IMX_I2C_I2CR); > > +} > > + > > +static void i2c_imx_stop(struct imx_i2c_struct *i2c_imx) > > +{ > > + unsigned int temp = 0; > > + > > + /* Stop I2C transaction */ > > + dev_dbg(&i2c_imx->adapter.dev, "<%s>\n", __func__); > > + temp = readb(i2c_imx->base + IMX_I2C_I2CR); > > + temp &= ~I2CR_MSTA; > > + writeb(temp, i2c_imx->base + IMX_I2C_I2CR); > > + /* setup chip registers to defaults */ > > + writeb(I2CR_IEN, i2c_imx->base + IMX_I2C_I2CR); > > + writeb(0, i2c_imx->base + IMX_I2C_I2SR); > > + /* > > + * This delay caused by an i.MXL hardware bug. > > + * If no (or too short) delay, no "STOP" bit will be generated. > > + */ > > + udelay(disable_delay); > > + /* Disable I2C controller */ > > + writeb(0, i2c_imx->base + IMX_I2C_I2CR); > > +} > > + > > +static void __init i2c_imx_set_clk(struct imx_i2c_struct *i2c_imx, > > + unsigned int rate) > > +{ > > + unsigned int i2c_clk_rate; > > + unsigned int div; > > + int i; > > + > > + /* Divider value calculation */ > > + i2c_clk_rate = clk_get_rate(i2c_imx->clk); > > + div = (i2c_clk_rate + rate - 1) / rate; > > + if (div < i2c_clk_div[0][0]) > > + i = 0; > > + else if (div > i2c_clk_div[ARRAY_SIZE(i2c_clk_div) - 1][0]) > > + i = ARRAY_SIZE(i2c_clk_div) - 1; > > + else > > + for (i = 0; i2c_clk_div[i][0] < div; i++); > > + > > + /* Write divider value to register */ > > + writeb(i2c_clk_div[i][1], i2c_imx->base + IMX_I2C_IFDR); > > + > > + /* > > + * There dummy delay is calculated. > > + * It should be about one I2C clock period long. > > + * This delay is used in I2C bus disable function > > + * to fix chip hardware bug. > > + */ > > + disable_delay = (500000U * i2c_clk_div[i][0] > > + + (i2c_clk_rate / 2) - 1) / (i2c_clk_rate / 2); > > + > > + /* dev_dbg() can't be used, because adapter is not yet registered */ > > +#ifdef CONFIG_I2C_DEBUG_BUS > > + printk(KERN_DEBUG "I2C: <%s> I2C_CLK=%d, REQ DIV=%d\n", > > + __func__, i2c_clk_rate, div); > > + printk(KERN_DEBUG "I2C: <%s> IFDR[IC]=0x%x, REAL DIV=%d\n", > > + __func__, i2c_clk_div[i][1], i2c_clk_div[i][0]); > > +#endif not using dev_xxx() here? > > +} > > + > > +static irqreturn_t i2c_imx_isr(int irq, void *dev_id) > > +{ > > + struct imx_i2c_struct *i2c_imx = dev_id; > > + unsigned int temp; > > + > > + temp = readb(i2c_imx->base + IMX_I2C_I2SR); > > + if (temp & I2SR_IIF) { > > + /* save status register */ > > + i2c_imx->i2csr = temp; > > + temp &= ~I2SR_IIF; > > + writeb(temp, i2c_imx->base + IMX_I2C_I2SR); > > + wake_up_interruptible(&i2c_imx->queue); > > + } > > + > > + return IRQ_HANDLED; > > +} it would be nice if we returned IRQ_NONE if we didn't have anything to handle. > > + > > +static int i2c_imx_write(struct imx_i2c_struct *i2c_imx, struct i2c_msg *msgs) > > +{ > > + int i, result; > > + > > + dev_dbg(&i2c_imx->adapter.dev, "<%s> write slave address: addr=0x%x\n", > > + __func__, msgs->addr); > > + > > + /* write slave address */ > > + writeb(msgs->addr, i2c_imx->base + IMX_I2C_I2DR); > > This is wrong! You have to double the address before writing to the > register. do you have a reference to the datasheet? > > + result = i2c_imx_trx_complete(i2c_imx); > > + if (result) > > + return result; > > + result = i2c_imx_acked(i2c_imx); > > + if (result) > > + return result; > > + dev_dbg(&i2c_imx->adapter.dev, "<%s> write data\n", __func__); > > + > > + /* write data */ > > + for (i = 0; i < msgs->len; i++) { > > + dev_dbg(&i2c_imx->adapter.dev, > > + "<%s> write byte: B%d=0x%X\n", > > + __func__, i, msgs->buf[i]); > > + writeb(msgs->buf[i], i2c_imx->base + IMX_I2C_I2DR); > > + result = i2c_imx_trx_complete(i2c_imx); > > + if (result) > > + return result; > > + result = i2c_imx_acked(i2c_imx); > > + if (result) > > + return result; > > + } > > + return 0; > > +} > > + > > +static int i2c_imx_read(struct imx_i2c_struct *i2c_imx, struct i2c_msg *msgs) > > +{ > > + int i, result; > > + unsigned int temp; > > + > > + dev_dbg(&i2c_imx->adapter.dev, > > + "<%s> write slave address: addr=0x%x\n", > > + __func__, msgs->addr | 0x01); > > + > > + /* write slave address */ > > + writeb(msgs->addr | 0x01, i2c_imx->base + IMX_I2C_I2DR); > > Same here - (msgs->addr << 1) | 0x01. good catch, missed that myself. > > + result = i2c_imx_trx_complete(i2c_imx); > > + if (result) > > + return result; > > + result = i2c_imx_acked(i2c_imx); > > + if (result) > > + return result; > > + > > + dev_dbg(&i2c_imx->adapter.dev, "<%s> setup bus\n", __func__); > > + > > + /* setup bus to read data */ > > + temp = readb(i2c_imx->base + IMX_I2C_I2CR); > > + temp &= ~I2CR_MTX; > > + if (msgs->len - 1) > > + temp &= ~I2CR_TXAK; > > + writeb(temp, i2c_imx->base + IMX_I2C_I2CR); > > + readb(i2c_imx->base + IMX_I2C_I2DR); /* dummy read */ > > + > > + dev_dbg(&i2c_imx->adapter.dev, "<%s> read data\n", __func__); > > + > > + /* read data */ > > + for (i = 0; i < msgs->len; i++) { > > + result = i2c_imx_trx_complete(i2c_imx); > > + if (result) > > + return result; > > + if (i == (msgs->len - 1)) { > > + dev_dbg(&i2c_imx->adapter.dev, > > + "<%s> clear MSTA\n", __func__); > > + temp = readb(i2c_imx->base + IMX_I2C_I2CR); > > + temp &= ~I2CR_MSTA; > > + writeb(temp, i2c_imx->base + IMX_I2C_I2CR); > > + } else if (i == (msgs->len - 2)) { > > + dev_dbg(&i2c_imx->adapter.dev, > > + "<%s> set TXAK\n", __func__); > > + temp = readb(i2c_imx->base + IMX_I2C_I2CR); > > + temp |= I2CR_TXAK; > > + writeb(temp, i2c_imx->base + IMX_I2C_I2CR); > > + } > > + msgs->buf[i] = readb(i2c_imx->base + IMX_I2C_I2DR); > > + dev_dbg(&i2c_imx->adapter.dev, > > + "<%s> read byte: B%d=0x%X\n", > > + __func__, i, msgs->buf[i]); > > + } > > + return 0; > > +} > > + > > +static int i2c_imx_xfer(struct i2c_adapter *adapter, > > + struct i2c_msg *msgs, int num) > > +{ > > + unsigned int i, temp; > > + int result; > > + struct imx_i2c_struct *i2c_imx = i2c_get_adapdata(adapter); > > + > > + dev_dbg(&i2c_imx->adapter.dev, "<%s>\n", __func__); > > + > > + /* Check or i2c bus is not busy */ > > + result = i2c_imx_bus_busy(i2c_imx); > > + if (result) > > + goto fail0; > > + > > + /* Start I2C transfer */ > > + i2c_imx_start(i2c_imx); > > + > > + /* read/write data */ > > + for (i = 0; i < num; i++) { > > + if (i) { > > + dev_dbg(&i2c_imx->adapter.dev, > > + "<%s> repeated start\n", __func__); > > + temp = readb(i2c_imx->base + IMX_I2C_I2CR); > > + temp |= I2CR_RSTA; > > + writeb(temp, i2c_imx->base + IMX_I2C_I2CR); > > + } > > + dev_dbg(&i2c_imx->adapter.dev, > > + "<%s> transfer message: %d\n", __func__, i); > > + /* write/read data */ > > +#ifdef CONFIG_I2C_DEBUG_BUS > > + temp = readb(i2c_imx->base + IMX_I2C_I2CR); > > + dev_dbg(&i2c_imx->adapter.dev, "<%s> CONTROL: IEN=%d, IIEN=%d, " > > + "MSTA=%d, MTX=%d, TXAK=%d, RSTA=%d\n", __func__, > > + (temp & I2CR_IEN ? 1 : 0), (temp & I2CR_IIEN ? 1 : 0), > > + (temp & I2CR_MSTA ? 1 : 0), (temp & I2CR_MTX ? 1 : 0), > > + (temp & I2CR_TXAK ? 1 : 0), (temp & I2CR_RSTA ? 1 : 0)); > > + temp = readb(i2c_imx->base + IMX_I2C_I2SR); > > + dev_dbg(&i2c_imx->adapter.dev, > > + "<%s> STATUS: ICF=%d, IAAS=%d, IBB=%d, " > > + "IAL=%d, SRW=%d, IIF=%d, RXAK=%d\n", __func__, > > + (temp & I2SR_ICF ? 1 : 0), (temp & I2SR_IAAS ? 1 : 0), > > + (temp & I2SR_IBB ? 1 : 0), (temp & I2SR_IAL ? 1 : 0), > > + (temp & I2SR_SRW ? 1 : 0), (temp & I2SR_IIF ? 1 : 0), > > + (temp & I2SR_RXAK ? 1 : 0)); > > +#endif > > + if (msgs[i].flags & I2C_M_RD) > > + result = i2c_imx_read(i2c_imx, &msgs[i]); > > + else > > + result = i2c_imx_write(i2c_imx, &msgs[i]); > > + } > > + > > +fail0: > > + /* Stop I2C transfer */ > > + i2c_imx_stop(i2c_imx); > > + > > + dev_dbg(&i2c_imx->adapter.dev, "<%s> exit with: %s: %d\n", __func__, > > + (result < 0) ? "error" : "success msg", > > + (result < 0) ? result : num); > > + return (result < 0) ? result : num; > > +} > > + > > +static u32 i2c_imx_func(struct i2c_adapter *adapter) > > +{ > > + return I2C_FUNC_I2C | I2C_FUNC_SMBUS_EMUL; > > +} > > + > > +static struct i2c_algorithm i2c_imx_algo = { > > + .master_xfer = i2c_imx_xfer, > > + .functionality = i2c_imx_func, > > +}; > > + > > +static int __init i2c_imx_probe(struct platform_device *pdev) > > +{ > > + struct imx_i2c_struct *i2c_imx; > > + struct resource *res; > > + struct imxi2c_platform_data *pdata; > > + void __iomem *base; > > + int irq; > > + int res_size; > > + int ret; > > + > > + dev_dbg(&pdev->dev, "<%s>\n", __func__); > > + > > + res = platform_get_resource(pdev, IORESOURCE_MEM, 0); > > + if (!res) { > > + dev_err(&pdev->dev, "can't get device resources\n"); > > + return -ENODEV; > > + } > > + irq = platform_get_irq(pdev, 0); > > + if (irq < 0) { > > + dev_err(&pdev->dev, "can't get irq number\n"); > > + return -ENODEV; > > + } > > + > > + pdata = pdev->dev.platform_data; > > + if (!pdata) { > > + dev_err(&pdev->dev, "I2C driver needs platform data\n"); > > + return -ENODEV; > > + } > > + > > + res_size = res->end - res->start + 1; > > You can use resource_size() > > > + if (!request_mem_region(res->start, res_size, res->name)) { > > + dev_err(&pdev->dev, "can't allocate %d bytes at %d address\n", > > + res_size, res->start); > > + return -ENOMEM; > > + } > > I was once told, one doesn't need request_mem_region() on regions from > platform data resources - this is already done implicitly. I belive that request_mem_region() is the correct thing to do, IIRC, the platform_device_register() calls insert_resource() on all the resource regions of the device so that it is guarnateed to appear in the /proc/ioport or /proc/iomem map. However I belive this does not actually make a reservation of that resource. > > + > > + if (pdata->init) { > > + ret = pdata->init(&pdev->dev); > > + if (ret) > > + goto fail0; > > + } > > + > > + base = ioremap(res->start, res_size); > > + if (!base) { > > + dev_err(&pdev->dev, "ioremap failed\n"); > > + ret = -EIO; > > + goto fail1; > > + } > > + > > + i2c_imx = kzalloc(sizeof(struct imx_i2c_struct), GFP_KERNEL); > > + if (!i2c_imx) { > > + dev_err(&pdev->dev, "can't allocate interface\n"); > > + ret = -ENOMEM; > > + goto fail2; > > + } > > + > > + /* Setup i2c_imx driver structure */ > > + strcpy(i2c_imx->adapter.name, pdev->name); > > + i2c_imx->adapter.owner = THIS_MODULE; > > + i2c_imx->adapter.algo = &i2c_imx_algo; > > + i2c_imx->adapter.dev.parent = &pdev->dev; > > + i2c_imx->adapter.nr = pdev->id; > > + i2c_imx->irq = irq; > > + i2c_imx->base = base; > > + i2c_imx->res = res; > > + > > + /* Get I2C clock */ > > + i2c_imx->clk = clk_get(NULL, "i2c_clk"); > > There can be several i2c busses on the system, so you want: > > + i2c_imx->clk = clk_get(&pdev->dev, "i2c_clk"); tbh, the bus clock should be defined as NULL, any extra clocks for a device get names. However this may need fixing in the platform as well. I belive RMK is already making these changes for AMBA and I will probably follow suit with the s3c architectures as soon as the rest of the development work is out of the way. > > + if (IS_ERR(i2c_imx->clk)) { > > + ret = PTR_ERR(i2c_imx->clk); > > + dev_err(&pdev->dev, "can't get I2C clock\n"); > > + goto fail3; > > + } > > + clk_enable(i2c_imx->clk); > > + > > + /* Request IRQ */ > > + ret = request_irq(i2c_imx->irq, i2c_imx_isr, > > + IRQF_DISABLED, pdev->name, i2c_imx); do you really need to run your i2c isr with interrupts disabled? > > + if (ret) { > > + dev_err(&pdev->dev, "can't claim irq %d\n", i2c_imx->irq); > > + goto fail4; > > + } > > + > > + /* Init queue */ > > + init_waitqueue_head(&i2c_imx->queue); > > + > > + /* Set up adapter data */ > > + i2c_set_adapdata(&i2c_imx->adapter, i2c_imx); > > + > > + /* Set up clock divider */ > > + i2c_imx_set_clk(i2c_imx, clkfreq); > > + > > + /* Set up chip registers to defaults */ > > + writeb(0, i2c_imx->base + IMX_I2C_I2CR); > > + writeb(0, i2c_imx->base + IMX_I2C_I2SR); > > + > > + /* Add I2C adapter */ > > + ret = i2c_add_numbered_adapter(&i2c_imx->adapter); > > + if (ret < 0) { > > + dev_err(&pdev->dev, "registration failed\n"); > > + goto fail5; > > + } > > + > > + /* Set up platform driver data */ > > + platform_set_drvdata(pdev, i2c_imx); > > + > > + dev_dbg(&i2c_imx->adapter.dev, "claimed irq %d\n", i2c_imx->irq); > > + dev_dbg(&i2c_imx->adapter.dev, "device resources from 0x%x to 0x%x\n", > > + i2c_imx->res->start, i2c_imx->res->end); > > + dev_dbg(&i2c_imx->adapter.dev, "allocated %d bytes at 0x%x \n", > > + res_size, i2c_imx->res->start); > > + dev_dbg(&i2c_imx->adapter.dev, "adapter name: \"%s\"\n", > > + i2c_imx->adapter.name); > > + dev_dbg(&i2c_imx->adapter.dev, "IMX I2C adapter registered\n"); > > + > > + return 0; /* Return OK */ > > + > > +fail5: > > + free_irq(i2c_imx->irq, i2c_imx); > > +fail4: > > + clk_disable(i2c_imx->clk); > > + clk_put(i2c_imx->clk); > > +fail3: > > + kfree(i2c_imx); > > +fail2: > > + iounmap(base); > > +fail1: > > + if (pdata->exit) > > + pdata->exit(&pdev->dev); > > +fail0: > > + release_mem_region(res->start, res_size); > > + return ret; /* Return error number */ > > +} > > + > > +static int __exit i2c_imx_remove(struct platform_device *pdev) > > +{ > > + struct imx_i2c_struct *i2c_imx = platform_get_drvdata(pdev); > > + struct imxi2c_platform_data *pdata = pdev->dev.platform_data; > > + > > + /* remove adapter */ > > + dev_dbg(&i2c_imx->adapter.dev, "adapter removed\n"); > > + i2c_del_adapter(&i2c_imx->adapter); > > + platform_set_drvdata(pdev, NULL); > > + > > + /* free interrupt */ > > + free_irq(i2c_imx->irq, i2c_imx); > > + > > + /* setup chip registers to defaults */ > > + writeb(0, i2c_imx->base + IMX_I2C_IADR); > > + writeb(0, i2c_imx->base + IMX_I2C_IFDR); > > + writeb(0, i2c_imx->base + IMX_I2C_I2CR); > > + writeb(0, i2c_imx->base + IMX_I2C_I2SR); > > + > > + /* Shut down hardware */ > > + if (pdata->exit) > > + pdata->exit(&pdev->dev); > > + > > + /* Disable I2C clock */ > > + clk_disable(i2c_imx->clk); > > + clk_put(i2c_imx->clk); > > + > > + /* Release memory */ > > + release_mem_region(i2c_imx->res->start, > > + i2c_imx->res->end - i2c_imx->res->start + 1); > > + iounmap(i2c_imx->base); > > + kfree(i2c_imx); > > + return 0; > > +} > > + > > +static struct platform_driver i2c_imx_driver = { > > + .probe = i2c_imx_probe, > > + .remove = __exit_p(i2c_imx_remove), > > + .driver = { > > + .name = DRIVER_NAME, > > + .owner = THIS_MODULE, > > + } > > +}; > > + > > +static int __init i2c_adap_imx_init(void) > > +{ > > + return platform_driver_probe(&i2c_imx_driver, i2c_imx_probe); > > +} > > + > > +static void __exit i2c_adap_imx_exit(void) > > +{ > > + platform_driver_unregister(&i2c_imx_driver); > > +} > > + > > +module_init(i2c_adap_imx_init); > > +module_exit(i2c_adap_imx_exit); > > + > > +module_param(clkfreq, uint, S_IRUGO); > > +MODULE_PARM_DESC(clkfreq, "desired IMX I2C Clock Rate in Hz"); > > Making clkfreq a module parameter you force the same frequency on all i2c > busses. On my i.MX31 system 2 busses are internal and one goes to a > connector, to which a camera is connected. This third bus can only handle > a lower frequency, which, however, doesn't mean we also have to throttle > the other two busses. Can we put this into platform data? > > > + > > +MODULE_LICENSE("GPL"); > > +MODULE_AUTHOR("Darius Augulis"); > > +MODULE_DESCRIPTION("I2C adapter driver for IMX I2C bus"); > > +MODULE_ALIAS("platform:" DRIVER_NAME); > > Index: linux-2.6.28-rc6/drivers/i2c/busses/Kconfig > > =================================================================== > > --- linux-2.6.28-rc6.orig/drivers/i2c/busses/Kconfig > > +++ linux-2.6.28-rc6/drivers/i2c/busses/Kconfig > > @@ -355,6 +355,16 @@ config I2C_IBM_IIC > > This driver can also be built as a module. If so, the module > > will be called i2c-ibm_iic. > > > > +config I2C_IMX > > + tristate "IMX I2C interface" > > + depends on ARCH_MXC > > + help > > + Say Y here if you want to use the IIC bus controller on > > + the Freescale i.MX/MXC processors. > > + > > + This driver can also be built as a module. If so, the module > > + will be called i2c-imx. > > + > > config I2C_IOP3XX > > tristate "Intel IOPx3xx and IXP4xx on-chip I2C interface" > > depends on ARCH_IOP32X || ARCH_IOP33X || ARCH_IXP4XX || ARCH_IOP13XX > > Index: linux-2.6.28-rc6/drivers/i2c/busses/Makefile > > =================================================================== > > --- linux-2.6.28-rc6.orig/drivers/i2c/busses/Makefile > > +++ linux-2.6.28-rc6/drivers/i2c/busses/Makefile > > @@ -33,6 +33,7 @@ obj-$(CONFIG_I2C_DAVINCI) += i2c-davinci > > obj-$(CONFIG_I2C_GPIO) += i2c-gpio.o > > obj-$(CONFIG_I2C_HIGHLANDER) += i2c-highlander.o > > obj-$(CONFIG_I2C_IBM_IIC) += i2c-ibm_iic.o > > +obj-$(CONFIG_I2C_IMX) += i2c-imx.o > > obj-$(CONFIG_I2C_IOP3XX) += i2c-iop3xx.o > > obj-$(CONFIG_I2C_IXP2000) += i2c-ixp2000.o > > obj-$(CONFIG_I2C_MPC) += i2c-mpc.o > > Index: linux-2.6.28-rc6/arch/arm/plat-mxc/include/mach/i2c.h > > =================================================================== > > --- /dev/null > > +++ linux-2.6.28-rc6/arch/arm/plat-mxc/include/mach/i2c.h > > @@ -0,0 +1,17 @@ > > +/* > > + * i2c.h - i.MX I2C driver header file > > + * > > + * Copyright (c) 2008, Darius Augulis <augulis.darius-Re5JQEeQqe8AvxtiuMwx3w@public.gmane.org> > > + * > > + * This file is released under the GPLv2 > > + */ > > + > > +#ifndef __ASM_ARCH_I2C_H_ > > +#define __ASM_ARCH_I2C_H_ > > + kernel doc comments would be nice for this to actually say what is expected of the implementation and what these calls are there for. > > +struct imxi2c_platform_data { > > + int (*init)(struct device *dev); > > + int (*exit)(struct device *dev); > > What are you going to use .exit() for? Is it really needed? Even if it is, > it can easily return void I guess? > > > +}; > > + > > +#endif /* __ASM_ARCH_I2C_H_ */ > > Thanks > Guennadi > --- > Guennadi Liakhovetski, Ph.D. > Freelance Open-Source Software Developer > -- > To unsubscribe from this list: send the line "unsubscribe linux-i2c" in > the body of a message to majordomo-u79uwXL29TY76Z2rM5mHXA@public.gmane.org > More majordomo info at http://vger.kernel.org/majordomo-info.html -- Ben (ben-elnMNo+KYs3YtjvyW6yDsg@public.gmane.org, http://www.fluff.org/) 'a smiley only costs 4 bytes' ^ permalink raw reply [flat|nested] 21+ messages in thread
[parent not found: <20081203234317.GA4256-elnMNo+KYs3pIgCt6eIbzw@public.gmane.org>]
* Re: [PATCH][resend] iMX/MXC support for I2C [not found] ` <20081203234317.GA4256-elnMNo+KYs3pIgCt6eIbzw@public.gmane.org> @ 2008-12-04 9:59 ` Darius [not found] ` <4937AA1C.8050004-Re5JQEeQqe8AvxtiuMwx3w@public.gmane.org> 0 siblings, 1 reply; 21+ messages in thread From: Darius @ 2008-12-04 9:59 UTC (permalink / raw) To: linux-i2c-u79uwXL29TY76Z2rM5mHXA Cc: linux-arm-kernel-xIg/pKzrS19vn6HldHNs0ANdhmdF6hFW >>> +static int i2c_imx_acked(struct imx_i2c_struct *i2c_imx) >>> +{ >>> + unsigned long orig_jiffies = jiffies; >>> + >>> + /* Wait for ack */ >>> + while (readb(i2c_imx->base + IMX_I2C_I2SR) & I2SR_RXAK) { >>> + if (signal_pending(current)) { >>> + dev_dbg(&i2c_imx->adapter.dev, >>> + "<%s> I2C Interrupted\n", __func__); >>> + return -EINTR; >>> + } >>> + if (time_after(jiffies, orig_jiffies + HZ)) { >>> + dev_dbg(&i2c_imx->adapter.dev, >>> + "<%s> No ACK\n", __func__); >>> + return -EIO; >>> + } >>> + schedule(); >>> + } >> 1s is way too long here! i2cdetect takes ages to complete. The mxc driver >> doesn't wait here at all - as soon as a tx is complete, it checks the ack >> status. Could you verify if this would work on your MX1 too? > > That does seem a problem, if we can detect a NAK immediately then all > the better. I tested on MX1. There delay is not needed at all. > >>> + >>> + dev_dbg(&i2c_imx->adapter.dev, "<%s> ACK received\n", __func__); >>> + return 0; /* No ACK */ >>> +} >>> + >>> +static void i2c_imx_start(struct imx_i2c_struct *i2c_imx) >>> +{ >>> + unsigned int temp = 0; >>> + >>> + dev_dbg(&i2c_imx->adapter.dev, "<%s>\n", __func__); >>> + >>> + /* Enable I2C controller */ >>> + writeb(I2CR_IEN, i2c_imx->base + IMX_I2C_I2CR); >>> + /* Start I2C transaction */ >>> + temp = readb(i2c_imx->base + IMX_I2C_I2CR); >>> + temp |= I2CR_MSTA; >>> + writeb(temp, i2c_imx->base + IMX_I2C_I2CR); >>> + temp |= I2CR_IIEN | I2CR_MTX | I2CR_TXAK; >>> + writeb(temp, i2c_imx->base + IMX_I2C_I2CR); >>> +} >>> + >>> +static void i2c_imx_stop(struct imx_i2c_struct *i2c_imx) >>> +{ >>> + unsigned int temp = 0; >>> + >>> + /* Stop I2C transaction */ >>> + dev_dbg(&i2c_imx->adapter.dev, "<%s>\n", __func__); >>> + temp = readb(i2c_imx->base + IMX_I2C_I2CR); >>> + temp &= ~I2CR_MSTA; >>> + writeb(temp, i2c_imx->base + IMX_I2C_I2CR); >>> + /* setup chip registers to defaults */ >>> + writeb(I2CR_IEN, i2c_imx->base + IMX_I2C_I2CR); >>> + writeb(0, i2c_imx->base + IMX_I2C_I2SR); >>> + /* >>> + * This delay caused by an i.MXL hardware bug. >>> + * If no (or too short) delay, no "STOP" bit will be generated. >>> + */ >>> + udelay(disable_delay); >>> + /* Disable I2C controller */ >>> + writeb(0, i2c_imx->base + IMX_I2C_I2CR); >>> +} >>> + >>> +static void __init i2c_imx_set_clk(struct imx_i2c_struct *i2c_imx, >>> + unsigned int rate) >>> +{ >>> + unsigned int i2c_clk_rate; >>> + unsigned int div; >>> + int i; >>> + >>> + /* Divider value calculation */ >>> + i2c_clk_rate = clk_get_rate(i2c_imx->clk); >>> + div = (i2c_clk_rate + rate - 1) / rate; >>> + if (div < i2c_clk_div[0][0]) >>> + i = 0; >>> + else if (div > i2c_clk_div[ARRAY_SIZE(i2c_clk_div) - 1][0]) >>> + i = ARRAY_SIZE(i2c_clk_div) - 1; >>> + else >>> + for (i = 0; i2c_clk_div[i][0] < div; i++); >>> + >>> + /* Write divider value to register */ >>> + writeb(i2c_clk_div[i][1], i2c_imx->base + IMX_I2C_IFDR); >>> + >>> + /* >>> + * There dummy delay is calculated. >>> + * It should be about one I2C clock period long. >>> + * This delay is used in I2C bus disable function >>> + * to fix chip hardware bug. >>> + */ >>> + disable_delay = (500000U * i2c_clk_div[i][0] >>> + + (i2c_clk_rate / 2) - 1) / (i2c_clk_rate / 2); >>> + >>> + /* dev_dbg() can't be used, because adapter is not yet registered */ >>> +#ifdef CONFIG_I2C_DEBUG_BUS >>> + printk(KERN_DEBUG "I2C: <%s> I2C_CLK=%d, REQ DIV=%d\n", >>> + __func__, i2c_clk_rate, div); >>> + printk(KERN_DEBUG "I2C: <%s> IFDR[IC]=0x%x, REAL DIV=%d\n", >>> + __func__, i2c_clk_div[i][1], i2c_clk_div[i][0]); >>> +#endif > > not using dev_xxx() here? because adapter is not yet registered. > >>> +} >>> + >>> +static irqreturn_t i2c_imx_isr(int irq, void *dev_id) >>> +{ >>> + struct imx_i2c_struct *i2c_imx = dev_id; >>> + unsigned int temp; >>> + >>> + temp = readb(i2c_imx->base + IMX_I2C_I2SR); >>> + if (temp & I2SR_IIF) { >>> + /* save status register */ >>> + i2c_imx->i2csr = temp; >>> + temp &= ~I2SR_IIF; >>> + writeb(temp, i2c_imx->base + IMX_I2C_I2SR); >>> + wake_up_interruptible(&i2c_imx->queue); >>> + } >>> + >>> + return IRQ_HANDLED; >>> +} > > it would be nice if we returned IRQ_NONE if we didn't have anything > to handle. I will add this. >>> + if (!request_mem_region(res->start, res_size, res->name)) { >>> + dev_err(&pdev->dev, "can't allocate %d bytes at %d address\n", >>> + res_size, res->start); >>> + return -ENOMEM; >>> + } >> I was once told, one doesn't need request_mem_region() on regions from >> platform data resources - this is already done implicitly. > > I belive that request_mem_region() is the correct thing to do, IIRC, > the platform_device_register() calls insert_resource() on all the > resource regions of the device so that it is guarnateed to appear in > the /proc/ioport or /proc/iomem map. However I belive this does not > actually make a reservation of that resource. > confused... Guennadi, can you please explain more your mention? >>> + >>> + if (pdata->init) { >>> + ret = pdata->init(&pdev->dev); >>> + if (ret) >>> + goto fail0; >>> + } >>> + >>> + base = ioremap(res->start, res_size); >>> + if (!base) { >>> + dev_err(&pdev->dev, "ioremap failed\n"); >>> + ret = -EIO; >>> + goto fail1; >>> + } >>> + >>> + i2c_imx = kzalloc(sizeof(struct imx_i2c_struct), GFP_KERNEL); >>> + if (!i2c_imx) { >>> + dev_err(&pdev->dev, "can't allocate interface\n"); >>> + ret = -ENOMEM; >>> + goto fail2; >>> + } >>> + >>> + /* Setup i2c_imx driver structure */ >>> + strcpy(i2c_imx->adapter.name, pdev->name); >>> + i2c_imx->adapter.owner = THIS_MODULE; >>> + i2c_imx->adapter.algo = &i2c_imx_algo; >>> + i2c_imx->adapter.dev.parent = &pdev->dev; >>> + i2c_imx->adapter.nr = pdev->id; >>> + i2c_imx->irq = irq; >>> + i2c_imx->base = base; >>> + i2c_imx->res = res; >>> + >>> + /* Get I2C clock */ >>> + i2c_imx->clk = clk_get(NULL, "i2c_clk"); >> There can be several i2c busses on the system, so you want: >> >> + i2c_imx->clk = clk_get(&pdev->dev, "i2c_clk"); > > tbh, the bus clock should be defined as NULL, any extra clocks for > a device get names. However this may need fixing in the platform > as well. I belive RMK is already making these changes for AMBA and > I will probably follow suit with the s3c architectures as soon as > the rest of the development work is out of the way. > >>> + if (IS_ERR(i2c_imx->clk)) { >>> + ret = PTR_ERR(i2c_imx->clk); >>> + dev_err(&pdev->dev, "can't get I2C clock\n"); >>> + goto fail3; >>> + } >>> + clk_enable(i2c_imx->clk); >>> + >>> + /* Request IRQ */ >>> + ret = request_irq(i2c_imx->irq, i2c_imx_isr, >>> + IRQF_DISABLED, pdev->name, i2c_imx); > > do you really need to run your i2c isr with interrupts disabled? no, interrupts can be enabled. will fix. > > kernel doc comments would be nice for this to actually say what is > expected of the implementation and what these calls are there for. > >>> +struct imxi2c_platform_data { >>> + int (*init)(struct device *dev); >>> + int (*exit)(struct device *dev); >> What are you going to use .exit() for? Is it really needed? Even if it is, >> it can easily return void I guess? >> Where to put commnets? Right here in driver code or is better in /Documentation? ^ permalink raw reply [flat|nested] 21+ messages in thread
[parent not found: <4937AA1C.8050004-Re5JQEeQqe8AvxtiuMwx3w@public.gmane.org>]
* Re: [PATCH][resend] iMX/MXC support for I2C [not found] ` <4937AA1C.8050004-Re5JQEeQqe8AvxtiuMwx3w@public.gmane.org> @ 2008-12-04 10:20 ` Guennadi Liakhovetski 2008-12-05 15:17 ` Guennadi Liakhovetski 1 sibling, 0 replies; 21+ messages in thread From: Guennadi Liakhovetski @ 2008-12-04 10:20 UTC (permalink / raw) To: Darius; +Cc: linux-i2c-u79uwXL29TY76Z2rM5mHXA, Ben Dooks On Thu, 4 Dec 2008, Darius wrote: > > > > + if (!request_mem_region(res->start, res_size, res->name)) { > > > > + dev_err(&pdev->dev, "can't allocate %d bytes at %d address\n", > > > > + res_size, res->start); > > > > + return -ENOMEM; > > > > + } > > > I was once told, one doesn't need request_mem_region() on regions from > > > platform data resources - this is already done implicitly. > > > > I belive that request_mem_region() is the correct thing to do, IIRC, > > the platform_device_register() calls insert_resource() on all the > > resource regions of the device so that it is guarnateed to appear in > > the /proc/ioport or /proc/iomem map. However I belive this does not > > actually make a reservation of that resource. > > > > confused... Guennadi, can you please explain more your mention? Look at this: http://marc.info/?l=linux-sh&m=121642007016402&w=2 > > > > + /* Get I2C clock */ > > > > + i2c_imx->clk = clk_get(NULL, "i2c_clk"); > > > There can be several i2c busses on the system, so you want: > > > > > > + i2c_imx->clk = clk_get(&pdev->dev, "i2c_clk"); > > > > tbh, the bus clock should be defined as NULL, any extra clocks for > > a device get names. However this may need fixing in the platform > > as well. I belive RMK is already making these changes for AMBA and > > I will probably follow suit with the s3c architectures as soon as > > the rest of the development work is out of the way. Ben, what do you mean here? I thought Russell was against platform-specific clock names and for function names. But i2c_clk is a function name. I don't understand the difference between "bus clock" and "extra clocks" - on i.MX31 we have 3 i2c busses with separate clock sources, so, we have to use the platform device in clk_get to request a specific i2c clock matching the platform-device index. That's all what I meant. > > > > +struct imxi2c_platform_data { > > > > + int (*init)(struct device *dev); > > > > + int (*exit)(struct device *dev); > > > What are you going to use .exit() for? Is it really needed? Even if it is, > > > it can easily return void I guess? > > > > > Where to put commnets? Right here in driver code or is better in > /Documentation? Something like this: +/** + * @init: initialise the interface + * @exit: free interface resources + */ +struct imxi2c_platform_data { + int (*init)(struct device *dev); + int (*exit)(struct device *dev); Thanks Guennadi --- Guennadi Liakhovetski, Ph.D. Freelance Open-Source Software Developer ^ permalink raw reply [flat|nested] 21+ messages in thread
* Re: [PATCH][resend] iMX/MXC support for I2C [not found] ` <4937AA1C.8050004-Re5JQEeQqe8AvxtiuMwx3w@public.gmane.org> 2008-12-04 10:20 ` Guennadi Liakhovetski @ 2008-12-05 15:17 ` Guennadi Liakhovetski [not found] ` <Pine.LNX.4.64.0812051612500.5840-0199iw4Nj15frtckUFj5Ag@public.gmane.org> 1 sibling, 1 reply; 21+ messages in thread From: Guennadi Liakhovetski @ 2008-12-05 15:17 UTC (permalink / raw) To: Darius; +Cc: linux-i2c-u79uwXL29TY76Z2rM5mHXA On Thu, 4 Dec 2008, Darius wrote: > I tested on MX1. There delay is not needed at all. Darius, how exactly you tested on MX1? I just found out, that also i2c_smbus_{read,write}_word_data() doesn't work with imx driver on i.MX31, whereas the mxc driver does. Thanks Guennadi --- Guennadi Liakhovetski, Ph.D. Freelance Open-Source Software Developer ^ permalink raw reply [flat|nested] 21+ messages in thread
[parent not found: <Pine.LNX.4.64.0812051612500.5840-0199iw4Nj15frtckUFj5Ag@public.gmane.org>]
* Re: [PATCH][resend] iMX/MXC support for I2C [not found] ` <Pine.LNX.4.64.0812051612500.5840-0199iw4Nj15frtckUFj5Ag@public.gmane.org> @ 2008-12-08 9:18 ` Darius 0 siblings, 0 replies; 21+ messages in thread From: Darius @ 2008-12-08 9:18 UTC (permalink / raw) To: linux-i2c-u79uwXL29TY76Z2rM5mHXA Guennadi Liakhovetski wrote: > On Thu, 4 Dec 2008, Darius wrote: > >> I tested on MX1. There delay is not needed at all. > > Darius, how exactly you tested on MX1? I just found out, that also > i2c_smbus_{read,write}_word_data() doesn't work with imx driver on i.MX31, > whereas the mxc driver does. Hi Guennadi, I tested with OV7720, OV7670 and MT9V111 drivers. Last one uses smbus data transfers - all works 100% ok. Today I will post new patch version with most of your and Ben comments fixed. Darius. > > Thanks > Guennadi > --- > Guennadi Liakhovetski, Ph.D. > Freelance Open-Source Software Developer ^ permalink raw reply [flat|nested] 21+ messages in thread
* Re: [PATCH][resend] iMX/MXC support for I2C 2008-12-02 14:00 [PATCH][resend] iMX/MXC support for I2C Darius 2008-12-02 14:04 ` Darius 2008-12-03 11:17 ` Guennadi Liakhovetski @ 2008-12-03 12:56 ` Guennadi Liakhovetski [not found] ` <Pine.LNX.4.64.0812031355120.4717-0199iw4Nj15frtckUFj5Ag@public.gmane.org> 2 siblings, 1 reply; 21+ messages in thread From: Guennadi Liakhovetski @ 2008-12-03 12:56 UTC (permalink / raw) To: Darius Cc: linux-i2c-u79uwXL29TY76Z2rM5mHXA, linux-arm-kernel-xIg/pKzrS19vn6HldHNs0ANdhmdF6hFW One more thing: you also want to do something like this i2c_imx->adapter.class = I2C_CLASS_HWMON | I2C_CLASS_SPD; in i2c_imx_probe(). If you want, I can send you my current version of your driver, so you can compare. Thanks Guennadi --- Guennadi Liakhovetski, Ph.D. Freelance Open-Source Software Developer ^ permalink raw reply [flat|nested] 21+ messages in thread
[parent not found: <Pine.LNX.4.64.0812031355120.4717-0199iw4Nj15frtckUFj5Ag@public.gmane.org>]
* Re: [PATCH][resend] iMX/MXC support for I2C [not found] ` <Pine.LNX.4.64.0812031355120.4717-0199iw4Nj15frtckUFj5Ag@public.gmane.org> @ 2008-12-03 13:07 ` Darius 2008-12-03 13:27 ` Wolfram Sang 2008-12-03 20:58 ` Guennadi Liakhovetski 0 siblings, 2 replies; 21+ messages in thread From: Darius @ 2008-12-03 13:07 UTC (permalink / raw) To: linux-i2c-u79uwXL29TY76Z2rM5mHXA Cc: linux-arm-kernel-xIg/pKzrS19vn6HldHNs0ANdhmdF6hFW Guennadi Liakhovetski wrote: > One more thing: you also want to do something like this > > i2c_imx->adapter.class = I2C_CLASS_HWMON | I2C_CLASS_SPD; some time ago, it was I2C_CLASS_ALL, but Jean told, that it will be removed at all :) > > in i2c_imx_probe(). > > If you want, I can send you my current version of your driver, so you can > compare. yes, please send it. > > Thanks > Guennadi > --- > Guennadi Liakhovetski, Ph.D. > Freelance Open-Source Software Developer ^ permalink raw reply [flat|nested] 21+ messages in thread
* Re: [PATCH][resend] iMX/MXC support for I2C 2008-12-03 13:07 ` Darius @ 2008-12-03 13:27 ` Wolfram Sang 2008-12-03 20:58 ` Guennadi Liakhovetski 1 sibling, 0 replies; 21+ messages in thread From: Wolfram Sang @ 2008-12-03 13:27 UTC (permalink / raw) To: Darius Cc: linux-i2c-u79uwXL29TY76Z2rM5mHXA, linux-arm-kernel-xIg/pKzrS19vn6HldHNs0ANdhmdF6hFW [-- Attachment #1: Type: text/plain, Size: 740 bytes --] On Wed, Dec 03, 2008 at 03:07:08PM +0200, Darius wrote: > Guennadi Liakhovetski wrote: >> One more thing: you also want to do something like this >> >> i2c_imx->adapter.class = I2C_CLASS_HWMON | I2C_CLASS_SPD; > > some time ago, it was I2C_CLASS_ALL, but Jean told, that it will be removed at all :) I also think it is better to not ask for probing. Those flags were recently removed for some platforms, see for example commit 0a346dacee18ff69f6162d9860d723a058f47321 or commit 618b26d52843c0f85b8eb143cf2695d7f6fd072d. It usually causes more problems on embedded boards. Kind regards, Wolfram -- Dipl.-Ing. Wolfram Sang | http://www.pengutronix.de Pengutronix - Linux Solutions for Science and Industry [-- Attachment #2: Digital signature --] [-- Type: application/pgp-signature, Size: 197 bytes --] ^ permalink raw reply [flat|nested] 21+ messages in thread
* Re: [PATCH][resend] iMX/MXC support for I2C 2008-12-03 13:07 ` Darius 2008-12-03 13:27 ` Wolfram Sang @ 2008-12-03 20:58 ` Guennadi Liakhovetski [not found] ` <Pine.LNX.4.64.0812032155350.7961-0199iw4Nj15frtckUFj5Ag@public.gmane.org> 1 sibling, 1 reply; 21+ messages in thread From: Guennadi Liakhovetski @ 2008-12-03 20:58 UTC (permalink / raw) To: Darius Cc: linux-i2c-u79uwXL29TY76Z2rM5mHXA, linux-arm-kernel-xIg/pKzrS19vn6HldHNs0ANdhmdF6hFW Hi Darius, it would be better, if you didn't drop my email from cc, thanks. On Wed, 3 Dec 2008, Darius wrote: > Guennadi Liakhovetski wrote: > > One more thing: you also want to do something like this > > > > i2c_imx->adapter.class = I2C_CLASS_HWMON | I2C_CLASS_SPD; > > some time ago, it was I2C_CLASS_ALL, but Jean told, that it will be removed at > all :) > > > > > in i2c_imx_probe(). > > > > If you want, I can send you my current version of your driver, so you can > > compare. > > yes, please send it. Below. Thanks Guennadi --- Guennadi Liakhovetski, Ph.D. Freelance Open-Source Software Developer >From 1d592e3a94b3e13537a680e3e0198c345248692d Mon Sep 17 00:00:00 2001 From: Guennadi Liakhovetski <g.liakhovetski-Mmb7MZpHnFY@public.gmane.org> Date: Wed, 3 Dec 2008 17:02:37 +0100 Subject: [PATCH] i2x-imx: fix client addresses, remove needless delays, support multiple busses With above bugs fixed now also tested to work on i.MX31. bus frequency is now configured over platform data, which allows different frequencies on different busses. Add an adapter.class assignment. Signed-off-by: Guennadi Liakhovetski <lg-ynQEQJNshbs@public.gmane.org> --- arch/arm/plat-mxc/include/mach/i2c.h | 8 ++ drivers/i2c/busses/i2c-imx.c | 143 ++++++++++++++++------------------ 2 files changed, 76 insertions(+), 75 deletions(-) diff --git a/arch/arm/plat-mxc/include/mach/i2c.h b/arch/arm/plat-mxc/include/mach/i2c.h index 9e6a9d8..37d068b 100644 --- a/arch/arm/plat-mxc/include/mach/i2c.h +++ b/arch/arm/plat-mxc/include/mach/i2c.h @@ -10,8 +10,16 @@ #define __ASM_ARCH_I2C_H_ struct imxi2c_platform_data { + u32 i2c_clk; int (*init)(struct device *dev); int (*exit)(struct device *dev); }; +struct imxi2c_board_data { + struct imxi2c_platform_data data; + unsigned int id; +}; + +extern int mxc_i2c_register_adapters(struct imxi2c_board_data *data, int n); + #endif /* __ASM_ARCH_I2C_H_ */ diff --git a/drivers/i2c/busses/i2c-imx.c b/drivers/i2c/busses/i2c-imx.c index fd2d16d..eeea7d7 100644 --- a/drivers/i2c/busses/i2c-imx.c +++ b/drivers/i2c/busses/i2c-imx.c @@ -23,12 +23,6 @@ * Implementation of I2C Adapter/Algorithm Driver * for I2C Bus integrated in Freescale i.MX/MXC processors * - * module parameters: - * - clkfreq: - * Sets the desired clock rate - * The default value is 100000 - * Max value is 400000 - * * Derived from Motorola GSG China I2C example driver * * Copyright (C) 2005 Torsten Koschorrek <koschorrek at synertronixx.de @@ -64,14 +58,6 @@ /* This will be the driver name the kernel reports */ #define DRIVER_NAME "imx-i2c" -/* Default values of module parameters */ -#define IMX_I2C_BIT_RATE 100000 /* 100kHz */ - -/* Timeouts */ -#define I2C_IMX_TIME_BUSY 2000 /* loop count */ -#define I2C_IMX_TIME_ACK 2000 /* loop count */ -#define I2C_IMX_TIME_TRX 5 /* seconds */ - /* IMX I2C registers */ #define IMX_I2C_IADR 0x00 /* i2c slave address */ #define IMX_I2C_IFDR 0x04 /* i2c frequency divider */ @@ -97,7 +83,6 @@ /** Variables ****************************************************************** *******************************************************************************/ -static unsigned int clkfreq = IMX_I2C_BIT_RATE; static unsigned int disable_delay; /* Dummy delay */ /* @@ -135,6 +120,21 @@ struct imx_i2c_struct { unsigned long i2csr; }; +/* + * i.MX has 32-bit registers, i.MX31 16-bit, but only 8 bits are used. So far + * 8-bit access seems to work on both. + */ +static inline unsigned int reg_read(const void __iomem *p) +{ + return __raw_readb(p); +} + +static inline void reg_write(unsigned int value, void __iomem *p) +{ + __raw_writeb(value, p); +} + + /** Functions for IMX I2C adapter driver *************************************** *******************************************************************************/ @@ -145,7 +145,7 @@ static int i2c_imx_bus_busy(struct imx_i2c_struct *i2c_imx) dev_dbg(&i2c_imx->adapter.dev, "<%s>\n", __func__); /* wait for bus not busy */ - while (readb(i2c_imx->base + IMX_I2C_I2SR) & I2SR_IBB) { + while (reg_read(i2c_imx->base + IMX_I2C_I2SR) & I2SR_IBB) { if (signal_pending(current)) { dev_dbg(&i2c_imx->adapter.dev, "<%s> I2C Interrupted\n", __func__); @@ -167,7 +167,7 @@ static int i2c_imx_trx_complete(struct imx_i2c_struct *i2c_imx) int result; result = wait_event_interruptible_timeout(i2c_imx->queue, - i2c_imx->i2csr & I2SR_IIF, I2C_IMX_TIME_TRX * HZ); + i2c_imx->i2csr & I2SR_IIF, i2c_imx->adapter.timeout); if (unlikely(result < 0)) { dev_dbg(&i2c_imx->adapter.dev, "<%s> result < 0\n", __func__); @@ -183,62 +183,50 @@ static int i2c_imx_trx_complete(struct imx_i2c_struct *i2c_imx) static int i2c_imx_acked(struct imx_i2c_struct *i2c_imx) { - unsigned long orig_jiffies = jiffies; - - /* Wait for ack */ - while (readb(i2c_imx->base + IMX_I2C_I2SR) & I2SR_RXAK) { - if (signal_pending(current)) { - dev_dbg(&i2c_imx->adapter.dev, - "<%s> I2C Interrupted\n", __func__); - return -EINTR; - } - if (time_after(jiffies, orig_jiffies + HZ)) { - dev_dbg(&i2c_imx->adapter.dev, - "<%s> No ACK\n", __func__); - return -EIO; - } - schedule(); + if (reg_read(i2c_imx->base + IMX_I2C_I2SR) & I2SR_RXAK) { + dev_dbg(&i2c_imx->adapter.dev, "<%s> No ACK\n", __func__); + return -EIO; /* No ACK */ } dev_dbg(&i2c_imx->adapter.dev, "<%s> ACK received\n", __func__); - return 0; /* No ACK */ + return 0; } static void i2c_imx_start(struct imx_i2c_struct *i2c_imx) { - unsigned int temp = 0; + unsigned int temp; dev_dbg(&i2c_imx->adapter.dev, "<%s>\n", __func__); /* Enable I2C controller */ - writeb(I2CR_IEN, i2c_imx->base + IMX_I2C_I2CR); + reg_write(I2CR_IEN, i2c_imx->base + IMX_I2C_I2CR); /* Start I2C transaction */ - temp = readb(i2c_imx->base + IMX_I2C_I2CR); + temp = reg_read(i2c_imx->base + IMX_I2C_I2CR); temp |= I2CR_MSTA; - writeb(temp, i2c_imx->base + IMX_I2C_I2CR); + reg_write(temp, i2c_imx->base + IMX_I2C_I2CR); temp |= I2CR_IIEN | I2CR_MTX | I2CR_TXAK; - writeb(temp, i2c_imx->base + IMX_I2C_I2CR); + reg_write(temp, i2c_imx->base + IMX_I2C_I2CR); } static void i2c_imx_stop(struct imx_i2c_struct *i2c_imx) { - unsigned int temp = 0; + unsigned int temp; /* Stop I2C transaction */ dev_dbg(&i2c_imx->adapter.dev, "<%s>\n", __func__); - temp = readb(i2c_imx->base + IMX_I2C_I2CR); + temp = reg_read(i2c_imx->base + IMX_I2C_I2CR); temp &= ~I2CR_MSTA; - writeb(temp, i2c_imx->base + IMX_I2C_I2CR); + reg_write(temp, i2c_imx->base + IMX_I2C_I2CR); /* setup chip registers to defaults */ - writeb(I2CR_IEN, i2c_imx->base + IMX_I2C_I2CR); - writeb(0, i2c_imx->base + IMX_I2C_I2SR); + reg_write(I2CR_IEN, i2c_imx->base + IMX_I2C_I2CR); + reg_write(0, i2c_imx->base + IMX_I2C_I2SR); /* * This delay caused by an i.MXL hardware bug. * If no (or too short) delay, no "STOP" bit will be generated. */ udelay(disable_delay); /* Disable I2C controller */ - writeb(0, i2c_imx->base + IMX_I2C_I2CR); + reg_write(0, i2c_imx->base + IMX_I2C_I2CR); } static void __init i2c_imx_set_clk(struct imx_i2c_struct *i2c_imx, @@ -259,7 +247,7 @@ static void __init i2c_imx_set_clk(struct imx_i2c_struct *i2c_imx, for (i = 0; i2c_clk_div[i][0] < div; i++); /* Write divider value to register */ - writeb(i2c_clk_div[i][1], i2c_imx->base + IMX_I2C_IFDR); + reg_write(i2c_clk_div[i][1], i2c_imx->base + IMX_I2C_IFDR); /* * There dummy delay is calculated. @@ -284,12 +272,12 @@ static irqreturn_t i2c_imx_isr(int irq, void *dev_id) struct imx_i2c_struct *i2c_imx = dev_id; unsigned int temp; - temp = readb(i2c_imx->base + IMX_I2C_I2SR); + temp = reg_read(i2c_imx->base + IMX_I2C_I2SR); if (temp & I2SR_IIF) { /* save status register */ i2c_imx->i2csr = temp; temp &= ~I2SR_IIF; - writeb(temp, i2c_imx->base + IMX_I2C_I2SR); + reg_write(temp, i2c_imx->base + IMX_I2C_I2SR); wake_up_interruptible(&i2c_imx->queue); } @@ -299,12 +287,15 @@ static irqreturn_t i2c_imx_isr(int irq, void *dev_id) static int i2c_imx_write(struct imx_i2c_struct *i2c_imx, struct i2c_msg *msgs) { int i, result; + unsigned int addr_trans; dev_dbg(&i2c_imx->adapter.dev, "<%s> write slave address: addr=0x%x\n", __func__, msgs->addr); + addr_trans = msgs->addr << 1; + /* write slave address */ - writeb(msgs->addr, i2c_imx->base + IMX_I2C_I2DR); + reg_write(addr_trans, i2c_imx->base + IMX_I2C_I2DR); result = i2c_imx_trx_complete(i2c_imx); if (result) return result; @@ -318,7 +309,7 @@ static int i2c_imx_write(struct imx_i2c_struct *i2c_imx, struct i2c_msg *msgs) dev_dbg(&i2c_imx->adapter.dev, "<%s> write byte: B%d=0x%X\n", __func__, i, msgs->buf[i]); - writeb(msgs->buf[i], i2c_imx->base + IMX_I2C_I2DR); + reg_write(msgs->buf[i], i2c_imx->base + IMX_I2C_I2DR); result = i2c_imx_trx_complete(i2c_imx); if (result) return result; @@ -333,13 +324,16 @@ static int i2c_imx_read(struct imx_i2c_struct *i2c_imx, struct i2c_msg *msgs) { int i, result; unsigned int temp; + unsigned int addr_trans; dev_dbg(&i2c_imx->adapter.dev, "<%s> write slave address: addr=0x%x\n", __func__, msgs->addr | 0x01); + addr_trans = (msgs->addr << 1) | 0x01; + /* write slave address */ - writeb(msgs->addr | 0x01, i2c_imx->base + IMX_I2C_I2DR); + reg_write(addr_trans, i2c_imx->base + IMX_I2C_I2DR); result = i2c_imx_trx_complete(i2c_imx); if (result) return result; @@ -350,12 +344,12 @@ static int i2c_imx_read(struct imx_i2c_struct *i2c_imx, struct i2c_msg *msgs) dev_dbg(&i2c_imx->adapter.dev, "<%s> setup bus\n", __func__); /* setup bus to read data */ - temp = readb(i2c_imx->base + IMX_I2C_I2CR); + temp = reg_read(i2c_imx->base + IMX_I2C_I2CR); temp &= ~I2CR_MTX; if (msgs->len - 1) temp &= ~I2CR_TXAK; - writeb(temp, i2c_imx->base + IMX_I2C_I2CR); - readb(i2c_imx->base + IMX_I2C_I2DR); /* dummy read */ + reg_write(temp, i2c_imx->base + IMX_I2C_I2CR); + reg_read(i2c_imx->base + IMX_I2C_I2DR); /* dummy read */ dev_dbg(&i2c_imx->adapter.dev, "<%s> read data\n", __func__); @@ -367,17 +361,17 @@ static int i2c_imx_read(struct imx_i2c_struct *i2c_imx, struct i2c_msg *msgs) if (i == (msgs->len - 1)) { dev_dbg(&i2c_imx->adapter.dev, "<%s> clear MSTA\n", __func__); - temp = readb(i2c_imx->base + IMX_I2C_I2CR); + temp = reg_read(i2c_imx->base + IMX_I2C_I2CR); temp &= ~I2CR_MSTA; - writeb(temp, i2c_imx->base + IMX_I2C_I2CR); + reg_write(temp, i2c_imx->base + IMX_I2C_I2CR); } else if (i == (msgs->len - 2)) { dev_dbg(&i2c_imx->adapter.dev, "<%s> set TXAK\n", __func__); - temp = readb(i2c_imx->base + IMX_I2C_I2CR); + temp = reg_read(i2c_imx->base + IMX_I2C_I2CR); temp |= I2CR_TXAK; - writeb(temp, i2c_imx->base + IMX_I2C_I2CR); + reg_write(temp, i2c_imx->base + IMX_I2C_I2CR); } - msgs->buf[i] = readb(i2c_imx->base + IMX_I2C_I2DR); + msgs->buf[i] = reg_read(i2c_imx->base + IMX_I2C_I2DR); dev_dbg(&i2c_imx->adapter.dev, "<%s> read byte: B%d=0x%X\n", __func__, i, msgs->buf[i]); @@ -407,21 +401,21 @@ static int i2c_imx_xfer(struct i2c_adapter *adapter, if (i) { dev_dbg(&i2c_imx->adapter.dev, "<%s> repeated start\n", __func__); - temp = readb(i2c_imx->base + IMX_I2C_I2CR); + temp = reg_read(i2c_imx->base + IMX_I2C_I2CR); temp |= I2CR_RSTA; - writeb(temp, i2c_imx->base + IMX_I2C_I2CR); + reg_write(temp, i2c_imx->base + IMX_I2C_I2CR); } dev_dbg(&i2c_imx->adapter.dev, "<%s> transfer message: %d\n", __func__, i); /* write/read data */ #ifdef CONFIG_I2C_DEBUG_BUS - temp = readb(i2c_imx->base + IMX_I2C_I2CR); + temp = reg_read(i2c_imx->base + IMX_I2C_I2CR); dev_dbg(&i2c_imx->adapter.dev, "<%s> CONTROL: IEN=%d, IIEN=%d, " "MSTA=%d, MTX=%d, TXAK=%d, RSTA=%d\n", __func__, (temp & I2CR_IEN ? 1 : 0), (temp & I2CR_IIEN ? 1 : 0), (temp & I2CR_MSTA ? 1 : 0), (temp & I2CR_MTX ? 1 : 0), (temp & I2CR_TXAK ? 1 : 0), (temp & I2CR_RSTA ? 1 : 0)); - temp = readb(i2c_imx->base + IMX_I2C_I2SR); + temp = reg_read(i2c_imx->base + IMX_I2C_I2SR); dev_dbg(&i2c_imx->adapter.dev, "<%s> STATUS: ICF=%d, IAAS=%d, IBB=%d, " "IAL=%d, SRW=%d, IIF=%d, RXAK=%d\n", __func__, @@ -442,8 +436,8 @@ fail0: dev_dbg(&i2c_imx->adapter.dev, "<%s> exit with: %s: %d\n", __func__, (result < 0) ? "error" : "success msg", - (result < 0) ? result : num); - return (result < 0) ? result : num; + result < 0 ? result : num); + return result < 0 ? result : num; } static u32 i2c_imx_func(struct i2c_adapter *adapter) @@ -518,12 +512,14 @@ static int __init i2c_imx_probe(struct platform_device *pdev) i2c_imx->adapter.algo = &i2c_imx_algo; i2c_imx->adapter.dev.parent = &pdev->dev; i2c_imx->adapter.nr = pdev->id; + i2c_imx->adapter.class = I2C_CLASS_HWMON | I2C_CLASS_SPD; + i2c_imx->adapter.timeout = 1; i2c_imx->irq = irq; i2c_imx->base = base; i2c_imx->res = res; /* Get I2C clock */ - i2c_imx->clk = clk_get(NULL, "i2c_clk"); + i2c_imx->clk = clk_get(&pdev->dev, "i2c_clk"); if (IS_ERR(i2c_imx->clk)) { ret = PTR_ERR(i2c_imx->clk); dev_err(&pdev->dev, "can't get I2C clock\n"); @@ -546,11 +542,11 @@ static int __init i2c_imx_probe(struct platform_device *pdev) i2c_set_adapdata(&i2c_imx->adapter, i2c_imx); /* Set up clock divider */ - i2c_imx_set_clk(i2c_imx, clkfreq); + i2c_imx_set_clk(i2c_imx, pdata->i2c_clk); /* Set up chip registers to defaults */ - writeb(0, i2c_imx->base + IMX_I2C_I2CR); - writeb(0, i2c_imx->base + IMX_I2C_I2SR); + reg_write(0, i2c_imx->base + IMX_I2C_I2CR); + reg_write(0, i2c_imx->base + IMX_I2C_I2SR); /* Add I2C adapter */ ret = i2c_add_numbered_adapter(&i2c_imx->adapter); @@ -604,10 +600,10 @@ static int __exit i2c_imx_remove(struct platform_device *pdev) free_irq(i2c_imx->irq, i2c_imx); /* setup chip registers to defaults */ - writeb(0, i2c_imx->base + IMX_I2C_IADR); - writeb(0, i2c_imx->base + IMX_I2C_IFDR); - writeb(0, i2c_imx->base + IMX_I2C_I2CR); - writeb(0, i2c_imx->base + IMX_I2C_I2SR); + reg_write(0, i2c_imx->base + IMX_I2C_IADR); + reg_write(0, i2c_imx->base + IMX_I2C_IFDR); + reg_write(0, i2c_imx->base + IMX_I2C_I2CR); + reg_write(0, i2c_imx->base + IMX_I2C_I2SR); /* Shut down hardware */ if (pdata->exit) @@ -647,9 +643,6 @@ static void __exit i2c_adap_imx_exit(void) module_init(i2c_adap_imx_init); module_exit(i2c_adap_imx_exit); -module_param(clkfreq, uint, S_IRUGO); -MODULE_PARM_DESC(clkfreq, "desired IMX I2C Clock Rate in Hz"); - MODULE_LICENSE("GPL"); MODULE_AUTHOR("Darius Augulis"); MODULE_DESCRIPTION("I2C adapter driver for IMX I2C bus"); -- 1.5.4 ^ permalink raw reply related [flat|nested] 21+ messages in thread
[parent not found: <Pine.LNX.4.64.0812032155350.7961-0199iw4Nj15frtckUFj5Ag@public.gmane.org>]
* Re: [PATCH][resend] iMX/MXC support for I2C [not found] ` <Pine.LNX.4.64.0812032155350.7961-0199iw4Nj15frtckUFj5Ag@public.gmane.org> @ 2008-12-03 23:48 ` Ben Dooks [not found] ` <20081203234833.GC4256-elnMNo+KYs3pIgCt6eIbzw@public.gmane.org> 0 siblings, 1 reply; 21+ messages in thread From: Ben Dooks @ 2008-12-03 23:48 UTC (permalink / raw) To: Guennadi Liakhovetski Cc: Darius, linux-i2c-u79uwXL29TY76Z2rM5mHXA, linux-arm-kernel-xIg/pKzrS19vn6HldHNs0ANdhmdF6hFW On Wed, Dec 03, 2008 at 09:58:10PM +0100, Guennadi Liakhovetski wrote: > Hi Darius, > > it would be better, if you didn't drop my email from cc, thanks. > > On Wed, 3 Dec 2008, Darius wrote: > > > Guennadi Liakhovetski wrote: > > > One more thing: you also want to do something like this > > > > > > i2c_imx->adapter.class = I2C_CLASS_HWMON | I2C_CLASS_SPD; > > > > some time ago, it was I2C_CLASS_ALL, but Jean told, that it will be removed at > > all :) > > > > > > > > in i2c_imx_probe(). > > > > > > If you want, I can send you my current version of your driver, so you can > > > compare. > > > > yes, please send it. > > Below. > > Thanks > Guennadi > --- > Guennadi Liakhovetski, Ph.D. > Freelance Open-Source Software Developer > > >From 1d592e3a94b3e13537a680e3e0198c345248692d Mon Sep 17 00:00:00 2001 > From: Guennadi Liakhovetski <g.liakhovetski-Mmb7MZpHnFY@public.gmane.org> > Date: Wed, 3 Dec 2008 17:02:37 +0100 > Subject: [PATCH] i2x-imx: fix client addresses, remove needless delays, support multiple busses > > With above bugs fixed now also tested to work on i.MX31. bus frequency is now > configured over platform data, which allows different frequencies on different > busses. Add an adapter.class assignment. > > Signed-off-by: Guennadi Liakhovetski <lg-ynQEQJNshbs@public.gmane.org> > --- > arch/arm/plat-mxc/include/mach/i2c.h | 8 ++ > drivers/i2c/busses/i2c-imx.c | 143 ++++++++++++++++------------------ > 2 files changed, 76 insertions(+), 75 deletions(-) > > diff --git a/arch/arm/plat-mxc/include/mach/i2c.h b/arch/arm/plat-mxc/include/mach/i2c.h > index 9e6a9d8..37d068b 100644 > --- a/arch/arm/plat-mxc/include/mach/i2c.h > +++ b/arch/arm/plat-mxc/include/mach/i2c.h > @@ -10,8 +10,16 @@ > #define __ASM_ARCH_I2C_H_ > > struct imxi2c_platform_data { > + u32 i2c_clk; > int (*init)(struct device *dev); > int (*exit)(struct device *dev); > }; > > +struct imxi2c_board_data { > + struct imxi2c_platform_data data; > + unsigned int id; > +}; > + > +extern int mxc_i2c_register_adapters(struct imxi2c_board_data *data, int n); if there are only 1 or two busses, why not do mx2_i2c_register_board(int i2c_channel_nr, struct imx_i2c_platform_data *pd); > #endif /* __ASM_ARCH_I2C_H_ */ > diff --git a/drivers/i2c/busses/i2c-imx.c b/drivers/i2c/busses/i2c-imx.c > index fd2d16d..eeea7d7 100644 > --- a/drivers/i2c/busses/i2c-imx.c > +++ b/drivers/i2c/busses/i2c-imx.c > @@ -23,12 +23,6 @@ > * Implementation of I2C Adapter/Algorithm Driver > * for I2C Bus integrated in Freescale i.MX/MXC processors > * > - * module parameters: > - * - clkfreq: > - * Sets the desired clock rate > - * The default value is 100000 > - * Max value is 400000 > - * > * Derived from Motorola GSG China I2C example driver > * > * Copyright (C) 2005 Torsten Koschorrek <koschorrek at synertronixx.de > @@ -64,14 +58,6 @@ > /* This will be the driver name the kernel reports */ > #define DRIVER_NAME "imx-i2c" > > -/* Default values of module parameters */ > -#define IMX_I2C_BIT_RATE 100000 /* 100kHz */ > - > -/* Timeouts */ > -#define I2C_IMX_TIME_BUSY 2000 /* loop count */ > -#define I2C_IMX_TIME_ACK 2000 /* loop count */ > -#define I2C_IMX_TIME_TRX 5 /* seconds */ > - > /* IMX I2C registers */ > #define IMX_I2C_IADR 0x00 /* i2c slave address */ > #define IMX_I2C_IFDR 0x04 /* i2c frequency divider */ > @@ -97,7 +83,6 @@ > /** Variables ****************************************************************** > *******************************************************************************/ > > -static unsigned int clkfreq = IMX_I2C_BIT_RATE; > static unsigned int disable_delay; /* Dummy delay */ > > /* > @@ -135,6 +120,21 @@ struct imx_i2c_struct { > unsigned long i2csr; > }; > > +/* > + * i.MX has 32-bit registers, i.MX31 16-bit, but only 8 bits are used. So far > + * 8-bit access seems to work on both. > + */ > +static inline unsigned int reg_read(const void __iomem *p) > +{ > + return __raw_readb(p); > +} > + > +static inline void reg_write(unsigned int value, void __iomem *p) > +{ > + __raw_writeb(value, p); > +} readb and writeb() are the right verions to use here. > + > + > /** Functions for IMX I2C adapter driver *************************************** > *******************************************************************************/ > > @@ -145,7 +145,7 @@ static int i2c_imx_bus_busy(struct imx_i2c_struct *i2c_imx) > dev_dbg(&i2c_imx->adapter.dev, "<%s>\n", __func__); > > /* wait for bus not busy */ > - while (readb(i2c_imx->base + IMX_I2C_I2SR) & I2SR_IBB) { > + while (reg_read(i2c_imx->base + IMX_I2C_I2SR) & I2SR_IBB) { > if (signal_pending(current)) { > dev_dbg(&i2c_imx->adapter.dev, > "<%s> I2C Interrupted\n", __func__); > @@ -167,7 +167,7 @@ static int i2c_imx_trx_complete(struct imx_i2c_struct *i2c_imx) > int result; > > result = wait_event_interruptible_timeout(i2c_imx->queue, > - i2c_imx->i2csr & I2SR_IIF, I2C_IMX_TIME_TRX * HZ); > + i2c_imx->i2csr & I2SR_IIF, i2c_imx->adapter.timeout); > > if (unlikely(result < 0)) { > dev_dbg(&i2c_imx->adapter.dev, "<%s> result < 0\n", __func__); > @@ -183,62 +183,50 @@ static int i2c_imx_trx_complete(struct imx_i2c_struct *i2c_imx) > > static int i2c_imx_acked(struct imx_i2c_struct *i2c_imx) > { > - unsigned long orig_jiffies = jiffies; > - > - /* Wait for ack */ > - while (readb(i2c_imx->base + IMX_I2C_I2SR) & I2SR_RXAK) { > - if (signal_pending(current)) { > - dev_dbg(&i2c_imx->adapter.dev, > - "<%s> I2C Interrupted\n", __func__); > - return -EINTR; > - } > - if (time_after(jiffies, orig_jiffies + HZ)) { > - dev_dbg(&i2c_imx->adapter.dev, > - "<%s> No ACK\n", __func__); > - return -EIO; > - } > - schedule(); > + if (reg_read(i2c_imx->base + IMX_I2C_I2SR) & I2SR_RXAK) { > + dev_dbg(&i2c_imx->adapter.dev, "<%s> No ACK\n", __func__); > + return -EIO; /* No ACK */ > } > > dev_dbg(&i2c_imx->adapter.dev, "<%s> ACK received\n", __func__); > - return 0; /* No ACK */ > + return 0; > } > > static void i2c_imx_start(struct imx_i2c_struct *i2c_imx) > { > - unsigned int temp = 0; > + unsigned int temp; > > dev_dbg(&i2c_imx->adapter.dev, "<%s>\n", __func__); > > /* Enable I2C controller */ > - writeb(I2CR_IEN, i2c_imx->base + IMX_I2C_I2CR); > + reg_write(I2CR_IEN, i2c_imx->base + IMX_I2C_I2CR); > /* Start I2C transaction */ > - temp = readb(i2c_imx->base + IMX_I2C_I2CR); > + temp = reg_read(i2c_imx->base + IMX_I2C_I2CR); > temp |= I2CR_MSTA; > - writeb(temp, i2c_imx->base + IMX_I2C_I2CR); > + reg_write(temp, i2c_imx->base + IMX_I2C_I2CR); > temp |= I2CR_IIEN | I2CR_MTX | I2CR_TXAK; > - writeb(temp, i2c_imx->base + IMX_I2C_I2CR); > + reg_write(temp, i2c_imx->base + IMX_I2C_I2CR); > } > > static void i2c_imx_stop(struct imx_i2c_struct *i2c_imx) > { > - unsigned int temp = 0; > + unsigned int temp; > > /* Stop I2C transaction */ > dev_dbg(&i2c_imx->adapter.dev, "<%s>\n", __func__); > - temp = readb(i2c_imx->base + IMX_I2C_I2CR); > + temp = reg_read(i2c_imx->base + IMX_I2C_I2CR); > temp &= ~I2CR_MSTA; > - writeb(temp, i2c_imx->base + IMX_I2C_I2CR); > + reg_write(temp, i2c_imx->base + IMX_I2C_I2CR); > /* setup chip registers to defaults */ > - writeb(I2CR_IEN, i2c_imx->base + IMX_I2C_I2CR); > - writeb(0, i2c_imx->base + IMX_I2C_I2SR); > + reg_write(I2CR_IEN, i2c_imx->base + IMX_I2C_I2CR); > + reg_write(0, i2c_imx->base + IMX_I2C_I2SR); > /* > * This delay caused by an i.MXL hardware bug. > * If no (or too short) delay, no "STOP" bit will be generated. > */ > udelay(disable_delay); > /* Disable I2C controller */ > - writeb(0, i2c_imx->base + IMX_I2C_I2CR); > + reg_write(0, i2c_imx->base + IMX_I2C_I2CR); > } > > static void __init i2c_imx_set_clk(struct imx_i2c_struct *i2c_imx, > @@ -259,7 +247,7 @@ static void __init i2c_imx_set_clk(struct imx_i2c_struct *i2c_imx, > for (i = 0; i2c_clk_div[i][0] < div; i++); > > /* Write divider value to register */ > - writeb(i2c_clk_div[i][1], i2c_imx->base + IMX_I2C_IFDR); > + reg_write(i2c_clk_div[i][1], i2c_imx->base + IMX_I2C_IFDR); > > /* > * There dummy delay is calculated. > @@ -284,12 +272,12 @@ static irqreturn_t i2c_imx_isr(int irq, void *dev_id) > struct imx_i2c_struct *i2c_imx = dev_id; > unsigned int temp; > > - temp = readb(i2c_imx->base + IMX_I2C_I2SR); > + temp = reg_read(i2c_imx->base + IMX_I2C_I2SR); > if (temp & I2SR_IIF) { > /* save status register */ > i2c_imx->i2csr = temp; > temp &= ~I2SR_IIF; > - writeb(temp, i2c_imx->base + IMX_I2C_I2SR); > + reg_write(temp, i2c_imx->base + IMX_I2C_I2SR); > wake_up_interruptible(&i2c_imx->queue); > } > > @@ -299,12 +287,15 @@ static irqreturn_t i2c_imx_isr(int irq, void *dev_id) > static int i2c_imx_write(struct imx_i2c_struct *i2c_imx, struct i2c_msg *msgs) > { > int i, result; > + unsigned int addr_trans; > > dev_dbg(&i2c_imx->adapter.dev, "<%s> write slave address: addr=0x%x\n", > __func__, msgs->addr); > > + addr_trans = msgs->addr << 1; > + > /* write slave address */ > - writeb(msgs->addr, i2c_imx->base + IMX_I2C_I2DR); > + reg_write(addr_trans, i2c_imx->base + IMX_I2C_I2DR); > result = i2c_imx_trx_complete(i2c_imx); > if (result) > return result; > @@ -318,7 +309,7 @@ static int i2c_imx_write(struct imx_i2c_struct *i2c_imx, struct i2c_msg *msgs) > dev_dbg(&i2c_imx->adapter.dev, > "<%s> write byte: B%d=0x%X\n", > __func__, i, msgs->buf[i]); > - writeb(msgs->buf[i], i2c_imx->base + IMX_I2C_I2DR); > + reg_write(msgs->buf[i], i2c_imx->base + IMX_I2C_I2DR); > result = i2c_imx_trx_complete(i2c_imx); > if (result) > return result; > @@ -333,13 +324,16 @@ static int i2c_imx_read(struct imx_i2c_struct *i2c_imx, struct i2c_msg *msgs) > { > int i, result; > unsigned int temp; > + unsigned int addr_trans; > > dev_dbg(&i2c_imx->adapter.dev, > "<%s> write slave address: addr=0x%x\n", > __func__, msgs->addr | 0x01); > > + addr_trans = (msgs->addr << 1) | 0x01; > + > /* write slave address */ > - writeb(msgs->addr | 0x01, i2c_imx->base + IMX_I2C_I2DR); > + reg_write(addr_trans, i2c_imx->base + IMX_I2C_I2DR); > result = i2c_imx_trx_complete(i2c_imx); > if (result) > return result; > @@ -350,12 +344,12 @@ static int i2c_imx_read(struct imx_i2c_struct *i2c_imx, struct i2c_msg *msgs) > dev_dbg(&i2c_imx->adapter.dev, "<%s> setup bus\n", __func__); > > /* setup bus to read data */ > - temp = readb(i2c_imx->base + IMX_I2C_I2CR); > + temp = reg_read(i2c_imx->base + IMX_I2C_I2CR); > temp &= ~I2CR_MTX; > if (msgs->len - 1) > temp &= ~I2CR_TXAK; > - writeb(temp, i2c_imx->base + IMX_I2C_I2CR); > - readb(i2c_imx->base + IMX_I2C_I2DR); /* dummy read */ > + reg_write(temp, i2c_imx->base + IMX_I2C_I2CR); > + reg_read(i2c_imx->base + IMX_I2C_I2DR); /* dummy read */ > > dev_dbg(&i2c_imx->adapter.dev, "<%s> read data\n", __func__); > > @@ -367,17 +361,17 @@ static int i2c_imx_read(struct imx_i2c_struct *i2c_imx, struct i2c_msg *msgs) > if (i == (msgs->len - 1)) { > dev_dbg(&i2c_imx->adapter.dev, > "<%s> clear MSTA\n", __func__); > - temp = readb(i2c_imx->base + IMX_I2C_I2CR); > + temp = reg_read(i2c_imx->base + IMX_I2C_I2CR); > temp &= ~I2CR_MSTA; > - writeb(temp, i2c_imx->base + IMX_I2C_I2CR); > + reg_write(temp, i2c_imx->base + IMX_I2C_I2CR); > } else if (i == (msgs->len - 2)) { > dev_dbg(&i2c_imx->adapter.dev, > "<%s> set TXAK\n", __func__); > - temp = readb(i2c_imx->base + IMX_I2C_I2CR); > + temp = reg_read(i2c_imx->base + IMX_I2C_I2CR); > temp |= I2CR_TXAK; > - writeb(temp, i2c_imx->base + IMX_I2C_I2CR); > + reg_write(temp, i2c_imx->base + IMX_I2C_I2CR); > } > - msgs->buf[i] = readb(i2c_imx->base + IMX_I2C_I2DR); > + msgs->buf[i] = reg_read(i2c_imx->base + IMX_I2C_I2DR); > dev_dbg(&i2c_imx->adapter.dev, > "<%s> read byte: B%d=0x%X\n", > __func__, i, msgs->buf[i]); > @@ -407,21 +401,21 @@ static int i2c_imx_xfer(struct i2c_adapter *adapter, > if (i) { > dev_dbg(&i2c_imx->adapter.dev, > "<%s> repeated start\n", __func__); > - temp = readb(i2c_imx->base + IMX_I2C_I2CR); > + temp = reg_read(i2c_imx->base + IMX_I2C_I2CR); > temp |= I2CR_RSTA; > - writeb(temp, i2c_imx->base + IMX_I2C_I2CR); > + reg_write(temp, i2c_imx->base + IMX_I2C_I2CR); > } > dev_dbg(&i2c_imx->adapter.dev, > "<%s> transfer message: %d\n", __func__, i); > /* write/read data */ > #ifdef CONFIG_I2C_DEBUG_BUS > - temp = readb(i2c_imx->base + IMX_I2C_I2CR); > + temp = reg_read(i2c_imx->base + IMX_I2C_I2CR); > dev_dbg(&i2c_imx->adapter.dev, "<%s> CONTROL: IEN=%d, IIEN=%d, " > "MSTA=%d, MTX=%d, TXAK=%d, RSTA=%d\n", __func__, > (temp & I2CR_IEN ? 1 : 0), (temp & I2CR_IIEN ? 1 : 0), > (temp & I2CR_MSTA ? 1 : 0), (temp & I2CR_MTX ? 1 : 0), > (temp & I2CR_TXAK ? 1 : 0), (temp & I2CR_RSTA ? 1 : 0)); > - temp = readb(i2c_imx->base + IMX_I2C_I2SR); > + temp = reg_read(i2c_imx->base + IMX_I2C_I2SR); > dev_dbg(&i2c_imx->adapter.dev, > "<%s> STATUS: ICF=%d, IAAS=%d, IBB=%d, " > "IAL=%d, SRW=%d, IIF=%d, RXAK=%d\n", __func__, > @@ -442,8 +436,8 @@ fail0: > > dev_dbg(&i2c_imx->adapter.dev, "<%s> exit with: %s: %d\n", __func__, > (result < 0) ? "error" : "success msg", > - (result < 0) ? result : num); > - return (result < 0) ? result : num; > + result < 0 ? result : num); > + return result < 0 ? result : num; > } > > static u32 i2c_imx_func(struct i2c_adapter *adapter) > @@ -518,12 +512,14 @@ static int __init i2c_imx_probe(struct platform_device *pdev) > i2c_imx->adapter.algo = &i2c_imx_algo; > i2c_imx->adapter.dev.parent = &pdev->dev; > i2c_imx->adapter.nr = pdev->id; > + i2c_imx->adapter.class = I2C_CLASS_HWMON | I2C_CLASS_SPD; > + i2c_imx->adapter.timeout = 1; > i2c_imx->irq = irq; > i2c_imx->base = base; > i2c_imx->res = res; > > /* Get I2C clock */ > - i2c_imx->clk = clk_get(NULL, "i2c_clk"); > + i2c_imx->clk = clk_get(&pdev->dev, "i2c_clk"); > if (IS_ERR(i2c_imx->clk)) { > ret = PTR_ERR(i2c_imx->clk); > dev_err(&pdev->dev, "can't get I2C clock\n"); > @@ -546,11 +542,11 @@ static int __init i2c_imx_probe(struct platform_device *pdev) > i2c_set_adapdata(&i2c_imx->adapter, i2c_imx); > > /* Set up clock divider */ > - i2c_imx_set_clk(i2c_imx, clkfreq); > + i2c_imx_set_clk(i2c_imx, pdata->i2c_clk); > > /* Set up chip registers to defaults */ > - writeb(0, i2c_imx->base + IMX_I2C_I2CR); > - writeb(0, i2c_imx->base + IMX_I2C_I2SR); > + reg_write(0, i2c_imx->base + IMX_I2C_I2CR); > + reg_write(0, i2c_imx->base + IMX_I2C_I2SR); > > /* Add I2C adapter */ > ret = i2c_add_numbered_adapter(&i2c_imx->adapter); > @@ -604,10 +600,10 @@ static int __exit i2c_imx_remove(struct platform_device *pdev) > free_irq(i2c_imx->irq, i2c_imx); > > /* setup chip registers to defaults */ > - writeb(0, i2c_imx->base + IMX_I2C_IADR); > - writeb(0, i2c_imx->base + IMX_I2C_IFDR); > - writeb(0, i2c_imx->base + IMX_I2C_I2CR); > - writeb(0, i2c_imx->base + IMX_I2C_I2SR); > + reg_write(0, i2c_imx->base + IMX_I2C_IADR); > + reg_write(0, i2c_imx->base + IMX_I2C_IFDR); > + reg_write(0, i2c_imx->base + IMX_I2C_I2CR); > + reg_write(0, i2c_imx->base + IMX_I2C_I2SR); > > /* Shut down hardware */ > if (pdata->exit) > @@ -647,9 +643,6 @@ static void __exit i2c_adap_imx_exit(void) > module_init(i2c_adap_imx_init); > module_exit(i2c_adap_imx_exit); > > -module_param(clkfreq, uint, S_IRUGO); > -MODULE_PARM_DESC(clkfreq, "desired IMX I2C Clock Rate in Hz"); > - > MODULE_LICENSE("GPL"); > MODULE_AUTHOR("Darius Augulis"); > MODULE_DESCRIPTION("I2C adapter driver for IMX I2C bus"); > -- > 1.5.4 > > -- > To unsubscribe from this list: send the line "unsubscribe linux-i2c" in > the body of a message to majordomo-u79uwXL29TY76Z2rM5mHXA@public.gmane.org > More majordomo info at http://vger.kernel.org/majordomo-info.html -- Ben (ben-elnMNo+KYs3YtjvyW6yDsg@public.gmane.org, http://www.fluff.org/) 'a smiley only costs 4 bytes' ^ permalink raw reply [flat|nested] 21+ messages in thread
[parent not found: <20081203234833.GC4256-elnMNo+KYs3pIgCt6eIbzw@public.gmane.org>]
* Re: [PATCH][resend] iMX/MXC support for I2C [not found] ` <20081203234833.GC4256-elnMNo+KYs3pIgCt6eIbzw@public.gmane.org> @ 2008-12-04 0:21 ` Guennadi Liakhovetski 0 siblings, 0 replies; 21+ messages in thread From: Guennadi Liakhovetski @ 2008-12-04 0:21 UTC (permalink / raw) To: Ben Dooks Cc: Darius, linux-i2c-u79uwXL29TY76Z2rM5mHXA, linux-arm-kernel-xIg/pKzrS19vn6HldHNs0ANdhmdF6hFW On Wed, 3 Dec 2008, Ben Dooks wrote: > On Wed, Dec 03, 2008 at 09:58:10PM +0100, Guennadi Liakhovetski wrote: > > > > struct imxi2c_platform_data { > > + u32 i2c_clk; > > int (*init)(struct device *dev); > > int (*exit)(struct device *dev); > > }; > > > > +struct imxi2c_board_data { > > + struct imxi2c_platform_data data; > > + unsigned int id; > > +}; > > + > > +extern int mxc_i2c_register_adapters(struct imxi2c_board_data *data, int n); > > if there are only 1 or two busses, why not do > > mx2_i2c_register_board(int i2c_channel_nr, struct > imx_i2c_platform_data *pd); on i.MX31 there are 3. Thanks Guennadi --- Guennadi Liakhovetski, Ph.D. Freelance Open-Source Software Developer ^ permalink raw reply [flat|nested] 21+ messages in thread
end of thread, other threads:[~2008-12-08 9:18 UTC | newest]
Thread overview: 21+ messages (download: mbox.gz follow: Atom feed
-- links below jump to the message on this page --
2008-12-02 14:00 [PATCH][resend] iMX/MXC support for I2C Darius
2008-12-02 14:04 ` Darius
2008-12-03 8:28 ` Jean Delvare
2008-12-03 11:17 ` Guennadi Liakhovetski
[not found] ` <Pine.LNX.4.64.0812030917440.4717-0199iw4Nj15frtckUFj5Ag@public.gmane.org>
2008-12-03 12:52 ` Darius
2008-12-03 15:19 ` Mark Brown
2008-12-03 21:19 ` Guennadi Liakhovetski
[not found] ` <Pine.LNX.4.64.0812032158150.7961-0199iw4Nj15frtckUFj5Ag@public.gmane.org>
2008-12-03 23:45 ` Ben Dooks
2008-12-04 7:47 ` Darius
[not found] ` <49378B1A.7030909-Re5JQEeQqe8AvxtiuMwx3w@public.gmane.org>
2008-12-04 13:30 ` Mark Brown
2008-12-03 23:43 ` Ben Dooks
[not found] ` <20081203234317.GA4256-elnMNo+KYs3pIgCt6eIbzw@public.gmane.org>
2008-12-04 9:59 ` Darius
[not found] ` <4937AA1C.8050004-Re5JQEeQqe8AvxtiuMwx3w@public.gmane.org>
2008-12-04 10:20 ` Guennadi Liakhovetski
2008-12-05 15:17 ` Guennadi Liakhovetski
[not found] ` <Pine.LNX.4.64.0812051612500.5840-0199iw4Nj15frtckUFj5Ag@public.gmane.org>
2008-12-08 9:18 ` Darius
2008-12-03 12:56 ` Guennadi Liakhovetski
[not found] ` <Pine.LNX.4.64.0812031355120.4717-0199iw4Nj15frtckUFj5Ag@public.gmane.org>
2008-12-03 13:07 ` Darius
2008-12-03 13:27 ` Wolfram Sang
2008-12-03 20:58 ` Guennadi Liakhovetski
[not found] ` <Pine.LNX.4.64.0812032155350.7961-0199iw4Nj15frtckUFj5Ag@public.gmane.org>
2008-12-03 23:48 ` Ben Dooks
[not found] ` <20081203234833.GC4256-elnMNo+KYs3pIgCt6eIbzw@public.gmane.org>
2008-12-04 0:21 ` Guennadi Liakhovetski
This is a public inbox, see mirroring instructions for how to clone and mirror all data and code used for this inbox