Re: [PATCH] i2c-designware: Add support for ACPI i2c device

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

 



On Fri, Aug 29, 2014 at 10:39:17AM +0200, Wolfram Sang wrote:
> On Fri, Aug 29, 2014 at 11:09:12AM +0800, Carl Peng wrote:
> > AMD platform i2c bus controllers are ACPI devices,
> > this patch is to add a ACPI glue for Designware
> > core, make it support i2c bus controller with
> > ACPI interface.
> > 
> > Signed-off-by: Carl Peng <carlpeng008@xxxxxxxxx>
> 
> i2c-designware-platdrv.c has ACPI support, can't you reuse that?
> CCing Mika for ACPI expertise...

Indeed, you should just update the i2c-designware-platdrv.c with your
ACPI IDs.

> 
> > ---
> >  drivers/i2c/busses/Kconfig                  |  12 +
> >  drivers/i2c/busses/Makefile                 |   1 +
> >  drivers/i2c/busses/i2c-designware-acpidrv.c | 355 ++++++++++++++++++++++++++++
> >  drivers/i2c/busses/i2c-designware-core.h    |   4 +
> >  4 files changed, 372 insertions(+)
> >  create mode 100644 drivers/i2c/busses/i2c-designware-acpidrv.c
> > 
> > diff --git a/drivers/i2c/busses/Kconfig b/drivers/i2c/busses/Kconfig
> > index 2ac87fa..974700c 100644
> > --- a/drivers/i2c/busses/Kconfig
> > +++ b/drivers/i2c/busses/Kconfig
> > @@ -441,6 +441,18 @@ config I2C_DESIGNWARE_PCI
> >  	  This driver can also be built as a module.  If so, the module
> >  	  will be called i2c-designware-pci.
> >  
> > +config I2C_DESIGNWARE_ACPI
> > +	tristate "Synopsys DesignWare ACPI"
> > +	depends on ACPI
> > +	select I2C_DESIGNWARE_CORE
> > +	help
> > +	  AMD platform i2c bus controller is ACPI device, this driver is
> > +	  to add ACPI glue for designware core, make it support ACPI
> > +	  interface.
> > +
> > +	  If you say yes to this option, support will be included for the
> > +	  Synopsys DesignWare I2C adapter. Only master mode is supported.
> > +
> >  config I2C_EFM32
> >  	tristate "EFM32 I2C controller"
> >  	depends on ARCH_EFM32 || COMPILE_TEST
> > diff --git a/drivers/i2c/busses/Makefile b/drivers/i2c/busses/Makefile
> > index 49bf07e..ce86081 100644
> > --- a/drivers/i2c/busses/Makefile
> > +++ b/drivers/i2c/busses/Makefile
> > @@ -42,6 +42,7 @@ obj-$(CONFIG_I2C_DESIGNWARE_PLATFORM)	+= i2c-designware-platform.o
> >  i2c-designware-platform-objs := i2c-designware-platdrv.o
> >  obj-$(CONFIG_I2C_DESIGNWARE_PCI)	+= i2c-designware-pci.o
> >  i2c-designware-pci-objs := i2c-designware-pcidrv.o
> > +obj-$(CONFIG_I2C_DESIGNWARE_ACPI)	+= i2c-designware-acpidrv.o
> >  obj-$(CONFIG_I2C_EFM32)		+= i2c-efm32.o
> >  obj-$(CONFIG_I2C_EG20T)		+= i2c-eg20t.o
> >  obj-$(CONFIG_I2C_EXYNOS5)	+= i2c-exynos5.o
> > diff --git a/drivers/i2c/busses/i2c-designware-acpidrv.c b/drivers/i2c/busses/i2c-designware-acpidrv.c
> > new file mode 100644
> > index 0000000..ebe3a51
> > --- /dev/null
> > +++ b/drivers/i2c/busses/i2c-designware-acpidrv.c
> > @@ -0,0 +1,355 @@
> > +/*
> > + * Synopsys DesignWare I2C adapter driver (master only).
> > + *
> > + * Based on the TI DAVINCI I2C adapter driver.
> > + *
> > + * Copyright (C) 2006 Texas Instruments.
> > + * Copyright (C) 2007 MontaVista Software Inc.
> > + * Copyright (C) 2009 Provigent Ltd.
> > + * Copyright (C) 2011 Intel corporation.
> > + *
> > + * ----------------------------------------------------------------------------
> > + *
> > + * This program is free software; you can redistribute it and/or modify
> > + * it under the terms of the GNU General Public License as published by
> > + * the Free Software Foundation; either version 2 of the License, or
> > + * (at your option) any later version.
> > + *
> > + * This program is distributed in the hope that it will be useful,
> > + * but WITHOUT ANY WARRANTY; without even the implied warranty of
> > + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
> > + * GNU General Public License for more details.
> > + *
> > + * ----------------------------------------------------------------------------
> > + *
> > + */
> > +
> > +#include <linux/kernel.h>
> > +#include <linux/module.h>
> > +#include <linux/delay.h>
> > +#include <linux/i2c.h>
> > +#include <linux/errno.h>
> > +#include <linux/sched.h>
> > +#include <linux/err.h>
> > +#include <linux/interrupt.h>
> > +#include <linux/io.h>
> > +#include <linux/slab.h>
> > +#include <linux/pm_runtime.h>
> > +#include <linux/acpi.h>
> > +
> > +#include "i2c-designware-core.h"
> > +
> > +#define CLK_KHZ	(133 * 1024)
> > +#define TX_FIFO_DEPTH	128
> > +#define RX_FIFO_DEPTH	128
> > +
> > +#define DRIVER_NAME	"i2c-designware-acpi"
> > +
> > +#define INTEL_MID_STD_CFG	(DW_IC_CON_MASTER |			\
> > +				DW_IC_CON_SLAVE_DISABLE |	\
> > +				DW_IC_CON_RESTART_EN)
> > +
> > +enum dw_acpi_ctl_id_t {
> > +	I2C_BUS_A = 0,
> > +	I2C_BUS_B,
> > +	I2C_BUS_C,
> > +	I2C_BUS_D,
> > +};
> > +
> > +struct dw_acpi_controller {
> > +	u32 bus_num;
> > +	u32 bus_cfg;
> > +	u32 tx_fifo_depth;
> > +	u32 rx_fifo_depth;
> > +	u32 clk_khz;
> > +};
> > +
> > +/*i2c controller parameter*/
> > +static struct  dw_acpi_controller  dw_acpi_controllers[] = {
> > +	[I2C_BUS_A] = {
> > +		.bus_num     = 1,
> > +		.bus_cfg   = INTEL_MID_STD_CFG | DW_IC_CON_SPEED_STD,
> > +		.tx_fifo_depth = TX_FIFO_DEPTH,
> > +		.rx_fifo_depth = RX_FIFO_DEPTH,
> > +		.clk_khz      =  CLK_KHZ,
> > +	},
> > +	[I2C_BUS_B] = {
> > +		.bus_num     = 2,
> > +		.bus_cfg   = INTEL_MID_STD_CFG | DW_IC_CON_SPEED_STD,
> > +		.tx_fifo_depth = TX_FIFO_DEPTH,
> > +		.rx_fifo_depth = RX_FIFO_DEPTH,
> > +		.clk_khz      =  CLK_KHZ,
> > +	},
> > +	[I2C_BUS_C] = {
> > +		.bus_num     = 3,
> > +		.bus_cfg   = INTEL_MID_STD_CFG | DW_IC_CON_SPEED_STD,
> > +		.tx_fifo_depth = TX_FIFO_DEPTH,
> > +		.rx_fifo_depth = RX_FIFO_DEPTH,
> > +		.clk_khz      =  CLK_KHZ,
> > +	},
> > +	[I2C_BUS_D] = {
> > +		.bus_num     = 4,
> > +		.bus_cfg   = INTEL_MID_STD_CFG | DW_IC_CON_SPEED_STD,
> > +		.tx_fifo_depth = TX_FIFO_DEPTH,
> > +		.rx_fifo_depth = RX_FIFO_DEPTH,
> > +		.clk_khz      =  CLK_KHZ,
> > +	},
> > +};
> > +
> > +static struct i2c_algorithm i2c_dw_algo = {
> > +	.master_xfer	= i2c_dw_xfer,
> > +	.functionality	= i2c_dw_func,
> > +};
> > +
> > +static int i2c_dw_acpi_suspend(struct device *dev)
> > +{
> > +	struct dw_i2c_dev *i2c = acpi_driver_data(to_acpi_device(dev));
> > +
> > +	i2c_dw_disable(i2c);
> > +
> > +	return 0;
> > +}
> > +
> > +static int i2c_dw_acpi_resume(struct device *dev)
> > +{
> > +	struct dw_i2c_dev *i2c = acpi_driver_data(to_acpi_device(dev));
> > +	u32 enabled;
> > +
> > +	enabled = i2c_dw_is_enabled(i2c);
> > +	if (enabled)
> > +		return 0;
> > +
> > +	i2c_dw_init(i2c);
> > +
> > +	return 0;
> > +}
> > +
> > +static int i2c_dw_acpi_runtime_idle(struct device *dev)
> > +{
> > +	int err = pm_schedule_suspend(dev, 500);
> > +
> > +	dev_dbg(dev, "runtime_idle called\n");
> > +
> > +	if (err != 0)
> > +		return 0;
> > +
> > +	return -EBUSY;
> > +}
> > +
> > +static int get_irq_flag(int trigger, int polarity, int share)
> > +{
> > +	int flags;
> > +
> > +	if (trigger == ACPI_LEVEL_SENSITIVE)
> > +		flags = (polarity == ACPI_ACTIVE_LOW) ? IRQF_TRIGGER_LOW
> > +							: IRQF_TRIGGER_HIGH;
> > +	else
> > +		flags = (polarity == ACPI_ACTIVE_LOW) ? IRQF_TRIGGER_FALLING
> > +							: IRQF_TRIGGER_RISING;
> > +
> > +	if (share == ACPI_SHARED)
> > +		flags |= IRQF_SHARED;
> > +
> > +	return flags;
> > +}
> > +
> > +static acpi_status
> > +acpi_i2c_parse_resource(struct acpi_resource *res, void *context)
> > +{
> > +	int ret;
> > +	struct dw_i2c_dev *dev = context;
> > +	struct acpi_resource_fixed_memory32 *fixmem32;
> > +
> > +	switch (res->type) {
> > +	case ACPI_RESOURCE_TYPE_FIXED_MEMORY32:
> > +		fixmem32 = &res->data.fixed_memory32;
> > +		if (!fixmem32)
> > +			return AE_NO_MEMORY;
> > +
> > +		dev->io_base = fixmem32->address;
> > +		dev->io_length = fixmem32->address_length;
> > +		ret = AE_OK;
> > +		break;
> > +
> > +	case ACPI_RESOURCE_TYPE_IRQ:
> > +		dev->irq = res->data.irq.interrupts[0];
> > +		dev->irq_flag = get_irq_flag(res->data.irq.triggering,
> > +			res->data.irq.polarity, res->data.irq.sharable);
> > +		ret = AE_OK;
> > +		break;
> > +
> > +	default:
> > +		if (dev->io_base && dev->irq)
> > +			ret = AE_OK;
> > +		else
> > +			ret = AE_NOT_FOUND;
> > +		break;
> > +	}
> > +
> > +	return ret;
> > +}
> > +
> > +static const struct dev_pm_ops i2c_dw_pm_ops = {
> > +	.resume         = i2c_dw_acpi_resume,
> > +	.suspend        = i2c_dw_acpi_suspend,
> > +	SET_RUNTIME_PM_OPS(i2c_dw_acpi_suspend, i2c_dw_acpi_resume,
> > +			   i2c_dw_acpi_runtime_idle)
> > +};
> > +
> > +static u32 i2c_dw_get_clk_rate_khz(struct dw_i2c_dev *dev)
> > +{
> > +	return dev->acpi_controller->clk_khz;
> > +}
> > +
> > +static u32 i2c_get_uid(struct acpi_device *dev)
> > +{
> > +	return kstrtoul(dev->pnp.unique_id, 10, NULL);
> > +}
> > +
> > +static int i2c_dw_acpi_add(struct acpi_device *pdev)
> > +{
> > +	int r;
> > +	acpi_status status;
> > +	struct dw_i2c_dev *dev;
> > +	struct i2c_adapter *adap;
> > +	struct dw_acpi_controller *controller;
> > +	u32 controller_id;
> > +
> > +	controller_id = i2c_get_uid(pdev);
> > +
> > +	if (controller_id >= ARRAY_SIZE(dw_acpi_controllers)) {
> > +		dev_err(&pdev->dev, "invalid controller number %d\n",
> > +			controller_id);
> > +		return  -EINVAL;
> > +	}
> > +
> > +	controller = &dw_acpi_controllers[controller_id];
> > +
> > +	dev = devm_kzalloc(&pdev->dev, sizeof(struct dw_i2c_dev), GFP_KERNEL);
> > +	if (!dev)
> > +		return  -ENOMEM;
> > +
> > +	status = acpi_walk_resources(pdev->handle, METHOD_NAME__CRS,
> > +				     acpi_i2c_parse_resource, dev);
> > +	if (ACPI_FAILURE(status) || !dev->io_base || !dev->irq) {
> > +		dev_err(&pdev->dev, "failure getting resource\n");
> > +		return -ENODEV;
> > +	}
> > +
> > +	init_completion(&dev->cmd_complete);
> > +	mutex_init(&dev->lock);
> > +	dev->clk = NULL;
> > +	dev->acpi_controller = controller;
> > +	dev->get_clk_rate_khz = i2c_dw_get_clk_rate_khz;
> > +	dev->base = devm_ioremap(&pdev->dev, dev->io_base, dev->io_length);
> > +	if (dev->base == NULL) {
> > +		dev_err(&pdev->dev, "failure mapping I/O memory\n");
> > +		return -EBUSY;
> > +	}
> > +	dev->dev = &pdev->dev;
> > +	dev->functionality =
> > +		I2C_FUNC_I2C |
> > +		I2C_FUNC_SMBUS_BYTE |
> > +		I2C_FUNC_SMBUS_BYTE_DATA |
> > +		I2C_FUNC_SMBUS_WORD_DATA |
> > +		I2C_FUNC_SMBUS_I2C_BLOCK;
> > +	dev->master_cfg =  controller->bus_cfg;
> > +
> > +	pdev->driver_data = dev;
> > +
> > +	dev->tx_fifo_depth = controller->tx_fifo_depth;
> > +	dev->rx_fifo_depth = controller->rx_fifo_depth;
> > +
> > +	r = i2c_dw_init(dev);
> > +	if (r) {
> > +		dev_err(&pdev->dev, "failure initiating i2c controller\n");
> > +		return -EINVAL;
> > +	}
> > +
> > +	adap = &dev->adapter;
> > +	i2c_set_adapdata(adap, dev);
> > +	adap->owner = THIS_MODULE;
> > +	adap->class = I2C_CLASS_HWMON;
> > +	adap->algo = &i2c_dw_algo;
> > +	adap->dev.parent = &pdev->dev;
> > +	ACPI_COMPANION_SET(adap->dev.parent, pdev);
> > +	adap->nr = controller->bus_num;
> > +	snprintf(adap->name, sizeof(adap->name), "i2c-designware-acpi-%d",
> > +		 adap->nr);
> > +
> > +	r = devm_request_irq(&pdev->dev, dev->irq, i2c_dw_isr, dev->irq_flag,
> > +			     adap->name, dev);
> > +	if (r) {
> > +		dev_err(&pdev->dev, "failure requesting irq %d\n", dev->irq);
> > +		return r;
> > +	}
> > +
> > +	i2c_dw_disable_int(dev);
> > +	i2c_dw_clear_int(dev);
> > +
> > +	r = i2c_add_numbered_adapter(adap);
> > +	if (r) {
> > +		dev_err(&pdev->dev, "failure adding adapter\n");
> > +		return r;
> > +	}
> > +
> > +	pm_runtime_set_autosuspend_delay(&pdev->dev, 1000);
> > +	pm_runtime_use_autosuspend(&pdev->dev);
> > +	pm_runtime_allow(&pdev->dev);
> > +
> > +	return 0;
> > +}
> > +
> > +static int i2c_dw_acpi_remove(struct acpi_device *pdev)
> > +{
> > +	struct dw_i2c_dev *dev = acpi_driver_data(pdev);
> > +
> > +	i2c_dw_disable(dev);
> > +	pm_runtime_forbid(&pdev->dev);
> > +	pm_runtime_get_noresume(&pdev->dev);
> > +
> > +	i2c_del_adapter(&dev->adapter);
> > +
> > +	return 0;
> > +}
> > +
> > +/* work with hotplug and coldplug */
> > +MODULE_ALIAS("i2c_designware-acpi");
> > +
> > +static const struct acpi_device_id i2_designware_acpi_ids[] = {
> > +	{ "IFC0000", 0},
> > +	{ "AMD0010", 0},
> > +	{ "", 0},
> > +};
> > +MODULE_DEVICE_TABLE(acpi, i2_designware_acpi_ids);
> > +
> > +static struct acpi_driver dw_i2c_driver = {
> > +	.name =         DRIVER_NAME,
> > +	.class =        DRIVER_NAME,
> > +	.ids =          i2_designware_acpi_ids,
> > +	.ops =          {
> > +				.add =  i2c_dw_acpi_add,
> > +				.remove = i2c_dw_acpi_remove,
> > +			},
> > +	.drv.pm = &i2c_dw_pm_ops,
> > +};
> > +
> > +static int __init dw_i2c_module_init(void)
> > +{
> > +	int ret;
> > +
> > +	ret = acpi_bus_register_driver(&dw_i2c_driver);
> > +
> > +	return ret;
> > +}
> > +module_init(dw_i2c_module_init);
> > +
> > +static void __exit dw_i2c_module_exit(void)
> > +{
> > +	acpi_bus_unregister_driver(&dw_i2c_driver);
> > +}
> > +module_exit(dw_i2c_module_exit);
> > +
> > +MODULE_AUTHOR("Carl Peng <carlpeng008@xxxxxxxxx>");
> > +MODULE_DESCRIPTION("Synopsys DesignWare ACPI I2C bus adapter");
> > +MODULE_LICENSE("GPL");
> > diff --git a/drivers/i2c/busses/i2c-designware-core.h b/drivers/i2c/busses/i2c-designware-core.h
> > index d66b6cb..b88dbb8 100644
> > --- a/drivers/i2c/busses/i2c-designware-core.h
> > +++ b/drivers/i2c/busses/i2c-designware-core.h
> > @@ -73,11 +73,14 @@
> >  struct dw_i2c_dev {
> >  	struct device		*dev;
> >  	void __iomem		*base;
> > +	resource_size_t		io_base;
> > +	u32			io_length;
> >  	struct completion	cmd_complete;
> >  	struct mutex		lock;
> >  	struct clk		*clk;
> >  	u32			(*get_clk_rate_khz) (struct dw_i2c_dev *dev);
> >  	struct dw_pci_controller *controller;
> > +	struct dw_acpi_controller *acpi_controller;
> >  	int			cmd_err;
> >  	struct i2c_msg		*msgs;
> >  	int			msgs_num;
> > @@ -91,6 +94,7 @@ struct dw_i2c_dev {
> >  	unsigned int		status;
> >  	u32			abort_source;
> >  	int			irq;
> > +	unsigned int		irq_flag;
> >  	u32			accessor_flags;
> >  	struct i2c_adapter	adapter;
> >  	u32			functionality;
> > -- 
> > 1.9.1
> > 
> > --
> > 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


--
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