[RFC PATCH 2/5] i2c-dmec: add i2c bus support for dmec

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

 



This is support for the i2c bus functionality of the Data Modul embedded
controllers.

Signed-off-by: Zahari Doychev <zahari.doychev@xxxxxxxxx>
---
 drivers/staging/dmec/Kconfig    |  10 +-
 drivers/staging/dmec/Makefile   |   1 +-
 drivers/staging/dmec/dmec.h     |   9 +-
 drivers/staging/dmec/i2c-dmec.c | 524 +++++++++++++++++++++++++++++++++-
 4 files changed, 544 insertions(+), 0 deletions(-)
 create mode 100644 drivers/staging/dmec/i2c-dmec.c

diff --git a/drivers/staging/dmec/Kconfig b/drivers/staging/dmec/Kconfig
index 3641907..0067b0b 100644
--- a/drivers/staging/dmec/Kconfig
+++ b/drivers/staging/dmec/Kconfig
@@ -7,3 +7,13 @@ config MFD_DMEC
 
 	    To compile this driver as a module, say M here: the module will be
             called dmec
+
+config I2C_DMEC
+	tristate "Data Modul I2C"
+	depends on MFD_DMEC && I2C
+	help
+	  Say Y here to enable support for a i2c bus on the Data Modul
+	  embedded controller.
+
+	  To compile this driver as a module, say M here: the module will be
+          called i2c-dmec
diff --git a/drivers/staging/dmec/Makefile b/drivers/staging/dmec/Makefile
index 859163b..c51a37e 100644
--- a/drivers/staging/dmec/Makefile
+++ b/drivers/staging/dmec/Makefile
@@ -1 +1,2 @@
 obj-$(CONFIG_MFD_DMEC)		+= dmec-core.o
+obj-$(CONFIG_I2C_DMEC)		+= i2c-dmec.o
diff --git a/drivers/staging/dmec/dmec.h b/drivers/staging/dmec/dmec.h
index 4d8712d..178937d 100644
--- a/drivers/staging/dmec/dmec.h
+++ b/drivers/staging/dmec/dmec.h
@@ -1,6 +1,15 @@
 #ifndef _LINUX_MFD_DMEC_H
 #define _LINUX_MFD_DMEC_H
 
+struct dmec_i2c_platform_data {
+	u32 reg_shift; /* register offset shift value */
+	u32 reg_io_width; /* register io read/write width */
+	u32 clock_khz; /* input clock in kHz */
+	bool big_endian; /* registers are big endian */
+	u8 num_devices; /* number of devices in the devices list */
+	struct i2c_board_info const *devices; /* devices connected to the bus */
+};
+
 struct regmap *dmec_get_regmap(struct device *dev);
 
 #endif
diff --git a/drivers/staging/dmec/i2c-dmec.c b/drivers/staging/dmec/i2c-dmec.c
new file mode 100644
index 0000000..456e61e
--- /dev/null
+++ b/drivers/staging/dmec/i2c-dmec.c
@@ -0,0 +1,524 @@
+/*
+ * i2c-dmec.c: I2C bus driver for OpenCores I2C controller
+ * (http://www.opencores.org/projects.cgi/web/i2c/overview).
+ *
+ * Peter Korsgaard <jacmet@xxxxxxxxxx>
+ *
+ * Support for the GRLIB port of the controller by
+ * Andreas Larsson <andreas@xxxxxxxxxxx>
+ *
+ * This file is licensed under the terms of the GNU General Public License
+ * version 2.  This program is licensed "as is" without any warranty of any
+ * kind, whether express or implied.
+ */
+#include <linux/clk.h>
+#include <linux/err.h>
+#include <linux/kernel.h>
+#include <linux/module.h>
+#include <linux/errno.h>
+#include <linux/platform_device.h>
+#include <linux/regmap.h>
+#include <linux/wait.h>
+#include <linux/slab.h>
+#include <linux/io.h>
+#include <linux/log2.h>
+#include <linux/i2c.h>
+#include <linux/i2c-mux.h>
+#include <linux/interrupt.h>
+#include "dmec.h"
+
+/* registers */
+#define DMECI2C_PRELOW		0
+#define DMECI2C_PREHIGH		1
+#define DMECI2C_CONTROL		2
+#define DMECI2C_DATA		3
+#define DMECI2C_CMD		4 /* write only */
+#define DMECI2C_STATUS		4 /* read only, same address as DMECI2C_CMD */
+#define DMECI2C_VER		5
+#define DMECI2C_MUX		6 /* i2c demultiplezer */
+
+#define DMECI2C_CTRL_EN		BIT(7)
+#define DMECI2C_CTRL_IEN	BIT(6)
+#define DMECI2C_CTRL_VSSCL	BIT(5)
+#define DMECI2C_CTRL_MABCLR	BIT(4)
+#define DMECI2C_CTRL_MMDIS	BIT(3)
+#define DMECI2C_CTRL_VSSDA	BIT(2)
+
+#define DMECI2C_CMD_START	0x91
+#define DMECI2C_CMD_STOP	0x41
+#define DMECI2C_CMD_READ	0x21
+#define DMECI2C_CMD_WRITE	0x11
+#define DMECI2C_CMD_READ_ACK	0x21
+#define DMECI2C_CMD_READ_NACK	0x29
+#define DMECI2C_CMD_IACK	0x01
+
+#define DMECI2C_STAT_IF		0x01
+#define DMECI2C_STAT_TIP	0x02
+#define DMECI2C_STAT_ARBLOST	0x20
+#define DMECI2C_STAT_BUSY	0x40
+#define DMECI2C_STAT_NACK	0x80
+
+#define STATE_DONE		0
+#define STATE_START		1
+#define STATE_WRITE		2
+#define STATE_READ		3
+#define STATE_ERROR		4
+
+#define TYPE_OCORES		0
+#define TYPE_GRLIB		1
+
+#define DMEC_I2C_OFFSET		0x20
+#define DMEC_I2C_MAX_BUS_NUM	3
+
+struct dmec_i2c {
+	struct device *dev;
+	void __iomem *base;
+	u32 reg_shift;
+	u32 reg_io_width;
+	wait_queue_head_t wait;
+	struct i2c_adapter adap;
+	struct i2c_mux_core *mux;
+	struct i2c_msg *msg;
+	int pos;
+	int nmsgs;
+	int state; /* see STATE_ */
+	struct clk *clk;
+	int ip_clock_khz;
+	int bus_clock_khz;
+
+	u8 irq;
+	struct regmap *regmap;
+};
+
+static int flags;
+module_param(flags, int, 0644);
+MODULE_PARM_DESC(flags, "additional flags for the i2c bus configuration");
+
+static inline void dmec_i2c_setreg(struct dmec_i2c *i2c, int reg, u8 value)
+{
+	struct regmap *regmap = i2c->regmap;
+
+	regmap_write(regmap, DMEC_I2C_OFFSET + reg, value);
+}
+
+static inline u8 dmec_i2c_getreg(struct dmec_i2c *i2c, int reg)
+{
+	struct regmap *regmap = i2c->regmap;
+	unsigned int val;
+
+	regmap_read(regmap, DMEC_I2C_OFFSET + reg, &val);
+
+	return val;
+}
+
+static int dmec_i2c_dmx_select(struct i2c_mux_core *mux, u32 chan)
+{
+	struct dmec_i2c *i2c = mux->priv;
+	u8 bus = chan & 0x3;
+
+	dmec_i2c_setreg(i2c, DMECI2C_MUX, bus);
+
+	return 0;
+}
+
+static void dmec_i2c_dmx_del(struct dmec_i2c *i2c)
+{
+	if (i2c->mux)
+		i2c_mux_del_adapters(i2c->mux);
+}
+
+static int dmec_i2c_dmx_add(struct dmec_i2c *i2c)
+{
+	u8 bus_mask;
+	int i, ret = 0;
+
+	bus_mask = dmec_i2c_getreg(i2c, DMECI2C_MUX);
+	bus_mask = (bus_mask & 0x70) >> 4;
+
+	i2c->mux = i2c_mux_alloc(&i2c->adap,
+				 i2c->dev,
+				 DMEC_I2C_MAX_BUS_NUM, 0, 0,
+				 dmec_i2c_dmx_select,
+				 NULL);
+	if (!i2c->mux)
+		return -ENOMEM;
+
+	i2c->mux->priv = i2c;
+
+	for (i = 0; i < DMEC_I2C_MAX_BUS_NUM; i++) {
+		if (!(bus_mask & (i + 1)))
+			/* bus is not present so skip */
+			continue;
+		ret = i2c_mux_add_adapter(i2c->mux, 0, i, 0);
+		if (ret) {
+			ret = -ENODEV;
+			dev_err(i2c->dev,
+				"i2c dmx failed to register adapter %d\n", i);
+			goto dmec_i2c_dmx_add_failed;
+		}
+	}
+
+	return 0;
+
+dmec_i2c_dmx_add_failed:
+	dmec_i2c_dmx_del(i2c);
+	return ret;
+}
+
+static void dmec_i2c_process(struct dmec_i2c *i2c)
+{
+	struct i2c_msg *msg = i2c->msg;
+	u8 stat = dmec_i2c_getreg(i2c, DMECI2C_STATUS);
+
+	if ((i2c->state == STATE_DONE) || (i2c->state == STATE_ERROR)) {
+		/* stop has been sent */
+		dmec_i2c_setreg(i2c, DMECI2C_CMD, DMECI2C_CMD_IACK);
+		wake_up(&i2c->wait);
+		return;
+	}
+
+	/* error? */
+	if (stat & DMECI2C_STAT_ARBLOST) {
+		i2c->state = STATE_ERROR;
+		dmec_i2c_setreg(i2c, DMECI2C_CMD, DMECI2C_CMD_STOP);
+		return;
+	}
+
+	if ((i2c->state == STATE_START) || (i2c->state == STATE_WRITE)) {
+		i2c->state =
+			(msg->flags & I2C_M_RD) ? STATE_READ : STATE_WRITE;
+
+		if (stat & DMECI2C_STAT_NACK) {
+			i2c->state = STATE_ERROR;
+			dmec_i2c_setreg(i2c, DMECI2C_CMD, DMECI2C_CMD_STOP);
+			return;
+		}
+	} else {
+		msg->buf[i2c->pos++] = dmec_i2c_getreg(i2c, DMECI2C_DATA);
+	}
+
+	/* end of msg? */
+	if (i2c->pos == msg->len) {
+		i2c->nmsgs--;
+		i2c->msg++;
+		i2c->pos = 0;
+		msg = i2c->msg;
+
+		if (i2c->nmsgs) {	/* end? */
+			/* send start? */
+			if (!(msg->flags & I2C_M_NOSTART)) {
+				u8 addr = (msg->addr << 1);
+
+				if (msg->flags & I2C_M_RD)
+					addr |= 1;
+
+				i2c->state = STATE_START;
+
+				dmec_i2c_setreg(i2c, DMECI2C_DATA, addr);
+				dmec_i2c_setreg(i2c, DMECI2C_CMD,
+						DMECI2C_CMD_START);
+				return;
+			}
+			i2c->state = (msg->flags & I2C_M_RD)
+					? STATE_READ : STATE_WRITE;
+		} else {
+			i2c->state = STATE_DONE;
+			dmec_i2c_setreg(i2c, DMECI2C_CMD, DMECI2C_CMD_STOP);
+			return;
+		}
+	}
+
+	if (i2c->state == STATE_READ) {
+		dmec_i2c_setreg(i2c, DMECI2C_CMD, i2c->pos == (msg->len - 1) ?
+			  DMECI2C_CMD_READ_NACK : DMECI2C_CMD_READ_ACK);
+	} else {
+		dmec_i2c_setreg(i2c, DMECI2C_DATA, msg->buf[i2c->pos++]);
+		dmec_i2c_setreg(i2c, DMECI2C_CMD, DMECI2C_CMD_WRITE);
+	}
+}
+
+static irqreturn_t dmec_i2c_isr(int irq, void *dev_id)
+{
+	struct dmec_i2c *i2c = dev_id;
+
+	dmec_i2c_process(i2c);
+
+	return IRQ_HANDLED;
+}
+
+static int dmec_i2c_xfer(struct i2c_adapter *adap,
+			 struct i2c_msg *msgs, int num)
+{
+	struct dmec_i2c *i2c = i2c_get_adapdata(adap);
+
+	i2c->msg = msgs;
+	i2c->pos = 0;
+	i2c->nmsgs = num;
+	i2c->state = STATE_START;
+
+	dmec_i2c_setreg(i2c, DMECI2C_DATA,
+			(i2c->msg->addr << 1) |
+			((i2c->msg->flags & I2C_M_RD) ? 1 : 0));
+
+	dmec_i2c_setreg(i2c, DMECI2C_CMD, DMECI2C_CMD_START);
+
+	if (wait_event_timeout(i2c->wait, (i2c->state == STATE_ERROR) ||
+			       (i2c->state == STATE_DONE), HZ))
+		return (i2c->state == STATE_DONE) ? num : -EIO;
+	else
+		return -ETIMEDOUT;
+}
+
+static int dmec_i2c_init(struct device *dev, struct dmec_i2c *i2c)
+{
+	int prescale;
+	int diff;
+	u8 stat;
+	u8 ctrl = dmec_i2c_getreg(i2c, DMECI2C_CONTROL);
+
+	/* make sure the device is disabled */
+	ctrl &= ~(DMECI2C_CTRL_EN | DMECI2C_CTRL_IEN |
+		  DMECI2C_CTRL_VSSCL | DMECI2C_CTRL_VSSDA |
+		  DMECI2C_CTRL_MABCLR | DMECI2C_CTRL_MMDIS);
+	dmec_i2c_setreg(i2c, DMECI2C_CONTROL, ctrl);
+
+	prescale = (i2c->ip_clock_khz / (8 * i2c->bus_clock_khz)) - 2;
+	prescale = clamp(prescale, 0, 0xffff);
+
+	diff = i2c->ip_clock_khz / (8 * (prescale + 2)) - i2c->bus_clock_khz;
+	if (abs(diff) > i2c->bus_clock_khz / 10) {
+		dev_err(dev,
+			"Unsupported clock: core: %d KHz, bus: %d KHz\n",
+			i2c->ip_clock_khz, i2c->bus_clock_khz);
+		return -EINVAL;
+	}
+
+	dmec_i2c_setreg(i2c, DMECI2C_PRELOW, prescale & 0xff);
+	dmec_i2c_setreg(i2c, DMECI2C_PREHIGH, prescale >> 8);
+
+	/* default to first bus */
+	dmec_i2c_setreg(i2c, DMECI2C_MUX, 0x0);
+
+	/* Init the device */
+	dmec_i2c_setreg(i2c, DMECI2C_CMD, DMECI2C_CMD_IACK);
+	ctrl |= DMECI2C_CTRL_EN;
+	if (!flags)
+		ctrl |= DMECI2C_CTRL_MABCLR | DMECI2C_CTRL_MMDIS;
+	else
+		ctrl |= (flags & 0x3c);
+	dmec_i2c_setreg(i2c, DMECI2C_CONTROL, ctrl);
+
+	stat = dmec_i2c_getreg(i2c, DMECI2C_STATUS);
+	if (stat & DMECI2C_STAT_BUSY) {
+		dev_warn(dev,
+			 "I2C bus is busy - generating stop signal\n");
+		dmec_i2c_setreg(i2c, DMECI2C_CMD, DMECI2C_CMD_STOP);
+	}
+
+	stat = dmec_i2c_getreg(i2c, DMECI2C_VER);
+	dev_info(dev, "v%u.%u\n", (stat >> 4) & 0xf, stat & 0xf);
+
+	return 0;
+}
+
+static int dmec_i2c_irq_enable(struct dmec_i2c *i2c)
+{
+	u8 ctrl;
+	unsigned int irq;
+	int ret;
+
+	irq = i2c->irq;
+
+	/* Initialize interrupt handlers if not already done */
+	init_waitqueue_head(&i2c->wait);
+
+	ret = devm_request_threaded_irq(i2c->dev, irq, NULL, dmec_i2c_isr,
+					IRQF_ONESHOT | IRQF_SHARED,
+					i2c->adap.name, i2c);
+	if (ret) {
+		dev_err(i2c->dev,
+			"Unable to claim IRQ\n");
+		return ret;
+	}
+
+	/* Now enable interrupts in the controller */
+	ctrl = dmec_i2c_getreg(i2c, DMECI2C_CONTROL);
+	ctrl |= DMECI2C_CTRL_IEN;
+	dmec_i2c_setreg(i2c, DMECI2C_CONTROL, ctrl);
+
+	return 0;
+}
+
+static int dmec_i2c_irq_disable(struct dmec_i2c *i2c)
+{
+	u8 ctrl;
+
+	ctrl = dmec_i2c_getreg(i2c, DMECI2C_CONTROL);
+	ctrl &= ~DMECI2C_CTRL_IEN;
+	dmec_i2c_setreg(i2c, DMECI2C_CONTROL, ctrl);
+
+	devm_free_irq(i2c->dev, i2c->irq, i2c);
+
+	return 0;
+}
+
+static u32 dmec_i2c_func(struct i2c_adapter *adap)
+{
+	return I2C_FUNC_I2C | I2C_FUNC_SMBUS_EMUL;
+}
+
+static const struct i2c_algorithm dmec_algorithm = {
+	.master_xfer = dmec_i2c_xfer,
+	.functionality = dmec_i2c_func,
+};
+
+static struct i2c_adapter dmec_adapter = {
+	.owner = THIS_MODULE,
+	.name = "i2c-dmec",
+	.class = I2C_CLASS_DEPRECATED,
+	.algo = &dmec_algorithm,
+};
+
+#define dmec_i2c_of_probe(pdev, i2c) -ENODEV
+
+static int dmec_i2c_probe(struct platform_device *pdev)
+{
+	struct dmec_i2c *i2c;
+	struct dmec_i2c_platform_data *pdata;
+	int ret;
+
+	i2c = devm_kzalloc(&pdev->dev, sizeof(*i2c), GFP_KERNEL);
+	if (!i2c)
+		return -ENOMEM;
+
+	i2c->dev = &pdev->dev;
+	pdata = dev_get_platdata(&pdev->dev);
+	if (pdata) {
+		i2c->reg_shift = pdata->reg_shift;
+		i2c->reg_io_width = pdata->reg_io_width;
+		i2c->ip_clock_khz = pdata->clock_khz;
+		i2c->bus_clock_khz = 100;
+		i2c->regmap = dmec_get_regmap(pdev->dev.parent);
+	} else {
+		ret = dmec_i2c_of_probe(pdev, i2c);
+		if (ret)
+			return ret;
+	}
+
+	if (i2c->reg_io_width == 0)
+		i2c->reg_io_width = 1; /* Set to default value */
+
+	i2c->irq = platform_get_irq(pdev, 0);
+
+	ret = dmec_i2c_init(&pdev->dev, i2c);
+	if (ret)
+		return ret;
+
+	/* hook up driver to tree */
+	platform_set_drvdata(pdev, i2c);
+	i2c->adap = dmec_adapter;
+	i2c_set_adapdata(&i2c->adap, i2c);
+	i2c->adap.dev.parent = &pdev->dev;
+	i2c->adap.dev.of_node = pdev->dev.of_node;
+
+	if (dmec_i2c_irq_enable(i2c)) {
+		dev_err(&pdev->dev, "Cannot claim IRQ\n");
+		return ret;
+	}
+
+	/* add i2c adapter to i2c tree */
+	ret = i2c_add_adapter(&i2c->adap);
+	if (ret) {
+		dev_err(&pdev->dev, "Failed to add adapter\n");
+		return ret;
+	}
+
+	ret = dmec_i2c_dmx_add(i2c);
+
+	return ret;
+}
+
+static int dmec_i2c_remove(struct platform_device *pdev)
+{
+	struct dmec_i2c *i2c = platform_get_drvdata(pdev);
+
+	/* disable i2c logic */
+	dmec_i2c_setreg(i2c, DMECI2C_CONTROL,
+			dmec_i2c_getreg(i2c, DMECI2C_CONTROL)
+			& ~(DMECI2C_CTRL_EN | DMECI2C_CTRL_IEN));
+
+	/* remove adapter & data */
+	dmec_i2c_dmx_del(i2c);
+	i2c_del_adapter(&i2c->adap);
+
+	if (!IS_ERR(i2c->clk))
+		clk_disable_unprepare(i2c->clk);
+
+	return 0;
+}
+
+#ifdef CONFIG_PM_SLEEP
+static int dmec_i2c_suspend(struct device *dev)
+{
+	struct dmec_i2c *i2c = dev_get_drvdata(dev);
+	u8 ctrl = dmec_i2c_getreg(i2c, DMECI2C_CONTROL);
+
+	dmec_i2c_irq_disable(i2c);
+
+	/* make sure the device is disabled */
+	dmec_i2c_setreg(i2c, DMECI2C_CONTROL,
+			ctrl & ~DMECI2C_CTRL_EN);
+
+	if (!IS_ERR(i2c->clk))
+		clk_disable_unprepare(i2c->clk);
+	return 0;
+}
+
+static int dmec_i2c_resume(struct device *dev)
+{
+	struct dmec_i2c *i2c = dev_get_drvdata(dev);
+	int ret;
+
+	if (!IS_ERR(i2c->clk)) {
+		unsigned long rate;
+
+		ret = clk_prepare_enable(i2c->clk);
+
+		if (ret) {
+			dev_err(dev,
+				"clk_prepare_enable failed: %d\n", ret);
+			return ret;
+		}
+		rate = clk_get_rate(i2c->clk) / 1000;
+		if (rate)
+			i2c->ip_clock_khz = rate;
+	}
+
+	ret = dmec_i2c_init(dev, i2c);
+	if (ret)
+		return ret;
+	return dmec_i2c_irq_enable(i2c);
+}
+
+static SIMPLE_DEV_PM_OPS(dmec_i2c_pm, dmec_i2c_suspend, dmec_i2c_resume);
+#define DMEC_I2C_PM	(&dmec_i2c_pm)
+#else
+#define DMEC_I2C_PM	NULL
+#endif
+
+static struct platform_driver dmec_i2c_driver = {
+	.probe   = dmec_i2c_probe,
+	.remove  = dmec_i2c_remove,
+	.driver  = {
+		.name = "dmec-i2c",
+		.pm = DMEC_I2C_PM,
+	},
+};
+
+module_platform_driver(dmec_i2c_driver);
+
+MODULE_AUTHOR("Peter Korsgaard <jacmet@xxxxxxxxxx>");
+MODULE_AUTHOR("Zahari Doychev <zahari.doychev@xxxxxxxxx>");
+MODULE_DESCRIPTION("DMO OpenCores based I2C bus driver");
+MODULE_LICENSE("GPL");
+MODULE_ALIAS("platform:dmec-i2c");
-- 
git-series 0.8.10
--
To unsubscribe from this list: send the line "unsubscribe linux-i2c" in
the body of a message to majordomo@xxxxxxxxxxxxxxx
More majordomo info at  http://vger.kernel.org/majordomo-info.html



[Index of Archives]     [Linux GPIO]     [Linux SPI]     [Linux Hardward Monitoring]     [LM Sensors]     [Linux USB Devel]     [Linux Media]     [Video for Linux]     [Linux Audio Users]     [Yosemite News]     [Linux Kernel]     [Linux SCSI]

  Powered by Linux