* [PATCH] i2c: SuperH Mobile I2C Bus Controller
@ 2008-03-20 14:18 Magnus Damm
2008-03-21 6:23 ` Paul Mundt
` (4 more replies)
0 siblings, 5 replies; 14+ messages in thread
From: Magnus Damm @ 2008-03-20 14:18 UTC (permalink / raw)
To: i2c; +Cc: khali, Magnus Damm, lethal, linux-sh
This patch adds support for the I2C block found in SuperH Mobile processors
such as sh7343, sh7722 and sh7723. Tested on a sh7722 MigoR board using the
rs5c732b rtc chip.
Signed-off-by: Magnus Damm <damm@igel.co.jp>
---
drivers/i2c/busses/Kconfig | 8
drivers/i2c/busses/Makefile | 1
drivers/i2c/busses/i2c-sh_mobile.c | 493 ++++++++++++++++++++++++++++++++++++
3 files changed, 502 insertions(+)
--- 0001/drivers/i2c/busses/Kconfig
+++ work/drivers/i2c/busses/Kconfig 2008-03-20 22:40:56.000000000 +0900
@@ -672,4 +672,12 @@ config I2C_PMCMSP
This driver can also be built as module. If so, the module
will be called i2c-pmcmsp.
+config I2C_SH_MOBILE
+ tristate "SuperH Mobile I2C Controller"
+ help
+ If you say yes to this option, support will be included for the
+ built-in I2C interface on the Renesas SH-Mobile processor.
+
+ This driver can also be built as a module. If so, the module
+ will be called i2c-sh_mobile.
endmenu
--- 0001/drivers/i2c/busses/Makefile
+++ work/drivers/i2c/busses/Makefile 2008-03-20 22:40:56.000000000 +0900
@@ -52,6 +52,7 @@ obj-$(CONFIG_I2C_VIAPRO) += i2c-viapro.o
obj-$(CONFIG_I2C_VOODOO3) += i2c-voodoo3.o
obj-$(CONFIG_SCx200_ACB) += scx200_acb.o
obj-$(CONFIG_SCx200_I2C) += scx200_i2c.o
+obj-$(CONFIG_I2C_SH_MOBILE) += i2c-sh_mobile.o
ifeq ($(CONFIG_I2C_DEBUG_BUS),y)
EXTRA_CFLAGS += -DDEBUG
--- /dev/null
+++ work/drivers/i2c/busses/i2c-sh_mobile.c 2008-03-20 22:45:33.000000000 +0900
@@ -0,0 +1,493 @@
+/*
+ * SuperH Mobile I2C Controller
+ *
+ * Copyright (C) 2008 Magnus Damm
+ *
+ * Based on i2c-simtec.c by Ben Dooks <ben@simtec.co.uk>
+ * Copyright (C) 2005 Simtec Electronics
+ *
+ * Portions of the code based on out-of-tree driver i2c-sh7343.c
+ * Copyright (c) 2006 Carlos Munoz <carlos@kenati.com>
+ *
+ * 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
+ *
+ * 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
+*/
+
+#include <linux/kernel.h>
+#include <linux/module.h>
+#include <linux/init.h>
+#include <linux/delay.h>
+#include <linux/platform_device.h>
+#include <linux/interrupt.h>
+#include <linux/i2c.h>
+#include <asm/io.h>
+
+enum {
+ OP_START = 0,
+ OP_TX_ONLY,
+ OP_TX_STOP,
+ OP_TX_TO_RX,
+ OP_RX_ONLY,
+ OP_RX_STOP,
+};
+
+struct sh_mobile_i2c_data {
+ struct resource *ioarea;
+ void __iomem *reg;
+ struct i2c_adapter adap;
+
+ u_int8_t iccl;
+ u_int8_t icch;
+
+ int fast_mode;
+ int rx_ack_high;
+
+ spinlock_t lock;
+ wait_queue_head_t wait;
+ struct i2c_msg *msg;
+ int pos;
+ int sr;
+};
+
+#define NORMAL_SPEED 100000
+#define FAST_SPEED 400000
+
+/* Register offsets */
+#define ICDR(pd) (pd->reg + 0x00)
+#define ICCR(pd) (pd->reg + 0x04)
+#define ICSR(pd) (pd->reg + 0x08)
+#define ICIC(pd) (pd->reg + 0x0c)
+#define ICCL(pd) (pd->reg + 0x10)
+#define ICCH(pd) (pd->reg + 0x14)
+
+/* Register bits */
+#define ICCR_ICE 0x80
+#define ICCR_RACK 0x40
+#define ICCR_TRS 0x10
+#define ICCR_BBSY 0x04
+#define ICCR_SCP 0x01
+
+#define ICSR_SCLM 0x80
+#define ICSR_SDAM 0x40
+#define SW_DONE 0x20
+#define ICSR_BUSY 0x10
+#define ICSR_AL 0x08
+#define ICSR_TACK 0x04
+#define ICSR_WAIT 0x02
+#define ICSR_DTE 0x01
+
+#define ICIC_ALE 0x08
+#define ICIC_TACKE 0x04
+#define ICIC_WAITE 0x02
+#define ICIC_DTEE 0x01
+
+static void activate_ch(struct sh_mobile_i2c_data *pd)
+{
+ /* Enable channel and configure rx ack */
+ iowrite8(ioread8(ICCR(pd)) | ICCR_ICE |
+ (pd->rx_ack_high ? ICCR_RACK : 0), ICCR(pd));
+
+ /* Mask all interrupts */
+ iowrite8(0, ICIC(pd));
+
+ /* Set the clock */
+ iowrite8(pd->iccl, ICCL(pd));
+ iowrite8(pd->icch, ICCH(pd));
+}
+
+static void deactivate_ch(struct sh_mobile_i2c_data *pd)
+{
+ /* Clear/disable interrupts */
+ iowrite8(0, ICSR(pd));
+ iowrite8(0, ICIC(pd));
+
+ /* Initialize channel */
+ iowrite8(ioread8(ICCR(pd)) & ~ICCR_ICE, ICCR(pd));
+}
+
+static unsigned char i2c_op(struct sh_mobile_i2c_data *pd,
+ int op, unsigned char data)
+{
+ unsigned char ret = 0;
+
+ pr_debug("op %d, data in 0x%02x\n", op, data);
+
+ spin_lock_irq(&pd->lock);
+
+ switch (op) {
+ case OP_START:
+ iowrite8(0x94, ICCR(pd));
+ break;
+ case OP_TX_ONLY:
+ iowrite8(data, ICDR(pd));
+ break;
+ case OP_TX_STOP:
+ iowrite8(data, ICDR(pd));
+ iowrite8(0x90, ICCR(pd));
+ iowrite8(ICIC_ALE | ICIC_TACKE, ICIC(pd));
+ break;
+ case OP_TX_TO_RX:
+ iowrite8(data, ICDR(pd));
+ iowrite8(0x81, ICCR(pd));
+ break;
+ case OP_RX_ONLY:
+ ret = ioread8(ICDR(pd));
+ break;
+ case OP_RX_STOP:
+ ret = ioread8(ICDR(pd));
+ iowrite8(0xc0, ICCR(pd));
+ break;
+ }
+
+ spin_unlock_irq(&pd->lock);
+
+ pr_debug("op %d, data out 0x%02x\n", op, ret);
+ return ret;
+}
+
+static irqreturn_t sh_mobile_i2c_isr(int irq, void *dev_id)
+{
+ struct platform_device *dev = dev_id;
+ struct sh_mobile_i2c_data *pd = platform_get_drvdata(dev);
+ struct i2c_msg *msg = pd->msg;
+ unsigned char data, sr;
+ int wakeup = 0;
+
+ sr = ioread8(ICSR(pd));
+ pd->sr |= sr;
+
+ pr_debug("i2c_isr 0x%02x 0x%02x %s %d %d!\n", sr, pd->sr,
+ (msg->flags & I2C_M_RD) ? "read" : "write",
+ pd->pos, msg->len);
+
+ if (sr & (ICSR_AL | ICSR_TACK)) {
+ iowrite8(0, ICIC(pd)); /* disable interrupts */
+ wakeup = 1;
+ goto do_wakeup;
+ }
+
+ if (pd->pos = msg->len) {
+ i2c_op(pd, OP_RX_ONLY, 0);
+ wakeup = 1;
+ goto do_wakeup;
+ }
+
+ if (pd->pos = -1) {
+ data = (msg->addr & 0x7f) << 1;
+ data |= (msg->flags & I2C_M_RD) ? 1 : 0;
+ } else
+ data = msg->buf[pd->pos];
+
+ if ((pd->pos = -1) || !(msg->flags & I2C_M_RD)) {
+ if (msg->flags & I2C_M_RD)
+ i2c_op(pd, OP_TX_TO_RX, data);
+ else if (pd->pos = (msg->len - 1)) {
+ i2c_op(pd, OP_TX_STOP, data);
+ wakeup = 1;
+ } else
+ i2c_op(pd, OP_TX_ONLY, data);
+ } else {
+ if (pd->pos = (msg->len - 1))
+ data = i2c_op(pd, OP_RX_STOP, 0);
+ else
+ data = i2c_op(pd, OP_RX_ONLY, 0);
+
+ msg->buf[pd->pos] = data;
+ }
+ pd->pos++;
+
+ do_wakeup:
+ if (wakeup) {
+ pd->sr |= SW_DONE;
+ wake_up(&pd->wait);
+ }
+
+ return IRQ_HANDLED;
+}
+
+static int start_ch(struct sh_mobile_i2c_data *pd, struct i2c_msg *usr_msg)
+{
+ /* Initialize channel registers */
+ iowrite8(ioread8(ICCR(pd)) & ~ICCR_ICE, ICCR(pd));
+
+ /* Enable channel and configure rx ack */
+ iowrite8(ioread8(ICCR(pd)) | ICCR_ICE |
+ (pd->rx_ack_high ? ICCR_RACK : 0), ICCR(pd));
+
+ /* Set the clock */
+ iowrite8(pd->iccl, ICCL(pd));
+ iowrite8(pd->icch, ICCH(pd));
+
+ pd->msg = usr_msg;
+ pd->pos = -1;
+ pd->sr = 0;
+
+ /* Enable all interrupts except wait */
+ iowrite8(ioread8(ICIC(pd)) | ICIC_ALE | ICIC_TACKE | ICIC_DTEE,
+ ICIC(pd));
+ return 0;
+}
+
+static int sh_mobile_i2c_xfer(struct i2c_adapter *adapter,
+ struct i2c_msg *msgs,
+ int num)
+{
+ struct sh_mobile_i2c_data *pd = adapter->algo_data;
+ struct i2c_msg *msg;
+ int err = 0;
+ u_int8_t val;
+ int i, k;
+
+ activate_ch(pd);
+
+ /* Process all messages */
+ for (i = 0; i < num; i++) {
+ msg = &msgs[i];
+
+ err = start_ch(pd, msg);
+ if (err)
+ break;
+
+ i2c_op(pd, OP_START, 0);
+
+ /* The interrupt handler takes care of the rest... */
+ k = wait_event_timeout(pd->wait,
+ pd->sr & (ICSR_TACK | SW_DONE),
+ 5 * HZ);
+ if (!k)
+ pr_err("Transfer request timed out\n");
+
+ k = 10;
+ again:
+ val = ioread8(ICSR(pd));
+
+ pr_debug("val 0x%02x pd->sr 0x%02x!\n", val, pd->sr);
+
+ if ((val | pd->sr) & (ICSR_TACK | ICSR_AL)) {
+ err = -EIO;
+ break;
+ }
+
+ if (!(!(val & ICSR_BUSY) && (val & ICSR_SCLM) &&
+ (val & ICSR_SDAM))) {
+ msleep(1);
+ if (k--)
+ goto again;
+
+ err = -EIO;
+ pr_err("Polling timed out\n");
+ break;
+ }
+ }
+
+ deactivate_ch(pd);
+
+ if (!err)
+ err = num;
+ return err;
+}
+
+u32 sh_mobile_i2c_func(struct i2c_adapter *adapter)
+{
+ return I2C_FUNC_I2C;
+}
+
+static struct i2c_algorithm sh_mobile_i2c_algorithm = {
+ .functionality = sh_mobile_i2c_func,
+ .master_xfer = sh_mobile_i2c_xfer,
+};
+
+static void sh_mobile_i2c_setup_channel(struct platform_device *dev)
+{
+ struct sh_mobile_i2c_data *pd = platform_get_drvdata(dev);
+ unsigned long peripheral_clk = CONFIG_SH_PCLK_FREQ;
+ u_int32_t num;
+ u_int32_t denom;
+ u_int32_t tmp;
+
+ spin_lock_init(&pd->lock);
+ init_waitqueue_head(&pd->wait);
+
+ /* Calculate the value for iccl. From the data sheet:
+ * iccl = (p clock ÷ transfer rate) × (L ÷ (L + H))
+ * where L and H are the SCL low/high ratio (5/4 in this case).
+ * We also round off the result.
+ */
+ num = peripheral_clk * 5;
+ denom = pd->fast_mode ? (FAST_SPEED * 9) : (NORMAL_SPEED * 9);
+ tmp = num * 10 / denom;
+ if (tmp % 10 >= 5)
+ pd->iccl = (u_int8_t)((num/denom) + 1);
+ else
+ pd->iccl = (u_int8_t)(num/denom);
+
+ /* Calculate the value for icch. From the data sheet:
+ icch = (p clock ÷ transfer rate) × (H ÷ (L + H)) */
+ num = peripheral_clk * 4;
+ tmp = num * 10 / denom;
+ if (tmp % 10 >= 5)
+ pd->icch = (u_int8_t)((num/denom) + 1);
+ else
+ pd->icch = (u_int8_t)(num/denom);
+
+ deactivate_ch(pd);
+}
+
+static int sh_mobile_i2c_hook_irqs(struct platform_device *dev, int hook)
+{
+ struct resource *res;
+ int ret = -ENXIO;
+ int q, m;
+ int k = 0;
+ int n = 0;
+
+ while ((res = platform_get_resource(dev, IORESOURCE_IRQ, k))) {
+ for (n = res->start; hook && n <= res->end; n++) {
+ if (request_irq(n, sh_mobile_i2c_isr, 0,
+ dev->name, dev))
+ goto rollback;
+ }
+ k++;
+ }
+
+ if (hook)
+ return k > 0 ? 0 : -ENOENT;
+
+ k--;
+ ret = 0;
+
+ rollback:
+ for (q = k; k >= 0; k--) {
+ for (m = n; m >= res->start; m--)
+ free_irq(m, dev);
+
+ res = platform_get_resource(dev, IORESOURCE_IRQ, k - 1);
+ m = res->end;
+ }
+
+ return ret;
+}
+
+static int sh_mobile_i2c_probe(struct platform_device *dev)
+{
+ struct sh_mobile_i2c_data *pd;
+ struct resource *res;
+ int size;
+ int ret;
+
+ pd = kzalloc(sizeof(struct sh_mobile_i2c_data), GFP_KERNEL);
+ if (pd = NULL) {
+ dev_err(&dev->dev, "cannot allocate private data\n");
+ return -ENOMEM;
+ }
+
+ ret = sh_mobile_i2c_hook_irqs(dev, 1);
+ if (ret) {
+ dev_err(&dev->dev, "cannot request IRQ\n");
+ goto err;
+ }
+
+ platform_set_drvdata(dev, pd);
+
+ res = platform_get_resource(dev, IORESOURCE_MEM, 0);
+ if (res = NULL) {
+ dev_err(&dev->dev, "cannot find IO resource\n");
+ ret = -ENOENT;
+ goto err_irq;
+ }
+
+ size = (res->end - res->start) + 1;
+
+ pd->ioarea = request_mem_region(res->start, size, dev->name);
+ if (pd->ioarea = NULL) {
+ dev_err(&dev->dev, "cannot request IO\n");
+ ret = -ENXIO;
+ goto err_irq;
+ }
+
+ pd->reg = ioremap(res->start, size);
+ if (pd->reg = NULL) {
+ dev_err(&dev->dev, "cannot map IO\n");
+ ret = -ENXIO;
+ goto err_res;
+ }
+
+ /* setup the private data */
+
+ pd->adap.owner = THIS_MODULE;
+ pd->adap.algo = &sh_mobile_i2c_algorithm;
+ pd->adap.algo_data = pd;
+ pd->adap.dev.parent = &dev->dev;
+ pd->adap.retries = 5;
+ pd->adap.nr = dev->id;
+
+ strlcpy(pd->adap.name, dev->name, sizeof(pd->adap.name));
+
+ sh_mobile_i2c_setup_channel(dev);
+
+ ret = i2c_add_numbered_adapter(&pd->adap);
+ if (ret < 0)
+ goto err_all;
+
+ return 0;
+
+ err_all:
+ iounmap(pd->reg);
+
+ err_res:
+ release_resource(pd->ioarea);
+
+ err_irq:
+ sh_mobile_i2c_hook_irqs(dev, 0);
+ err:
+ kfree(pd);
+ return ret;
+}
+
+static int sh_mobile_i2c_remove(struct platform_device *dev)
+{
+ struct sh_mobile_i2c_data *pd = platform_get_drvdata(dev);
+
+ i2c_del_adapter(&pd->adap);
+ iounmap(pd->reg);
+ release_resource(pd->ioarea);
+ sh_mobile_i2c_hook_irqs(dev, 0);
+ kfree(pd);
+ return 0;
+}
+
+static struct platform_driver sh_mobile_i2c_driver = {
+ .driver = {
+ .name = "i2c-sh_mobile",
+ .owner = THIS_MODULE,
+ },
+ .probe = sh_mobile_i2c_probe,
+ .remove = sh_mobile_i2c_remove,
+};
+
+static int __init sh_mobile_i2c_adap_init(void)
+{
+ return platform_driver_register(&sh_mobile_i2c_driver);
+}
+
+static void __exit sh_mobile_i2c_adap_exit(void)
+{
+ platform_driver_unregister(&sh_mobile_i2c_driver);
+}
+
+module_init(sh_mobile_i2c_adap_init);
+module_exit(sh_mobile_i2c_adap_exit);
+
+MODULE_DESCRIPTION("SuperH Mobile I2C Bus Controller driver");
+MODULE_AUTHOR("Magnus Damm <damm@opensource.se>");
+MODULE_LICENSE("GPL");
^ permalink raw reply [flat|nested] 14+ messages in thread
* Re: [PATCH] i2c: SuperH Mobile I2C Bus Controller
2008-03-20 14:18 [PATCH] i2c: SuperH Mobile I2C Bus Controller Magnus Damm
@ 2008-03-21 6:23 ` Paul Mundt
2008-03-21 8:14 ` Magnus Damm
2008-03-21 12:07 ` [PATCH] i2c: SuperH Mobile I2C Bus Controller V2 Magnus Damm
` (3 subsequent siblings)
4 siblings, 1 reply; 14+ messages in thread
From: Paul Mundt @ 2008-03-21 6:23 UTC (permalink / raw)
To: Magnus Damm; +Cc: i2c, khali, linux-sh
On Thu, Mar 20, 2008 at 11:18:40PM +0900, Magnus Damm wrote:
> +static void sh_mobile_i2c_setup_channel(struct platform_device *dev)
> +{
> + struct sh_mobile_i2c_data *pd = platform_get_drvdata(dev);
> + unsigned long peripheral_clk = CONFIG_SH_PCLK_FREQ;
> + u_int32_t num;
> + u_int32_t denom;
> + u_int32_t tmp;
> +
Use clk_get() and friends here for the module clock. This should
absolutely not be hardcoded, especially as it can be scaled by the clock
framework, and the i2c clock will need to rebalance itself to account for
that. Any driver that references CONFIG_SH_PCLK_FREQ is utterly broken.
^ permalink raw reply [flat|nested] 14+ messages in thread
* Re: [PATCH] i2c: SuperH Mobile I2C Bus Controller
2008-03-21 6:23 ` Paul Mundt
@ 2008-03-21 8:14 ` Magnus Damm
0 siblings, 0 replies; 14+ messages in thread
From: Magnus Damm @ 2008-03-21 8:14 UTC (permalink / raw)
To: Paul Mundt, Magnus Damm, i2c, khali, linux-sh
On Fri, Mar 21, 2008 at 3:23 PM, Paul Mundt <lethal@linux-sh.org> wrote:
> On Thu, Mar 20, 2008 at 11:18:40PM +0900, Magnus Damm wrote:
> > +static void sh_mobile_i2c_setup_channel(struct platform_device *dev)
> > +{
> > + struct sh_mobile_i2c_data *pd = platform_get_drvdata(dev);
> > + unsigned long peripheral_clk = CONFIG_SH_PCLK_FREQ;
> > + u_int32_t num;
> > + u_int32_t denom;
> > + u_int32_t tmp;
> > +
> Use clk_get() and friends here for the module clock. This should
> absolutely not be hardcoded, especially as it can be scaled by the clock
> framework, and the i2c clock will need to rebalance itself to account for
> that. Any driver that references CONFIG_SH_PCLK_FREQ is utterly broken.
Thanks for spotting this. I'll fix this and repost V2 in a bit.
/ magnus
^ permalink raw reply [flat|nested] 14+ messages in thread
* [PATCH] i2c: SuperH Mobile I2C Bus Controller V2
2008-03-20 14:18 [PATCH] i2c: SuperH Mobile I2C Bus Controller Magnus Damm
2008-03-21 6:23 ` Paul Mundt
@ 2008-03-21 12:07 ` Magnus Damm
2008-03-25 10:55 ` [PATCH] i2c: SuperH Mobile I2C Bus Controller V3 Paul Mundt
` (2 subsequent siblings)
4 siblings, 0 replies; 14+ messages in thread
From: Magnus Damm @ 2008-03-21 12:07 UTC (permalink / raw)
To: i2c; +Cc: khali, Magnus Damm, lethal, linux-sh
This is V2 of the SuperH Mobile I2C Controller Driver. A simple Master
only driver for the I2C block included in processors such as sh7343,
sh7722 and sh7723. Tested on a sh7722 MigoR using a rs5c732b rtc.
Changes since V1:
- Use clk_get()/clk_put()/clk_get_rate() to get peripheral clock rate.
- Use pdev->dev.bus_id instead of dev->name
Signed-off-by: Magnus Damm <damm@igel.co.jp>
---
drivers/i2c/busses/Kconfig | 8
drivers/i2c/busses/Makefile | 1
drivers/i2c/busses/i2c-sh_mobile.c | 504 ++++++++++++++++++++++++++++++++++++
3 files changed, 513 insertions(+)
--- 0001/drivers/i2c/busses/Kconfig
+++ work/drivers/i2c/busses/Kconfig 2008-03-21 20:22:54.000000000 +0900
@@ -672,4 +672,12 @@ config I2C_PMCMSP
This driver can also be built as module. If so, the module
will be called i2c-pmcmsp.
+config I2C_SH_MOBILE
+ tristate "SuperH Mobile I2C Controller"
+ help
+ If you say yes to this option, support will be included for the
+ built-in I2C interface on the Renesas SH-Mobile processor.
+
+ This driver can also be built as a module. If so, the module
+ will be called i2c-sh_mobile.
endmenu
--- 0001/drivers/i2c/busses/Makefile
+++ work/drivers/i2c/busses/Makefile 2008-03-21 20:22:54.000000000 +0900
@@ -52,6 +52,7 @@ obj-$(CONFIG_I2C_VIAPRO) += i2c-viapro.o
obj-$(CONFIG_I2C_VOODOO3) += i2c-voodoo3.o
obj-$(CONFIG_SCx200_ACB) += scx200_acb.o
obj-$(CONFIG_SCx200_I2C) += scx200_i2c.o
+obj-$(CONFIG_I2C_SH_MOBILE) += i2c-sh_mobile.o
ifeq ($(CONFIG_I2C_DEBUG_BUS),y)
EXTRA_CFLAGS += -DDEBUG
--- /dev/null
+++ work/drivers/i2c/busses/i2c-sh_mobile.c 2008-03-21 20:48:36.000000000 +0900
@@ -0,0 +1,504 @@
+/*
+ * SuperH Mobile I2C Controller
+ *
+ * Copyright (C) 2008 Magnus Damm
+ *
+ * Based on i2c-simtec.c by Ben Dooks <ben@simtec.co.uk>
+ * Copyright (C) 2005 Simtec Electronics
+ *
+ * Portions of the code based on out-of-tree driver i2c-sh7343.c
+ * Copyright (c) 2006 Carlos Munoz <carlos@kenati.com>
+ *
+ * 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
+ *
+ * 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
+*/
+
+#include <linux/kernel.h>
+#include <linux/module.h>
+#include <linux/init.h>
+#include <linux/delay.h>
+#include <linux/platform_device.h>
+#include <linux/interrupt.h>
+#include <linux/i2c.h>
+#include <linux/err.h>
+#include <linux/clk.h>
+#include <asm/io.h>
+
+enum {
+ OP_START = 0,
+ OP_TX_ONLY,
+ OP_TX_STOP,
+ OP_TX_TO_RX,
+ OP_RX_ONLY,
+ OP_RX_STOP,
+};
+
+struct sh_mobile_i2c_data {
+ struct resource *ioarea;
+ void __iomem *reg;
+ struct i2c_adapter adap;
+
+ struct clk *clk;
+ u_int8_t iccl;
+ u_int8_t icch;
+
+ int fast_mode;
+ int rx_ack_high;
+
+ spinlock_t lock;
+ wait_queue_head_t wait;
+ struct i2c_msg *msg;
+ int pos;
+ int sr;
+};
+
+#define NORMAL_SPEED 100000
+#define FAST_SPEED 400000
+
+/* Register offsets */
+#define ICDR(pd) (pd->reg + 0x00)
+#define ICCR(pd) (pd->reg + 0x04)
+#define ICSR(pd) (pd->reg + 0x08)
+#define ICIC(pd) (pd->reg + 0x0c)
+#define ICCL(pd) (pd->reg + 0x10)
+#define ICCH(pd) (pd->reg + 0x14)
+
+/* Register bits */
+#define ICCR_ICE 0x80
+#define ICCR_RACK 0x40
+#define ICCR_TRS 0x10
+#define ICCR_BBSY 0x04
+#define ICCR_SCP 0x01
+
+#define ICSR_SCLM 0x80
+#define ICSR_SDAM 0x40
+#define SW_DONE 0x20
+#define ICSR_BUSY 0x10
+#define ICSR_AL 0x08
+#define ICSR_TACK 0x04
+#define ICSR_WAIT 0x02
+#define ICSR_DTE 0x01
+
+#define ICIC_ALE 0x08
+#define ICIC_TACKE 0x04
+#define ICIC_WAITE 0x02
+#define ICIC_DTEE 0x01
+
+static void activate_ch(struct sh_mobile_i2c_data *pd)
+{
+ /* Enable channel and configure rx ack */
+ iowrite8(ioread8(ICCR(pd)) | ICCR_ICE |
+ (pd->rx_ack_high ? ICCR_RACK : 0), ICCR(pd));
+
+ /* Mask all interrupts */
+ iowrite8(0, ICIC(pd));
+
+ /* Set the clock */
+ iowrite8(pd->iccl, ICCL(pd));
+ iowrite8(pd->icch, ICCH(pd));
+}
+
+static void deactivate_ch(struct sh_mobile_i2c_data *pd)
+{
+ /* Clear/disable interrupts */
+ iowrite8(0, ICSR(pd));
+ iowrite8(0, ICIC(pd));
+
+ /* Initialize channel */
+ iowrite8(ioread8(ICCR(pd)) & ~ICCR_ICE, ICCR(pd));
+}
+
+static unsigned char i2c_op(struct sh_mobile_i2c_data *pd,
+ int op, unsigned char data)
+{
+ unsigned char ret = 0;
+
+ pr_debug("op %d, data in 0x%02x\n", op, data);
+
+ spin_lock_irq(&pd->lock);
+
+ switch (op) {
+ case OP_START:
+ iowrite8(0x94, ICCR(pd));
+ break;
+ case OP_TX_ONLY:
+ iowrite8(data, ICDR(pd));
+ break;
+ case OP_TX_STOP:
+ iowrite8(data, ICDR(pd));
+ iowrite8(0x90, ICCR(pd));
+ iowrite8(ICIC_ALE | ICIC_TACKE, ICIC(pd));
+ break;
+ case OP_TX_TO_RX:
+ iowrite8(data, ICDR(pd));
+ iowrite8(0x81, ICCR(pd));
+ break;
+ case OP_RX_ONLY:
+ ret = ioread8(ICDR(pd));
+ break;
+ case OP_RX_STOP:
+ ret = ioread8(ICDR(pd));
+ iowrite8(0xc0, ICCR(pd));
+ break;
+ }
+
+ spin_unlock_irq(&pd->lock);
+
+ pr_debug("op %d, data out 0x%02x\n", op, ret);
+ return ret;
+}
+
+static irqreturn_t sh_mobile_i2c_isr(int irq, void *dev_id)
+{
+ struct platform_device *dev = dev_id;
+ struct sh_mobile_i2c_data *pd = platform_get_drvdata(dev);
+ struct i2c_msg *msg = pd->msg;
+ unsigned char data, sr;
+ int wakeup = 0;
+
+ sr = ioread8(ICSR(pd));
+ pd->sr |= sr;
+
+ pr_debug("i2c_isr 0x%02x 0x%02x %s %d %d!\n", sr, pd->sr,
+ (msg->flags & I2C_M_RD) ? "read" : "write",
+ pd->pos, msg->len);
+
+ if (sr & (ICSR_AL | ICSR_TACK)) {
+ iowrite8(0, ICIC(pd)); /* disable interrupts */
+ wakeup = 1;
+ goto do_wakeup;
+ }
+
+ if (pd->pos = msg->len) {
+ i2c_op(pd, OP_RX_ONLY, 0);
+ wakeup = 1;
+ goto do_wakeup;
+ }
+
+ if (pd->pos = -1) {
+ data = (msg->addr & 0x7f) << 1;
+ data |= (msg->flags & I2C_M_RD) ? 1 : 0;
+ } else
+ data = msg->buf[pd->pos];
+
+ if ((pd->pos = -1) || !(msg->flags & I2C_M_RD)) {
+ if (msg->flags & I2C_M_RD)
+ i2c_op(pd, OP_TX_TO_RX, data);
+ else if (pd->pos = (msg->len - 1)) {
+ i2c_op(pd, OP_TX_STOP, data);
+ wakeup = 1;
+ } else
+ i2c_op(pd, OP_TX_ONLY, data);
+ } else {
+ if (pd->pos = (msg->len - 1))
+ data = i2c_op(pd, OP_RX_STOP, 0);
+ else
+ data = i2c_op(pd, OP_RX_ONLY, 0);
+
+ msg->buf[pd->pos] = data;
+ }
+ pd->pos++;
+
+ do_wakeup:
+ if (wakeup) {
+ pd->sr |= SW_DONE;
+ wake_up(&pd->wait);
+ }
+
+ return IRQ_HANDLED;
+}
+
+static int start_ch(struct sh_mobile_i2c_data *pd, struct i2c_msg *usr_msg)
+{
+ /* Initialize channel registers */
+ iowrite8(ioread8(ICCR(pd)) & ~ICCR_ICE, ICCR(pd));
+
+ /* Enable channel and configure rx ack */
+ iowrite8(ioread8(ICCR(pd)) | ICCR_ICE |
+ (pd->rx_ack_high ? ICCR_RACK : 0), ICCR(pd));
+
+ /* Set the clock */
+ iowrite8(pd->iccl, ICCL(pd));
+ iowrite8(pd->icch, ICCH(pd));
+
+ pd->msg = usr_msg;
+ pd->pos = -1;
+ pd->sr = 0;
+
+ /* Enable all interrupts except wait */
+ iowrite8(ioread8(ICIC(pd)) | ICIC_ALE | ICIC_TACKE | ICIC_DTEE,
+ ICIC(pd));
+ return 0;
+}
+
+static int sh_mobile_i2c_xfer(struct i2c_adapter *adapter,
+ struct i2c_msg *msgs,
+ int num)
+{
+ struct sh_mobile_i2c_data *pd = adapter->algo_data;
+ struct i2c_msg *msg;
+ int err = 0;
+ u_int8_t val;
+ int i, k;
+
+ activate_ch(pd);
+
+ /* Process all messages */
+ for (i = 0; i < num; i++) {
+ msg = &msgs[i];
+
+ err = start_ch(pd, msg);
+ if (err)
+ break;
+
+ i2c_op(pd, OP_START, 0);
+
+ /* The interrupt handler takes care of the rest... */
+ k = wait_event_timeout(pd->wait,
+ pd->sr & (ICSR_TACK | SW_DONE),
+ 5 * HZ);
+ if (!k)
+ pr_err("Transfer request timed out\n");
+
+ k = 10;
+ again:
+ val = ioread8(ICSR(pd));
+
+ pr_debug("val 0x%02x pd->sr 0x%02x!\n", val, pd->sr);
+
+ if ((val | pd->sr) & (ICSR_TACK | ICSR_AL)) {
+ err = -EIO;
+ break;
+ }
+
+ if (!(!(val & ICSR_BUSY) && (val & ICSR_SCLM) &&
+ (val & ICSR_SDAM))) {
+ msleep(1);
+ if (k--)
+ goto again;
+
+ err = -EIO;
+ pr_err("Polling timed out\n");
+ break;
+ }
+ }
+
+ deactivate_ch(pd);
+
+ if (!err)
+ err = num;
+ return err;
+}
+
+u32 sh_mobile_i2c_func(struct i2c_adapter *adapter)
+{
+ return I2C_FUNC_I2C;
+}
+
+static struct i2c_algorithm sh_mobile_i2c_algorithm = {
+ .functionality = sh_mobile_i2c_func,
+ .master_xfer = sh_mobile_i2c_xfer,
+};
+
+static void sh_mobile_i2c_setup_channel(struct platform_device *dev)
+{
+ struct sh_mobile_i2c_data *pd = platform_get_drvdata(dev);
+ unsigned long peripheral_clk = clk_get_rate(pd->clk);
+ u_int32_t num;
+ u_int32_t denom;
+ u_int32_t tmp;
+
+ spin_lock_init(&pd->lock);
+ init_waitqueue_head(&pd->wait);
+
+ /* Calculate the value for iccl. From the data sheet:
+ * iccl = (p clock ÷ transfer rate) × (L ÷ (L + H))
+ * where L and H are the SCL low/high ratio (5/4 in this case).
+ * We also round off the result.
+ */
+ num = peripheral_clk * 5;
+ denom = pd->fast_mode ? (FAST_SPEED * 9) : (NORMAL_SPEED * 9);
+ tmp = num * 10 / denom;
+ if (tmp % 10 >= 5)
+ pd->iccl = (u_int8_t)((num/denom) + 1);
+ else
+ pd->iccl = (u_int8_t)(num/denom);
+
+ /* Calculate the value for icch. From the data sheet:
+ icch = (p clock ÷ transfer rate) × (H ÷ (L + H)) */
+ num = peripheral_clk * 4;
+ tmp = num * 10 / denom;
+ if (tmp % 10 >= 5)
+ pd->icch = (u_int8_t)((num/denom) + 1);
+ else
+ pd->icch = (u_int8_t)(num/denom);
+
+ deactivate_ch(pd);
+}
+
+static int sh_mobile_i2c_hook_irqs(struct platform_device *dev, int hook)
+{
+ struct resource *res;
+ int ret = -ENXIO;
+ int q, m;
+ int k = 0;
+ int n = 0;
+
+ while ((res = platform_get_resource(dev, IORESOURCE_IRQ, k))) {
+ for (n = res->start; hook && n <= res->end; n++) {
+ if (request_irq(n, sh_mobile_i2c_isr, 0,
+ dev->dev.bus_id, dev))
+ goto rollback;
+ }
+ k++;
+ }
+
+ if (hook)
+ return k > 0 ? 0 : -ENOENT;
+
+ k--;
+ ret = 0;
+
+ rollback:
+ for (q = k; k >= 0; k--) {
+ for (m = n; m >= res->start; m--)
+ free_irq(m, dev);
+
+ res = platform_get_resource(dev, IORESOURCE_IRQ, k - 1);
+ m = res->end;
+ }
+
+ return ret;
+}
+
+static int sh_mobile_i2c_probe(struct platform_device *dev)
+{
+ struct sh_mobile_i2c_data *pd;
+ struct resource *res;
+ int size;
+ int ret;
+
+ pd = kzalloc(sizeof(struct sh_mobile_i2c_data), GFP_KERNEL);
+ if (pd = NULL) {
+ dev_err(&dev->dev, "cannot allocate private data\n");
+ return -ENOMEM;
+ }
+
+ pd->clk = clk_get(&dev->dev, "peripheral_clk");
+ if (IS_ERR(pd->clk)) {
+ dev_err(&dev->dev, "cannot get peripheral clock\n");
+ ret = PTR_ERR(pd->clk);
+ goto err;
+ }
+
+ ret = sh_mobile_i2c_hook_irqs(dev, 1);
+ if (ret) {
+ dev_err(&dev->dev, "cannot request IRQ\n");
+ goto err_clk;
+ }
+
+ platform_set_drvdata(dev, pd);
+
+ res = platform_get_resource(dev, IORESOURCE_MEM, 0);
+ if (res = NULL) {
+ dev_err(&dev->dev, "cannot find IO resource\n");
+ ret = -ENOENT;
+ goto err_irq;
+ }
+
+ size = (res->end - res->start) + 1;
+
+ pd->ioarea = request_mem_region(res->start, size, dev->dev.bus_id);
+ if (pd->ioarea = NULL) {
+ dev_err(&dev->dev, "cannot request IO\n");
+ ret = -ENXIO;
+ goto err_irq;
+ }
+
+ pd->reg = ioremap(res->start, size);
+ if (pd->reg = NULL) {
+ dev_err(&dev->dev, "cannot map IO\n");
+ ret = -ENXIO;
+ goto err_res;
+ }
+
+ /* setup the private data */
+
+ pd->adap.owner = THIS_MODULE;
+ pd->adap.algo = &sh_mobile_i2c_algorithm;
+ pd->adap.algo_data = pd;
+ pd->adap.dev.parent = &dev->dev;
+ pd->adap.retries = 5;
+ pd->adap.nr = dev->id;
+
+ strlcpy(pd->adap.name, dev->name, sizeof(pd->adap.name));
+
+ sh_mobile_i2c_setup_channel(dev);
+
+ ret = i2c_add_numbered_adapter(&pd->adap);
+ if (ret < 0)
+ goto err_all;
+
+ return 0;
+
+ err_all:
+ iounmap(pd->reg);
+ err_res:
+ release_resource(pd->ioarea);
+ err_irq:
+ sh_mobile_i2c_hook_irqs(dev, 0);
+ err_clk:
+ clk_put(pd->clk);
+ err:
+ kfree(pd);
+ return ret;
+}
+
+static int sh_mobile_i2c_remove(struct platform_device *dev)
+{
+ struct sh_mobile_i2c_data *pd = platform_get_drvdata(dev);
+
+ i2c_del_adapter(&pd->adap);
+ iounmap(pd->reg);
+ release_resource(pd->ioarea);
+ sh_mobile_i2c_hook_irqs(dev, 0);
+ clk_put(pd->clk);
+ kfree(pd);
+ return 0;
+}
+
+static struct platform_driver sh_mobile_i2c_driver = {
+ .driver = {
+ .name = "i2c-sh_mobile",
+ .owner = THIS_MODULE,
+ },
+ .probe = sh_mobile_i2c_probe,
+ .remove = sh_mobile_i2c_remove,
+};
+
+static int __init sh_mobile_i2c_adap_init(void)
+{
+ return platform_driver_register(&sh_mobile_i2c_driver);
+}
+
+static void __exit sh_mobile_i2c_adap_exit(void)
+{
+ platform_driver_unregister(&sh_mobile_i2c_driver);
+}
+
+module_init(sh_mobile_i2c_adap_init);
+module_exit(sh_mobile_i2c_adap_exit);
+
+MODULE_DESCRIPTION("SuperH Mobile I2C Bus Controller driver");
+MODULE_AUTHOR("Magnus Damm");
+MODULE_LICENSE("GPL");
^ permalink raw reply [flat|nested] 14+ messages in thread
* [PATCH] i2c: SuperH Mobile I2C Bus Controller V3
2008-03-20 14:18 [PATCH] i2c: SuperH Mobile I2C Bus Controller Magnus Damm
2008-03-21 6:23 ` Paul Mundt
2008-03-21 12:07 ` [PATCH] i2c: SuperH Mobile I2C Bus Controller V2 Magnus Damm
@ 2008-03-25 10:55 ` Paul Mundt
2008-03-26 19:56 ` Andrew Morton
2008-03-27 22:07 ` Andrew Morton
2008-03-28 9:35 ` [PATCH] i2c: SuperH Mobile I2C Bus Controller V4 Magnus Damm
2008-04-02 2:59 ` [PATCH] i2c: SuperH Mobile I2C Bus Controller V5 Magnus Damm
4 siblings, 2 replies; 14+ messages in thread
From: Paul Mundt @ 2008-03-25 10:55 UTC (permalink / raw)
To: i2c, khali; +Cc: Magnus Damm, Andrew Morton, linux-sh
This is V3 of the SuperH Mobile I2C Controller Driver. A simple Master
only driver for the I2C block included in processors such as sh7343,
sh7722 and sh7723. Tested on a sh7722 MigoR using a rs5c372b rtc.
Changes since V2:
- dev_xxx() use
- Kill off superfluous ioarea resource
- Add SMBus emulation
Changes since V1:
- Use clk_get()/clk_put()/clk_get_rate() to get peripheral clock rate.
- Use pdev->dev.bus_id instead of dev->name
Verified with the rtc-rs5c372 SMBus conversion patches currently in -mm.
Signed-off-by: Magnus Damm <damm@igel.co.jp>
Signed-off-by: Paul Mundt <lethal@linux-sh.org>
---
drivers/i2c/busses/Kconfig | 9
drivers/i2c/busses/Makefile | 1
drivers/i2c/busses/i2c-sh_mobile.c | 504 +++++++++++++++++++++++++++++++++++++
3 files changed, 513 insertions(+)
diff --git a/drivers/i2c/busses/Kconfig b/drivers/i2c/busses/Kconfig
index 3376d53..80b426c 100644
--- a/drivers/i2c/busses/Kconfig
+++ b/drivers/i2c/busses/Kconfig
@@ -684,4 +684,13 @@ config I2C_PMCMSP
This driver can also be built as module. If so, the module
will be called i2c-pmcmsp.
+config I2C_SH_MOBILE
+ tristate "SuperH Mobile I2C Controller"
+ help
+ If you say yes to this option, support will be included for the
+ built-in I2C interface on the Renesas SH-Mobile processor.
+
+ This driver can also be built as a module. If so, the module
+ will be called i2c-sh_mobile.
+
endmenu
diff --git a/drivers/i2c/busses/Makefile b/drivers/i2c/busses/Makefile
index d179bce..5fe799d 100644
--- a/drivers/i2c/busses/Makefile
+++ b/drivers/i2c/busses/Makefile
@@ -53,6 +53,7 @@ obj-$(CONFIG_I2C_VIAPRO) += i2c-viapro.o
obj-$(CONFIG_I2C_VOODOO3) += i2c-voodoo3.o
obj-$(CONFIG_SCx200_ACB) += scx200_acb.o
obj-$(CONFIG_SCx200_I2C) += scx200_i2c.o
+obj-$(CONFIG_I2C_SH_MOBILE) += i2c-sh_mobile.o
ifeq ($(CONFIG_I2C_DEBUG_BUS),y)
EXTRA_CFLAGS += -DDEBUG
diff --git a/drivers/i2c/busses/i2c-sh_mobile.c b/drivers/i2c/busses/i2c-sh_mobile.c
new file mode 100644
index 0000000..ff7f69e
--- /dev/null
+++ b/drivers/i2c/busses/i2c-sh_mobile.c
@@ -0,0 +1,497 @@
+/*
+ * SuperH Mobile I2C Controller
+ *
+ * Copyright (C) 2008 Magnus Damm
+ *
+ * Based on i2c-simtec.c by Ben Dooks <ben@simtec.co.uk>
+ * Copyright (C) 2005 Simtec Electronics
+ *
+ * Portions of the code based on out-of-tree driver i2c-sh7343.c
+ * Copyright (c) 2006 Carlos Munoz <carlos@kenati.com>
+ *
+ * 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
+ *
+ * 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
+ */
+
+#include <linux/kernel.h>
+#include <linux/module.h>
+#include <linux/init.h>
+#include <linux/delay.h>
+#include <linux/platform_device.h>
+#include <linux/interrupt.h>
+#include <linux/i2c.h>
+#include <linux/err.h>
+#include <linux/clk.h>
+#include <asm/io.h>
+
+enum {
+ OP_START = 0,
+ OP_TX_ONLY,
+ OP_TX_STOP,
+ OP_TX_TO_RX,
+ OP_RX_ONLY,
+ OP_RX_STOP,
+};
+
+struct sh_mobile_i2c_data {
+ struct device *dev;
+ void __iomem *reg;
+ struct i2c_adapter adap;
+
+ struct clk *clk;
+ u_int8_t iccl;
+ u_int8_t icch;
+
+ int fast_mode;
+ int rx_ack_high;
+
+ spinlock_t lock;
+ wait_queue_head_t wait;
+ struct i2c_msg *msg;
+ int pos;
+ int sr;
+};
+
+#define NORMAL_SPEED 100000
+#define FAST_SPEED 400000
+
+/* Register offsets */
+#define ICDR(pd) (pd->reg + 0x00)
+#define ICCR(pd) (pd->reg + 0x04)
+#define ICSR(pd) (pd->reg + 0x08)
+#define ICIC(pd) (pd->reg + 0x0c)
+#define ICCL(pd) (pd->reg + 0x10)
+#define ICCH(pd) (pd->reg + 0x14)
+
+/* Register bits */
+#define ICCR_ICE 0x80
+#define ICCR_RACK 0x40
+#define ICCR_TRS 0x10
+#define ICCR_BBSY 0x04
+#define ICCR_SCP 0x01
+
+#define ICSR_SCLM 0x80
+#define ICSR_SDAM 0x40
+#define SW_DONE 0x20
+#define ICSR_BUSY 0x10
+#define ICSR_AL 0x08
+#define ICSR_TACK 0x04
+#define ICSR_WAIT 0x02
+#define ICSR_DTE 0x01
+
+#define ICIC_ALE 0x08
+#define ICIC_TACKE 0x04
+#define ICIC_WAITE 0x02
+#define ICIC_DTEE 0x01
+
+static void activate_ch(struct sh_mobile_i2c_data *pd)
+{
+ /* Enable channel and configure rx ack */
+ iowrite8(ioread8(ICCR(pd)) | ICCR_ICE |
+ (pd->rx_ack_high ? ICCR_RACK : 0), ICCR(pd));
+
+ /* Mask all interrupts */
+ iowrite8(0, ICIC(pd));
+
+ /* Set the clock */
+ iowrite8(pd->iccl, ICCL(pd));
+ iowrite8(pd->icch, ICCH(pd));
+}
+
+static void deactivate_ch(struct sh_mobile_i2c_data *pd)
+{
+ /* Clear/disable interrupts */
+ iowrite8(0, ICSR(pd));
+ iowrite8(0, ICIC(pd));
+
+ /* Initialize channel */
+ iowrite8(ioread8(ICCR(pd)) & ~ICCR_ICE, ICCR(pd));
+}
+
+static unsigned char i2c_op(struct sh_mobile_i2c_data *pd,
+ int op, unsigned char data)
+{
+ unsigned char ret = 0;
+
+ dev_dbg(pd->dev, "op %d, data in 0x%02x\n", op, data);
+
+ spin_lock_irq(&pd->lock);
+
+ switch (op) {
+ case OP_START:
+ iowrite8(0x94, ICCR(pd));
+ break;
+ case OP_TX_ONLY:
+ iowrite8(data, ICDR(pd));
+ break;
+ case OP_TX_STOP:
+ iowrite8(data, ICDR(pd));
+ iowrite8(0x90, ICCR(pd));
+ iowrite8(ICIC_ALE | ICIC_TACKE, ICIC(pd));
+ break;
+ case OP_TX_TO_RX:
+ iowrite8(data, ICDR(pd));
+ iowrite8(0x81, ICCR(pd));
+ break;
+ case OP_RX_ONLY:
+ ret = ioread8(ICDR(pd));
+ break;
+ case OP_RX_STOP:
+ ret = ioread8(ICDR(pd));
+ iowrite8(0xc0, ICCR(pd));
+ break;
+ }
+
+ spin_unlock_irq(&pd->lock);
+
+ dev_dbg(pd->dev, "op %d, data out 0x%02x\n", op, ret);
+ return ret;
+}
+
+static irqreturn_t sh_mobile_i2c_isr(int irq, void *dev_id)
+{
+ struct platform_device *dev = dev_id;
+ struct sh_mobile_i2c_data *pd = platform_get_drvdata(dev);
+ struct i2c_msg *msg = pd->msg;
+ unsigned char data, sr;
+ int wakeup = 0;
+
+ sr = ioread8(ICSR(pd));
+ pd->sr |= sr;
+
+ dev_dbg(pd->dev, "i2c_isr 0x%02x 0x%02x %s %d %d!\n", sr, pd->sr,
+ (msg->flags & I2C_M_RD) ? "read" : "write",
+ pd->pos, msg->len);
+
+ if (sr & (ICSR_AL | ICSR_TACK)) {
+ iowrite8(0, ICIC(pd)); /* disable interrupts */
+ wakeup = 1;
+ goto do_wakeup;
+ }
+
+ if (pd->pos = msg->len) {
+ i2c_op(pd, OP_RX_ONLY, 0);
+ wakeup = 1;
+ goto do_wakeup;
+ }
+
+ if (pd->pos = -1) {
+ data = (msg->addr & 0x7f) << 1;
+ data |= (msg->flags & I2C_M_RD) ? 1 : 0;
+ } else
+ data = msg->buf[pd->pos];
+
+ if ((pd->pos = -1) || !(msg->flags & I2C_M_RD)) {
+ if (msg->flags & I2C_M_RD)
+ i2c_op(pd, OP_TX_TO_RX, data);
+ else if (pd->pos = (msg->len - 1)) {
+ i2c_op(pd, OP_TX_STOP, data);
+ wakeup = 1;
+ } else
+ i2c_op(pd, OP_TX_ONLY, data);
+ } else {
+ if (pd->pos = (msg->len - 1))
+ data = i2c_op(pd, OP_RX_STOP, 0);
+ else
+ data = i2c_op(pd, OP_RX_ONLY, 0);
+
+ msg->buf[pd->pos] = data;
+ }
+ pd->pos++;
+
+ do_wakeup:
+ if (wakeup) {
+ pd->sr |= SW_DONE;
+ wake_up(&pd->wait);
+ }
+
+ return IRQ_HANDLED;
+}
+
+static int start_ch(struct sh_mobile_i2c_data *pd, struct i2c_msg *usr_msg)
+{
+ /* Initialize channel registers */
+ iowrite8(ioread8(ICCR(pd)) & ~ICCR_ICE, ICCR(pd));
+
+ /* Enable channel and configure rx ack */
+ iowrite8(ioread8(ICCR(pd)) | ICCR_ICE |
+ (pd->rx_ack_high ? ICCR_RACK : 0), ICCR(pd));
+
+ /* Set the clock */
+ iowrite8(pd->iccl, ICCL(pd));
+ iowrite8(pd->icch, ICCH(pd));
+
+ pd->msg = usr_msg;
+ pd->pos = -1;
+ pd->sr = 0;
+
+ /* Enable all interrupts except wait */
+ iowrite8(ioread8(ICIC(pd)) | ICIC_ALE | ICIC_TACKE | ICIC_DTEE,
+ ICIC(pd));
+ return 0;
+}
+
+static int sh_mobile_i2c_xfer(struct i2c_adapter *adapter,
+ struct i2c_msg *msgs,
+ int num)
+{
+ struct sh_mobile_i2c_data *pd = i2c_get_adapdata(adapter);
+ struct i2c_msg *msg;
+ int err = 0;
+ u_int8_t val;
+ int i, k;
+
+ activate_ch(pd);
+
+ /* Process all messages */
+ for (i = 0; i < num; i++) {
+ msg = &msgs[i];
+
+ err = start_ch(pd, msg);
+ if (err)
+ break;
+
+ i2c_op(pd, OP_START, 0);
+
+ /* The interrupt handler takes care of the rest... */
+ k = wait_event_timeout(pd->wait,
+ pd->sr & (ICSR_TACK | SW_DONE),
+ 5 * HZ);
+ if (!k)
+ dev_err(pd->dev, "Transfer request timed out\n");
+
+ k = 10;
+ again:
+ val = ioread8(ICSR(pd));
+
+ dev_dbg(pd->dev, "val 0x%02x pd->sr 0x%02x!\n", val, pd->sr);
+
+ if ((val | pd->sr) & (ICSR_TACK | ICSR_AL)) {
+ err = -EIO;
+ break;
+ }
+
+ if (!(!(val & ICSR_BUSY) && (val & ICSR_SCLM) &&
+ (val & ICSR_SDAM))) {
+ msleep(1);
+ if (k--)
+ goto again;
+
+ err = -EIO;
+ dev_err(pd->dev, "Polling timed out\n");
+ break;
+ }
+ }
+
+ deactivate_ch(pd);
+
+ if (!err)
+ err = num;
+ return err;
+}
+
+static u32 sh_mobile_i2c_func(struct i2c_adapter *adapter)
+{
+ return I2C_FUNC_I2C | I2C_FUNC_SMBUS_EMUL;
+}
+
+static struct i2c_algorithm sh_mobile_i2c_algorithm = {
+ .functionality = sh_mobile_i2c_func,
+ .master_xfer = sh_mobile_i2c_xfer,
+};
+
+static void sh_mobile_i2c_setup_channel(struct platform_device *dev)
+{
+ struct sh_mobile_i2c_data *pd = platform_get_drvdata(dev);
+ unsigned long peripheral_clk = clk_get_rate(pd->clk);
+ u_int32_t num;
+ u_int32_t denom;
+ u_int32_t tmp;
+
+ spin_lock_init(&pd->lock);
+ init_waitqueue_head(&pd->wait);
+
+ /* Calculate the value for iccl. From the data sheet:
+ * iccl = (p clock ÷ transfer rate) × (L ÷ (L + H))
+ * where L and H are the SCL low/high ratio (5/4 in this case).
+ * We also round off the result.
+ */
+ num = peripheral_clk * 5;
+ denom = pd->fast_mode ? (FAST_SPEED * 9) : (NORMAL_SPEED * 9);
+ tmp = num * 10 / denom;
+ if (tmp % 10 >= 5)
+ pd->iccl = (u_int8_t)((num/denom) + 1);
+ else
+ pd->iccl = (u_int8_t)(num/denom);
+
+ /* Calculate the value for icch. From the data sheet:
+ icch = (p clock ÷ transfer rate) × (H ÷ (L + H)) */
+ num = peripheral_clk * 4;
+ tmp = num * 10 / denom;
+ if (tmp % 10 >= 5)
+ pd->icch = (u_int8_t)((num/denom) + 1);
+ else
+ pd->icch = (u_int8_t)(num/denom);
+
+ deactivate_ch(pd);
+}
+
+static int sh_mobile_i2c_hook_irqs(struct platform_device *dev, int hook)
+{
+ struct resource *res;
+ int ret = -ENXIO;
+ int q, m;
+ int k = 0;
+ int n = 0;
+
+ while ((res = platform_get_resource(dev, IORESOURCE_IRQ, k))) {
+ for (n = res->start; hook && n <= res->end; n++) {
+ if (request_irq(n, sh_mobile_i2c_isr, 0,
+ dev->dev.bus_id, dev))
+ goto rollback;
+ }
+ k++;
+ }
+
+ if (hook)
+ return k > 0 ? 0 : -ENOENT;
+
+ k--;
+ ret = 0;
+
+ rollback:
+ for (q = k; k >= 0; k--) {
+ for (m = n; m >= res->start; m--)
+ free_irq(m, dev);
+
+ res = platform_get_resource(dev, IORESOURCE_IRQ, k - 1);
+ m = res->end;
+ }
+
+ return ret;
+}
+
+static int sh_mobile_i2c_probe(struct platform_device *dev)
+{
+ struct sh_mobile_i2c_data *pd;
+ struct i2c_adapter *adap;
+ struct resource *res;
+ int size;
+ int ret;
+
+ pd = kzalloc(sizeof(struct sh_mobile_i2c_data), GFP_KERNEL);
+ if (pd = NULL) {
+ dev_err(&dev->dev, "cannot allocate private data\n");
+ return -ENOMEM;
+ }
+
+ pd->clk = clk_get(&dev->dev, "peripheral_clk");
+ if (IS_ERR(pd->clk)) {
+ dev_err(&dev->dev, "cannot get peripheral clock\n");
+ ret = PTR_ERR(pd->clk);
+ goto err;
+ }
+
+ ret = sh_mobile_i2c_hook_irqs(dev, 1);
+ if (ret) {
+ dev_err(&dev->dev, "cannot request IRQ\n");
+ goto err_clk;
+ }
+
+ pd->dev = &dev->dev;
+ platform_set_drvdata(dev, pd);
+
+ res = platform_get_resource(dev, IORESOURCE_MEM, 0);
+ if (res = NULL) {
+ dev_err(&dev->dev, "cannot find IO resource\n");
+ ret = -ENOENT;
+ goto err_irq;
+ }
+
+ size = (res->end - res->start) + 1;
+
+ pd->reg = ioremap(res->start, size);
+ if (pd->reg = NULL) {
+ dev_err(&dev->dev, "cannot map IO\n");
+ ret = -ENXIO;
+ goto err_irq;
+ }
+
+ /* setup the private data */
+ adap = &pd->adap;
+ i2c_set_adapdata(adap, pd);
+
+ adap->owner = THIS_MODULE;
+ adap->algo = &sh_mobile_i2c_algorithm;
+ adap->dev.parent = &dev->dev;
+ adap->retries = 5;
+ adap->nr = dev->id;
+
+ strlcpy(adap->name, dev->name, sizeof(adap->name));
+
+ sh_mobile_i2c_setup_channel(dev);
+
+ ret = i2c_add_numbered_adapter(adap);
+ if (ret < 0)
+ goto err_all;
+
+ return 0;
+
+ err_all:
+ iounmap(pd->reg);
+ err_irq:
+ sh_mobile_i2c_hook_irqs(dev, 0);
+ err_clk:
+ clk_put(pd->clk);
+ err:
+ kfree(pd);
+ return ret;
+}
+
+static int sh_mobile_i2c_remove(struct platform_device *dev)
+{
+ struct sh_mobile_i2c_data *pd = platform_get_drvdata(dev);
+
+ i2c_del_adapter(&pd->adap);
+ iounmap(pd->reg);
+ sh_mobile_i2c_hook_irqs(dev, 0);
+ clk_put(pd->clk);
+ kfree(pd);
+ return 0;
+}
+
+static struct platform_driver sh_mobile_i2c_driver = {
+ .driver = {
+ .name = "i2c-sh_mobile",
+ .owner = THIS_MODULE,
+ },
+ .probe = sh_mobile_i2c_probe,
+ .remove = sh_mobile_i2c_remove,
+};
+
+static int __init sh_mobile_i2c_adap_init(void)
+{
+ return platform_driver_register(&sh_mobile_i2c_driver);
+}
+
+static void __exit sh_mobile_i2c_adap_exit(void)
+{
+ platform_driver_unregister(&sh_mobile_i2c_driver);
+}
+
+module_init(sh_mobile_i2c_adap_init);
+module_exit(sh_mobile_i2c_adap_exit);
+
+MODULE_DESCRIPTION("SuperH Mobile I2C Bus Controller driver");
+MODULE_AUTHOR("Magnus Damm");
+MODULE_LICENSE("GPL");
^ permalink raw reply related [flat|nested] 14+ messages in thread
* Re: [PATCH] i2c: SuperH Mobile I2C Bus Controller V3
2008-03-25 10:55 ` [PATCH] i2c: SuperH Mobile I2C Bus Controller V3 Paul Mundt
@ 2008-03-26 19:56 ` Andrew Morton
2008-03-27 22:07 ` Andrew Morton
1 sibling, 0 replies; 14+ messages in thread
From: Andrew Morton @ 2008-03-26 19:56 UTC (permalink / raw)
To: Paul Mundt; +Cc: i2c, khali, magnus.damm, linux-sh
On Tue, 25 Mar 2008 19:55:59 +0900
Paul Mundt <lethal@linux-sh.org> wrote:
> This is V3 of the SuperH Mobile I2C Controller Driver. A simple Master
> only driver for the I2C block included in processors such as sh7343,
> sh7722 and sh7723. Tested on a sh7722 MigoR using a rs5c372b rtc.
>
> +#include <asm/io.h>
please use checkpatch.
> +static unsigned char i2c_op(struct sh_mobile_i2c_data *pd,
> + int op, unsigned char data)
> +{
> + unsigned char ret = 0;
> +
> + dev_dbg(pd->dev, "op %d, data in 0x%02x\n", op, data);
> +
> + spin_lock_irq(&pd->lock);
> +
> + switch (op) {
> + case OP_START:
> + iowrite8(0x94, ICCR(pd));
> + break;
> + case OP_TX_ONLY:
> + iowrite8(data, ICDR(pd));
> + break;
> + case OP_TX_STOP:
> + iowrite8(data, ICDR(pd));
> + iowrite8(0x90, ICCR(pd));
> + iowrite8(ICIC_ALE | ICIC_TACKE, ICIC(pd));
> + break;
> + case OP_TX_TO_RX:
> + iowrite8(data, ICDR(pd));
> + iowrite8(0x81, ICCR(pd));
> + break;
> + case OP_RX_ONLY:
> + ret = ioread8(ICDR(pd));
> + break;
> + case OP_RX_STOP:
> + ret = ioread8(ICDR(pd));
> + iowrite8(0xc0, ICCR(pd));
> + break;
> + }
> +
> + spin_unlock_irq(&pd->lock);
> +
> + dev_dbg(pd->dev, "op %d, data out 0x%02x\n", op, ret);
> + return ret;
> +}
I'd be a bit concerned about reenabling interrupts...
> +static irqreturn_t sh_mobile_i2c_isr(int irq, void *dev_id)
> +{
> + struct platform_device *dev = dev_id;
> + struct sh_mobile_i2c_data *pd = platform_get_drvdata(dev);
> + struct i2c_msg *msg = pd->msg;
> + unsigned char data, sr;
> + int wakeup = 0;
> +
> + sr = ioread8(ICSR(pd));
> + pd->sr |= sr;
> +
> + dev_dbg(pd->dev, "i2c_isr 0x%02x 0x%02x %s %d %d!\n", sr, pd->sr,
> + (msg->flags & I2C_M_RD) ? "read" : "write",
> + pd->pos, msg->len);
> +
> + if (sr & (ICSR_AL | ICSR_TACK)) {
> + iowrite8(0, ICIC(pd)); /* disable interrupts */
> + wakeup = 1;
> + goto do_wakeup;
> + }
> +
> + if (pd->pos = msg->len) {
> + i2c_op(pd, OP_RX_ONLY, 0);
from the interrupt handler...
> +static int sh_mobile_i2c_hook_irqs(struct platform_device *dev, int hook)
> +{
> + struct resource *res;
> + int ret = -ENXIO;
> + int q, m;
> + int k = 0;
> + int n = 0;
> +
> + while ((res = platform_get_resource(dev, IORESOURCE_IRQ, k))) {
> + for (n = res->start; hook && n <= res->end; n++) {
> + if (request_irq(n, sh_mobile_i2c_isr, 0,
> + dev->dev.bus_id, dev))
even though it isn't using IRQF_DISABLED here,
: int request_irq(unsigned int irq, irq_handler_t handler,
: unsigned long irqflags, const char *devname, void *dev_id)
: {
: struct irqaction *action;
: int retval;
:
: #ifdef CONFIG_LOCKDEP
: /*
: * Lockdep wants atomic interrupt handlers:
: */
: irqflags |= IRQF_DISABLED;
: #endif
it will do so if lockdep is supported.
^ permalink raw reply [flat|nested] 14+ messages in thread
* Re: [PATCH] i2c: SuperH Mobile I2C Bus Controller V3
2008-03-25 10:55 ` [PATCH] i2c: SuperH Mobile I2C Bus Controller V3 Paul Mundt
2008-03-26 19:56 ` Andrew Morton
@ 2008-03-27 22:07 ` Andrew Morton
1 sibling, 0 replies; 14+ messages in thread
From: Andrew Morton @ 2008-03-27 22:07 UTC (permalink / raw)
To: Paul Mundt; +Cc: i2c, khali, magnus.damm, linux-sh
On Tue, 25 Mar 2008 19:55:59 +0900
Paul Mundt <lethal@linux-sh.org> wrote:
> This is V3 of the SuperH Mobile I2C Controller Driver.
i386 allmodconfig:
ERROR: "clk_get_rate" [drivers/i2c/busses/i2c-sh_mobile.ko] undefined!
ERROR: "clk_get" [drivers/i2c/busses/i2c-sh_mobile.ko] undefined!
ERROR: "clk_put" [drivers/i2c/busses/i2c-sh_mobile.ko] undefined!
^ permalink raw reply [flat|nested] 14+ messages in thread
* [PATCH] i2c: SuperH Mobile I2C Bus Controller V4
2008-03-20 14:18 [PATCH] i2c: SuperH Mobile I2C Bus Controller Magnus Damm
` (2 preceding siblings ...)
2008-03-25 10:55 ` [PATCH] i2c: SuperH Mobile I2C Bus Controller V3 Paul Mundt
@ 2008-03-28 9:35 ` Magnus Damm
2008-03-29 17:57 ` [i2c] " Ben Dooks
2008-04-02 2:59 ` [PATCH] i2c: SuperH Mobile I2C Bus Controller V5 Magnus Damm
4 siblings, 1 reply; 14+ messages in thread
From: Magnus Damm @ 2008-03-28 9:35 UTC (permalink / raw)
To: i2c; +Cc: khali, Magnus Damm, lethal, akpm, linux-sh
This is V4 of the SuperH Mobile I2C Controller Driver. A simple Master
only driver for the I2C block included in processors such as sh7343,
sh7722 and sh7723. Tested on a sh7722 MigoR using a rs5c732b rtc.
Changes since V3:
- Added SUPERH Kconfig dependency
- Use spin_lock_irqsave() and spin_unlock_irqrestore()
- Use IRQF_DISABLED
- Checkpatch fixes
Changes since V2:
- dev_xxx() use
- Kill off superfluous ioarea resource
- Add SMBus emulation
Changes since V1:
- Use clk_get()/clk_put()/clk_get_rate() to get peripheral clock rate.
- Use pdev->dev.bus_id instead of dev->name
Verified with the rtc-rs5c372 SMBus conversion patches currently in -mm.
Signed-off-by: Magnus Damm <damm@igel.co.jp>
Signed-off-by: Paul Mundt <lethal@linux-sh.org>
---
drivers/i2c/busses/Kconfig | 10
drivers/i2c/busses/Makefile | 1
drivers/i2c/busses/i2c-sh_mobile.c | 498 ++++++++++++++++++++++++++++++++++++
3 files changed, 509 insertions(+)
--- 0001/drivers/i2c/busses/Kconfig
+++ work/drivers/i2c/busses/Kconfig 2008-03-28 16:45:42.000000000 +0900
@@ -672,4 +672,14 @@ config I2C_PMCMSP
This driver can also be built as module. If so, the module
will be called i2c-pmcmsp.
+config I2C_SH_MOBILE
+ tristate "SuperH Mobile I2C Controller"
+ depends on SUPERH
+ help
+ If you say yes to this option, support will be included for the
+ built-in I2C interface on the Renesas SH-Mobile processor.
+
+ This driver can also be built as a module. If so, the module
+ will be called i2c-sh_mobile.
+
endmenu
--- 0001/drivers/i2c/busses/Makefile
+++ work/drivers/i2c/busses/Makefile 2008-03-28 16:35:37.000000000 +0900
@@ -52,6 +52,7 @@ obj-$(CONFIG_I2C_VIAPRO) += i2c-viapro.o
obj-$(CONFIG_I2C_VOODOO3) += i2c-voodoo3.o
obj-$(CONFIG_SCx200_ACB) += scx200_acb.o
obj-$(CONFIG_SCx200_I2C) += scx200_i2c.o
+obj-$(CONFIG_I2C_SH_MOBILE) += i2c-sh_mobile.o
ifeq ($(CONFIG_I2C_DEBUG_BUS),y)
EXTRA_CFLAGS += -DDEBUG
--- /dev/null
+++ work/drivers/i2c/busses/i2c-sh_mobile.c 2008-03-28 17:24:11.000000000 +0900
@@ -0,0 +1,498 @@
+/*
+ * SuperH Mobile I2C Controller
+ *
+ * Copyright (C) 2008 Magnus Damm
+ *
+ * Based on i2c-simtec.c by Ben Dooks <ben@simtec.co.uk>
+ * Copyright (C) 2005 Simtec Electronics
+ *
+ * Portions of the code based on out-of-tree driver i2c-sh7343.c
+ * Copyright (c) 2006 Carlos Munoz <carlos@kenati.com>
+ *
+ * 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
+ *
+ * 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
+ */
+
+#include <linux/kernel.h>
+#include <linux/module.h>
+#include <linux/init.h>
+#include <linux/delay.h>
+#include <linux/platform_device.h>
+#include <linux/interrupt.h>
+#include <linux/i2c.h>
+#include <linux/err.h>
+#include <linux/clk.h>
+#include <linux/io.h>
+
+enum {
+ OP_START = 0,
+ OP_TX_ONLY,
+ OP_TX_STOP,
+ OP_TX_TO_RX,
+ OP_RX_ONLY,
+ OP_RX_STOP,
+};
+
+struct sh_mobile_i2c_data {
+ struct device *dev;
+ void __iomem *reg;
+ struct i2c_adapter adap;
+
+ struct clk *clk;
+ u_int8_t iccl;
+ u_int8_t icch;
+
+ int fast_mode;
+ int rx_ack_high;
+
+ spinlock_t lock;
+ wait_queue_head_t wait;
+ struct i2c_msg *msg;
+ int pos;
+ int sr;
+};
+
+#define NORMAL_SPEED 100000
+#define FAST_SPEED 400000
+
+/* Register offsets */
+#define ICDR(pd) (pd->reg + 0x00)
+#define ICCR(pd) (pd->reg + 0x04)
+#define ICSR(pd) (pd->reg + 0x08)
+#define ICIC(pd) (pd->reg + 0x0c)
+#define ICCL(pd) (pd->reg + 0x10)
+#define ICCH(pd) (pd->reg + 0x14)
+
+/* Register bits */
+#define ICCR_ICE 0x80
+#define ICCR_RACK 0x40
+#define ICCR_TRS 0x10
+#define ICCR_BBSY 0x04
+#define ICCR_SCP 0x01
+
+#define ICSR_SCLM 0x80
+#define ICSR_SDAM 0x40
+#define SW_DONE 0x20
+#define ICSR_BUSY 0x10
+#define ICSR_AL 0x08
+#define ICSR_TACK 0x04
+#define ICSR_WAIT 0x02
+#define ICSR_DTE 0x01
+
+#define ICIC_ALE 0x08
+#define ICIC_TACKE 0x04
+#define ICIC_WAITE 0x02
+#define ICIC_DTEE 0x01
+
+static void activate_ch(struct sh_mobile_i2c_data *pd)
+{
+ /* Enable channel and configure rx ack */
+ iowrite8(ioread8(ICCR(pd)) | ICCR_ICE |
+ (pd->rx_ack_high ? ICCR_RACK : 0), ICCR(pd));
+
+ /* Mask all interrupts */
+ iowrite8(0, ICIC(pd));
+
+ /* Set the clock */
+ iowrite8(pd->iccl, ICCL(pd));
+ iowrite8(pd->icch, ICCH(pd));
+}
+
+static void deactivate_ch(struct sh_mobile_i2c_data *pd)
+{
+ /* Clear/disable interrupts */
+ iowrite8(0, ICSR(pd));
+ iowrite8(0, ICIC(pd));
+
+ /* Initialize channel */
+ iowrite8(ioread8(ICCR(pd)) & ~ICCR_ICE, ICCR(pd));
+}
+
+static unsigned char i2c_op(struct sh_mobile_i2c_data *pd,
+ int op, unsigned char data)
+{
+ unsigned char ret = 0;
+ unsigned long flags;
+
+ dev_dbg(pd->dev, "op %d, data in 0x%02x\n", op, data);
+
+ spin_lock_irqsave(&pd->lock, flags);
+
+ switch (op) {
+ case OP_START:
+ iowrite8(0x94, ICCR(pd));
+ break;
+ case OP_TX_ONLY:
+ iowrite8(data, ICDR(pd));
+ break;
+ case OP_TX_STOP:
+ iowrite8(data, ICDR(pd));
+ iowrite8(0x90, ICCR(pd));
+ iowrite8(ICIC_ALE | ICIC_TACKE, ICIC(pd));
+ break;
+ case OP_TX_TO_RX:
+ iowrite8(data, ICDR(pd));
+ iowrite8(0x81, ICCR(pd));
+ break;
+ case OP_RX_ONLY:
+ ret = ioread8(ICDR(pd));
+ break;
+ case OP_RX_STOP:
+ ret = ioread8(ICDR(pd));
+ iowrite8(0xc0, ICCR(pd));
+ break;
+ }
+
+ spin_unlock_irqrestore(&pd->lock, flags);
+
+ dev_dbg(pd->dev, "op %d, data out 0x%02x\n", op, ret);
+ return ret;
+}
+
+static irqreturn_t sh_mobile_i2c_isr(int irq, void *dev_id)
+{
+ struct platform_device *dev = dev_id;
+ struct sh_mobile_i2c_data *pd = platform_get_drvdata(dev);
+ struct i2c_msg *msg = pd->msg;
+ unsigned char data, sr;
+ int wakeup = 0;
+
+ sr = ioread8(ICSR(pd));
+ pd->sr |= sr;
+
+ dev_dbg(pd->dev, "i2c_isr 0x%02x 0x%02x %s %d %d!\n", sr, pd->sr,
+ (msg->flags & I2C_M_RD) ? "read" : "write",
+ pd->pos, msg->len);
+
+ if (sr & (ICSR_AL | ICSR_TACK)) {
+ iowrite8(0, ICIC(pd)); /* disable interrupts */
+ wakeup = 1;
+ goto do_wakeup;
+ }
+
+ if (pd->pos = msg->len) {
+ i2c_op(pd, OP_RX_ONLY, 0);
+ wakeup = 1;
+ goto do_wakeup;
+ }
+
+ if (pd->pos = -1) {
+ data = (msg->addr & 0x7f) << 1;
+ data |= (msg->flags & I2C_M_RD) ? 1 : 0;
+ } else
+ data = msg->buf[pd->pos];
+
+ if ((pd->pos = -1) || !(msg->flags & I2C_M_RD)) {
+ if (msg->flags & I2C_M_RD)
+ i2c_op(pd, OP_TX_TO_RX, data);
+ else if (pd->pos = (msg->len - 1)) {
+ i2c_op(pd, OP_TX_STOP, data);
+ wakeup = 1;
+ } else
+ i2c_op(pd, OP_TX_ONLY, data);
+ } else {
+ if (pd->pos = (msg->len - 1))
+ data = i2c_op(pd, OP_RX_STOP, 0);
+ else
+ data = i2c_op(pd, OP_RX_ONLY, 0);
+
+ msg->buf[pd->pos] = data;
+ }
+ pd->pos++;
+
+ do_wakeup:
+ if (wakeup) {
+ pd->sr |= SW_DONE;
+ wake_up(&pd->wait);
+ }
+
+ return IRQ_HANDLED;
+}
+
+static int start_ch(struct sh_mobile_i2c_data *pd, struct i2c_msg *usr_msg)
+{
+ /* Initialize channel registers */
+ iowrite8(ioread8(ICCR(pd)) & ~ICCR_ICE, ICCR(pd));
+
+ /* Enable channel and configure rx ack */
+ iowrite8(ioread8(ICCR(pd)) | ICCR_ICE |
+ (pd->rx_ack_high ? ICCR_RACK : 0), ICCR(pd));
+
+ /* Set the clock */
+ iowrite8(pd->iccl, ICCL(pd));
+ iowrite8(pd->icch, ICCH(pd));
+
+ pd->msg = usr_msg;
+ pd->pos = -1;
+ pd->sr = 0;
+
+ /* Enable all interrupts except wait */
+ iowrite8(ioread8(ICIC(pd)) | ICIC_ALE | ICIC_TACKE | ICIC_DTEE,
+ ICIC(pd));
+ return 0;
+}
+
+static int sh_mobile_i2c_xfer(struct i2c_adapter *adapter,
+ struct i2c_msg *msgs,
+ int num)
+{
+ struct sh_mobile_i2c_data *pd = i2c_get_adapdata(adapter);
+ struct i2c_msg *msg;
+ int err = 0;
+ u_int8_t val;
+ int i, k;
+
+ activate_ch(pd);
+
+ /* Process all messages */
+ for (i = 0; i < num; i++) {
+ msg = &msgs[i];
+
+ err = start_ch(pd, msg);
+ if (err)
+ break;
+
+ i2c_op(pd, OP_START, 0);
+
+ /* The interrupt handler takes care of the rest... */
+ k = wait_event_timeout(pd->wait,
+ pd->sr & (ICSR_TACK | SW_DONE),
+ 5 * HZ);
+ if (!k)
+ dev_err(pd->dev, "Transfer request timed out\n");
+
+ k = 10;
+again:
+ val = ioread8(ICSR(pd));
+
+ dev_dbg(pd->dev, "val 0x%02x pd->sr 0x%02x!\n", val, pd->sr);
+
+ if ((val | pd->sr) & (ICSR_TACK | ICSR_AL)) {
+ err = -EIO;
+ break;
+ }
+
+ if (!(!(val & ICSR_BUSY) && (val & ICSR_SCLM) &&
+ (val & ICSR_SDAM))) {
+ msleep(1);
+ if (k--)
+ goto again;
+
+ err = -EIO;
+ dev_err(pd->dev, "Polling timed out\n");
+ break;
+ }
+ }
+
+ deactivate_ch(pd);
+
+ if (!err)
+ err = num;
+ return err;
+}
+
+static u32 sh_mobile_i2c_func(struct i2c_adapter *adapter)
+{
+ return I2C_FUNC_I2C | I2C_FUNC_SMBUS_EMUL;
+}
+
+static struct i2c_algorithm sh_mobile_i2c_algorithm = {
+ .functionality = sh_mobile_i2c_func,
+ .master_xfer = sh_mobile_i2c_xfer,
+};
+
+static void sh_mobile_i2c_setup_channel(struct platform_device *dev)
+{
+ struct sh_mobile_i2c_data *pd = platform_get_drvdata(dev);
+ unsigned long peripheral_clk = clk_get_rate(pd->clk);
+ u_int32_t num;
+ u_int32_t denom;
+ u_int32_t tmp;
+
+ spin_lock_init(&pd->lock);
+ init_waitqueue_head(&pd->wait);
+
+ /* Calculate the value for iccl. From the data sheet:
+ * iccl = (p clock ÷ transfer rate) × (L ÷ (L + H))
+ * where L and H are the SCL low/high ratio (5/4 in this case).
+ * We also round off the result.
+ */
+ num = peripheral_clk * 5;
+ denom = pd->fast_mode ? (FAST_SPEED * 9) : (NORMAL_SPEED * 9);
+ tmp = num * 10 / denom;
+ if (tmp % 10 >= 5)
+ pd->iccl = (u_int8_t)((num/denom) + 1);
+ else
+ pd->iccl = (u_int8_t)(num/denom);
+
+ /* Calculate the value for icch. From the data sheet:
+ icch = (p clock ÷ transfer rate) × (H ÷ (L + H)) */
+ num = peripheral_clk * 4;
+ tmp = num * 10 / denom;
+ if (tmp % 10 >= 5)
+ pd->icch = (u_int8_t)((num/denom) + 1);
+ else
+ pd->icch = (u_int8_t)(num/denom);
+
+ deactivate_ch(pd);
+}
+
+static int sh_mobile_i2c_hook_irqs(struct platform_device *dev, int hook)
+{
+ struct resource *res;
+ int ret = -ENXIO;
+ int q, m;
+ int k = 0;
+ int n = 0;
+
+ while ((res = platform_get_resource(dev, IORESOURCE_IRQ, k))) {
+ for (n = res->start; hook && n <= res->end; n++) {
+ if (request_irq(n, sh_mobile_i2c_isr, IRQF_DISABLED,
+ dev->dev.bus_id, dev))
+ goto rollback;
+ }
+ k++;
+ }
+
+ if (hook)
+ return k > 0 ? 0 : -ENOENT;
+
+ k--;
+ ret = 0;
+
+ rollback:
+ for (q = k; k >= 0; k--) {
+ for (m = n; m >= res->start; m--)
+ free_irq(m, dev);
+
+ res = platform_get_resource(dev, IORESOURCE_IRQ, k - 1);
+ m = res->end;
+ }
+
+ return ret;
+}
+
+static int sh_mobile_i2c_probe(struct platform_device *dev)
+{
+ struct sh_mobile_i2c_data *pd;
+ struct i2c_adapter *adap;
+ struct resource *res;
+ int size;
+ int ret;
+
+ pd = kzalloc(sizeof(struct sh_mobile_i2c_data), GFP_KERNEL);
+ if (pd = NULL) {
+ dev_err(&dev->dev, "cannot allocate private data\n");
+ return -ENOMEM;
+ }
+
+ pd->clk = clk_get(&dev->dev, "peripheral_clk");
+ if (IS_ERR(pd->clk)) {
+ dev_err(&dev->dev, "cannot get peripheral clock\n");
+ ret = PTR_ERR(pd->clk);
+ goto err;
+ }
+
+ ret = sh_mobile_i2c_hook_irqs(dev, 1);
+ if (ret) {
+ dev_err(&dev->dev, "cannot request IRQ\n");
+ goto err_clk;
+ }
+
+ pd->dev = &dev->dev;
+ platform_set_drvdata(dev, pd);
+
+ res = platform_get_resource(dev, IORESOURCE_MEM, 0);
+ if (res = NULL) {
+ dev_err(&dev->dev, "cannot find IO resource\n");
+ ret = -ENOENT;
+ goto err_irq;
+ }
+
+ size = (res->end - res->start) + 1;
+
+ pd->reg = ioremap(res->start, size);
+ if (pd->reg = NULL) {
+ dev_err(&dev->dev, "cannot map IO\n");
+ ret = -ENXIO;
+ goto err_irq;
+ }
+
+ /* setup the private data */
+ adap = &pd->adap;
+ i2c_set_adapdata(adap, pd);
+
+ adap->owner = THIS_MODULE;
+ adap->algo = &sh_mobile_i2c_algorithm;
+ adap->dev.parent = &dev->dev;
+ adap->retries = 5;
+ adap->nr = dev->id;
+
+ strlcpy(adap->name, dev->name, sizeof(adap->name));
+
+ sh_mobile_i2c_setup_channel(dev);
+
+ ret = i2c_add_numbered_adapter(adap);
+ if (ret < 0)
+ goto err_all;
+
+ return 0;
+
+ err_all:
+ iounmap(pd->reg);
+ err_irq:
+ sh_mobile_i2c_hook_irqs(dev, 0);
+ err_clk:
+ clk_put(pd->clk);
+ err:
+ kfree(pd);
+ return ret;
+}
+
+static int sh_mobile_i2c_remove(struct platform_device *dev)
+{
+ struct sh_mobile_i2c_data *pd = platform_get_drvdata(dev);
+
+ i2c_del_adapter(&pd->adap);
+ iounmap(pd->reg);
+ sh_mobile_i2c_hook_irqs(dev, 0);
+ clk_put(pd->clk);
+ kfree(pd);
+ return 0;
+}
+
+static struct platform_driver sh_mobile_i2c_driver = {
+ .driver = {
+ .name = "i2c-sh_mobile",
+ .owner = THIS_MODULE,
+ },
+ .probe = sh_mobile_i2c_probe,
+ .remove = sh_mobile_i2c_remove,
+};
+
+static int __init sh_mobile_i2c_adap_init(void)
+{
+ return platform_driver_register(&sh_mobile_i2c_driver);
+}
+
+static void __exit sh_mobile_i2c_adap_exit(void)
+{
+ platform_driver_unregister(&sh_mobile_i2c_driver);
+}
+
+module_init(sh_mobile_i2c_adap_init);
+module_exit(sh_mobile_i2c_adap_exit);
+
+MODULE_DESCRIPTION("SuperH Mobile I2C Bus Controller driver");
+MODULE_AUTHOR("Magnus Damm");
+MODULE_LICENSE("GPL");
^ permalink raw reply [flat|nested] 14+ messages in thread
* Re: [i2c] [PATCH] i2c: SuperH Mobile I2C Bus Controller V4
2008-03-28 9:35 ` [PATCH] i2c: SuperH Mobile I2C Bus Controller V4 Magnus Damm
@ 2008-03-29 17:57 ` Ben Dooks
[not found] ` <20080329175732.GA9638-elnMNo+KYs3pIgCt6eIbzw@public.gmane.org>
` (2 more replies)
0 siblings, 3 replies; 14+ messages in thread
From: Ben Dooks @ 2008-03-29 17:57 UTC (permalink / raw)
To: Magnus Damm; +Cc: i2c, akpm, lethal, linux-sh
On Fri, Mar 28, 2008 at 06:35:30PM +0900, Magnus Damm wrote:
> This is V4 of the SuperH Mobile I2C Controller Driver. A simple Master
> only driver for the I2C block included in processors such as sh7343,
> sh7722 and sh7723. Tested on a sh7722 MigoR using a rs5c732b rtc.
>
> Changes since V3:
> - Added SUPERH Kconfig dependency
> - Use spin_lock_irqsave() and spin_unlock_irqrestore()
> - Use IRQF_DISABLED
> - Checkpatch fixes
> Changes since V2:
> - dev_xxx() use
> - Kill off superfluous ioarea resource
> - Add SMBus emulation
> Changes since V1:
> - Use clk_get()/clk_put()/clk_get_rate() to get peripheral clock rate.
> - Use pdev->dev.bus_id instead of dev->name
Do we really need these in the change message, we tend to assume that
the changes before submission aren't really relevant. I would advise
on removing these.
> Verified with the rtc-rs5c372 SMBus conversion patches currently in -mm.
>
> Signed-off-by: Magnus Damm <damm@igel.co.jp>
> Signed-off-by: Paul Mundt <lethal@linux-sh.org>
> ---
>
> drivers/i2c/busses/Kconfig | 10
> drivers/i2c/busses/Makefile | 1
> drivers/i2c/busses/i2c-sh_mobile.c | 498 ++++++++++++++++++++++++++++++++++++
> 3 files changed, 509 insertions(+)
>
> --- 0001/drivers/i2c/busses/Kconfig
> +++ work/drivers/i2c/busses/Kconfig 2008-03-28 16:45:42.000000000 +0900
> @@ -672,4 +672,14 @@ config I2C_PMCMSP
> This driver can also be built as module. If so, the module
> will be called i2c-pmcmsp.
>
> +config I2C_SH_MOBILE
> + tristate "SuperH Mobile I2C Controller"
> + depends on SUPERH
> + help
> + If you say yes to this option, support will be included for the
> + built-in I2C interface on the Renesas SH-Mobile processor.
> +
> + This driver can also be built as a module. If so, the module
> + will be called i2c-sh_mobile.
> +
> endmenu
> --- 0001/drivers/i2c/busses/Makefile
> +++ work/drivers/i2c/busses/Makefile 2008-03-28 16:35:37.000000000 +0900
> @@ -52,6 +52,7 @@ obj-$(CONFIG_I2C_VIAPRO) += i2c-viapro.o
> obj-$(CONFIG_I2C_VOODOO3) += i2c-voodoo3.o
> obj-$(CONFIG_SCx200_ACB) += scx200_acb.o
> obj-$(CONFIG_SCx200_I2C) += scx200_i2c.o
> +obj-$(CONFIG_I2C_SH_MOBILE) += i2c-sh_mobile.o
>
> ifeq ($(CONFIG_I2C_DEBUG_BUS),y)
> EXTRA_CFLAGS += -DDEBUG
> --- /dev/null
> +++ work/drivers/i2c/busses/i2c-sh_mobile.c 2008-03-28 17:24:11.000000000 +0900
> @@ -0,0 +1,498 @@
> +/*
> + * SuperH Mobile I2C Controller
> + *
> + * Copyright (C) 2008 Magnus Damm
> + *
> + * Based on i2c-simtec.c by Ben Dooks <ben@simtec.co.uk>
> + * Copyright (C) 2005 Simtec Electronics
> + *
I think you can probably entirely remove that, I don't see enough of
a similarity with my i2c-simtec driver (a cpld based gpio style
adaptor).
> + * Portions of the code based on out-of-tree driver i2c-sh7343.c
> + * Copyright (c) 2006 Carlos Munoz <carlos@kenati.com>
> + *
> + * 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
> + *
> + * 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
> + */
> +
> +#include <linux/kernel.h>
> +#include <linux/module.h>
> +#include <linux/init.h>
> +#include <linux/delay.h>
> +#include <linux/platform_device.h>
> +#include <linux/interrupt.h>
> +#include <linux/i2c.h>
> +#include <linux/err.h>
> +#include <linux/clk.h>
> +#include <linux/io.h>
> +
> +enum {
> + OP_START = 0,
> + OP_TX_ONLY,
> + OP_TX_STOP,
> + OP_TX_TO_RX,
> + OP_RX_ONLY,
> + OP_RX_STOP,
> +};
Please name the enum something like sh_mobile_i2c_op, and use the
named enum when you are using this in the rest of the code, this way
gcc will catch missed cases in case statements.
> +
> +struct sh_mobile_i2c_data {
> + struct device *dev;
> + void __iomem *reg;
> + struct i2c_adapter adap;
> +
> + struct clk *clk;
> + u_int8_t iccl;
> + u_int8_t icch;
> +
> + int fast_mode;
> + int rx_ack_high;
interesting, why not use smaller variables or flags for these
variables?
> +
> + spinlock_t lock;
> + wait_queue_head_t wait;
> + struct i2c_msg *msg;
> + int pos;
> + int sr;
> +};
> +
> +#define NORMAL_SPEED 100000
> +#define FAST_SPEED 400000
> +
> +/* Register offsets */
> +#define ICDR(pd) (pd->reg + 0x00)
> +#define ICCR(pd) (pd->reg + 0x04)
> +#define ICSR(pd) (pd->reg + 0x08)
> +#define ICIC(pd) (pd->reg + 0x0c)
> +#define ICCL(pd) (pd->reg + 0x10)
> +#define ICCH(pd) (pd->reg + 0x14)
> +
> +/* Register bits */
> +#define ICCR_ICE 0x80
> +#define ICCR_RACK 0x40
> +#define ICCR_TRS 0x10
> +#define ICCR_BBSY 0x04
> +#define ICCR_SCP 0x01
> +
> +#define ICSR_SCLM 0x80
> +#define ICSR_SDAM 0x40
> +#define SW_DONE 0x20
> +#define ICSR_BUSY 0x10
> +#define ICSR_AL 0x08
> +#define ICSR_TACK 0x04
> +#define ICSR_WAIT 0x02
> +#define ICSR_DTE 0x01
> +
> +#define ICIC_ALE 0x08
> +#define ICIC_TACKE 0x04
> +#define ICIC_WAITE 0x02
> +#define ICIC_DTEE 0x01
> +
> +static void activate_ch(struct sh_mobile_i2c_data *pd)
> +{
> + /* Enable channel and configure rx ack */
> + iowrite8(ioread8(ICCR(pd)) | ICCR_ICE |
> + (pd->rx_ack_high ? ICCR_RACK : 0), ICCR(pd));
> +
> + /* Mask all interrupts */
> + iowrite8(0, ICIC(pd));
> +
> + /* Set the clock */
> + iowrite8(pd->iccl, ICCL(pd));
> + iowrite8(pd->icch, ICCH(pd));
> +}
> +
> +static void deactivate_ch(struct sh_mobile_i2c_data *pd)
> +{
> + /* Clear/disable interrupts */
> + iowrite8(0, ICSR(pd));
> + iowrite8(0, ICIC(pd));
> +
> + /* Initialize channel */
> + iowrite8(ioread8(ICCR(pd)) & ~ICCR_ICE, ICCR(pd));
> +}
> +
> +static unsigned char i2c_op(struct sh_mobile_i2c_data *pd,
> + int op, unsigned char data)
see above comment on the use of the named enum for the op field.
> +{
> + unsigned char ret = 0;
> + unsigned long flags;
> +
> + dev_dbg(pd->dev, "op %d, data in 0x%02x\n", op, data);
> +
> + spin_lock_irqsave(&pd->lock, flags);
> +
> + switch (op) {
> + case OP_START:
> + iowrite8(0x94, ICCR(pd));
> + break;
> + case OP_TX_ONLY:
> + iowrite8(data, ICDR(pd));
> + break;
> + case OP_TX_STOP:
> + iowrite8(data, ICDR(pd));
> + iowrite8(0x90, ICCR(pd));
> + iowrite8(ICIC_ALE | ICIC_TACKE, ICIC(pd));
> + break;
> + case OP_TX_TO_RX:
> + iowrite8(data, ICDR(pd));
> + iowrite8(0x81, ICCR(pd));
> + break;
> + case OP_RX_ONLY:
> + ret = ioread8(ICDR(pd));
> + break;
> + case OP_RX_STOP:
> + ret = ioread8(ICDR(pd));
> + iowrite8(0xc0, ICCR(pd));
> + break;
> + }
> +
> + spin_unlock_irqrestore(&pd->lock, flags);
> +
> + dev_dbg(pd->dev, "op %d, data out 0x%02x\n", op, ret);
> + return ret;
> +}
> +
> +static irqreturn_t sh_mobile_i2c_isr(int irq, void *dev_id)
> +{
> + struct platform_device *dev = dev_id;
> + struct sh_mobile_i2c_data *pd = platform_get_drvdata(dev);
> + struct i2c_msg *msg = pd->msg;
> + unsigned char data, sr;
> + int wakeup = 0;
> +
> + sr = ioread8(ICSR(pd));
> + pd->sr |= sr;
> +
> + dev_dbg(pd->dev, "i2c_isr 0x%02x 0x%02x %s %d %d!\n", sr, pd->sr,
> + (msg->flags & I2C_M_RD) ? "read" : "write",
> + pd->pos, msg->len);
> +
> + if (sr & (ICSR_AL | ICSR_TACK)) {
> + iowrite8(0, ICIC(pd)); /* disable interrupts */
> + wakeup = 1;
> + goto do_wakeup;
> + }
> +
> + if (pd->pos = msg->len) {
> + i2c_op(pd, OP_RX_ONLY, 0);
> + wakeup = 1;
> + goto do_wakeup;
> + }
> +
> + if (pd->pos = -1) {
> + data = (msg->addr & 0x7f) << 1;
> + data |= (msg->flags & I2C_M_RD) ? 1 : 0;
> + } else
> + data = msg->buf[pd->pos];
> +
> + if ((pd->pos = -1) || !(msg->flags & I2C_M_RD)) {
> + if (msg->flags & I2C_M_RD)
> + i2c_op(pd, OP_TX_TO_RX, data);
> + else if (pd->pos = (msg->len - 1)) {
> + i2c_op(pd, OP_TX_STOP, data);
> + wakeup = 1;
> + } else
> + i2c_op(pd, OP_TX_ONLY, data);
> + } else {
> + if (pd->pos = (msg->len - 1))
> + data = i2c_op(pd, OP_RX_STOP, 0);
> + else
> + data = i2c_op(pd, OP_RX_ONLY, 0);
> +
> + msg->buf[pd->pos] = data;
> + }
> + pd->pos++;
> +
> + do_wakeup:
> + if (wakeup) {
> + pd->sr |= SW_DONE;
> + wake_up(&pd->wait);
> + }
> +
> + return IRQ_HANDLED;
> +}
> +
> +static int start_ch(struct sh_mobile_i2c_data *pd, struct i2c_msg *usr_msg)
> +{
> + /* Initialize channel registers */
> + iowrite8(ioread8(ICCR(pd)) & ~ICCR_ICE, ICCR(pd));
> +
> + /* Enable channel and configure rx ack */
> + iowrite8(ioread8(ICCR(pd)) | ICCR_ICE |
> + (pd->rx_ack_high ? ICCR_RACK : 0), ICCR(pd));
> +
> + /* Set the clock */
> + iowrite8(pd->iccl, ICCL(pd));
> + iowrite8(pd->icch, ICCH(pd));
> +
> + pd->msg = usr_msg;
> + pd->pos = -1;
> + pd->sr = 0;
> +
> + /* Enable all interrupts except wait */
> + iowrite8(ioread8(ICIC(pd)) | ICIC_ALE | ICIC_TACKE | ICIC_DTEE,
> + ICIC(pd));
> + return 0;
> +}
> +
> +static int sh_mobile_i2c_xfer(struct i2c_adapter *adapter,
> + struct i2c_msg *msgs,
> + int num)
> +{
> + struct sh_mobile_i2c_data *pd = i2c_get_adapdata(adapter);
> + struct i2c_msg *msg;
> + int err = 0;
> + u_int8_t val;
> + int i, k;
> +
> + activate_ch(pd);
> +
> + /* Process all messages */
> + for (i = 0; i < num; i++) {
> + msg = &msgs[i];
> +
> + err = start_ch(pd, msg);
> + if (err)
> + break;
> +
> + i2c_op(pd, OP_START, 0);
> +
> + /* The interrupt handler takes care of the rest... */
> + k = wait_event_timeout(pd->wait,
> + pd->sr & (ICSR_TACK | SW_DONE),
> + 5 * HZ);
> + if (!k)
> + dev_err(pd->dev, "Transfer request timed out\n");
> +
Recycling k like this isn't pleasant, but isn't really a show stopper
on getting this driver, however it isn't helpful when trying to work
out what is going on in this loop, so it might be better to try and
split the usage and give them seperate variables such as 'ret' and
'retry_count'.
> + k = 10;
> +again:
> + val = ioread8(ICSR(pd));
> +
> + dev_dbg(pd->dev, "val 0x%02x pd->sr 0x%02x!\n", val, pd->sr);
do you really need that exclamation mark?
> +
> + if ((val | pd->sr) & (ICSR_TACK | ICSR_AL)) {
> + err = -EIO;
> + break;
> + }
>
IIRC, the error for not receiving an ACK is -ECONNREFUSED when
transfering data and -EREMOTEIO when sending the device address,
and you should also check the message flags appropriately for
I2C_M_IGNORE_ACK.
> +
> + if (!(!(val & ICSR_BUSY) && (val & ICSR_SCLM) &&
> + (val & ICSR_SDAM))) {
> + msleep(1);
I assume this is the reason why you aren't trying to go to the next
message in the ISR. Out of interest, why is this here, adding a
comment would explain what is going on here.
> + if (k--)
> + goto again;
> +
> + err = -EIO;
> + dev_err(pd->dev, "Polling timed out\n");
> + break;
> + }
> + }
> +
> + deactivate_ch(pd);
> +
> + if (!err)
> + err = num;
> + return err;
> +}
> +
> +static u32 sh_mobile_i2c_func(struct i2c_adapter *adapter)
> +{
> + return I2C_FUNC_I2C | I2C_FUNC_SMBUS_EMUL;
> +}
> +
> +static struct i2c_algorithm sh_mobile_i2c_algorithm = {
> + .functionality = sh_mobile_i2c_func,
> + .master_xfer = sh_mobile_i2c_xfer,
> +};
> +
> +static void sh_mobile_i2c_setup_channel(struct platform_device *dev)
> +{
> + struct sh_mobile_i2c_data *pd = platform_get_drvdata(dev);
> + unsigned long peripheral_clk = clk_get_rate(pd->clk);
> + u_int32_t num;
> + u_int32_t denom;
> + u_int32_t tmp;
> +
> + spin_lock_init(&pd->lock);
> + init_waitqueue_head(&pd->wait);
> +
> + /* Calculate the value for iccl. From the data sheet:
> + * iccl = (p clock ÷ transfer rate) × (L ÷ (L + H))
> + * where L and H are the SCL low/high ratio (5/4 in this case).
> + * We also round off the result.
> + */
there are some weird encoding characters in this message, making it
difficult to work out what is going on here.
> + num = peripheral_clk * 5;
> + denom = pd->fast_mode ? (FAST_SPEED * 9) : (NORMAL_SPEED * 9);
> + tmp = num * 10 / denom;
> + if (tmp % 10 >= 5)
> + pd->iccl = (u_int8_t)((num/denom) + 1);
> + else
> + pd->iccl = (u_int8_t)(num/denom);
Is this a case of using the division macros that round up?
> +
> + /* Calculate the value for icch. From the data sheet:
> + icch = (p clock ÷ transfer rate) × (H ÷ (L + H)) */
> + num = peripheral_clk * 4;
> + tmp = num * 10 / denom;
> + if (tmp % 10 >= 5)
> + pd->icch = (u_int8_t)((num/denom) + 1);
> + else
> + pd->icch = (u_int8_t)(num/denom);
> +
> + deactivate_ch(pd);
> +}
> +
> +static int sh_mobile_i2c_hook_irqs(struct platform_device *dev, int hook)
> +{
> + struct resource *res;
> + int ret = -ENXIO;
> + int q, m;
> + int k = 0;
> + int n = 0;
> +
> + while ((res = platform_get_resource(dev, IORESOURCE_IRQ, k))) {
> + for (n = res->start; hook && n <= res->end; n++) {
> + if (request_irq(n, sh_mobile_i2c_isr, IRQF_DISABLED,
> + dev->dev.bus_id, dev))
> + goto rollback;
> + }
> + k++;
> + }
> +
> + if (hook)
> + return k > 0 ? 0 : -ENOENT;
> +
> + k--;
> + ret = 0;
> +
> + rollback:
> + for (q = k; k >= 0; k--) {
> + for (m = n; m >= res->start; m--)
> + free_irq(m, dev);
> +
> + res = platform_get_resource(dev, IORESOURCE_IRQ, k - 1);
> + m = res->end;
> + }
> +
> + return ret;
> +}
How many IRQs can one controller have?
> +
> +static int sh_mobile_i2c_probe(struct platform_device *dev)
> +{
> + struct sh_mobile_i2c_data *pd;
> + struct i2c_adapter *adap;
> + struct resource *res;
> + int size;
> + int ret;
> +
> + pd = kzalloc(sizeof(struct sh_mobile_i2c_data), GFP_KERNEL);
> + if (pd = NULL) {
> + dev_err(&dev->dev, "cannot allocate private data\n");
> + return -ENOMEM;
> + }
> +
> + pd->clk = clk_get(&dev->dev, "peripheral_clk");
> + if (IS_ERR(pd->clk)) {
> + dev_err(&dev->dev, "cannot get peripheral clock\n");
> + ret = PTR_ERR(pd->clk);
> + goto err;
> + }
I note that you do not call clk_enable() or clk_disable() for this
clock, is this some form of global always-on clock, or is an
assumption that some other driver is holding the clock open for you?
> +
> + ret = sh_mobile_i2c_hook_irqs(dev, 1);
> + if (ret) {
> + dev_err(&dev->dev, "cannot request IRQ\n");
> + goto err_clk;
> + }
> +
> + pd->dev = &dev->dev;
> + platform_set_drvdata(dev, pd);
> +
> + res = platform_get_resource(dev, IORESOURCE_MEM, 0);
> + if (res = NULL) {
> + dev_err(&dev->dev, "cannot find IO resource\n");
> + ret = -ENOENT;
> + goto err_irq;
> + }
> +
> + size = (res->end - res->start) + 1;
hmm, has no-one done as res_size() macro yet?
> +
> + pd->reg = ioremap(res->start, size);
> + if (pd->reg = NULL) {
> + dev_err(&dev->dev, "cannot map IO\n");
> + ret = -ENXIO;
> + goto err_irq;
> + }
> +
> + /* setup the private data */
> + adap = &pd->adap;
> + i2c_set_adapdata(adap, pd);
> +
> + adap->owner = THIS_MODULE;
> + adap->algo = &sh_mobile_i2c_algorithm;
> + adap->dev.parent = &dev->dev;
> + adap->retries = 5;
> + adap->nr = dev->id;
> +
> + strlcpy(adap->name, dev->name, sizeof(adap->name));
> +
> + sh_mobile_i2c_setup_channel(dev);
> +
> + ret = i2c_add_numbered_adapter(adap);
> + if (ret < 0)
> + goto err_all;
Adding a dev_err() statement here to say the adaptor attach failed
would be useful, especially as there are dev_err() statements on all
the other probe points.
> +
> + return 0;
> +
> + err_all:
> + iounmap(pd->reg);
> + err_irq:
> + sh_mobile_i2c_hook_irqs(dev, 0);
> + err_clk:
> + clk_put(pd->clk);
> + err:
> + kfree(pd);
> + return ret;
> +}
> +
> +static int sh_mobile_i2c_remove(struct platform_device *dev)
> +{
> + struct sh_mobile_i2c_data *pd = platform_get_drvdata(dev);
> +
> + i2c_del_adapter(&pd->adap);
> + iounmap(pd->reg);
> + sh_mobile_i2c_hook_irqs(dev, 0);
> + clk_put(pd->clk);
> + kfree(pd);
> + return 0;
> +}
> +
> +static struct platform_driver sh_mobile_i2c_driver = {
> + .driver = {
> + .name = "i2c-sh_mobile",
> + .owner = THIS_MODULE,
> + },
> + .probe = sh_mobile_i2c_probe,
> + .remove = sh_mobile_i2c_remove,
> +};
Possibly you should make the .remove function be __devexit and the
the .remove entry be an __devexit_p() macro.
No suspend and resume support? Not a show stopper, but would have
been nice to see.
> +static int __init sh_mobile_i2c_adap_init(void)
> +{
> + return platform_driver_register(&sh_mobile_i2c_driver);
> +}
> +
> +static void __exit sh_mobile_i2c_adap_exit(void)
> +{
> + platform_driver_unregister(&sh_mobile_i2c_driver);
> +}
> +
> +module_init(sh_mobile_i2c_adap_init);
> +module_exit(sh_mobile_i2c_adap_exit);
> +
> +MODULE_DESCRIPTION("SuperH Mobile I2C Bus Controller driver");
> +MODULE_AUTHOR("Magnus Damm");
> +MODULE_LICENSE("GPL");
IIRC, your header license says GPL v2, so this should be changed to
agree with that statement.
Overall, the code looks fine other than a few comments could be added
and possibly some function descriptions before their declerations so
that you can get an idea of what is going on in them.
--
Ben (ben@fluff.org, http://www.fluff.org/)
'a smiley only costs 4 bytes'
^ permalink raw reply [flat|nested] 14+ messages in thread
* Re: [PATCH] i2c: SuperH Mobile I2C Bus Controller V4
[not found] ` <20080329175732.GA9638-elnMNo+KYs3pIgCt6eIbzw@public.gmane.org>
@ 2008-03-29 19:57 ` Jean Delvare
0 siblings, 0 replies; 14+ messages in thread
From: Jean Delvare @ 2008-03-29 19:57 UTC (permalink / raw)
To: Ben Dooks
Cc: Paul Mundt, i2c-GZX6beZjE8VD60Wz+7aTrA,
linux-sh-u79uwXL29TY76Z2rM5mHXA
Hi Ben, Magnus, Paul,
This is Ben's first review as the new i2c subsystem co-maintainer. I
will just make a couple of comments on top of that to clarify a few
things. Thanks _a lot_ Ben for stepping in and doing that work :)
On Sat, 29 Mar 2008 17:57:32 +0000, Ben Dooks wrote:
> On Fri, Mar 28, 2008 at 06:35:30PM +0900, Magnus Damm wrote:
> > This is V4 of the SuperH Mobile I2C Controller Driver. A simple Master
> > only driver for the I2C block included in processors such as sh7343,
> > sh7722 and sh7723. Tested on a sh7722 MigoR using a rs5c732b rtc.
> >
> > Changes since V3:
> > - Added SUPERH Kconfig dependency
> > - Use spin_lock_irqsave() and spin_unlock_irqrestore()
> > - Use IRQF_DISABLED
> > - Checkpatch fixes
> > Changes since V2:
> > - dev_xxx() use
> > - Kill off superfluous ioarea resource
> > - Add SMBus emulation
> > Changes since V1:
> > - Use clk_get()/clk_put()/clk_get_rate() to get peripheral clock rate.
> > - Use pdev->dev.bus_id instead of dev->name
>
> Do we really need these in the change message, we tend to assume that
> the changes before submission aren't really relevant. I would advise
> on removing these.
In fact the right place for these (useful) temporary changelog entries
is before the diffstat, right below, so that the reviewers can see them,
but they don't go in git.
> > Verified with the rtc-rs5c372 SMBus conversion patches currently in -mm.
> >
> > Signed-off-by: Magnus Damm <damm@igel.co.jp>
> > Signed-off-by: Paul Mundt <lethal@linux-sh.org>
> > ---
> >
> > drivers/i2c/busses/Kconfig | 10
> > drivers/i2c/busses/Makefile | 1
> > drivers/i2c/busses/i2c-sh_mobile.c | 498 ++++++++++++++++++++++++++++++++++++
> > 3 files changed, 509 insertions(+)
> >
> > --- 0001/drivers/i2c/busses/Kconfig
> > +++ work/drivers/i2c/busses/Kconfig 2008-03-28 16:45:42.000000000 +0900
> > (...)
> (...)
> IIRC, the error for not receiving an ACK is -ECONNREFUSED when
> transfering data and -EREMOTEIO when sending the device address,
> and you should also check the message flags appropriately for
> I2C_M_IGNORE_ACK.
Actually not. I2C_M_IGNORE_ACK's implementation is purely optional, and
as this is a violation of the I2C standard, the less implemented it
gets, the happier I am.
> > (...)
> > + /* Calculate the value for iccl. From the data sheet:
> > + * iccl = (p clock ÷ transfer rate) × (L ÷ (L + H))
> > + * where L and H are the SCL low/high ratio (5/4 in this case).
> > + * We also round off the result.
> > + */
>
> there are some weird encoding characters in this message, making it
> difficult to work out what is going on here.
I totally second this. Please stick to ASCII as much as possible, it's
way easier for everyone.
Thanks,
--
Jean Delvare
^ permalink raw reply [flat|nested] 14+ messages in thread
* Re: [i2c] [PATCH] i2c: SuperH Mobile I2C Bus Controller V4
2008-03-29 17:57 ` [i2c] " Ben Dooks
[not found] ` <20080329175732.GA9638-elnMNo+KYs3pIgCt6eIbzw@public.gmane.org>
@ 2008-03-31 22:36 ` Andrew Morton
2008-04-02 1:20 ` Magnus Damm
2 siblings, 0 replies; 14+ messages in thread
From: Andrew Morton @ 2008-03-31 22:36 UTC (permalink / raw)
To: Ben Dooks; +Cc: magnus.damm, i2c, lethal, linux-sh
On Sat, 29 Mar 2008 17:57:32 +0000
Ben Dooks <ben-linux@fluff.org> wrote:
> On Fri, Mar 28, 2008 at 06:35:30PM +0900, Magnus Damm wrote:
> > This is V4 of the SuperH Mobile I2C Controller Driver. A simple Master
> > only driver for the I2C block included in processors such as sh7343,
> > sh7722 and sh7723. Tested on a sh7722 MigoR using a rs5c732b rtc.
> >
> > Changes since V3:
> > - Added SUPERH Kconfig dependency
> > - Use spin_lock_irqsave() and spin_unlock_irqrestore()
> > - Use IRQF_DISABLED
> > - Checkpatch fixes
> > Changes since V2:
> > - dev_xxx() use
> > - Kill off superfluous ioarea resource
> > - Add SMBus emulation
> > Changes since V1:
> > - Use clk_get()/clk_put()/clk_get_rate() to get peripheral clock rate.
> > - Use pdev->dev.bus_id instead of dev->name
>
> Do we really need these in the change message, we tend to assume that
> the changes before submission aren't really relevant. I would advise
> on removing these.
They are valuable for people who have reviewed earlier versions of the
patchset so yes, please do maintain them.
I will snip them out when I merge them however - this short-term
inter-version information isn't appropriate to the final git commit.
^ permalink raw reply [flat|nested] 14+ messages in thread
* Re: [i2c] [PATCH] i2c: SuperH Mobile I2C Bus Controller V4
2008-03-29 17:57 ` [i2c] " Ben Dooks
[not found] ` <20080329175732.GA9638-elnMNo+KYs3pIgCt6eIbzw@public.gmane.org>
2008-03-31 22:36 ` [i2c] " Andrew Morton
@ 2008-04-02 1:20 ` Magnus Damm
2 siblings, 0 replies; 14+ messages in thread
From: Magnus Damm @ 2008-04-02 1:20 UTC (permalink / raw)
To: Ben Dooks; +Cc: i2c, akpm, lethal, linux-sh
Hi Ben,
On Sun, Mar 30, 2008 at 2:57 AM, Ben Dooks <ben-linux@fluff.org> wrote:
> On Fri, Mar 28, 2008 at 06:35:30PM +0900, Magnus Damm wrote:
> > This is V4 of the SuperH Mobile I2C Controller Driver. A simple Master
> > only driver for the I2C block included in processors such as sh7343,
> > sh7722 and sh7723. Tested on a sh7722 MigoR using a rs5c732b rtc.
> How many IRQs can one controller have?
Four IRQs per channel for this particular version of I2C controller.
> I note that you do not call clk_enable() or clk_disable() for this
> clock, is this some form of global always-on clock, or is an
> assumption that some other driver is holding the clock open for you?
Nice catch. I'll add code that does clk_enable() and clk_disable().
> > + size = (res->end - res->start) + 1;
>
> hmm, has no-one done as res_size() macro yet?
It's been on my TODO list forever. =)
> > +MODULE_LICENSE("GPL");
>
> IIRC, your header license says GPL v2, so this should be changed to
> agree with that statement.
Uhm, right. Thanks for pointing that out.
> Overall, the code looks fine other than a few comments could be added
> and possibly some function descriptions before their declerations so
> that you can get an idea of what is going on in them.
I'll fix up things and repost. Thanks for your review!
/ magnus
^ permalink raw reply [flat|nested] 14+ messages in thread
* [PATCH] i2c: SuperH Mobile I2C Bus Controller V5
2008-03-20 14:18 [PATCH] i2c: SuperH Mobile I2C Bus Controller Magnus Damm
` (3 preceding siblings ...)
2008-03-28 9:35 ` [PATCH] i2c: SuperH Mobile I2C Bus Controller V4 Magnus Damm
@ 2008-04-02 2:59 ` Magnus Damm
2008-04-05 8:50 ` Jean Delvare
4 siblings, 1 reply; 14+ messages in thread
From: Magnus Damm @ 2008-04-02 2:59 UTC (permalink / raw)
To: i2c; +Cc: linux-sh, lethal, ben-linux, khali, Magnus Damm, akpm
This is V5 of the SuperH Mobile I2C Controller Driver. A simple Master
only driver for the I2C block included in processors such as sh7343,
sh7722 and sh7723. Tested on a sh7722 MigoR using a rs5c732b rtc.
Signed-off-by: Magnus Damm <damm@igel.co.jp>
Signed-off-by: Paul Mundt <lethal@linux-sh.org>
---
Changes since V4:
- Use sh_mobile_i2c_op enum
- Remove fast_mode and rx_ack_high code
- Use variable name retry_count instead of k
- Encoding fixes
- Use clk_enable() and clk_disable()
- Remove deactivate_ch() call from setup_channel()
- Use dev_err() if i2c_add_numbered_adapter() fails
- MODULE_LICENSE("GPL v2")
- Comment fix, cleaned up header
Changes since V3:
- Added SUPERH Kconfig dependency
- Use spin_lock_irqsave() and spin_unlock_irqrestore()
- Use IRQF_DISABLED
- Checkpatch fixes
Changes since V2:
- dev_xxx() use
- Kill off superfluous ioarea resource
- Add SMBus emulation
Changes since V1:
- Use clk_get()/clk_put()/clk_get_rate() to get peripheral clock rate.
- Use pdev->dev.bus_id instead of dev->name
Verified with the rtc-rs5c372 SMBus conversion patches currently in -mm.
drivers/i2c/busses/Kconfig | 10
drivers/i2c/busses/Makefile | 1
drivers/i2c/busses/i2c-sh_mobile.c | 500 ++++++++++++++++++++++++++++++++++++
3 files changed, 511 insertions(+)
--- 0001/drivers/i2c/busses/Kconfig
+++ work/drivers/i2c/busses/Kconfig 2008-04-01 20:16:55.000000000 +0900
@@ -672,4 +672,14 @@ config I2C_PMCMSP
This driver can also be built as module. If so, the module
will be called i2c-pmcmsp.
+config I2C_SH_MOBILE
+ tristate "SuperH Mobile I2C Controller"
+ depends on SUPERH
+ help
+ If you say yes to this option, support will be included for the
+ built-in I2C interface on the Renesas SH-Mobile processor.
+
+ This driver can also be built as a module. If so, the module
+ will be called i2c-sh_mobile.
+
endmenu
--- 0001/drivers/i2c/busses/Makefile
+++ work/drivers/i2c/busses/Makefile 2008-04-01 20:16:55.000000000 +0900
@@ -52,6 +52,7 @@ obj-$(CONFIG_I2C_VIAPRO) += i2c-viapro.o
obj-$(CONFIG_I2C_VOODOO3) += i2c-voodoo3.o
obj-$(CONFIG_SCx200_ACB) += scx200_acb.o
obj-$(CONFIG_SCx200_I2C) += scx200_i2c.o
+obj-$(CONFIG_I2C_SH_MOBILE) += i2c-sh_mobile.o
ifeq ($(CONFIG_I2C_DEBUG_BUS),y)
EXTRA_CFLAGS += -DDEBUG
--- /dev/null
+++ work/drivers/i2c/busses/i2c-sh_mobile.c 2008-04-02 10:28:42.000000000 +0900
@@ -0,0 +1,500 @@
+/*
+ * SuperH Mobile I2C Controller
+ *
+ * Copyright (C) 2008 Magnus Damm
+ *
+ * Portions of the code based on out-of-tree driver i2c-sh7343.c
+ * Copyright (c) 2006 Carlos Munoz <carlos@kenati.com>
+ *
+ * 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
+ *
+ * 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
+ */
+
+#include <linux/kernel.h>
+#include <linux/module.h>
+#include <linux/init.h>
+#include <linux/delay.h>
+#include <linux/platform_device.h>
+#include <linux/interrupt.h>
+#include <linux/i2c.h>
+#include <linux/err.h>
+#include <linux/clk.h>
+#include <linux/io.h>
+
+enum sh_mobile_i2c_op {
+ OP_START = 0,
+ OP_TX_ONLY,
+ OP_TX_STOP,
+ OP_TX_TO_RX,
+ OP_RX_ONLY,
+ OP_RX_STOP,
+};
+
+struct sh_mobile_i2c_data {
+ struct device *dev;
+ void __iomem *reg;
+ struct i2c_adapter adap;
+
+ struct clk *clk;
+ u_int8_t iccl;
+ u_int8_t icch;
+
+ spinlock_t lock;
+ wait_queue_head_t wait;
+ struct i2c_msg *msg;
+ int pos;
+ int sr;
+};
+
+#define NORMAL_SPEED 100000 /* FAST_SPEED 400000 */
+
+/* Register offsets */
+#define ICDR(pd) (pd->reg + 0x00)
+#define ICCR(pd) (pd->reg + 0x04)
+#define ICSR(pd) (pd->reg + 0x08)
+#define ICIC(pd) (pd->reg + 0x0c)
+#define ICCL(pd) (pd->reg + 0x10)
+#define ICCH(pd) (pd->reg + 0x14)
+
+/* Register bits */
+#define ICCR_ICE 0x80
+#define ICCR_RACK 0x40
+#define ICCR_TRS 0x10
+#define ICCR_BBSY 0x04
+#define ICCR_SCP 0x01
+
+#define ICSR_SCLM 0x80
+#define ICSR_SDAM 0x40
+#define SW_DONE 0x20
+#define ICSR_BUSY 0x10
+#define ICSR_AL 0x08
+#define ICSR_TACK 0x04
+#define ICSR_WAIT 0x02
+#define ICSR_DTE 0x01
+
+#define ICIC_ALE 0x08
+#define ICIC_TACKE 0x04
+#define ICIC_WAITE 0x02
+#define ICIC_DTEE 0x01
+
+static void activate_ch(struct sh_mobile_i2c_data *pd)
+{
+ /* Make sure the clock is enabled */
+ clk_enable(pd->clk);
+
+ /* Enable channel and configure rx ack */
+ iowrite8(ioread8(ICCR(pd)) | ICCR_ICE, ICCR(pd));
+
+ /* Mask all interrupts */
+ iowrite8(0, ICIC(pd));
+
+ /* Set the clock */
+ iowrite8(pd->iccl, ICCL(pd));
+ iowrite8(pd->icch, ICCH(pd));
+}
+
+static void deactivate_ch(struct sh_mobile_i2c_data *pd)
+{
+ /* Clear/disable interrupts */
+ iowrite8(0, ICSR(pd));
+ iowrite8(0, ICIC(pd));
+
+ /* Disable channel */
+ iowrite8(ioread8(ICCR(pd)) & ~ICCR_ICE, ICCR(pd));
+
+ /* Disable clock */
+ clk_disable(pd->clk);
+}
+
+static unsigned char i2c_op(struct sh_mobile_i2c_data *pd,
+ enum sh_mobile_i2c_op op, unsigned char data)
+{
+ unsigned char ret = 0;
+ unsigned long flags;
+
+ dev_dbg(pd->dev, "op %d, data in 0x%02x\n", op, data);
+
+ spin_lock_irqsave(&pd->lock, flags);
+
+ switch (op) {
+ case OP_START:
+ iowrite8(0x94, ICCR(pd));
+ break;
+ case OP_TX_ONLY:
+ iowrite8(data, ICDR(pd));
+ break;
+ case OP_TX_STOP:
+ iowrite8(data, ICDR(pd));
+ iowrite8(0x90, ICCR(pd));
+ iowrite8(ICIC_ALE | ICIC_TACKE, ICIC(pd));
+ break;
+ case OP_TX_TO_RX:
+ iowrite8(data, ICDR(pd));
+ iowrite8(0x81, ICCR(pd));
+ break;
+ case OP_RX_ONLY:
+ ret = ioread8(ICDR(pd));
+ break;
+ case OP_RX_STOP:
+ ret = ioread8(ICDR(pd));
+ iowrite8(0xc0, ICCR(pd));
+ break;
+ }
+
+ spin_unlock_irqrestore(&pd->lock, flags);
+
+ dev_dbg(pd->dev, "op %d, data out 0x%02x\n", op, ret);
+ return ret;
+}
+
+static irqreturn_t sh_mobile_i2c_isr(int irq, void *dev_id)
+{
+ struct platform_device *dev = dev_id;
+ struct sh_mobile_i2c_data *pd = platform_get_drvdata(dev);
+ struct i2c_msg *msg = pd->msg;
+ unsigned char data, sr;
+ int wakeup = 0;
+
+ sr = ioread8(ICSR(pd));
+ pd->sr |= sr;
+
+ dev_dbg(pd->dev, "i2c_isr 0x%02x 0x%02x %s %d %d!\n", sr, pd->sr,
+ (msg->flags & I2C_M_RD) ? "read" : "write",
+ pd->pos, msg->len);
+
+ if (sr & (ICSR_AL | ICSR_TACK)) {
+ iowrite8(0, ICIC(pd)); /* disable interrupts */
+ wakeup = 1;
+ goto do_wakeup;
+ }
+
+ if (pd->pos = msg->len) {
+ i2c_op(pd, OP_RX_ONLY, 0);
+ wakeup = 1;
+ goto do_wakeup;
+ }
+
+ if (pd->pos = -1) {
+ data = (msg->addr & 0x7f) << 1;
+ data |= (msg->flags & I2C_M_RD) ? 1 : 0;
+ } else
+ data = msg->buf[pd->pos];
+
+ if ((pd->pos = -1) || !(msg->flags & I2C_M_RD)) {
+ if (msg->flags & I2C_M_RD)
+ i2c_op(pd, OP_TX_TO_RX, data);
+ else if (pd->pos = (msg->len - 1)) {
+ i2c_op(pd, OP_TX_STOP, data);
+ wakeup = 1;
+ } else
+ i2c_op(pd, OP_TX_ONLY, data);
+ } else {
+ if (pd->pos = (msg->len - 1))
+ data = i2c_op(pd, OP_RX_STOP, 0);
+ else
+ data = i2c_op(pd, OP_RX_ONLY, 0);
+
+ msg->buf[pd->pos] = data;
+ }
+ pd->pos++;
+
+ do_wakeup:
+ if (wakeup) {
+ pd->sr |= SW_DONE;
+ wake_up(&pd->wait);
+ }
+
+ return IRQ_HANDLED;
+}
+
+static int start_ch(struct sh_mobile_i2c_data *pd, struct i2c_msg *usr_msg)
+{
+ /* Initialize channel registers */
+ iowrite8(ioread8(ICCR(pd)) & ~ICCR_ICE, ICCR(pd));
+
+ /* Enable channel and configure rx ack */
+ iowrite8(ioread8(ICCR(pd)) | ICCR_ICE, ICCR(pd));
+
+ /* Set the clock */
+ iowrite8(pd->iccl, ICCL(pd));
+ iowrite8(pd->icch, ICCH(pd));
+
+ pd->msg = usr_msg;
+ pd->pos = -1;
+ pd->sr = 0;
+
+ /* Enable all interrupts except wait */
+ iowrite8(ioread8(ICIC(pd)) | ICIC_ALE | ICIC_TACKE | ICIC_DTEE,
+ ICIC(pd));
+ return 0;
+}
+
+static int sh_mobile_i2c_xfer(struct i2c_adapter *adapter,
+ struct i2c_msg *msgs,
+ int num)
+{
+ struct sh_mobile_i2c_data *pd = i2c_get_adapdata(adapter);
+ struct i2c_msg *msg;
+ int err = 0;
+ u_int8_t val;
+ int i, k, retry_count;
+
+ activate_ch(pd);
+
+ /* Process all messages */
+ for (i = 0; i < num; i++) {
+ msg = &msgs[i];
+
+ err = start_ch(pd, msg);
+ if (err)
+ break;
+
+ i2c_op(pd, OP_START, 0);
+
+ /* The interrupt handler takes care of the rest... */
+ k = wait_event_timeout(pd->wait,
+ pd->sr & (ICSR_TACK | SW_DONE),
+ 5 * HZ);
+ if (!k)
+ dev_err(pd->dev, "Transfer request timed out\n");
+
+ retry_count = 10;
+again:
+ val = ioread8(ICSR(pd));
+
+ dev_dbg(pd->dev, "val 0x%02x pd->sr 0x%02x\n", val, pd->sr);
+
+ if ((val | pd->sr) & (ICSR_TACK | ICSR_AL)) {
+ err = -EIO;
+ break;
+ }
+
+ /* the interrupt handler may wake us up before the
+ * transfer is finished, so poll the hardware
+ * until we're done.
+ */
+
+ if (!(!(val & ICSR_BUSY) && (val & ICSR_SCLM) &&
+ (val & ICSR_SDAM))) {
+ msleep(1);
+ if (retry_count--)
+ goto again;
+
+ err = -EIO;
+ dev_err(pd->dev, "Polling timed out\n");
+ break;
+ }
+ }
+
+ deactivate_ch(pd);
+
+ if (!err)
+ err = num;
+ return err;
+}
+
+static u32 sh_mobile_i2c_func(struct i2c_adapter *adapter)
+{
+ return I2C_FUNC_I2C | I2C_FUNC_SMBUS_EMUL;
+}
+
+static struct i2c_algorithm sh_mobile_i2c_algorithm = {
+ .functionality = sh_mobile_i2c_func,
+ .master_xfer = sh_mobile_i2c_xfer,
+};
+
+static void sh_mobile_i2c_setup_channel(struct platform_device *dev)
+{
+ struct sh_mobile_i2c_data *pd = platform_get_drvdata(dev);
+ unsigned long peripheral_clk = clk_get_rate(pd->clk);
+ u_int32_t num;
+ u_int32_t denom;
+ u_int32_t tmp;
+
+ spin_lock_init(&pd->lock);
+ init_waitqueue_head(&pd->wait);
+
+ /* Calculate the value for iccl. From the data sheet:
+ * iccl = (p clock / transfer rate) * (L / (L + H))
+ * where L and H are the SCL low/high ratio (5/4 in this case).
+ * We also round off the result.
+ */
+ num = peripheral_clk * 5;
+ denom = NORMAL_SPEED * 9;
+ tmp = num * 10 / denom;
+ if (tmp % 10 >= 5)
+ pd->iccl = (u_int8_t)((num/denom) + 1);
+ else
+ pd->iccl = (u_int8_t)(num/denom);
+
+ /* Calculate the value for icch. From the data sheet:
+ icch = (p clock / transfer rate) * (H / (L + H)) */
+ num = peripheral_clk * 4;
+ tmp = num * 10 / denom;
+ if (tmp % 10 >= 5)
+ pd->icch = (u_int8_t)((num/denom) + 1);
+ else
+ pd->icch = (u_int8_t)(num/denom);
+}
+
+static int sh_mobile_i2c_hook_irqs(struct platform_device *dev, int hook)
+{
+ struct resource *res;
+ int ret = -ENXIO;
+ int q, m;
+ int k = 0;
+ int n = 0;
+
+ while ((res = platform_get_resource(dev, IORESOURCE_IRQ, k))) {
+ for (n = res->start; hook && n <= res->end; n++) {
+ if (request_irq(n, sh_mobile_i2c_isr, IRQF_DISABLED,
+ dev->dev.bus_id, dev))
+ goto rollback;
+ }
+ k++;
+ }
+
+ if (hook)
+ return k > 0 ? 0 : -ENOENT;
+
+ k--;
+ ret = 0;
+
+ rollback:
+ for (q = k; k >= 0; k--) {
+ for (m = n; m >= res->start; m--)
+ free_irq(m, dev);
+
+ res = platform_get_resource(dev, IORESOURCE_IRQ, k - 1);
+ m = res->end;
+ }
+
+ return ret;
+}
+
+static int sh_mobile_i2c_probe(struct platform_device *dev)
+{
+ struct sh_mobile_i2c_data *pd;
+ struct i2c_adapter *adap;
+ struct resource *res;
+ int size;
+ int ret;
+
+ pd = kzalloc(sizeof(struct sh_mobile_i2c_data), GFP_KERNEL);
+ if (pd = NULL) {
+ dev_err(&dev->dev, "cannot allocate private data\n");
+ return -ENOMEM;
+ }
+
+ pd->clk = clk_get(&dev->dev, "peripheral_clk");
+ if (IS_ERR(pd->clk)) {
+ dev_err(&dev->dev, "cannot get peripheral clock\n");
+ ret = PTR_ERR(pd->clk);
+ goto err;
+ }
+
+ ret = sh_mobile_i2c_hook_irqs(dev, 1);
+ if (ret) {
+ dev_err(&dev->dev, "cannot request IRQ\n");
+ goto err_clk;
+ }
+
+ pd->dev = &dev->dev;
+ platform_set_drvdata(dev, pd);
+
+ res = platform_get_resource(dev, IORESOURCE_MEM, 0);
+ if (res = NULL) {
+ dev_err(&dev->dev, "cannot find IO resource\n");
+ ret = -ENOENT;
+ goto err_irq;
+ }
+
+ size = (res->end - res->start) + 1;
+
+ pd->reg = ioremap(res->start, size);
+ if (pd->reg = NULL) {
+ dev_err(&dev->dev, "cannot map IO\n");
+ ret = -ENXIO;
+ goto err_irq;
+ }
+
+ /* setup the private data */
+ adap = &pd->adap;
+ i2c_set_adapdata(adap, pd);
+
+ adap->owner = THIS_MODULE;
+ adap->algo = &sh_mobile_i2c_algorithm;
+ adap->dev.parent = &dev->dev;
+ adap->retries = 5;
+ adap->nr = dev->id;
+
+ strlcpy(adap->name, dev->name, sizeof(adap->name));
+
+ sh_mobile_i2c_setup_channel(dev);
+
+ ret = i2c_add_numbered_adapter(adap);
+ if (ret < 0) {
+ dev_err(&dev->dev, "cannot add numbered adapter\n");
+ goto err_all;
+ }
+
+ return 0;
+
+ err_all:
+ iounmap(pd->reg);
+ err_irq:
+ sh_mobile_i2c_hook_irqs(dev, 0);
+ err_clk:
+ clk_put(pd->clk);
+ err:
+ kfree(pd);
+ return ret;
+}
+
+static int sh_mobile_i2c_remove(struct platform_device *dev)
+{
+ struct sh_mobile_i2c_data *pd = platform_get_drvdata(dev);
+
+ i2c_del_adapter(&pd->adap);
+ iounmap(pd->reg);
+ sh_mobile_i2c_hook_irqs(dev, 0);
+ clk_put(pd->clk);
+ kfree(pd);
+ return 0;
+}
+
+static struct platform_driver sh_mobile_i2c_driver = {
+ .driver = {
+ .name = "i2c-sh_mobile",
+ .owner = THIS_MODULE,
+ },
+ .probe = sh_mobile_i2c_probe,
+ .remove = sh_mobile_i2c_remove,
+};
+
+static int __init sh_mobile_i2c_adap_init(void)
+{
+ return platform_driver_register(&sh_mobile_i2c_driver);
+}
+
+static void __exit sh_mobile_i2c_adap_exit(void)
+{
+ platform_driver_unregister(&sh_mobile_i2c_driver);
+}
+
+module_init(sh_mobile_i2c_adap_init);
+module_exit(sh_mobile_i2c_adap_exit);
+
+MODULE_DESCRIPTION("SuperH Mobile I2C Bus Controller driver");
+MODULE_AUTHOR("Magnus Damm");
+MODULE_LICENSE("GPL v2");
^ permalink raw reply [flat|nested] 14+ messages in thread
* Re: [PATCH] i2c: SuperH Mobile I2C Bus Controller V5
2008-04-02 2:59 ` [PATCH] i2c: SuperH Mobile I2C Bus Controller V5 Magnus Damm
@ 2008-04-05 8:50 ` Jean Delvare
0 siblings, 0 replies; 14+ messages in thread
From: Jean Delvare @ 2008-04-05 8:50 UTC (permalink / raw)
To: Magnus Damm; +Cc: i2c, linux-sh, Paul Mundt, Ben Dooks, Andrew Morton
Hi Magnus,
On Wed, 02 Apr 2008 11:59:50 +0900, Magnus Damm wrote:
> This is V5 of the SuperH Mobile I2C Controller Driver. A simple Master
> only driver for the I2C block included in processors such as sh7343,
> sh7722 and sh7723. Tested on a sh7722 MigoR using a rs5c732b rtc.
>
> Signed-off-by: Magnus Damm <damm@igel.co.jp>
> Signed-off-by: Paul Mundt <lethal@linux-sh.org>
> ---
>
> Changes since V4:
> - Use sh_mobile_i2c_op enum
> - Remove fast_mode and rx_ack_high code
> - Use variable name retry_count instead of k
> - Encoding fixes
> - Use clk_enable() and clk_disable()
> - Remove deactivate_ch() call from setup_channel()
> - Use dev_err() if i2c_add_numbered_adapter() fails
> - MODULE_LICENSE("GPL v2")
> - Comment fix, cleaned up header
> Changes since V3:
> - Added SUPERH Kconfig dependency
> - Use spin_lock_irqsave() and spin_unlock_irqrestore()
> - Use IRQF_DISABLED
> - Checkpatch fixes
> Changes since V2:
> - dev_xxx() use
> - Kill off superfluous ioarea resource
> - Add SMBus emulation
> Changes since V1:
> - Use clk_get()/clk_put()/clk_get_rate() to get peripheral clock rate.
> - Use pdev->dev.bus_id instead of dev->name
>
> Verified with the rtc-rs5c372 SMBus conversion patches currently in -mm.
>
> drivers/i2c/busses/Kconfig | 10
> drivers/i2c/busses/Makefile | 1
> drivers/i2c/busses/i2c-sh_mobile.c | 500 ++++++++++++++++++++++++++++++++++++
> 3 files changed, 511 insertions(+)
Patch added to my i2c tree and queued up for merge in 2.6.26-rc1.
Thanks to everyone who reviewed the previous versions of the driver.
--
Jean Delvare
^ permalink raw reply [flat|nested] 14+ messages in thread
end of thread, other threads:[~2008-04-05 8:50 UTC | newest]
Thread overview: 14+ messages (download: mbox.gz follow: Atom feed
-- links below jump to the message on this page --
2008-03-20 14:18 [PATCH] i2c: SuperH Mobile I2C Bus Controller Magnus Damm
2008-03-21 6:23 ` Paul Mundt
2008-03-21 8:14 ` Magnus Damm
2008-03-21 12:07 ` [PATCH] i2c: SuperH Mobile I2C Bus Controller V2 Magnus Damm
2008-03-25 10:55 ` [PATCH] i2c: SuperH Mobile I2C Bus Controller V3 Paul Mundt
2008-03-26 19:56 ` Andrew Morton
2008-03-27 22:07 ` Andrew Morton
2008-03-28 9:35 ` [PATCH] i2c: SuperH Mobile I2C Bus Controller V4 Magnus Damm
2008-03-29 17:57 ` [i2c] " Ben Dooks
[not found] ` <20080329175732.GA9638-elnMNo+KYs3pIgCt6eIbzw@public.gmane.org>
2008-03-29 19:57 ` Jean Delvare
2008-03-31 22:36 ` [i2c] " Andrew Morton
2008-04-02 1:20 ` Magnus Damm
2008-04-02 2:59 ` [PATCH] i2c: SuperH Mobile I2C Bus Controller V5 Magnus Damm
2008-04-05 8:50 ` Jean Delvare
This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox