mirror of
				https://kernel.googlesource.com/pub/scm/linux/kernel/git/torvalds/linux
				synced 2025-11-04 16:52:06 +10:00 
			
		
		
		
	Merge branch 'i2c/for-next' of git://git.kernel.org/pub/scm/linux/kernel/git/wsa/linux
Pull i2c updates from Wolfram Sang: "Highlights from the I2C subsystem for 3.18: - new drivers for Axxia AM55xx, and Hisilicon hix5hd2 SoC. - designware driver gained AMD support, exynos gained exynos7 support The rest is usual driver stuff. Hopefully no lowlights this time" * 'i2c/for-next' of git://git.kernel.org/pub/scm/linux/kernel/git/wsa/linux: i2c: i801: Add Device IDs for Intel Sunrise Point PCH i2c: hix5hd2: add i2c controller driver i2c-imx: Disable the clock on probe failure i2c: designware: Add support for AMD I2C controller i2c: designware: Rework probe() to get clock a bit later i2c: designware: Default to fast mode in case of ACPI i2c: axxia: Add I2C driver for AXM55xx i2c: exynos: add support for HSI2C module on Exynos7 i2c: mxs: detect No Slave Ack on SELECT in PIO mode i2c: cros_ec: Remove EC_I2C_FLAG_10BIT i2c: cros-ec-tunnel: Add of match table i2c: rcar: remove sign-compare flaw i2c: ismt: Use minimum descriptor size i2c: imx: Add arbitration lost check i2c: rk3x: Remove unlikely() annotations i2c: rcar: check for no IRQ in rcar_i2c_irq() i2c: rcar: make rcar_i2c_prepare_msg() *void* i2c: rcar: simplify check for last message i2c: designware: add support of platform data to set I2C mode i2c: designware: add support of I2C standard mode
This commit is contained in:
		
						commit
						278f1d0730
					
				
							
								
								
									
										30
									
								
								Documentation/devicetree/bindings/i2c/i2c-axxia.txt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										30
									
								
								Documentation/devicetree/bindings/i2c/i2c-axxia.txt
									
									
									
									
									
										Normal file
									
								
							@ -0,0 +1,30 @@
 | 
			
		||||
LSI Axxia I2C
 | 
			
		||||
 | 
			
		||||
Required properties :
 | 
			
		||||
- compatible : Must be "lsi,api2c"
 | 
			
		||||
- reg : Offset and length of the register set for the device
 | 
			
		||||
- interrupts : the interrupt specifier
 | 
			
		||||
- #address-cells : Must be <1>;
 | 
			
		||||
- #size-cells : Must be <0>;
 | 
			
		||||
- clock-names : Must contain "i2c".
 | 
			
		||||
- clocks: Must contain an entry for each name in clock-names. See the common
 | 
			
		||||
  clock bindings.
 | 
			
		||||
 | 
			
		||||
Optional properties :
 | 
			
		||||
- clock-frequency : Desired I2C bus clock frequency in Hz. If not specified,
 | 
			
		||||
  the default 100 kHz frequency will be used. As only Normal and Fast modes
 | 
			
		||||
  are supported, possible values are 100000 and 400000.
 | 
			
		||||
 | 
			
		||||
Example :
 | 
			
		||||
 | 
			
		||||
i2c@02010084000 {
 | 
			
		||||
	compatible = "lsi,api2c";
 | 
			
		||||
	device_type = "i2c";
 | 
			
		||||
	#address-cells = <1>;
 | 
			
		||||
	#size-cells = <0>;
 | 
			
		||||
	reg = <0x20 0x10084000 0x00 0x1000>;
 | 
			
		||||
	interrupts = <0 19 4>;
 | 
			
		||||
	clocks = <&clk_per>;
 | 
			
		||||
	clock-names = "i2c";
 | 
			
		||||
	clock-frequency = <400000>;
 | 
			
		||||
};
 | 
			
		||||
@ -12,6 +12,8 @@ Required properties:
 | 
			
		||||
				on Exynos5250 and Exynos5420 SoCs.
 | 
			
		||||
	-> "samsung,exynos5260-hsi2c", for i2c compatible with HSI2C available
 | 
			
		||||
				on Exynos5260 SoCs.
 | 
			
		||||
	-> "samsung,exynos7-hsi2c", for i2c compatible with HSI2C available
 | 
			
		||||
				on Exynos7 SoCs.
 | 
			
		||||
 | 
			
		||||
  - reg: physical base address of the controller and length of memory mapped
 | 
			
		||||
    region.
 | 
			
		||||
 | 
			
		||||
							
								
								
									
										24
									
								
								Documentation/devicetree/bindings/i2c/i2c-hix5hd2.txt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										24
									
								
								Documentation/devicetree/bindings/i2c/i2c-hix5hd2.txt
									
									
									
									
									
										Normal file
									
								
							@ -0,0 +1,24 @@
 | 
			
		||||
I2C for Hisilicon hix5hd2 chipset platform
 | 
			
		||||
 | 
			
		||||
Required properties:
 | 
			
		||||
- compatible: Must be "hisilicon,hix5hd2-i2c"
 | 
			
		||||
- reg: physical base address of the controller and length of memory mapped
 | 
			
		||||
     region.
 | 
			
		||||
- interrupts: interrupt number to the cpu.
 | 
			
		||||
- #address-cells = <1>;
 | 
			
		||||
- #size-cells = <0>;
 | 
			
		||||
- clocks: phandles to input clocks.
 | 
			
		||||
 | 
			
		||||
Optional properties:
 | 
			
		||||
- clock-frequency: Desired I2C bus frequency in Hz, otherwise defaults to 100000
 | 
			
		||||
- Child nodes conforming to i2c bus binding
 | 
			
		||||
 | 
			
		||||
Examples:
 | 
			
		||||
I2C0@f8b10000 {
 | 
			
		||||
	compatible = "hisilicon,hix5hd2-i2c";
 | 
			
		||||
	reg = <0xf8b10000 0x1000>;
 | 
			
		||||
	interrupts = <0 38 4>;
 | 
			
		||||
	clocks = <&clock HIX5HD2_I2C0_RST>;
 | 
			
		||||
	#address-cells = <1>;
 | 
			
		||||
	#size-cells = <0>;
 | 
			
		||||
}
 | 
			
		||||
@ -28,6 +28,7 @@ Supported adapters:
 | 
			
		||||
  * Intel Wildcat Point (PCH)
 | 
			
		||||
  * Intel Wildcat Point-LP (PCH)
 | 
			
		||||
  * Intel BayTrail (SOC)
 | 
			
		||||
  * Intel Sunrise Point-H (PCH)
 | 
			
		||||
   Datasheets: Publicly available at the Intel website
 | 
			
		||||
 | 
			
		||||
On Intel Patsburg and later chipsets, both the normal host SMBus controller
 | 
			
		||||
 | 
			
		||||
@ -77,6 +77,16 @@ config I2C_AMD8111
 | 
			
		||||
	  This driver can also be built as a module.  If so, the module
 | 
			
		||||
	  will be called i2c-amd8111.
 | 
			
		||||
 | 
			
		||||
config I2C_HIX5HD2
 | 
			
		||||
	tristate "Hix5hd2 high-speed I2C driver"
 | 
			
		||||
	depends on ARCH_HIX5HD2
 | 
			
		||||
	help
 | 
			
		||||
	  Say Y here to include support for high-speed I2C controller in the
 | 
			
		||||
	  Hisilicon based hix5hd2 SoCs.
 | 
			
		||||
 | 
			
		||||
	  This driver can also be built as a module.  If so, the module
 | 
			
		||||
	  will be called i2c-hix5hd2.
 | 
			
		||||
 | 
			
		||||
config I2C_I801
 | 
			
		||||
	tristate "Intel 82801 (ICH/PCH)"
 | 
			
		||||
	depends on PCI
 | 
			
		||||
@ -112,6 +122,7 @@ config I2C_I801
 | 
			
		||||
	    Wildcat Point (PCH)
 | 
			
		||||
	    Wildcat Point-LP (PCH)
 | 
			
		||||
	    BayTrail (SOC)
 | 
			
		||||
	    Sunrise Point-H (PCH)
 | 
			
		||||
 | 
			
		||||
	  This driver can also be built as a module.  If so, the module
 | 
			
		||||
	  will be called i2c-i801.
 | 
			
		||||
@ -337,6 +348,17 @@ config I2C_AU1550
 | 
			
		||||
	  This driver can also be built as a module.  If so, the module
 | 
			
		||||
	  will be called i2c-au1550.
 | 
			
		||||
 | 
			
		||||
config I2C_AXXIA
 | 
			
		||||
	tristate "Axxia I2C controller"
 | 
			
		||||
	depends on ARCH_AXXIA || COMPILE_TEST
 | 
			
		||||
	default ARCH_AXXIA
 | 
			
		||||
	help
 | 
			
		||||
	  Say yes if you want to support the I2C bus on Axxia platforms.
 | 
			
		||||
 | 
			
		||||
	  Please note that this controller is limited to transfers of maximum
 | 
			
		||||
	  255 bytes in length. Any attempt to to a larger transfer will return
 | 
			
		||||
	  an error.
 | 
			
		||||
 | 
			
		||||
config I2C_BCM2835
 | 
			
		||||
	tristate "Broadcom BCM2835 I2C controller"
 | 
			
		||||
	depends on ARCH_BCM2835
 | 
			
		||||
@ -423,6 +445,7 @@ config I2C_DESIGNWARE_CORE
 | 
			
		||||
config I2C_DESIGNWARE_PLATFORM
 | 
			
		||||
	tristate "Synopsys DesignWare Platform"
 | 
			
		||||
	select I2C_DESIGNWARE_CORE
 | 
			
		||||
	depends on (ACPI && COMMON_CLK) || !ACPI
 | 
			
		||||
	help
 | 
			
		||||
	  If you say yes to this option, support will be included for the
 | 
			
		||||
	  Synopsys DesignWare I2C adapter. Only master mode is supported.
 | 
			
		||||
@ -465,7 +488,7 @@ config I2C_EG20T
 | 
			
		||||
 | 
			
		||||
config I2C_EXYNOS5
 | 
			
		||||
	tristate "Exynos5 high-speed I2C driver"
 | 
			
		||||
	depends on ARCH_EXYNOS5 && OF
 | 
			
		||||
	depends on ARCH_EXYNOS && OF
 | 
			
		||||
	default y
 | 
			
		||||
	help
 | 
			
		||||
	  High-speed I2C controller on Exynos5 based Samsung SoCs.
 | 
			
		||||
 | 
			
		||||
@ -31,6 +31,7 @@ obj-$(CONFIG_I2C_POWERMAC)	+= i2c-powermac.o
 | 
			
		||||
# Embedded system I2C/SMBus host controller drivers
 | 
			
		||||
obj-$(CONFIG_I2C_AT91)		+= i2c-at91.o
 | 
			
		||||
obj-$(CONFIG_I2C_AU1550)	+= i2c-au1550.o
 | 
			
		||||
obj-$(CONFIG_I2C_AXXIA)		+= i2c-axxia.o
 | 
			
		||||
obj-$(CONFIG_I2C_BCM2835)	+= i2c-bcm2835.o
 | 
			
		||||
obj-$(CONFIG_I2C_BLACKFIN_TWI)	+= i2c-bfin-twi.o
 | 
			
		||||
obj-$(CONFIG_I2C_CADENCE)	+= i2c-cadence.o
 | 
			
		||||
@ -47,6 +48,7 @@ obj-$(CONFIG_I2C_EG20T)		+= i2c-eg20t.o
 | 
			
		||||
obj-$(CONFIG_I2C_EXYNOS5)	+= i2c-exynos5.o
 | 
			
		||||
obj-$(CONFIG_I2C_GPIO)		+= i2c-gpio.o
 | 
			
		||||
obj-$(CONFIG_I2C_HIGHLANDER)	+= i2c-highlander.o
 | 
			
		||||
obj-$(CONFIG_I2C_HIX5HD2)	+= i2c-hix5hd2.o
 | 
			
		||||
obj-$(CONFIG_I2C_IBM_IIC)	+= i2c-ibm_iic.o
 | 
			
		||||
obj-$(CONFIG_I2C_IMX)		+= i2c-imx.o
 | 
			
		||||
obj-$(CONFIG_I2C_IOP3XX)	+= i2c-iop3xx.o
 | 
			
		||||
 | 
			
		||||
							
								
								
									
										559
									
								
								drivers/i2c/busses/i2c-axxia.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										559
									
								
								drivers/i2c/busses/i2c-axxia.c
									
									
									
									
									
										Normal file
									
								
							@ -0,0 +1,559 @@
 | 
			
		||||
/*
 | 
			
		||||
 * This driver implements I2C master functionality using the LSI API2C
 | 
			
		||||
 * controller.
 | 
			
		||||
 *
 | 
			
		||||
 * NOTE: The controller has a limitation in that it can only do transfers of
 | 
			
		||||
 * maximum 255 bytes at a time. If a larger transfer is attempted, error code
 | 
			
		||||
 * (-EINVAL) is returned.
 | 
			
		||||
 *
 | 
			
		||||
 * This software is licensed under the terms of the GNU General Public
 | 
			
		||||
 * License version 2, as published by the Free Software Foundation, and
 | 
			
		||||
 * may be copied, distributed, and modified under those terms.
 | 
			
		||||
 */
 | 
			
		||||
#include <linux/clk.h>
 | 
			
		||||
#include <linux/clkdev.h>
 | 
			
		||||
#include <linux/err.h>
 | 
			
		||||
#include <linux/i2c.h>
 | 
			
		||||
#include <linux/init.h>
 | 
			
		||||
#include <linux/interrupt.h>
 | 
			
		||||
#include <linux/module.h>
 | 
			
		||||
#include <linux/io.h>
 | 
			
		||||
#include <linux/kernel.h>
 | 
			
		||||
#include <linux/platform_device.h>
 | 
			
		||||
 | 
			
		||||
#define SCL_WAIT_TIMEOUT_NS 25000000
 | 
			
		||||
#define I2C_XFER_TIMEOUT    (msecs_to_jiffies(250))
 | 
			
		||||
#define I2C_STOP_TIMEOUT    (msecs_to_jiffies(100))
 | 
			
		||||
#define FIFO_SIZE           8
 | 
			
		||||
 | 
			
		||||
#define GLOBAL_CONTROL		0x00
 | 
			
		||||
#define   GLOBAL_MST_EN         BIT(0)
 | 
			
		||||
#define   GLOBAL_SLV_EN         BIT(1)
 | 
			
		||||
#define   GLOBAL_IBML_EN        BIT(2)
 | 
			
		||||
#define INTERRUPT_STATUS	0x04
 | 
			
		||||
#define INTERRUPT_ENABLE	0x08
 | 
			
		||||
#define   INT_SLV               BIT(1)
 | 
			
		||||
#define   INT_MST               BIT(0)
 | 
			
		||||
#define WAIT_TIMER_CONTROL	0x0c
 | 
			
		||||
#define   WT_EN			BIT(15)
 | 
			
		||||
#define   WT_VALUE(_x)		((_x) & 0x7fff)
 | 
			
		||||
#define IBML_TIMEOUT		0x10
 | 
			
		||||
#define IBML_LOW_MEXT		0x14
 | 
			
		||||
#define IBML_LOW_SEXT		0x18
 | 
			
		||||
#define TIMER_CLOCK_DIV		0x1c
 | 
			
		||||
#define I2C_BUS_MONITOR		0x20
 | 
			
		||||
#define SOFT_RESET		0x24
 | 
			
		||||
#define MST_COMMAND		0x28
 | 
			
		||||
#define   CMD_BUSY		(1<<3)
 | 
			
		||||
#define   CMD_MANUAL		(0x00 | CMD_BUSY)
 | 
			
		||||
#define   CMD_AUTO		(0x01 | CMD_BUSY)
 | 
			
		||||
#define MST_RX_XFER		0x2c
 | 
			
		||||
#define MST_TX_XFER		0x30
 | 
			
		||||
#define MST_ADDR_1		0x34
 | 
			
		||||
#define MST_ADDR_2		0x38
 | 
			
		||||
#define MST_DATA		0x3c
 | 
			
		||||
#define MST_TX_FIFO		0x40
 | 
			
		||||
#define MST_RX_FIFO		0x44
 | 
			
		||||
#define MST_INT_ENABLE		0x48
 | 
			
		||||
#define MST_INT_STATUS		0x4c
 | 
			
		||||
#define   MST_STATUS_RFL	(1 << 13) /* RX FIFO serivce */
 | 
			
		||||
#define   MST_STATUS_TFL	(1 << 12) /* TX FIFO service */
 | 
			
		||||
#define   MST_STATUS_SNS	(1 << 11) /* Manual mode done */
 | 
			
		||||
#define   MST_STATUS_SS		(1 << 10) /* Automatic mode done */
 | 
			
		||||
#define   MST_STATUS_SCC	(1 << 9)  /* Stop complete */
 | 
			
		||||
#define   MST_STATUS_IP		(1 << 8)  /* Invalid parameter */
 | 
			
		||||
#define   MST_STATUS_TSS	(1 << 7)  /* Timeout */
 | 
			
		||||
#define   MST_STATUS_AL		(1 << 6)  /* Arbitration lost */
 | 
			
		||||
#define   MST_STATUS_ND		(1 << 5)  /* NAK on data phase */
 | 
			
		||||
#define   MST_STATUS_NA		(1 << 4)  /* NAK on address phase */
 | 
			
		||||
#define   MST_STATUS_NAK	(MST_STATUS_NA | \
 | 
			
		||||
				 MST_STATUS_ND)
 | 
			
		||||
#define   MST_STATUS_ERR	(MST_STATUS_NAK | \
 | 
			
		||||
				 MST_STATUS_AL  | \
 | 
			
		||||
				 MST_STATUS_IP  | \
 | 
			
		||||
				 MST_STATUS_TSS)
 | 
			
		||||
#define MST_TX_BYTES_XFRD	0x50
 | 
			
		||||
#define MST_RX_BYTES_XFRD	0x54
 | 
			
		||||
#define SCL_HIGH_PERIOD		0x80
 | 
			
		||||
#define SCL_LOW_PERIOD		0x84
 | 
			
		||||
#define SPIKE_FLTR_LEN		0x88
 | 
			
		||||
#define SDA_SETUP_TIME		0x8c
 | 
			
		||||
#define SDA_HOLD_TIME		0x90
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * axxia_i2c_dev - I2C device context
 | 
			
		||||
 * @base: pointer to register struct
 | 
			
		||||
 * @msg: pointer to current message
 | 
			
		||||
 * @msg_xfrd: number of bytes transferred in msg
 | 
			
		||||
 * @msg_err: error code for completed message
 | 
			
		||||
 * @msg_complete: xfer completion object
 | 
			
		||||
 * @dev: device reference
 | 
			
		||||
 * @adapter: core i2c abstraction
 | 
			
		||||
 * @i2c_clk: clock reference for i2c input clock
 | 
			
		||||
 * @bus_clk_rate: current i2c bus clock rate
 | 
			
		||||
 */
 | 
			
		||||
struct axxia_i2c_dev {
 | 
			
		||||
	void __iomem *base;
 | 
			
		||||
	struct i2c_msg *msg;
 | 
			
		||||
	size_t msg_xfrd;
 | 
			
		||||
	int msg_err;
 | 
			
		||||
	struct completion msg_complete;
 | 
			
		||||
	struct device *dev;
 | 
			
		||||
	struct i2c_adapter adapter;
 | 
			
		||||
	struct clk *i2c_clk;
 | 
			
		||||
	u32 bus_clk_rate;
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
static void i2c_int_disable(struct axxia_i2c_dev *idev, u32 mask)
 | 
			
		||||
{
 | 
			
		||||
	u32 int_en;
 | 
			
		||||
 | 
			
		||||
	int_en = readl(idev->base + MST_INT_ENABLE);
 | 
			
		||||
	writel(int_en & ~mask, idev->base + MST_INT_ENABLE);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static void i2c_int_enable(struct axxia_i2c_dev *idev, u32 mask)
 | 
			
		||||
{
 | 
			
		||||
	u32 int_en;
 | 
			
		||||
 | 
			
		||||
	int_en = readl(idev->base + MST_INT_ENABLE);
 | 
			
		||||
	writel(int_en | mask, idev->base + MST_INT_ENABLE);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * ns_to_clk - Convert time (ns) to clock cycles for the given clock frequency.
 | 
			
		||||
 */
 | 
			
		||||
static u32 ns_to_clk(u64 ns, u32 clk_mhz)
 | 
			
		||||
{
 | 
			
		||||
	return div_u64(ns * clk_mhz, 1000);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static int axxia_i2c_init(struct axxia_i2c_dev *idev)
 | 
			
		||||
{
 | 
			
		||||
	u32 divisor = clk_get_rate(idev->i2c_clk) / idev->bus_clk_rate;
 | 
			
		||||
	u32 clk_mhz = clk_get_rate(idev->i2c_clk) / 1000000;
 | 
			
		||||
	u32 t_setup;
 | 
			
		||||
	u32 t_high, t_low;
 | 
			
		||||
	u32 tmo_clk;
 | 
			
		||||
	u32 prescale;
 | 
			
		||||
	unsigned long timeout;
 | 
			
		||||
 | 
			
		||||
	dev_dbg(idev->dev, "rate=%uHz per_clk=%uMHz -> ratio=1:%u\n",
 | 
			
		||||
		idev->bus_clk_rate, clk_mhz, divisor);
 | 
			
		||||
 | 
			
		||||
	/* Reset controller */
 | 
			
		||||
	writel(0x01, idev->base + SOFT_RESET);
 | 
			
		||||
	timeout = jiffies + msecs_to_jiffies(100);
 | 
			
		||||
	while (readl(idev->base + SOFT_RESET) & 1) {
 | 
			
		||||
		if (time_after(jiffies, timeout)) {
 | 
			
		||||
			dev_warn(idev->dev, "Soft reset failed\n");
 | 
			
		||||
			break;
 | 
			
		||||
		}
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	/* Enable Master Mode */
 | 
			
		||||
	writel(0x1, idev->base + GLOBAL_CONTROL);
 | 
			
		||||
 | 
			
		||||
	if (idev->bus_clk_rate <= 100000) {
 | 
			
		||||
		/* Standard mode SCL 50/50, tSU:DAT = 250 ns */
 | 
			
		||||
		t_high = divisor * 1 / 2;
 | 
			
		||||
		t_low = divisor * 1 / 2;
 | 
			
		||||
		t_setup = ns_to_clk(250, clk_mhz);
 | 
			
		||||
	} else {
 | 
			
		||||
		/* Fast mode SCL 33/66, tSU:DAT = 100 ns */
 | 
			
		||||
		t_high = divisor * 1 / 3;
 | 
			
		||||
		t_low = divisor * 2 / 3;
 | 
			
		||||
		t_setup = ns_to_clk(100, clk_mhz);
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	/* SCL High Time */
 | 
			
		||||
	writel(t_high, idev->base + SCL_HIGH_PERIOD);
 | 
			
		||||
	/* SCL Low Time */
 | 
			
		||||
	writel(t_low, idev->base + SCL_LOW_PERIOD);
 | 
			
		||||
	/* SDA Setup Time */
 | 
			
		||||
	writel(t_setup, idev->base + SDA_SETUP_TIME);
 | 
			
		||||
	/* SDA Hold Time, 300ns */
 | 
			
		||||
	writel(ns_to_clk(300, clk_mhz), idev->base + SDA_HOLD_TIME);
 | 
			
		||||
	/* Filter <50ns spikes */
 | 
			
		||||
	writel(ns_to_clk(50, clk_mhz), idev->base + SPIKE_FLTR_LEN);
 | 
			
		||||
 | 
			
		||||
	/* Configure Time-Out Registers */
 | 
			
		||||
	tmo_clk = ns_to_clk(SCL_WAIT_TIMEOUT_NS, clk_mhz);
 | 
			
		||||
 | 
			
		||||
	/* Find prescaler value that makes tmo_clk fit in 15-bits counter. */
 | 
			
		||||
	for (prescale = 0; prescale < 15; ++prescale) {
 | 
			
		||||
		if (tmo_clk <= 0x7fff)
 | 
			
		||||
			break;
 | 
			
		||||
		tmo_clk >>= 1;
 | 
			
		||||
	}
 | 
			
		||||
	if (tmo_clk > 0x7fff)
 | 
			
		||||
		tmo_clk = 0x7fff;
 | 
			
		||||
 | 
			
		||||
	/* Prescale divider (log2) */
 | 
			
		||||
	writel(prescale, idev->base + TIMER_CLOCK_DIV);
 | 
			
		||||
	/* Timeout in divided clocks */
 | 
			
		||||
	writel(WT_EN | WT_VALUE(tmo_clk), idev->base + WAIT_TIMER_CONTROL);
 | 
			
		||||
 | 
			
		||||
	/* Mask all master interrupt bits */
 | 
			
		||||
	i2c_int_disable(idev, ~0);
 | 
			
		||||
 | 
			
		||||
	/* Interrupt enable */
 | 
			
		||||
	writel(0x01, idev->base + INTERRUPT_ENABLE);
 | 
			
		||||
 | 
			
		||||
	return 0;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static int i2c_m_rd(const struct i2c_msg *msg)
 | 
			
		||||
{
 | 
			
		||||
	return (msg->flags & I2C_M_RD) != 0;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static int i2c_m_ten(const struct i2c_msg *msg)
 | 
			
		||||
{
 | 
			
		||||
	return (msg->flags & I2C_M_TEN) != 0;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static int i2c_m_recv_len(const struct i2c_msg *msg)
 | 
			
		||||
{
 | 
			
		||||
	return (msg->flags & I2C_M_RECV_LEN) != 0;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * axxia_i2c_empty_rx_fifo - Fetch data from RX FIFO and update SMBus block
 | 
			
		||||
 * transfer length if this is the first byte of such a transfer.
 | 
			
		||||
 */
 | 
			
		||||
static int axxia_i2c_empty_rx_fifo(struct axxia_i2c_dev *idev)
 | 
			
		||||
{
 | 
			
		||||
	struct i2c_msg *msg = idev->msg;
 | 
			
		||||
	size_t rx_fifo_avail = readl(idev->base + MST_RX_FIFO);
 | 
			
		||||
	int bytes_to_transfer = min(rx_fifo_avail, msg->len - idev->msg_xfrd);
 | 
			
		||||
 | 
			
		||||
	while (bytes_to_transfer-- > 0) {
 | 
			
		||||
		int c = readl(idev->base + MST_DATA);
 | 
			
		||||
 | 
			
		||||
		if (idev->msg_xfrd == 0 && i2c_m_recv_len(msg)) {
 | 
			
		||||
			/*
 | 
			
		||||
			 * Check length byte for SMBus block read
 | 
			
		||||
			 */
 | 
			
		||||
			if (c <= 0 || c > I2C_SMBUS_BLOCK_MAX) {
 | 
			
		||||
				idev->msg_err = -EPROTO;
 | 
			
		||||
				i2c_int_disable(idev, ~0);
 | 
			
		||||
				complete(&idev->msg_complete);
 | 
			
		||||
				break;
 | 
			
		||||
			}
 | 
			
		||||
			msg->len = 1 + c;
 | 
			
		||||
			writel(msg->len, idev->base + MST_RX_XFER);
 | 
			
		||||
		}
 | 
			
		||||
		msg->buf[idev->msg_xfrd++] = c;
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	return 0;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * axxia_i2c_fill_tx_fifo - Fill TX FIFO from current message buffer.
 | 
			
		||||
 * @return: Number of bytes left to transfer.
 | 
			
		||||
 */
 | 
			
		||||
static int axxia_i2c_fill_tx_fifo(struct axxia_i2c_dev *idev)
 | 
			
		||||
{
 | 
			
		||||
	struct i2c_msg *msg = idev->msg;
 | 
			
		||||
	size_t tx_fifo_avail = FIFO_SIZE - readl(idev->base + MST_TX_FIFO);
 | 
			
		||||
	int bytes_to_transfer = min(tx_fifo_avail, msg->len - idev->msg_xfrd);
 | 
			
		||||
	int ret = msg->len - idev->msg_xfrd - bytes_to_transfer;
 | 
			
		||||
 | 
			
		||||
	while (bytes_to_transfer-- > 0)
 | 
			
		||||
		writel(msg->buf[idev->msg_xfrd++], idev->base + MST_DATA);
 | 
			
		||||
 | 
			
		||||
	return ret;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static irqreturn_t axxia_i2c_isr(int irq, void *_dev)
 | 
			
		||||
{
 | 
			
		||||
	struct axxia_i2c_dev *idev = _dev;
 | 
			
		||||
	u32 status;
 | 
			
		||||
 | 
			
		||||
	if (!(readl(idev->base + INTERRUPT_STATUS) & INT_MST))
 | 
			
		||||
		return IRQ_NONE;
 | 
			
		||||
 | 
			
		||||
	/* Read interrupt status bits */
 | 
			
		||||
	status = readl(idev->base + MST_INT_STATUS);
 | 
			
		||||
 | 
			
		||||
	if (!idev->msg) {
 | 
			
		||||
		dev_warn(idev->dev, "unexpected interrupt\n");
 | 
			
		||||
		goto out;
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	/* RX FIFO needs service? */
 | 
			
		||||
	if (i2c_m_rd(idev->msg) && (status & MST_STATUS_RFL))
 | 
			
		||||
		axxia_i2c_empty_rx_fifo(idev);
 | 
			
		||||
 | 
			
		||||
	/* TX FIFO needs service? */
 | 
			
		||||
	if (!i2c_m_rd(idev->msg) && (status & MST_STATUS_TFL)) {
 | 
			
		||||
		if (axxia_i2c_fill_tx_fifo(idev) == 0)
 | 
			
		||||
			i2c_int_disable(idev, MST_STATUS_TFL);
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	if (status & MST_STATUS_SCC) {
 | 
			
		||||
		/* Stop completed */
 | 
			
		||||
		i2c_int_disable(idev, ~0);
 | 
			
		||||
		complete(&idev->msg_complete);
 | 
			
		||||
	} else if (status & MST_STATUS_SNS) {
 | 
			
		||||
		/* Transfer done */
 | 
			
		||||
		i2c_int_disable(idev, ~0);
 | 
			
		||||
		if (i2c_m_rd(idev->msg) && idev->msg_xfrd < idev->msg->len)
 | 
			
		||||
			axxia_i2c_empty_rx_fifo(idev);
 | 
			
		||||
		complete(&idev->msg_complete);
 | 
			
		||||
	} else if (unlikely(status & MST_STATUS_ERR)) {
 | 
			
		||||
		/* Transfer error */
 | 
			
		||||
		i2c_int_disable(idev, ~0);
 | 
			
		||||
		if (status & MST_STATUS_AL)
 | 
			
		||||
			idev->msg_err = -EAGAIN;
 | 
			
		||||
		else if (status & MST_STATUS_NAK)
 | 
			
		||||
			idev->msg_err = -ENXIO;
 | 
			
		||||
		else
 | 
			
		||||
			idev->msg_err = -EIO;
 | 
			
		||||
		dev_dbg(idev->dev, "error %#x, addr=%#x rx=%u/%u tx=%u/%u\n",
 | 
			
		||||
			status,
 | 
			
		||||
			idev->msg->addr,
 | 
			
		||||
			readl(idev->base + MST_RX_BYTES_XFRD),
 | 
			
		||||
			readl(idev->base + MST_RX_XFER),
 | 
			
		||||
			readl(idev->base + MST_TX_BYTES_XFRD),
 | 
			
		||||
			readl(idev->base + MST_TX_XFER));
 | 
			
		||||
		complete(&idev->msg_complete);
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
out:
 | 
			
		||||
	/* Clear interrupt */
 | 
			
		||||
	writel(INT_MST, idev->base + INTERRUPT_STATUS);
 | 
			
		||||
 | 
			
		||||
	return IRQ_HANDLED;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static int axxia_i2c_xfer_msg(struct axxia_i2c_dev *idev, struct i2c_msg *msg)
 | 
			
		||||
{
 | 
			
		||||
	u32 int_mask = MST_STATUS_ERR | MST_STATUS_SNS;
 | 
			
		||||
	u32 rx_xfer, tx_xfer;
 | 
			
		||||
	u32 addr_1, addr_2;
 | 
			
		||||
	int ret;
 | 
			
		||||
 | 
			
		||||
	if (msg->len > 255) {
 | 
			
		||||
		dev_warn(idev->dev, "unsupported length %u\n", msg->len);
 | 
			
		||||
		return -EINVAL;
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	idev->msg = msg;
 | 
			
		||||
	idev->msg_xfrd = 0;
 | 
			
		||||
	idev->msg_err = 0;
 | 
			
		||||
	reinit_completion(&idev->msg_complete);
 | 
			
		||||
 | 
			
		||||
	if (i2c_m_ten(msg)) {
 | 
			
		||||
		/* 10-bit address
 | 
			
		||||
		 *   addr_1: 5'b11110 | addr[9:8] | (R/nW)
 | 
			
		||||
		 *   addr_2: addr[7:0]
 | 
			
		||||
		 */
 | 
			
		||||
		addr_1 = 0xF0 | ((msg->addr >> 7) & 0x06);
 | 
			
		||||
		addr_2 = msg->addr & 0xFF;
 | 
			
		||||
	} else {
 | 
			
		||||
		/* 7-bit address
 | 
			
		||||
		 *   addr_1: addr[6:0] | (R/nW)
 | 
			
		||||
		 *   addr_2: dont care
 | 
			
		||||
		 */
 | 
			
		||||
		addr_1 = (msg->addr << 1) & 0xFF;
 | 
			
		||||
		addr_2 = 0;
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	if (i2c_m_rd(msg)) {
 | 
			
		||||
		/* I2C read transfer */
 | 
			
		||||
		rx_xfer = i2c_m_recv_len(msg) ? I2C_SMBUS_BLOCK_MAX : msg->len;
 | 
			
		||||
		tx_xfer = 0;
 | 
			
		||||
		addr_1 |= 1;	/* Set the R/nW bit of the address */
 | 
			
		||||
	} else {
 | 
			
		||||
		/* I2C write transfer */
 | 
			
		||||
		rx_xfer = 0;
 | 
			
		||||
		tx_xfer = msg->len;
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	writel(rx_xfer, idev->base + MST_RX_XFER);
 | 
			
		||||
	writel(tx_xfer, idev->base + MST_TX_XFER);
 | 
			
		||||
	writel(addr_1, idev->base + MST_ADDR_1);
 | 
			
		||||
	writel(addr_2, idev->base + MST_ADDR_2);
 | 
			
		||||
 | 
			
		||||
	if (i2c_m_rd(msg))
 | 
			
		||||
		int_mask |= MST_STATUS_RFL;
 | 
			
		||||
	else if (axxia_i2c_fill_tx_fifo(idev) != 0)
 | 
			
		||||
		int_mask |= MST_STATUS_TFL;
 | 
			
		||||
 | 
			
		||||
	/* Start manual mode */
 | 
			
		||||
	writel(CMD_MANUAL, idev->base + MST_COMMAND);
 | 
			
		||||
 | 
			
		||||
	i2c_int_enable(idev, int_mask);
 | 
			
		||||
 | 
			
		||||
	ret = wait_for_completion_timeout(&idev->msg_complete,
 | 
			
		||||
					  I2C_XFER_TIMEOUT);
 | 
			
		||||
 | 
			
		||||
	i2c_int_disable(idev, int_mask);
 | 
			
		||||
 | 
			
		||||
	if (readl(idev->base + MST_COMMAND) & CMD_BUSY)
 | 
			
		||||
		dev_warn(idev->dev, "busy after xfer\n");
 | 
			
		||||
 | 
			
		||||
	if (ret == 0)
 | 
			
		||||
		idev->msg_err = -ETIMEDOUT;
 | 
			
		||||
 | 
			
		||||
	if (unlikely(idev->msg_err) && idev->msg_err != -ENXIO)
 | 
			
		||||
		axxia_i2c_init(idev);
 | 
			
		||||
 | 
			
		||||
	return idev->msg_err;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static int axxia_i2c_stop(struct axxia_i2c_dev *idev)
 | 
			
		||||
{
 | 
			
		||||
	u32 int_mask = MST_STATUS_ERR | MST_STATUS_SCC;
 | 
			
		||||
	int ret;
 | 
			
		||||
 | 
			
		||||
	reinit_completion(&idev->msg_complete);
 | 
			
		||||
 | 
			
		||||
	/* Issue stop */
 | 
			
		||||
	writel(0xb, idev->base + MST_COMMAND);
 | 
			
		||||
	i2c_int_enable(idev, int_mask);
 | 
			
		||||
	ret = wait_for_completion_timeout(&idev->msg_complete,
 | 
			
		||||
					  I2C_STOP_TIMEOUT);
 | 
			
		||||
	i2c_int_disable(idev, int_mask);
 | 
			
		||||
	if (ret == 0)
 | 
			
		||||
		return -ETIMEDOUT;
 | 
			
		||||
 | 
			
		||||
	if (readl(idev->base + MST_COMMAND) & CMD_BUSY)
 | 
			
		||||
		dev_warn(idev->dev, "busy after stop\n");
 | 
			
		||||
 | 
			
		||||
	return 0;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static int
 | 
			
		||||
axxia_i2c_xfer(struct i2c_adapter *adap, struct i2c_msg msgs[], int num)
 | 
			
		||||
{
 | 
			
		||||
	struct axxia_i2c_dev *idev = i2c_get_adapdata(adap);
 | 
			
		||||
	int i;
 | 
			
		||||
	int ret = 0;
 | 
			
		||||
 | 
			
		||||
	for (i = 0; ret == 0 && i < num; ++i)
 | 
			
		||||
		ret = axxia_i2c_xfer_msg(idev, &msgs[i]);
 | 
			
		||||
 | 
			
		||||
	axxia_i2c_stop(idev);
 | 
			
		||||
 | 
			
		||||
	return ret ? : i;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static u32 axxia_i2c_func(struct i2c_adapter *adap)
 | 
			
		||||
{
 | 
			
		||||
	u32 caps = (I2C_FUNC_I2C | I2C_FUNC_10BIT_ADDR |
 | 
			
		||||
		    I2C_FUNC_SMBUS_EMUL | I2C_FUNC_SMBUS_BLOCK_DATA);
 | 
			
		||||
	return caps;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static const struct i2c_algorithm axxia_i2c_algo = {
 | 
			
		||||
	.master_xfer = axxia_i2c_xfer,
 | 
			
		||||
	.functionality = axxia_i2c_func,
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
static int axxia_i2c_probe(struct platform_device *pdev)
 | 
			
		||||
{
 | 
			
		||||
	struct device_node *np = pdev->dev.of_node;
 | 
			
		||||
	struct axxia_i2c_dev *idev = NULL;
 | 
			
		||||
	struct resource *res;
 | 
			
		||||
	void __iomem *base;
 | 
			
		||||
	int irq;
 | 
			
		||||
	int ret = 0;
 | 
			
		||||
 | 
			
		||||
	idev = devm_kzalloc(&pdev->dev, sizeof(*idev), GFP_KERNEL);
 | 
			
		||||
	if (!idev)
 | 
			
		||||
		return -ENOMEM;
 | 
			
		||||
 | 
			
		||||
	res = platform_get_resource(pdev, IORESOURCE_MEM, 0);
 | 
			
		||||
	base = devm_ioremap_resource(&pdev->dev, res);
 | 
			
		||||
	if (IS_ERR(base))
 | 
			
		||||
		return PTR_ERR(base);
 | 
			
		||||
 | 
			
		||||
	irq = platform_get_irq(pdev, 0);
 | 
			
		||||
	if (irq < 0) {
 | 
			
		||||
		dev_err(&pdev->dev, "missing interrupt resource\n");
 | 
			
		||||
		return irq;
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	idev->i2c_clk = devm_clk_get(&pdev->dev, "i2c");
 | 
			
		||||
	if (IS_ERR(idev->i2c_clk)) {
 | 
			
		||||
		dev_err(&pdev->dev, "missing clock\n");
 | 
			
		||||
		return PTR_ERR(idev->i2c_clk);
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	idev->base = base;
 | 
			
		||||
	idev->dev = &pdev->dev;
 | 
			
		||||
	init_completion(&idev->msg_complete);
 | 
			
		||||
 | 
			
		||||
	of_property_read_u32(np, "clock-frequency", &idev->bus_clk_rate);
 | 
			
		||||
	if (idev->bus_clk_rate == 0)
 | 
			
		||||
		idev->bus_clk_rate = 100000;	/* default clock rate */
 | 
			
		||||
 | 
			
		||||
	ret = axxia_i2c_init(idev);
 | 
			
		||||
	if (ret) {
 | 
			
		||||
		dev_err(&pdev->dev, "failed to initialize\n");
 | 
			
		||||
		return ret;
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	ret = devm_request_irq(&pdev->dev, irq, axxia_i2c_isr, 0,
 | 
			
		||||
			       pdev->name, idev);
 | 
			
		||||
	if (ret) {
 | 
			
		||||
		dev_err(&pdev->dev, "failed to claim IRQ%d\n", irq);
 | 
			
		||||
		return ret;
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	clk_prepare_enable(idev->i2c_clk);
 | 
			
		||||
 | 
			
		||||
	i2c_set_adapdata(&idev->adapter, idev);
 | 
			
		||||
	strlcpy(idev->adapter.name, pdev->name, sizeof(idev->adapter.name));
 | 
			
		||||
	idev->adapter.owner = THIS_MODULE;
 | 
			
		||||
	idev->adapter.algo = &axxia_i2c_algo;
 | 
			
		||||
	idev->adapter.dev.parent = &pdev->dev;
 | 
			
		||||
	idev->adapter.dev.of_node = pdev->dev.of_node;
 | 
			
		||||
 | 
			
		||||
	platform_set_drvdata(pdev, idev);
 | 
			
		||||
 | 
			
		||||
	ret = i2c_add_adapter(&idev->adapter);
 | 
			
		||||
	if (ret) {
 | 
			
		||||
		dev_err(&pdev->dev, "failed to add adapter\n");
 | 
			
		||||
		return ret;
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	return 0;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static int axxia_i2c_remove(struct platform_device *pdev)
 | 
			
		||||
{
 | 
			
		||||
	struct axxia_i2c_dev *idev = platform_get_drvdata(pdev);
 | 
			
		||||
 | 
			
		||||
	clk_disable_unprepare(idev->i2c_clk);
 | 
			
		||||
	i2c_del_adapter(&idev->adapter);
 | 
			
		||||
 | 
			
		||||
	return 0;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/* Match table for of_platform binding */
 | 
			
		||||
static const struct of_device_id axxia_i2c_of_match[] = {
 | 
			
		||||
	{ .compatible = "lsi,api2c", },
 | 
			
		||||
	{},
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
MODULE_DEVICE_TABLE(of, axxia_i2c_of_match);
 | 
			
		||||
 | 
			
		||||
static struct platform_driver axxia_i2c_driver = {
 | 
			
		||||
	.probe = axxia_i2c_probe,
 | 
			
		||||
	.remove = axxia_i2c_remove,
 | 
			
		||||
	.driver = {
 | 
			
		||||
		.name = "axxia-i2c",
 | 
			
		||||
		.of_match_table = axxia_i2c_of_match,
 | 
			
		||||
	},
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
module_platform_driver(axxia_i2c_driver);
 | 
			
		||||
 | 
			
		||||
MODULE_DESCRIPTION("Axxia I2C Bus driver");
 | 
			
		||||
MODULE_AUTHOR("Anders Berg <anders.berg@lsi.com>");
 | 
			
		||||
MODULE_LICENSE("GPL v2");
 | 
			
		||||
@ -96,7 +96,7 @@ static int ec_i2c_construct_message(u8 *buf, const struct i2c_msg i2c_msgs[],
 | 
			
		||||
		msg->addr_flags = i2c_msg->addr;
 | 
			
		||||
 | 
			
		||||
		if (i2c_msg->flags & I2C_M_TEN)
 | 
			
		||||
			msg->addr_flags |= EC_I2C_FLAG_10BIT;
 | 
			
		||||
			return -EINVAL;
 | 
			
		||||
 | 
			
		||||
		if (i2c_msg->flags & I2C_M_RD) {
 | 
			
		||||
			msg->addr_flags |= EC_I2C_FLAG_READ;
 | 
			
		||||
@ -220,7 +220,9 @@ static int ec_i2c_xfer(struct i2c_adapter *adap, struct i2c_msg i2c_msgs[],
 | 
			
		||||
		}
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	ec_i2c_construct_message(request, i2c_msgs, num, bus_num);
 | 
			
		||||
	result = ec_i2c_construct_message(request, i2c_msgs, num, bus_num);
 | 
			
		||||
	if (result)
 | 
			
		||||
		goto exit;
 | 
			
		||||
 | 
			
		||||
	msg.version = 0;
 | 
			
		||||
	msg.command = EC_CMD_I2C_PASSTHRU;
 | 
			
		||||
@ -313,11 +315,20 @@ static int ec_i2c_remove(struct platform_device *dev)
 | 
			
		||||
	return 0;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
#ifdef CONFIG_OF
 | 
			
		||||
static const struct of_device_id cros_ec_i2c_of_match[] = {
 | 
			
		||||
	{ .compatible = "google,cros-ec-i2c-tunnel" },
 | 
			
		||||
	{},
 | 
			
		||||
};
 | 
			
		||||
MODULE_DEVICE_TABLE(of, cros_ec_i2c_of_match);
 | 
			
		||||
#endif
 | 
			
		||||
 | 
			
		||||
static struct platform_driver ec_i2c_tunnel_driver = {
 | 
			
		||||
	.probe = ec_i2c_probe,
 | 
			
		||||
	.remove = ec_i2c_remove,
 | 
			
		||||
	.driver = {
 | 
			
		||||
		.name = "cros-ec-i2c-tunnel",
 | 
			
		||||
		.of_match_table = of_match_ptr(cros_ec_i2c_of_match),
 | 
			
		||||
	},
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@ -30,6 +30,7 @@
 | 
			
		||||
#include <linux/delay.h>
 | 
			
		||||
#include <linux/i2c.h>
 | 
			
		||||
#include <linux/clk.h>
 | 
			
		||||
#include <linux/clk-provider.h>
 | 
			
		||||
#include <linux/errno.h>
 | 
			
		||||
#include <linux/sched.h>
 | 
			
		||||
#include <linux/err.h>
 | 
			
		||||
@ -41,6 +42,7 @@
 | 
			
		||||
#include <linux/io.h>
 | 
			
		||||
#include <linux/slab.h>
 | 
			
		||||
#include <linux/acpi.h>
 | 
			
		||||
#include <linux/platform_data/i2c-designware.h>
 | 
			
		||||
#include "i2c-designware-core.h"
 | 
			
		||||
 | 
			
		||||
static struct i2c_algorithm i2c_dw_algo = {
 | 
			
		||||
@ -79,10 +81,7 @@ static void dw_i2c_acpi_params(struct platform_device *pdev, char method[],
 | 
			
		||||
static int dw_i2c_acpi_configure(struct platform_device *pdev)
 | 
			
		||||
{
 | 
			
		||||
	struct dw_i2c_dev *dev = platform_get_drvdata(pdev);
 | 
			
		||||
	bool fs_mode = dev->master_cfg & DW_IC_CON_SPEED_FAST;
 | 
			
		||||
 | 
			
		||||
	if (!ACPI_HANDLE(&pdev->dev))
 | 
			
		||||
		return -ENODEV;
 | 
			
		||||
	const struct acpi_device_id *id;
 | 
			
		||||
 | 
			
		||||
	dev->adapter.nr = -1;
 | 
			
		||||
	dev->tx_fifo_depth = 32;
 | 
			
		||||
@ -92,14 +91,33 @@ static int dw_i2c_acpi_configure(struct platform_device *pdev)
 | 
			
		||||
	 * Try to get SDA hold time and *CNT values from an ACPI method if
 | 
			
		||||
	 * it exists for both supported speed modes.
 | 
			
		||||
	 */
 | 
			
		||||
	dw_i2c_acpi_params(pdev, "SSCN", &dev->ss_hcnt, &dev->ss_lcnt,
 | 
			
		||||
			   fs_mode ? NULL : &dev->sda_hold_time);
 | 
			
		||||
	dw_i2c_acpi_params(pdev, "SSCN", &dev->ss_hcnt, &dev->ss_lcnt, NULL);
 | 
			
		||||
	dw_i2c_acpi_params(pdev, "FMCN", &dev->fs_hcnt, &dev->fs_lcnt,
 | 
			
		||||
			   fs_mode ? &dev->sda_hold_time : NULL);
 | 
			
		||||
			   &dev->sda_hold_time);
 | 
			
		||||
 | 
			
		||||
	/*
 | 
			
		||||
	 * Provide a way for Designware I2C host controllers that are not
 | 
			
		||||
	 * based on Intel LPSS to specify their input clock frequency via
 | 
			
		||||
	 * id->driver_data.
 | 
			
		||||
	 */
 | 
			
		||||
	id = acpi_match_device(pdev->dev.driver->acpi_match_table, &pdev->dev);
 | 
			
		||||
	if (id && id->driver_data)
 | 
			
		||||
		clk_register_fixed_rate(&pdev->dev, dev_name(&pdev->dev), NULL,
 | 
			
		||||
					CLK_IS_ROOT, id->driver_data);
 | 
			
		||||
 | 
			
		||||
	return 0;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static void dw_i2c_acpi_unconfigure(struct platform_device *pdev)
 | 
			
		||||
{
 | 
			
		||||
	struct dw_i2c_dev *dev = platform_get_drvdata(pdev);
 | 
			
		||||
	const struct acpi_device_id *id;
 | 
			
		||||
 | 
			
		||||
	id = acpi_match_device(pdev->dev.driver->acpi_match_table, &pdev->dev);
 | 
			
		||||
	if (id && id->driver_data)
 | 
			
		||||
		clk_unregister(dev->clk);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static const struct acpi_device_id dw_i2c_acpi_match[] = {
 | 
			
		||||
	{ "INT33C2", 0 },
 | 
			
		||||
	{ "INT33C3", 0 },
 | 
			
		||||
@ -107,6 +125,7 @@ static const struct acpi_device_id dw_i2c_acpi_match[] = {
 | 
			
		||||
	{ "INT3433", 0 },
 | 
			
		||||
	{ "80860F41", 0 },
 | 
			
		||||
	{ "808622C1", 0 },
 | 
			
		||||
	{ "AMD0010", 133 * 1000 * 1000 },
 | 
			
		||||
	{ }
 | 
			
		||||
};
 | 
			
		||||
MODULE_DEVICE_TABLE(acpi, dw_i2c_acpi_match);
 | 
			
		||||
@ -115,6 +134,7 @@ static inline int dw_i2c_acpi_configure(struct platform_device *pdev)
 | 
			
		||||
{
 | 
			
		||||
	return -ENODEV;
 | 
			
		||||
}
 | 
			
		||||
static inline void dw_i2c_acpi_unconfigure(struct platform_device *pdev) { }
 | 
			
		||||
#endif
 | 
			
		||||
 | 
			
		||||
static int dw_i2c_probe(struct platform_device *pdev)
 | 
			
		||||
@ -122,7 +142,9 @@ static int dw_i2c_probe(struct platform_device *pdev)
 | 
			
		||||
	struct dw_i2c_dev *dev;
 | 
			
		||||
	struct i2c_adapter *adap;
 | 
			
		||||
	struct resource *mem;
 | 
			
		||||
	struct dw_i2c_platform_data *pdata;
 | 
			
		||||
	int irq, r;
 | 
			
		||||
	u32 clk_freq, ht = 0;
 | 
			
		||||
 | 
			
		||||
	irq = platform_get_irq(pdev, 0);
 | 
			
		||||
	if (irq < 0) {
 | 
			
		||||
@ -145,21 +167,14 @@ static int dw_i2c_probe(struct platform_device *pdev)
 | 
			
		||||
	dev->irq = irq;
 | 
			
		||||
	platform_set_drvdata(pdev, dev);
 | 
			
		||||
 | 
			
		||||
	dev->clk = devm_clk_get(&pdev->dev, NULL);
 | 
			
		||||
	dev->get_clk_rate_khz = i2c_dw_get_clk_rate_khz;
 | 
			
		||||
 | 
			
		||||
	if (IS_ERR(dev->clk))
 | 
			
		||||
		return PTR_ERR(dev->clk);
 | 
			
		||||
	clk_prepare_enable(dev->clk);
 | 
			
		||||
 | 
			
		||||
	if (pdev->dev.of_node) {
 | 
			
		||||
		u32 ht = 0;
 | 
			
		||||
		u32 ic_clk = dev->get_clk_rate_khz(dev);
 | 
			
		||||
	/* fast mode by default because of legacy reasons */
 | 
			
		||||
	clk_freq = 400000;
 | 
			
		||||
 | 
			
		||||
	if (ACPI_COMPANION(&pdev->dev)) {
 | 
			
		||||
		dw_i2c_acpi_configure(pdev);
 | 
			
		||||
	} else if (pdev->dev.of_node) {
 | 
			
		||||
		of_property_read_u32(pdev->dev.of_node,
 | 
			
		||||
					"i2c-sda-hold-time-ns", &ht);
 | 
			
		||||
		dev->sda_hold_time = div_u64((u64)ic_clk * ht + 500000,
 | 
			
		||||
					     1000000);
 | 
			
		||||
 | 
			
		||||
		of_property_read_u32(pdev->dev.of_node,
 | 
			
		||||
				     "i2c-sda-falling-time-ns",
 | 
			
		||||
@ -167,6 +182,21 @@ static int dw_i2c_probe(struct platform_device *pdev)
 | 
			
		||||
		of_property_read_u32(pdev->dev.of_node,
 | 
			
		||||
				     "i2c-scl-falling-time-ns",
 | 
			
		||||
				     &dev->scl_falling_time);
 | 
			
		||||
 | 
			
		||||
		of_property_read_u32(pdev->dev.of_node, "clock-frequency",
 | 
			
		||||
				     &clk_freq);
 | 
			
		||||
 | 
			
		||||
		/* Only standard mode at 100kHz and fast mode at 400kHz
 | 
			
		||||
		 * are supported.
 | 
			
		||||
		 */
 | 
			
		||||
		if (clk_freq != 100000 && clk_freq != 400000) {
 | 
			
		||||
			dev_err(&pdev->dev, "Only 100kHz and 400kHz supported");
 | 
			
		||||
			return -EINVAL;
 | 
			
		||||
		}
 | 
			
		||||
	} else {
 | 
			
		||||
		pdata = dev_get_platdata(&pdev->dev);
 | 
			
		||||
		if (pdata)
 | 
			
		||||
			clk_freq = pdata->i2c_scl_freq;
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	dev->functionality =
 | 
			
		||||
@ -176,12 +206,27 @@ static int dw_i2c_probe(struct platform_device *pdev)
 | 
			
		||||
		I2C_FUNC_SMBUS_BYTE_DATA |
 | 
			
		||||
		I2C_FUNC_SMBUS_WORD_DATA |
 | 
			
		||||
		I2C_FUNC_SMBUS_I2C_BLOCK;
 | 
			
		||||
	dev->master_cfg =  DW_IC_CON_MASTER | DW_IC_CON_SLAVE_DISABLE |
 | 
			
		||||
		DW_IC_CON_RESTART_EN | DW_IC_CON_SPEED_FAST;
 | 
			
		||||
	if (clk_freq == 100000)
 | 
			
		||||
		dev->master_cfg =  DW_IC_CON_MASTER | DW_IC_CON_SLAVE_DISABLE |
 | 
			
		||||
			DW_IC_CON_RESTART_EN | DW_IC_CON_SPEED_STD;
 | 
			
		||||
	else
 | 
			
		||||
		dev->master_cfg =  DW_IC_CON_MASTER | DW_IC_CON_SLAVE_DISABLE |
 | 
			
		||||
			DW_IC_CON_RESTART_EN | DW_IC_CON_SPEED_FAST;
 | 
			
		||||
 | 
			
		||||
	/* Try first if we can configure the device from ACPI */
 | 
			
		||||
	r = dw_i2c_acpi_configure(pdev);
 | 
			
		||||
	if (r) {
 | 
			
		||||
	dev->clk = devm_clk_get(&pdev->dev, NULL);
 | 
			
		||||
	dev->get_clk_rate_khz = i2c_dw_get_clk_rate_khz;
 | 
			
		||||
	if (IS_ERR(dev->clk))
 | 
			
		||||
		return PTR_ERR(dev->clk);
 | 
			
		||||
	clk_prepare_enable(dev->clk);
 | 
			
		||||
 | 
			
		||||
	if (!dev->sda_hold_time && ht) {
 | 
			
		||||
		u32 ic_clk = dev->get_clk_rate_khz(dev);
 | 
			
		||||
 | 
			
		||||
		dev->sda_hold_time = div_u64((u64)ic_clk * ht + 500000,
 | 
			
		||||
					     1000000);
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	if (!dev->tx_fifo_depth) {
 | 
			
		||||
		u32 param1 = i2c_dw_read_comp_param(dev);
 | 
			
		||||
 | 
			
		||||
		dev->tx_fifo_depth = ((param1 >> 16) & 0xff) + 1;
 | 
			
		||||
@ -237,6 +282,9 @@ static int dw_i2c_remove(struct platform_device *pdev)
 | 
			
		||||
	pm_runtime_put(&pdev->dev);
 | 
			
		||||
	pm_runtime_disable(&pdev->dev);
 | 
			
		||||
 | 
			
		||||
	if (ACPI_COMPANION(&pdev->dev))
 | 
			
		||||
		dw_i2c_acpi_unconfigure(pdev);
 | 
			
		||||
 | 
			
		||||
	return 0;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@ -83,7 +83,6 @@
 | 
			
		||||
#define HSI2C_INT_TX_ALMOSTEMPTY_EN		(1u << 0)
 | 
			
		||||
#define HSI2C_INT_RX_ALMOSTFULL_EN		(1u << 1)
 | 
			
		||||
#define HSI2C_INT_TRAILING_EN			(1u << 6)
 | 
			
		||||
#define HSI2C_INT_I2C_EN			(1u << 9)
 | 
			
		||||
 | 
			
		||||
/* I2C_INT_STAT Register bits */
 | 
			
		||||
#define HSI2C_INT_TX_ALMOSTEMPTY		(1u << 0)
 | 
			
		||||
@ -95,6 +94,17 @@
 | 
			
		||||
#define HSI2C_INT_TRAILING			(1u << 6)
 | 
			
		||||
#define HSI2C_INT_I2C				(1u << 9)
 | 
			
		||||
 | 
			
		||||
#define HSI2C_INT_TRANS_DONE			(1u << 7)
 | 
			
		||||
#define HSI2C_INT_TRANS_ABORT			(1u << 8)
 | 
			
		||||
#define HSI2C_INT_NO_DEV_ACK			(1u << 9)
 | 
			
		||||
#define HSI2C_INT_NO_DEV			(1u << 10)
 | 
			
		||||
#define HSI2C_INT_TIMEOUT			(1u << 11)
 | 
			
		||||
#define HSI2C_INT_I2C_TRANS			(HSI2C_INT_TRANS_DONE |	\
 | 
			
		||||
						HSI2C_INT_TRANS_ABORT |	\
 | 
			
		||||
						HSI2C_INT_NO_DEV_ACK |	\
 | 
			
		||||
						HSI2C_INT_NO_DEV |	\
 | 
			
		||||
						HSI2C_INT_TIMEOUT)
 | 
			
		||||
 | 
			
		||||
/* I2C_FIFO_STAT Register bits */
 | 
			
		||||
#define HSI2C_RX_FIFO_EMPTY			(1u << 24)
 | 
			
		||||
#define HSI2C_RX_FIFO_FULL			(1u << 23)
 | 
			
		||||
@ -143,6 +153,8 @@
 | 
			
		||||
 | 
			
		||||
#define EXYNOS5_I2C_TIMEOUT (msecs_to_jiffies(1000))
 | 
			
		||||
 | 
			
		||||
#define HSI2C_EXYNOS7	BIT(0)
 | 
			
		||||
 | 
			
		||||
struct exynos5_i2c {
 | 
			
		||||
	struct i2c_adapter	adap;
 | 
			
		||||
	unsigned int		suspended:1;
 | 
			
		||||
@ -192,6 +204,7 @@ struct exynos5_i2c {
 | 
			
		||||
 */
 | 
			
		||||
struct exynos_hsi2c_variant {
 | 
			
		||||
	unsigned int	fifo_depth;
 | 
			
		||||
	unsigned int	hw;
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
static const struct exynos_hsi2c_variant exynos5250_hsi2c_data = {
 | 
			
		||||
@ -202,6 +215,11 @@ static const struct exynos_hsi2c_variant exynos5260_hsi2c_data = {
 | 
			
		||||
	.fifo_depth	= 16,
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
static const struct exynos_hsi2c_variant exynos7_hsi2c_data = {
 | 
			
		||||
	.fifo_depth	= 16,
 | 
			
		||||
	.hw		= HSI2C_EXYNOS7,
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
static const struct of_device_id exynos5_i2c_match[] = {
 | 
			
		||||
	{
 | 
			
		||||
		.compatible = "samsung,exynos5-hsi2c",
 | 
			
		||||
@ -212,6 +230,9 @@ static const struct of_device_id exynos5_i2c_match[] = {
 | 
			
		||||
	}, {
 | 
			
		||||
		.compatible = "samsung,exynos5260-hsi2c",
 | 
			
		||||
		.data = &exynos5260_hsi2c_data
 | 
			
		||||
	}, {
 | 
			
		||||
		.compatible = "samsung,exynos7-hsi2c",
 | 
			
		||||
		.data = &exynos7_hsi2c_data
 | 
			
		||||
	}, {},
 | 
			
		||||
};
 | 
			
		||||
MODULE_DEVICE_TABLE(of, exynos5_i2c_match);
 | 
			
		||||
@ -256,13 +277,24 @@ static int exynos5_i2c_set_timing(struct exynos5_i2c *i2c, int mode)
 | 
			
		||||
				i2c->hs_clock : i2c->fs_clock;
 | 
			
		||||
 | 
			
		||||
	/*
 | 
			
		||||
	 * In case of HSI2C controller in Exynos5 series
 | 
			
		||||
	 * FPCLK / FI2C =
 | 
			
		||||
	 * (CLK_DIV + 1) * (TSCLK_L + TSCLK_H + 2) + 8 + 2 * FLT_CYCLE
 | 
			
		||||
	 *
 | 
			
		||||
	 * In case of HSI2C controllers in Exynos7 series
 | 
			
		||||
	 * FPCLK / FI2C =
 | 
			
		||||
	 * (CLK_DIV + 1) * (TSCLK_L + TSCLK_H + 2) + 8 + FLT_CYCLE
 | 
			
		||||
	 *
 | 
			
		||||
	 * utemp0 = (CLK_DIV + 1) * (TSCLK_L + TSCLK_H + 2)
 | 
			
		||||
	 * utemp1 = (TSCLK_L + TSCLK_H + 2)
 | 
			
		||||
	 */
 | 
			
		||||
	t_ftl_cycle = (readl(i2c->regs + HSI2C_CONF) >> 16) & 0x7;
 | 
			
		||||
	utemp0 = (clkin / op_clk) - 8 - 2 * t_ftl_cycle;
 | 
			
		||||
	utemp0 = (clkin / op_clk) - 8;
 | 
			
		||||
 | 
			
		||||
	if (i2c->variant->hw == HSI2C_EXYNOS7)
 | 
			
		||||
		utemp0 -= t_ftl_cycle;
 | 
			
		||||
	else
 | 
			
		||||
		utemp0 -= 2 * t_ftl_cycle;
 | 
			
		||||
 | 
			
		||||
	/* CLK_DIV max is 256 */
 | 
			
		||||
	for (div = 0; div < 256; div++) {
 | 
			
		||||
@ -407,7 +439,28 @@ static irqreturn_t exynos5_i2c_irq(int irqno, void *dev_id)
 | 
			
		||||
	writel(int_status, i2c->regs + HSI2C_INT_STATUS);
 | 
			
		||||
 | 
			
		||||
	/* handle interrupt related to the transfer status */
 | 
			
		||||
	if (int_status & HSI2C_INT_I2C) {
 | 
			
		||||
	if (i2c->variant->hw == HSI2C_EXYNOS7) {
 | 
			
		||||
		if (int_status & HSI2C_INT_TRANS_DONE) {
 | 
			
		||||
			i2c->trans_done = 1;
 | 
			
		||||
			i2c->state = 0;
 | 
			
		||||
		} else if (int_status & HSI2C_INT_TRANS_ABORT) {
 | 
			
		||||
			dev_dbg(i2c->dev, "Deal with arbitration lose\n");
 | 
			
		||||
			i2c->state = -EAGAIN;
 | 
			
		||||
			goto stop;
 | 
			
		||||
		} else if (int_status & HSI2C_INT_NO_DEV_ACK) {
 | 
			
		||||
			dev_dbg(i2c->dev, "No ACK from device\n");
 | 
			
		||||
			i2c->state = -ENXIO;
 | 
			
		||||
			goto stop;
 | 
			
		||||
		} else if (int_status & HSI2C_INT_NO_DEV) {
 | 
			
		||||
			dev_dbg(i2c->dev, "No device\n");
 | 
			
		||||
			i2c->state = -ENXIO;
 | 
			
		||||
			goto stop;
 | 
			
		||||
		} else if (int_status & HSI2C_INT_TIMEOUT) {
 | 
			
		||||
			dev_dbg(i2c->dev, "Accessing device timed out\n");
 | 
			
		||||
			i2c->state = -EAGAIN;
 | 
			
		||||
			goto stop;
 | 
			
		||||
		}
 | 
			
		||||
	} else if (int_status & HSI2C_INT_I2C) {
 | 
			
		||||
		trans_status = readl(i2c->regs + HSI2C_TRANS_STATUS);
 | 
			
		||||
		if (trans_status & HSI2C_NO_DEV_ACK) {
 | 
			
		||||
			dev_dbg(i2c->dev, "No ACK from device\n");
 | 
			
		||||
@ -512,12 +565,17 @@ static int exynos5_i2c_wait_bus_idle(struct exynos5_i2c *i2c)
 | 
			
		||||
static void exynos5_i2c_message_start(struct exynos5_i2c *i2c, int stop)
 | 
			
		||||
{
 | 
			
		||||
	u32 i2c_ctl;
 | 
			
		||||
	u32 int_en = HSI2C_INT_I2C_EN;
 | 
			
		||||
	u32 int_en = 0;
 | 
			
		||||
	u32 i2c_auto_conf = 0;
 | 
			
		||||
	u32 fifo_ctl;
 | 
			
		||||
	unsigned long flags;
 | 
			
		||||
	unsigned short trig_lvl;
 | 
			
		||||
 | 
			
		||||
	if (i2c->variant->hw == HSI2C_EXYNOS7)
 | 
			
		||||
		int_en |= HSI2C_INT_I2C_TRANS;
 | 
			
		||||
	else
 | 
			
		||||
		int_en |= HSI2C_INT_I2C;
 | 
			
		||||
 | 
			
		||||
	i2c_ctl = readl(i2c->regs + HSI2C_CTL);
 | 
			
		||||
	i2c_ctl &= ~(HSI2C_TXCHON | HSI2C_RXCHON);
 | 
			
		||||
	fifo_ctl = HSI2C_RXFIFO_EN | HSI2C_TXFIFO_EN;
 | 
			
		||||
@ -724,12 +782,13 @@ static int exynos5_i2c_probe(struct platform_device *pdev)
 | 
			
		||||
		goto err_clk;
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	/* Need to check the variant before setting up. */
 | 
			
		||||
	i2c->variant = exynos5_i2c_get_variant(pdev);
 | 
			
		||||
 | 
			
		||||
	ret = exynos5_hsi2c_clock_setup(i2c);
 | 
			
		||||
	if (ret)
 | 
			
		||||
		goto err_clk;
 | 
			
		||||
 | 
			
		||||
	i2c->variant = exynos5_i2c_get_variant(pdev);
 | 
			
		||||
 | 
			
		||||
	exynos5_i2c_reset(i2c);
 | 
			
		||||
 | 
			
		||||
	ret = i2c_add_adapter(&i2c->adap);
 | 
			
		||||
 | 
			
		||||
							
								
								
									
										557
									
								
								drivers/i2c/busses/i2c-hix5hd2.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										557
									
								
								drivers/i2c/busses/i2c-hix5hd2.c
									
									
									
									
									
										Normal file
									
								
							@ -0,0 +1,557 @@
 | 
			
		||||
/*
 | 
			
		||||
 * Copyright (c) 2014 Linaro Ltd.
 | 
			
		||||
 * Copyright (c) 2014 Hisilicon Limited.
 | 
			
		||||
 *
 | 
			
		||||
 * 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.
 | 
			
		||||
 *
 | 
			
		||||
 * Now only support 7 bit address.
 | 
			
		||||
 */
 | 
			
		||||
 | 
			
		||||
#include <linux/clk.h>
 | 
			
		||||
#include <linux/delay.h>
 | 
			
		||||
#include <linux/i2c.h>
 | 
			
		||||
#include <linux/io.h>
 | 
			
		||||
#include <linux/interrupt.h>
 | 
			
		||||
#include <linux/module.h>
 | 
			
		||||
#include <linux/of.h>
 | 
			
		||||
#include <linux/platform_device.h>
 | 
			
		||||
#include <linux/pm_runtime.h>
 | 
			
		||||
 | 
			
		||||
/* Register Map */
 | 
			
		||||
#define HIX5I2C_CTRL		0x00
 | 
			
		||||
#define HIX5I2C_COM		0x04
 | 
			
		||||
#define HIX5I2C_ICR		0x08
 | 
			
		||||
#define HIX5I2C_SR		0x0c
 | 
			
		||||
#define HIX5I2C_SCL_H		0x10
 | 
			
		||||
#define HIX5I2C_SCL_L		0x14
 | 
			
		||||
#define HIX5I2C_TXR		0x18
 | 
			
		||||
#define HIX5I2C_RXR		0x1c
 | 
			
		||||
 | 
			
		||||
/* I2C_CTRL_REG */
 | 
			
		||||
#define I2C_ENABLE		BIT(8)
 | 
			
		||||
#define I2C_UNMASK_TOTAL	BIT(7)
 | 
			
		||||
#define I2C_UNMASK_START	BIT(6)
 | 
			
		||||
#define I2C_UNMASK_END		BIT(5)
 | 
			
		||||
#define I2C_UNMASK_SEND		BIT(4)
 | 
			
		||||
#define I2C_UNMASK_RECEIVE	BIT(3)
 | 
			
		||||
#define I2C_UNMASK_ACK		BIT(2)
 | 
			
		||||
#define I2C_UNMASK_ARBITRATE	BIT(1)
 | 
			
		||||
#define I2C_UNMASK_OVER		BIT(0)
 | 
			
		||||
#define I2C_UNMASK_ALL		(I2C_UNMASK_ACK | I2C_UNMASK_OVER)
 | 
			
		||||
 | 
			
		||||
/* I2C_COM_REG */
 | 
			
		||||
#define I2C_NO_ACK		BIT(4)
 | 
			
		||||
#define I2C_START		BIT(3)
 | 
			
		||||
#define I2C_READ		BIT(2)
 | 
			
		||||
#define I2C_WRITE		BIT(1)
 | 
			
		||||
#define I2C_STOP		BIT(0)
 | 
			
		||||
 | 
			
		||||
/* I2C_ICR_REG */
 | 
			
		||||
#define I2C_CLEAR_START		BIT(6)
 | 
			
		||||
#define I2C_CLEAR_END		BIT(5)
 | 
			
		||||
#define I2C_CLEAR_SEND		BIT(4)
 | 
			
		||||
#define I2C_CLEAR_RECEIVE	BIT(3)
 | 
			
		||||
#define I2C_CLEAR_ACK		BIT(2)
 | 
			
		||||
#define I2C_CLEAR_ARBITRATE	BIT(1)
 | 
			
		||||
#define I2C_CLEAR_OVER		BIT(0)
 | 
			
		||||
#define I2C_CLEAR_ALL		(I2C_CLEAR_START | I2C_CLEAR_END | \
 | 
			
		||||
				I2C_CLEAR_SEND | I2C_CLEAR_RECEIVE | \
 | 
			
		||||
				I2C_CLEAR_ACK | I2C_CLEAR_ARBITRATE | \
 | 
			
		||||
				I2C_CLEAR_OVER)
 | 
			
		||||
 | 
			
		||||
/* I2C_SR_REG */
 | 
			
		||||
#define I2C_BUSY		BIT(7)
 | 
			
		||||
#define I2C_START_INTR		BIT(6)
 | 
			
		||||
#define I2C_END_INTR		BIT(5)
 | 
			
		||||
#define I2C_SEND_INTR		BIT(4)
 | 
			
		||||
#define I2C_RECEIVE_INTR	BIT(3)
 | 
			
		||||
#define I2C_ACK_INTR		BIT(2)
 | 
			
		||||
#define I2C_ARBITRATE_INTR	BIT(1)
 | 
			
		||||
#define I2C_OVER_INTR		BIT(0)
 | 
			
		||||
 | 
			
		||||
#define HIX5I2C_MAX_FREQ	400000		/* 400k */
 | 
			
		||||
#define HIX5I2C_READ_OPERATION	0x01
 | 
			
		||||
 | 
			
		||||
enum hix5hd2_i2c_state {
 | 
			
		||||
	HIX5I2C_STAT_RW_ERR = -1,
 | 
			
		||||
	HIX5I2C_STAT_INIT,
 | 
			
		||||
	HIX5I2C_STAT_RW,
 | 
			
		||||
	HIX5I2C_STAT_SND_STOP,
 | 
			
		||||
	HIX5I2C_STAT_RW_SUCCESS,
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
struct hix5hd2_i2c_priv {
 | 
			
		||||
	struct i2c_adapter adap;
 | 
			
		||||
	struct i2c_msg *msg;
 | 
			
		||||
	struct completion msg_complete;
 | 
			
		||||
	unsigned int msg_idx;
 | 
			
		||||
	unsigned int msg_len;
 | 
			
		||||
	int stop;
 | 
			
		||||
	void __iomem *regs;
 | 
			
		||||
	struct clk *clk;
 | 
			
		||||
	struct device *dev;
 | 
			
		||||
	spinlock_t lock;	/* IRQ synchronization */
 | 
			
		||||
	int err;
 | 
			
		||||
	unsigned int freq;
 | 
			
		||||
	enum hix5hd2_i2c_state state;
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
static u32 hix5hd2_i2c_clr_pend_irq(struct hix5hd2_i2c_priv *priv)
 | 
			
		||||
{
 | 
			
		||||
	u32 val = readl_relaxed(priv->regs + HIX5I2C_SR);
 | 
			
		||||
 | 
			
		||||
	writel_relaxed(val, priv->regs + HIX5I2C_ICR);
 | 
			
		||||
 | 
			
		||||
	return val;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static void hix5hd2_i2c_clr_all_irq(struct hix5hd2_i2c_priv *priv)
 | 
			
		||||
{
 | 
			
		||||
	writel_relaxed(I2C_CLEAR_ALL, priv->regs + HIX5I2C_ICR);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static void hix5hd2_i2c_disable_irq(struct hix5hd2_i2c_priv *priv)
 | 
			
		||||
{
 | 
			
		||||
	writel_relaxed(0, priv->regs + HIX5I2C_CTRL);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static void hix5hd2_i2c_enable_irq(struct hix5hd2_i2c_priv *priv)
 | 
			
		||||
{
 | 
			
		||||
	writel_relaxed(I2C_ENABLE | I2C_UNMASK_TOTAL | I2C_UNMASK_ALL,
 | 
			
		||||
		       priv->regs + HIX5I2C_CTRL);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static void hix5hd2_i2c_drv_setrate(struct hix5hd2_i2c_priv *priv)
 | 
			
		||||
{
 | 
			
		||||
	u32 rate, val;
 | 
			
		||||
	u32 scl, sysclock;
 | 
			
		||||
 | 
			
		||||
	/* close all i2c interrupt */
 | 
			
		||||
	val = readl_relaxed(priv->regs + HIX5I2C_CTRL);
 | 
			
		||||
	writel_relaxed(val & (~I2C_UNMASK_TOTAL), priv->regs + HIX5I2C_CTRL);
 | 
			
		||||
 | 
			
		||||
	rate = priv->freq;
 | 
			
		||||
	sysclock = clk_get_rate(priv->clk);
 | 
			
		||||
	scl = (sysclock / (rate * 2)) / 2 - 1;
 | 
			
		||||
	writel_relaxed(scl, priv->regs + HIX5I2C_SCL_H);
 | 
			
		||||
	writel_relaxed(scl, priv->regs + HIX5I2C_SCL_L);
 | 
			
		||||
 | 
			
		||||
	/* restore original interrupt*/
 | 
			
		||||
	writel_relaxed(val, priv->regs + HIX5I2C_CTRL);
 | 
			
		||||
 | 
			
		||||
	dev_dbg(priv->dev, "%s: sysclock=%d, rate=%d, scl=%d\n",
 | 
			
		||||
		__func__, sysclock, rate, scl);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static void hix5hd2_i2c_init(struct hix5hd2_i2c_priv *priv)
 | 
			
		||||
{
 | 
			
		||||
	hix5hd2_i2c_disable_irq(priv);
 | 
			
		||||
	hix5hd2_i2c_drv_setrate(priv);
 | 
			
		||||
	hix5hd2_i2c_clr_all_irq(priv);
 | 
			
		||||
	hix5hd2_i2c_enable_irq(priv);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static void hix5hd2_i2c_reset(struct hix5hd2_i2c_priv *priv)
 | 
			
		||||
{
 | 
			
		||||
	clk_disable_unprepare(priv->clk);
 | 
			
		||||
	msleep(20);
 | 
			
		||||
	clk_prepare_enable(priv->clk);
 | 
			
		||||
	hix5hd2_i2c_init(priv);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static int hix5hd2_i2c_wait_bus_idle(struct hix5hd2_i2c_priv *priv)
 | 
			
		||||
{
 | 
			
		||||
	unsigned long stop_time;
 | 
			
		||||
	u32 int_status;
 | 
			
		||||
 | 
			
		||||
	/* wait for 100 milli seconds for the bus to be idle */
 | 
			
		||||
	stop_time = jiffies + msecs_to_jiffies(100);
 | 
			
		||||
	do {
 | 
			
		||||
		int_status = hix5hd2_i2c_clr_pend_irq(priv);
 | 
			
		||||
		if (!(int_status & I2C_BUSY))
 | 
			
		||||
			return 0;
 | 
			
		||||
 | 
			
		||||
		usleep_range(50, 200);
 | 
			
		||||
	} while (time_before(jiffies, stop_time));
 | 
			
		||||
 | 
			
		||||
	return -EBUSY;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static void hix5hd2_rw_over(struct hix5hd2_i2c_priv *priv)
 | 
			
		||||
{
 | 
			
		||||
	if (priv->state == HIX5I2C_STAT_SND_STOP)
 | 
			
		||||
		dev_dbg(priv->dev, "%s: rw and send stop over\n", __func__);
 | 
			
		||||
	else
 | 
			
		||||
		dev_dbg(priv->dev, "%s: have not data to send\n", __func__);
 | 
			
		||||
 | 
			
		||||
	priv->state = HIX5I2C_STAT_RW_SUCCESS;
 | 
			
		||||
	priv->err = 0;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static void hix5hd2_rw_handle_stop(struct hix5hd2_i2c_priv *priv)
 | 
			
		||||
{
 | 
			
		||||
	if (priv->stop) {
 | 
			
		||||
		priv->state = HIX5I2C_STAT_SND_STOP;
 | 
			
		||||
		writel_relaxed(I2C_STOP, priv->regs + HIX5I2C_COM);
 | 
			
		||||
	} else {
 | 
			
		||||
		hix5hd2_rw_over(priv);
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static void hix5hd2_read_handle(struct hix5hd2_i2c_priv *priv)
 | 
			
		||||
{
 | 
			
		||||
	if (priv->msg_len == 1) {
 | 
			
		||||
		/* the last byte don't need send ACK */
 | 
			
		||||
		writel_relaxed(I2C_READ | I2C_NO_ACK, priv->regs + HIX5I2C_COM);
 | 
			
		||||
	} else if (priv->msg_len > 1) {
 | 
			
		||||
		/* if i2c master receive data will send ACK */
 | 
			
		||||
		writel_relaxed(I2C_READ, priv->regs + HIX5I2C_COM);
 | 
			
		||||
	} else {
 | 
			
		||||
		hix5hd2_rw_handle_stop(priv);
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static void hix5hd2_write_handle(struct hix5hd2_i2c_priv *priv)
 | 
			
		||||
{
 | 
			
		||||
	u8 data;
 | 
			
		||||
 | 
			
		||||
	if (priv->msg_len > 0) {
 | 
			
		||||
		data = priv->msg->buf[priv->msg_idx++];
 | 
			
		||||
		writel_relaxed(data, priv->regs + HIX5I2C_TXR);
 | 
			
		||||
		writel_relaxed(I2C_WRITE, priv->regs + HIX5I2C_COM);
 | 
			
		||||
	} else {
 | 
			
		||||
		hix5hd2_rw_handle_stop(priv);
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static int hix5hd2_rw_preprocess(struct hix5hd2_i2c_priv *priv)
 | 
			
		||||
{
 | 
			
		||||
	u8 data;
 | 
			
		||||
 | 
			
		||||
	if (priv->state == HIX5I2C_STAT_INIT) {
 | 
			
		||||
		priv->state = HIX5I2C_STAT_RW;
 | 
			
		||||
	} else if (priv->state == HIX5I2C_STAT_RW) {
 | 
			
		||||
		if (priv->msg->flags & I2C_M_RD) {
 | 
			
		||||
			data = readl_relaxed(priv->regs + HIX5I2C_RXR);
 | 
			
		||||
			priv->msg->buf[priv->msg_idx++] = data;
 | 
			
		||||
		}
 | 
			
		||||
		priv->msg_len--;
 | 
			
		||||
	} else {
 | 
			
		||||
		dev_dbg(priv->dev, "%s: error: priv->state = %d, msg_len = %d\n",
 | 
			
		||||
			__func__, priv->state, priv->msg_len);
 | 
			
		||||
		return -EAGAIN;
 | 
			
		||||
	}
 | 
			
		||||
	return 0;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static irqreturn_t hix5hd2_i2c_irq(int irqno, void *dev_id)
 | 
			
		||||
{
 | 
			
		||||
	struct hix5hd2_i2c_priv *priv = dev_id;
 | 
			
		||||
	u32 int_status;
 | 
			
		||||
	int ret;
 | 
			
		||||
 | 
			
		||||
	spin_lock(&priv->lock);
 | 
			
		||||
 | 
			
		||||
	int_status = hix5hd2_i2c_clr_pend_irq(priv);
 | 
			
		||||
 | 
			
		||||
	/* handle error */
 | 
			
		||||
	if (int_status & I2C_ARBITRATE_INTR) {
 | 
			
		||||
		/* bus error */
 | 
			
		||||
		dev_dbg(priv->dev, "ARB bus loss\n");
 | 
			
		||||
		priv->err = -EAGAIN;
 | 
			
		||||
		priv->state = HIX5I2C_STAT_RW_ERR;
 | 
			
		||||
		goto stop;
 | 
			
		||||
	} else if (int_status & I2C_ACK_INTR) {
 | 
			
		||||
		/* ack error */
 | 
			
		||||
		dev_dbg(priv->dev, "No ACK from device\n");
 | 
			
		||||
		priv->err = -ENXIO;
 | 
			
		||||
		priv->state = HIX5I2C_STAT_RW_ERR;
 | 
			
		||||
		goto stop;
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	if (int_status & I2C_OVER_INTR) {
 | 
			
		||||
		if (priv->msg_len > 0) {
 | 
			
		||||
			ret = hix5hd2_rw_preprocess(priv);
 | 
			
		||||
			if (ret) {
 | 
			
		||||
				priv->err = ret;
 | 
			
		||||
				priv->state = HIX5I2C_STAT_RW_ERR;
 | 
			
		||||
				goto stop;
 | 
			
		||||
			}
 | 
			
		||||
			if (priv->msg->flags & I2C_M_RD)
 | 
			
		||||
				hix5hd2_read_handle(priv);
 | 
			
		||||
			else
 | 
			
		||||
				hix5hd2_write_handle(priv);
 | 
			
		||||
		} else {
 | 
			
		||||
			hix5hd2_rw_over(priv);
 | 
			
		||||
		}
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
stop:
 | 
			
		||||
	if ((priv->state == HIX5I2C_STAT_RW_SUCCESS &&
 | 
			
		||||
	     priv->msg->len == priv->msg_idx) ||
 | 
			
		||||
	    (priv->state == HIX5I2C_STAT_RW_ERR)) {
 | 
			
		||||
		hix5hd2_i2c_disable_irq(priv);
 | 
			
		||||
		hix5hd2_i2c_clr_pend_irq(priv);
 | 
			
		||||
		complete(&priv->msg_complete);
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	spin_unlock(&priv->lock);
 | 
			
		||||
 | 
			
		||||
	return IRQ_HANDLED;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static void hix5hd2_i2c_message_start(struct hix5hd2_i2c_priv *priv, int stop)
 | 
			
		||||
{
 | 
			
		||||
	unsigned long flags;
 | 
			
		||||
 | 
			
		||||
	spin_lock_irqsave(&priv->lock, flags);
 | 
			
		||||
	hix5hd2_i2c_clr_all_irq(priv);
 | 
			
		||||
	hix5hd2_i2c_enable_irq(priv);
 | 
			
		||||
 | 
			
		||||
	if (priv->msg->flags & I2C_M_RD)
 | 
			
		||||
		writel_relaxed((priv->msg->addr << 1) | HIX5I2C_READ_OPERATION,
 | 
			
		||||
			       priv->regs + HIX5I2C_TXR);
 | 
			
		||||
	else
 | 
			
		||||
		writel_relaxed(priv->msg->addr << 1,
 | 
			
		||||
			       priv->regs + HIX5I2C_TXR);
 | 
			
		||||
 | 
			
		||||
	writel_relaxed(I2C_WRITE | I2C_START, priv->regs + HIX5I2C_COM);
 | 
			
		||||
	spin_unlock_irqrestore(&priv->lock, flags);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static int hix5hd2_i2c_xfer_msg(struct hix5hd2_i2c_priv *priv,
 | 
			
		||||
				struct i2c_msg *msgs, int stop)
 | 
			
		||||
{
 | 
			
		||||
	unsigned long timeout;
 | 
			
		||||
	int ret;
 | 
			
		||||
 | 
			
		||||
	priv->msg = msgs;
 | 
			
		||||
	priv->msg_idx = 0;
 | 
			
		||||
	priv->msg_len = priv->msg->len;
 | 
			
		||||
	priv->stop = stop;
 | 
			
		||||
	priv->err = 0;
 | 
			
		||||
	priv->state = HIX5I2C_STAT_INIT;
 | 
			
		||||
 | 
			
		||||
	reinit_completion(&priv->msg_complete);
 | 
			
		||||
	hix5hd2_i2c_message_start(priv, stop);
 | 
			
		||||
 | 
			
		||||
	timeout = wait_for_completion_timeout(&priv->msg_complete,
 | 
			
		||||
					      priv->adap.timeout);
 | 
			
		||||
	if (timeout == 0) {
 | 
			
		||||
		priv->state = HIX5I2C_STAT_RW_ERR;
 | 
			
		||||
		priv->err = -ETIMEDOUT;
 | 
			
		||||
		dev_warn(priv->dev, "%s timeout=%d\n",
 | 
			
		||||
			 msgs->flags & I2C_M_RD ? "rx" : "tx",
 | 
			
		||||
			 priv->adap.timeout);
 | 
			
		||||
	}
 | 
			
		||||
	ret = priv->state;
 | 
			
		||||
 | 
			
		||||
	/*
 | 
			
		||||
	 * If this is the last message to be transfered (stop == 1)
 | 
			
		||||
	 * Then check if the bus can be brought back to idle.
 | 
			
		||||
	 */
 | 
			
		||||
	if (priv->state == HIX5I2C_STAT_RW_SUCCESS && stop)
 | 
			
		||||
		ret = hix5hd2_i2c_wait_bus_idle(priv);
 | 
			
		||||
 | 
			
		||||
	if (ret < 0)
 | 
			
		||||
		hix5hd2_i2c_reset(priv);
 | 
			
		||||
 | 
			
		||||
	return priv->err;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static int hix5hd2_i2c_xfer(struct i2c_adapter *adap,
 | 
			
		||||
			    struct i2c_msg *msgs, int num)
 | 
			
		||||
{
 | 
			
		||||
	struct hix5hd2_i2c_priv *priv = i2c_get_adapdata(adap);
 | 
			
		||||
	int i, ret, stop;
 | 
			
		||||
 | 
			
		||||
	pm_runtime_get_sync(priv->dev);
 | 
			
		||||
 | 
			
		||||
	for (i = 0; i < num; i++, msgs++) {
 | 
			
		||||
		stop = (i == num - 1);
 | 
			
		||||
		ret = hix5hd2_i2c_xfer_msg(priv, msgs, stop);
 | 
			
		||||
		if (ret < 0)
 | 
			
		||||
			goto out;
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	if (i == num) {
 | 
			
		||||
		ret = num;
 | 
			
		||||
	} else {
 | 
			
		||||
		/* Only one message, cannot access the device */
 | 
			
		||||
		if (i == 1)
 | 
			
		||||
			ret = -EREMOTEIO;
 | 
			
		||||
		else
 | 
			
		||||
			ret = i;
 | 
			
		||||
 | 
			
		||||
		dev_warn(priv->dev, "xfer message failed\n");
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
out:
 | 
			
		||||
	pm_runtime_mark_last_busy(priv->dev);
 | 
			
		||||
	pm_runtime_put_autosuspend(priv->dev);
 | 
			
		||||
	return ret;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static u32 hix5hd2_i2c_func(struct i2c_adapter *adap)
 | 
			
		||||
{
 | 
			
		||||
	return I2C_FUNC_I2C | (I2C_FUNC_SMBUS_EMUL & ~I2C_FUNC_SMBUS_QUICK);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static const struct i2c_algorithm hix5hd2_i2c_algorithm = {
 | 
			
		||||
	.master_xfer		= hix5hd2_i2c_xfer,
 | 
			
		||||
	.functionality		= hix5hd2_i2c_func,
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
static int hix5hd2_i2c_probe(struct platform_device *pdev)
 | 
			
		||||
{
 | 
			
		||||
	struct device_node *np = pdev->dev.of_node;
 | 
			
		||||
	struct hix5hd2_i2c_priv *priv;
 | 
			
		||||
	struct resource *mem;
 | 
			
		||||
	unsigned int freq;
 | 
			
		||||
	int irq, ret;
 | 
			
		||||
 | 
			
		||||
	priv = devm_kzalloc(&pdev->dev, sizeof(*priv), GFP_KERNEL);
 | 
			
		||||
	if (!priv)
 | 
			
		||||
		return -ENOMEM;
 | 
			
		||||
 | 
			
		||||
	if (of_property_read_u32(np, "clock-frequency", &freq)) {
 | 
			
		||||
		/* use 100k as default value */
 | 
			
		||||
		priv->freq = 100000;
 | 
			
		||||
	} else {
 | 
			
		||||
		if (freq > HIX5I2C_MAX_FREQ) {
 | 
			
		||||
			priv->freq = HIX5I2C_MAX_FREQ;
 | 
			
		||||
			dev_warn(priv->dev, "use max freq %d instead\n",
 | 
			
		||||
				 HIX5I2C_MAX_FREQ);
 | 
			
		||||
		} else {
 | 
			
		||||
			priv->freq = freq;
 | 
			
		||||
		}
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	mem = platform_get_resource(pdev, IORESOURCE_MEM, 0);
 | 
			
		||||
	priv->regs = devm_ioremap_resource(&pdev->dev, mem);
 | 
			
		||||
	if (IS_ERR(priv->regs))
 | 
			
		||||
		return PTR_ERR(priv->regs);
 | 
			
		||||
 | 
			
		||||
	irq = platform_get_irq(pdev, 0);
 | 
			
		||||
	if (irq <= 0) {
 | 
			
		||||
		dev_err(&pdev->dev, "cannot find HS-I2C IRQ\n");
 | 
			
		||||
		return irq;
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	priv->clk = devm_clk_get(&pdev->dev, NULL);
 | 
			
		||||
	if (IS_ERR(priv->clk)) {
 | 
			
		||||
		dev_err(&pdev->dev, "cannot get clock\n");
 | 
			
		||||
		return PTR_ERR(priv->clk);
 | 
			
		||||
	}
 | 
			
		||||
	clk_prepare_enable(priv->clk);
 | 
			
		||||
 | 
			
		||||
	strlcpy(priv->adap.name, "hix5hd2-i2c", sizeof(priv->adap.name));
 | 
			
		||||
	priv->dev = &pdev->dev;
 | 
			
		||||
	priv->adap.owner = THIS_MODULE;
 | 
			
		||||
	priv->adap.algo = &hix5hd2_i2c_algorithm;
 | 
			
		||||
	priv->adap.retries = 3;
 | 
			
		||||
	priv->adap.dev.of_node = np;
 | 
			
		||||
	priv->adap.algo_data = priv;
 | 
			
		||||
	priv->adap.dev.parent = &pdev->dev;
 | 
			
		||||
	i2c_set_adapdata(&priv->adap, priv);
 | 
			
		||||
	platform_set_drvdata(pdev, priv);
 | 
			
		||||
	spin_lock_init(&priv->lock);
 | 
			
		||||
	init_completion(&priv->msg_complete);
 | 
			
		||||
 | 
			
		||||
	hix5hd2_i2c_init(priv);
 | 
			
		||||
 | 
			
		||||
	ret = devm_request_irq(&pdev->dev, irq, hix5hd2_i2c_irq,
 | 
			
		||||
			       IRQF_NO_SUSPEND | IRQF_ONESHOT,
 | 
			
		||||
			       dev_name(&pdev->dev), priv);
 | 
			
		||||
	if (ret != 0) {
 | 
			
		||||
		dev_err(&pdev->dev, "cannot request HS-I2C IRQ %d\n", irq);
 | 
			
		||||
		goto err_clk;
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	pm_suspend_ignore_children(&pdev->dev, true);
 | 
			
		||||
	pm_runtime_set_autosuspend_delay(priv->dev, MSEC_PER_SEC);
 | 
			
		||||
	pm_runtime_use_autosuspend(priv->dev);
 | 
			
		||||
	pm_runtime_set_active(priv->dev);
 | 
			
		||||
	pm_runtime_enable(priv->dev);
 | 
			
		||||
 | 
			
		||||
	ret = i2c_add_adapter(&priv->adap);
 | 
			
		||||
	if (ret < 0) {
 | 
			
		||||
		dev_err(&pdev->dev, "failed to add bus to i2c core\n");
 | 
			
		||||
		goto err_runtime;
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	return ret;
 | 
			
		||||
 | 
			
		||||
err_runtime:
 | 
			
		||||
	pm_runtime_disable(priv->dev);
 | 
			
		||||
	pm_runtime_set_suspended(priv->dev);
 | 
			
		||||
err_clk:
 | 
			
		||||
	clk_disable_unprepare(priv->clk);
 | 
			
		||||
	return ret;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static int hix5hd2_i2c_remove(struct platform_device *pdev)
 | 
			
		||||
{
 | 
			
		||||
	struct hix5hd2_i2c_priv *priv = platform_get_drvdata(pdev);
 | 
			
		||||
 | 
			
		||||
	i2c_del_adapter(&priv->adap);
 | 
			
		||||
	pm_runtime_disable(priv->dev);
 | 
			
		||||
	pm_runtime_set_suspended(priv->dev);
 | 
			
		||||
 | 
			
		||||
	return 0;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
#ifdef CONFIG_PM
 | 
			
		||||
static int hix5hd2_i2c_runtime_suspend(struct device *dev)
 | 
			
		||||
{
 | 
			
		||||
	struct platform_device *pdev = to_platform_device(dev);
 | 
			
		||||
	struct hix5hd2_i2c_priv *priv = platform_get_drvdata(pdev);
 | 
			
		||||
 | 
			
		||||
	clk_disable_unprepare(priv->clk);
 | 
			
		||||
 | 
			
		||||
	return 0;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static int hix5hd2_i2c_runtime_resume(struct device *dev)
 | 
			
		||||
{
 | 
			
		||||
	struct platform_device *pdev = to_platform_device(dev);
 | 
			
		||||
	struct hix5hd2_i2c_priv *priv = platform_get_drvdata(pdev);
 | 
			
		||||
 | 
			
		||||
	clk_prepare_enable(priv->clk);
 | 
			
		||||
	hix5hd2_i2c_init(priv);
 | 
			
		||||
 | 
			
		||||
	return 0;
 | 
			
		||||
}
 | 
			
		||||
#endif
 | 
			
		||||
 | 
			
		||||
static const struct dev_pm_ops hix5hd2_i2c_pm_ops = {
 | 
			
		||||
	SET_PM_RUNTIME_PM_OPS(hix5hd2_i2c_runtime_suspend,
 | 
			
		||||
			      hix5hd2_i2c_runtime_resume,
 | 
			
		||||
			      NULL)
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
static const struct of_device_id hix5hd2_i2c_match[] = {
 | 
			
		||||
	{ .compatible = "hisilicon,hix5hd2-i2c" },
 | 
			
		||||
	{},
 | 
			
		||||
};
 | 
			
		||||
MODULE_DEVICE_TABLE(of, hix5hd2_i2c_match);
 | 
			
		||||
 | 
			
		||||
static struct platform_driver hix5hd2_i2c_driver = {
 | 
			
		||||
	.probe		= hix5hd2_i2c_probe,
 | 
			
		||||
	.remove		= hix5hd2_i2c_remove,
 | 
			
		||||
	.driver		= {
 | 
			
		||||
		.name	= "hix5hd2-i2c",
 | 
			
		||||
		.pm	= &hix5hd2_i2c_pm_ops,
 | 
			
		||||
		.of_match_table = hix5hd2_i2c_match,
 | 
			
		||||
	},
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
module_platform_driver(hix5hd2_i2c_driver);
 | 
			
		||||
 | 
			
		||||
MODULE_DESCRIPTION("Hix5hd2 I2C Bus driver");
 | 
			
		||||
MODULE_AUTHOR("Wei Yan <sledge.yanwei@huawei.com>");
 | 
			
		||||
MODULE_LICENSE("GPL");
 | 
			
		||||
MODULE_ALIAS("platform:i2c-hix5hd2");
 | 
			
		||||
@ -62,6 +62,7 @@
 | 
			
		||||
 * Wildcat Point (PCH)		0x8ca2	32	hard	yes	yes	yes
 | 
			
		||||
 * Wildcat Point-LP (PCH)	0x9ca2	32	hard	yes	yes	yes
 | 
			
		||||
 * BayTrail (SOC)		0x0f12	32	hard	yes	yes	yes
 | 
			
		||||
 * Sunrise Point-H (PCH) 	0xa123  32	hard	yes	yes	yes
 | 
			
		||||
 *
 | 
			
		||||
 * Features supported by this driver:
 | 
			
		||||
 * Software PEC				no
 | 
			
		||||
@ -184,6 +185,7 @@
 | 
			
		||||
#define PCI_DEVICE_ID_INTEL_WELLSBURG_SMBUS_MS2		0x8d7f
 | 
			
		||||
#define PCI_DEVICE_ID_INTEL_LYNXPOINT_LP_SMBUS		0x9c22
 | 
			
		||||
#define PCI_DEVICE_ID_INTEL_WILDCATPOINT_LP_SMBUS	0x9ca2
 | 
			
		||||
#define PCI_DEVICE_ID_INTEL_SUNRISEPOINT_H_SMBUS	0xa123
 | 
			
		||||
 | 
			
		||||
struct i801_mux_config {
 | 
			
		||||
	char *gpio_chip;
 | 
			
		||||
@ -830,6 +832,7 @@ static const struct pci_device_id i801_ids[] = {
 | 
			
		||||
	{ PCI_DEVICE(PCI_VENDOR_ID_INTEL, PCI_DEVICE_ID_INTEL_WILDCATPOINT_LP_SMBUS) },
 | 
			
		||||
	{ PCI_DEVICE(PCI_VENDOR_ID_INTEL, PCI_DEVICE_ID_INTEL_BAYTRAIL_SMBUS) },
 | 
			
		||||
	{ PCI_DEVICE(PCI_VENDOR_ID_INTEL, PCI_DEVICE_ID_INTEL_BRASWELL_SMBUS) },
 | 
			
		||||
	{ PCI_DEVICE(PCI_VENDOR_ID_INTEL, PCI_DEVICE_ID_INTEL_SUNRISEPOINT_H_SMBUS) },
 | 
			
		||||
	{ 0, }
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@ -268,6 +268,14 @@ static int i2c_imx_bus_busy(struct imx_i2c_struct *i2c_imx, int for_busy)
 | 
			
		||||
 | 
			
		||||
	while (1) {
 | 
			
		||||
		temp = imx_i2c_read_reg(i2c_imx, IMX_I2C_I2SR);
 | 
			
		||||
 | 
			
		||||
		/* check for arbitration lost */
 | 
			
		||||
		if (temp & I2SR_IAL) {
 | 
			
		||||
			temp &= ~I2SR_IAL;
 | 
			
		||||
			imx_i2c_write_reg(temp, i2c_imx, IMX_I2C_I2SR);
 | 
			
		||||
			return -EAGAIN;
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		if (for_busy && (temp & I2SR_IBB))
 | 
			
		||||
			break;
 | 
			
		||||
		if (!for_busy && !(temp & I2SR_IBB))
 | 
			
		||||
@ -702,7 +710,7 @@ static int i2c_imx_probe(struct platform_device *pdev)
 | 
			
		||||
				pdev->name, i2c_imx);
 | 
			
		||||
	if (ret) {
 | 
			
		||||
		dev_err(&pdev->dev, "can't claim irq %d\n", irq);
 | 
			
		||||
		return ret;
 | 
			
		||||
		goto clk_disable;
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	/* Init queue */
 | 
			
		||||
@ -727,7 +735,7 @@ static int i2c_imx_probe(struct platform_device *pdev)
 | 
			
		||||
	ret = i2c_add_numbered_adapter(&i2c_imx->adapter);
 | 
			
		||||
	if (ret < 0) {
 | 
			
		||||
		dev_err(&pdev->dev, "registration failed\n");
 | 
			
		||||
		return ret;
 | 
			
		||||
		goto clk_disable;
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	/* Set up platform driver data */
 | 
			
		||||
@ -741,6 +749,10 @@ static int i2c_imx_probe(struct platform_device *pdev)
 | 
			
		||||
	dev_info(&i2c_imx->adapter.dev, "IMX I2C adapter registered\n");
 | 
			
		||||
 | 
			
		||||
	return 0;   /* Return OK */
 | 
			
		||||
 | 
			
		||||
clk_disable:
 | 
			
		||||
	clk_disable_unprepare(i2c_imx->clk);
 | 
			
		||||
	return ret;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static int i2c_imx_remove(struct platform_device *pdev)
 | 
			
		||||
 | 
			
		||||
@ -81,7 +81,7 @@
 | 
			
		||||
#define PCI_DEVICE_ID_INTEL_S1200_SMT1	0x0c5a
 | 
			
		||||
#define PCI_DEVICE_ID_INTEL_AVOTON_SMT	0x1f15
 | 
			
		||||
 | 
			
		||||
#define ISMT_DESC_ENTRIES	32	/* number of descriptor entries */
 | 
			
		||||
#define ISMT_DESC_ENTRIES	2	/* number of descriptor entries */
 | 
			
		||||
#define ISMT_MAX_RETRIES	3	/* number of SMBus retries to attempt */
 | 
			
		||||
 | 
			
		||||
/* Hardware Descriptor Constants - Control Field */
 | 
			
		||||
 | 
			
		||||
@ -307,6 +307,9 @@ static int mxs_i2c_pio_wait_xfer_end(struct mxs_i2c_dev *i2c)
 | 
			
		||||
	unsigned long timeout = jiffies + msecs_to_jiffies(1000);
 | 
			
		||||
 | 
			
		||||
	while (readl(i2c->regs + MXS_I2C_CTRL0) & MXS_I2C_CTRL0_RUN) {
 | 
			
		||||
		if (readl(i2c->regs + MXS_I2C_CTRL1) &
 | 
			
		||||
				MXS_I2C_CTRL1_NO_SLAVE_ACK_IRQ)
 | 
			
		||||
			return -ENXIO;
 | 
			
		||||
		if (time_after(jiffies, timeout))
 | 
			
		||||
			return -ETIMEDOUT;
 | 
			
		||||
		cond_resched();
 | 
			
		||||
 | 
			
		||||
@ -195,7 +195,7 @@ static int rcar_i2c_clock_calculate(struct rcar_i2c_priv *priv,
 | 
			
		||||
	 */
 | 
			
		||||
	rate = clk_get_rate(priv->clk);
 | 
			
		||||
	cdf = rate / 20000000;
 | 
			
		||||
	if (cdf >= 1 << cdf_width) {
 | 
			
		||||
	if (cdf >= 1U << cdf_width) {
 | 
			
		||||
		dev_err(dev, "Input clock %lu too high\n", rate);
 | 
			
		||||
		return -EIO;
 | 
			
		||||
	}
 | 
			
		||||
@ -245,7 +245,7 @@ scgd_find:
 | 
			
		||||
	return 0;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static int rcar_i2c_prepare_msg(struct rcar_i2c_priv *priv)
 | 
			
		||||
static void rcar_i2c_prepare_msg(struct rcar_i2c_priv *priv)
 | 
			
		||||
{
 | 
			
		||||
	int read = !!rcar_i2c_is_recv(priv);
 | 
			
		||||
 | 
			
		||||
@ -253,8 +253,6 @@ static int rcar_i2c_prepare_msg(struct rcar_i2c_priv *priv)
 | 
			
		||||
	rcar_i2c_write(priv, ICMSR, 0);
 | 
			
		||||
	rcar_i2c_write(priv, ICMCR, RCAR_BUS_PHASE_START);
 | 
			
		||||
	rcar_i2c_write(priv, ICMIER, read ? RCAR_IRQ_RECV : RCAR_IRQ_SEND);
 | 
			
		||||
 | 
			
		||||
	return 0;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/*
 | 
			
		||||
@ -365,6 +363,7 @@ static int rcar_i2c_irq_recv(struct rcar_i2c_priv *priv, u32 msr)
 | 
			
		||||
static irqreturn_t rcar_i2c_irq(int irq, void *ptr)
 | 
			
		||||
{
 | 
			
		||||
	struct rcar_i2c_priv *priv = ptr;
 | 
			
		||||
	irqreturn_t result = IRQ_HANDLED;
 | 
			
		||||
	u32 msr;
 | 
			
		||||
 | 
			
		||||
	/*-------------- spin lock -----------------*/
 | 
			
		||||
@ -374,6 +373,10 @@ static irqreturn_t rcar_i2c_irq(int irq, void *ptr)
 | 
			
		||||
 | 
			
		||||
	/* Only handle interrupts that are currently enabled */
 | 
			
		||||
	msr &= rcar_i2c_read(priv, ICMIER);
 | 
			
		||||
	if (!msr) {
 | 
			
		||||
		result = IRQ_NONE;
 | 
			
		||||
		goto exit;
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	/* Arbitration lost */
 | 
			
		||||
	if (msr & MAL) {
 | 
			
		||||
@ -408,10 +411,11 @@ out:
 | 
			
		||||
		wake_up(&priv->wait);
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
exit:
 | 
			
		||||
	spin_unlock(&priv->lock);
 | 
			
		||||
	/*-------------- spin unlock -----------------*/
 | 
			
		||||
 | 
			
		||||
	return IRQ_HANDLED;
 | 
			
		||||
	return result;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static int rcar_i2c_master_xfer(struct i2c_adapter *adap,
 | 
			
		||||
@ -453,17 +457,14 @@ static int rcar_i2c_master_xfer(struct i2c_adapter *adap,
 | 
			
		||||
		priv->msg	= &msgs[i];
 | 
			
		||||
		priv->pos	= 0;
 | 
			
		||||
		priv->flags	= 0;
 | 
			
		||||
		if (priv->msg == &msgs[num - 1])
 | 
			
		||||
		if (i == num - 1)
 | 
			
		||||
			rcar_i2c_flags_set(priv, ID_LAST_MSG);
 | 
			
		||||
 | 
			
		||||
		ret = rcar_i2c_prepare_msg(priv);
 | 
			
		||||
		rcar_i2c_prepare_msg(priv);
 | 
			
		||||
 | 
			
		||||
		spin_unlock_irqrestore(&priv->lock, flags);
 | 
			
		||||
		/*-------------- spin unlock -----------------*/
 | 
			
		||||
 | 
			
		||||
		if (ret < 0)
 | 
			
		||||
			break;
 | 
			
		||||
 | 
			
		||||
		timeout = wait_event_timeout(priv->wait,
 | 
			
		||||
					     rcar_i2c_flags_has(priv, ID_DONE),
 | 
			
		||||
					     5 * HZ);
 | 
			
		||||
 | 
			
		||||
@ -208,7 +208,7 @@ static void rk3x_i2c_prepare_read(struct rk3x_i2c *i2c)
 | 
			
		||||
	 * The hw can read up to 32 bytes at a time. If we need more than one
 | 
			
		||||
	 * chunk, send an ACK after the last byte of the current chunk.
 | 
			
		||||
	 */
 | 
			
		||||
	if (unlikely(len > 32)) {
 | 
			
		||||
	if (len > 32) {
 | 
			
		||||
		len = 32;
 | 
			
		||||
		con &= ~REG_CON_LASTACK;
 | 
			
		||||
	} else {
 | 
			
		||||
@ -403,7 +403,7 @@ static irqreturn_t rk3x_i2c_irq(int irqno, void *dev_id)
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	/* is there anything left to handle? */
 | 
			
		||||
	if (unlikely((ipd & REG_INT_ALL) == 0))
 | 
			
		||||
	if ((ipd & REG_INT_ALL) == 0)
 | 
			
		||||
		goto out;
 | 
			
		||||
 | 
			
		||||
	switch (i2c->state) {
 | 
			
		||||
 | 
			
		||||
@ -1928,9 +1928,6 @@ struct ec_response_power_info {
 | 
			
		||||
 | 
			
		||||
#define EC_CMD_I2C_PASSTHRU 0x9e
 | 
			
		||||
 | 
			
		||||
/* Slave address is 10 (not 7) bit */
 | 
			
		||||
#define EC_I2C_FLAG_10BIT	(1 << 16)
 | 
			
		||||
 | 
			
		||||
/* Read data; if not present, message is a write */
 | 
			
		||||
#define EC_I2C_FLAG_READ	(1 << 15)
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
							
								
								
									
										21
									
								
								include/linux/platform_data/i2c-designware.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										21
									
								
								include/linux/platform_data/i2c-designware.h
									
									
									
									
									
										Normal file
									
								
							@ -0,0 +1,21 @@
 | 
			
		||||
/*
 | 
			
		||||
 * Copyright(c) 2014 Intel Corporation.
 | 
			
		||||
 *
 | 
			
		||||
 * This program is free software; you can redistribute it and/or modify it
 | 
			
		||||
 * under the terms and conditions of the GNU General Public License,
 | 
			
		||||
 * version 2, as published by the Free Software Foundation.
 | 
			
		||||
 *
 | 
			
		||||
 * This program is distributed in the hope 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.
 | 
			
		||||
 */
 | 
			
		||||
 | 
			
		||||
#ifndef I2C_DESIGNWARE_H
 | 
			
		||||
#define I2C_DESIGNWARE_H
 | 
			
		||||
 | 
			
		||||
struct dw_i2c_platform_data {
 | 
			
		||||
	unsigned int i2c_scl_freq;
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
#endif
 | 
			
		||||
		Loading…
	
		Reference in New Issue
	
	Block a user