diff options
Diffstat (limited to 'drivers/i2c')
| -rw-r--r-- | drivers/i2c/Kconfig | 15 | ||||
| -rw-r--r-- | drivers/i2c/Makefile | 2 | ||||
| -rw-r--r-- | drivers/i2c/busses/i2c-at91.c | 32 | ||||
| -rw-r--r-- | drivers/i2c/busses/i2c-i801.c | 2 | ||||
| -rw-r--r-- | drivers/i2c/busses/i2c-mv64xxx.c | 3 | ||||
| -rw-r--r-- | drivers/i2c/busses/i2c-rcar.c | 37 | ||||
| -rw-r--r-- | drivers/i2c/busses/i2c-rk3x.c | 4 | ||||
| -rw-r--r-- | drivers/i2c/i2c-acpi.c | 2 | 
8 files changed, 73 insertions, 24 deletions
| diff --git a/drivers/i2c/Kconfig b/drivers/i2c/Kconfig index 3e3b680dc007..b51a402752c4 100644 --- a/drivers/i2c/Kconfig +++ b/drivers/i2c/Kconfig @@ -23,17 +23,14 @@ config I2C  	  This I2C support can also be built as a module.  If so, the module  	  will be called i2c-core. -config I2C_ACPI -	bool "I2C ACPI support" -	select I2C -	depends on ACPI +config ACPI_I2C_OPREGION +	bool "ACPI I2C Operation region support" +	depends on I2C=y && ACPI  	default y  	help -	  Say Y here if you want to enable ACPI I2C support. This includes support -	  for automatic enumeration of I2C slave devices and support for ACPI I2C -	  Operation Regions. Operation Regions allow firmware (BIOS) code to -	  access I2C slave devices, such as smart batteries through an I2C host -	  controller driver. +	  Say Y here if you want to enable ACPI I2C operation region support. +	  Operation Regions allow firmware (BIOS) code to access I2C slave devices, +	  such as smart batteries through an I2C host controller driver.  if I2C diff --git a/drivers/i2c/Makefile b/drivers/i2c/Makefile index a1f590cbb435..e0228b228256 100644 --- a/drivers/i2c/Makefile +++ b/drivers/i2c/Makefile @@ -3,7 +3,7 @@  #  i2ccore-y := i2c-core.o -i2ccore-$(CONFIG_I2C_ACPI)	+= i2c-acpi.o +i2ccore-$(CONFIG_ACPI)	 	+= i2c-acpi.o  obj-$(CONFIG_I2C_BOARDINFO)	+= i2c-boardinfo.o  obj-$(CONFIG_I2C)		+= i2ccore.o diff --git a/drivers/i2c/busses/i2c-at91.c b/drivers/i2c/busses/i2c-at91.c index 79a68999a696..917d54588d95 100644 --- a/drivers/i2c/busses/i2c-at91.c +++ b/drivers/i2c/busses/i2c-at91.c @@ -101,6 +101,7 @@ struct at91_twi_dev {  	unsigned twi_cwgr_reg;  	struct at91_twi_pdata *pdata;  	bool use_dma; +	bool recv_len_abort;  	struct at91_twi_dma dma;  }; @@ -267,12 +268,24 @@ static void at91_twi_read_next_byte(struct at91_twi_dev *dev)  	*dev->buf = at91_twi_read(dev, AT91_TWI_RHR) & 0xff;  	--dev->buf_len; +	/* return if aborting, we only needed to read RHR to clear RXRDY*/ +	if (dev->recv_len_abort) +		return; +  	/* handle I2C_SMBUS_BLOCK_DATA */  	if (unlikely(dev->msg->flags & I2C_M_RECV_LEN)) { -		dev->msg->flags &= ~I2C_M_RECV_LEN; -		dev->buf_len += *dev->buf; -		dev->msg->len = dev->buf_len + 1; -		dev_dbg(dev->dev, "received block length %d\n", dev->buf_len); +		/* ensure length byte is a valid value */ +		if (*dev->buf <= I2C_SMBUS_BLOCK_MAX && *dev->buf > 0) { +			dev->msg->flags &= ~I2C_M_RECV_LEN; +			dev->buf_len += *dev->buf; +			dev->msg->len = dev->buf_len + 1; +			dev_dbg(dev->dev, "received block length %d\n", +					 dev->buf_len); +		} else { +			/* abort and send the stop by reading one more byte */ +			dev->recv_len_abort = true; +			dev->buf_len = 1; +		}  	}  	/* send stop if second but last byte has been read */ @@ -421,8 +434,8 @@ static int at91_do_twi_transfer(struct at91_twi_dev *dev)  		}  	} -	ret = wait_for_completion_interruptible_timeout(&dev->cmd_complete, -							dev->adapter.timeout); +	ret = wait_for_completion_io_timeout(&dev->cmd_complete, +					     dev->adapter.timeout);  	if (ret == 0) {  		dev_err(dev->dev, "controller timed out\n");  		at91_init_twi_bus(dev); @@ -444,6 +457,12 @@ static int at91_do_twi_transfer(struct at91_twi_dev *dev)  		ret = -EIO;  		goto error;  	} +	if (dev->recv_len_abort) { +		dev_err(dev->dev, "invalid smbus block length recvd\n"); +		ret = -EPROTO; +		goto error; +	} +  	dev_dbg(dev->dev, "transfer complete\n");  	return 0; @@ -500,6 +519,7 @@ static int at91_twi_xfer(struct i2c_adapter *adap, struct i2c_msg *msg, int num)  	dev->buf_len = m_start->len;  	dev->buf = m_start->buf;  	dev->msg = m_start; +	dev->recv_len_abort = false;  	ret = at91_do_twi_transfer(dev); diff --git a/drivers/i2c/busses/i2c-i801.c b/drivers/i2c/busses/i2c-i801.c index 2994690b26e9..10467a327749 100644 --- a/drivers/i2c/busses/i2c-i801.c +++ b/drivers/i2c/busses/i2c-i801.c @@ -164,6 +164,7 @@  /* Older devices have their ID defined in <linux/pci_ids.h> */  #define PCI_DEVICE_ID_INTEL_BAYTRAIL_SMBUS		0x0f12 +#define PCI_DEVICE_ID_INTEL_BRASWELL_SMBUS		0x2292  #define PCI_DEVICE_ID_INTEL_COUGARPOINT_SMBUS		0x1c22  #define PCI_DEVICE_ID_INTEL_PATSBURG_SMBUS		0x1d22  /* Patsburg also has three 'Integrated Device Function' SMBus controllers */ @@ -828,6 +829,7 @@ static const struct pci_device_id i801_ids[] = {  	{ PCI_DEVICE(PCI_VENDOR_ID_INTEL, PCI_DEVICE_ID_INTEL_WILDCATPOINT_SMBUS) },  	{ 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) },  	{ 0, }  }; diff --git a/drivers/i2c/busses/i2c-mv64xxx.c b/drivers/i2c/busses/i2c-mv64xxx.c index 6dc5ded86f62..2f64273d3f2b 100644 --- a/drivers/i2c/busses/i2c-mv64xxx.c +++ b/drivers/i2c/busses/i2c-mv64xxx.c @@ -746,8 +746,7 @@ mv64xxx_of_config(struct mv64xxx_i2c_data *drv_data,  	}  	tclk = clk_get_rate(drv_data->clk); -	rc = of_property_read_u32(np, "clock-frequency", &bus_freq); -	if (rc) +	if (of_property_read_u32(np, "clock-frequency", &bus_freq))  		bus_freq = 100000; /* 100kHz by default */  	if (!mv64xxx_find_baud_factors(bus_freq, tclk, diff --git a/drivers/i2c/busses/i2c-rcar.c b/drivers/i2c/busses/i2c-rcar.c index f3c7139dfa25..1cc146cfc1f3 100644 --- a/drivers/i2c/busses/i2c-rcar.c +++ b/drivers/i2c/busses/i2c-rcar.c @@ -34,6 +34,7 @@  #include <linux/platform_device.h>  #include <linux/pm_runtime.h>  #include <linux/slab.h> +#include <linux/spinlock.h>  /* register offsets */  #define ICSCR	0x00	/* slave ctrl */ @@ -95,6 +96,7 @@ struct rcar_i2c_priv {  	struct i2c_msg	*msg;  	struct clk *clk; +	spinlock_t lock;  	wait_queue_head_t wait;  	int pos; @@ -365,20 +367,20 @@ static irqreturn_t rcar_i2c_irq(int irq, void *ptr)  	struct rcar_i2c_priv *priv = ptr;  	u32 msr; +	/*-------------- spin lock -----------------*/ +	spin_lock(&priv->lock); +  	msr = rcar_i2c_read(priv, ICMSR); +	/* Only handle interrupts that are currently enabled */ +	msr &= rcar_i2c_read(priv, ICMIER); +  	/* Arbitration lost */  	if (msr & MAL) {  		rcar_i2c_flags_set(priv, (ID_DONE | ID_ARBLOST));  		goto out;  	} -	/* Stop */ -	if (msr & MST) { -		rcar_i2c_flags_set(priv, ID_DONE); -		goto out; -	} -  	/* Nack */  	if (msr & MNR) {  		/* go to stop phase */ @@ -388,6 +390,12 @@ static irqreturn_t rcar_i2c_irq(int irq, void *ptr)  		goto out;  	} +	/* Stop */ +	if (msr & MST) { +		rcar_i2c_flags_set(priv, ID_DONE); +		goto out; +	} +  	if (rcar_i2c_is_recv(priv))  		rcar_i2c_flags_set(priv, rcar_i2c_irq_recv(priv, msr));  	else @@ -400,6 +408,9 @@ out:  		wake_up(&priv->wait);  	} +	spin_unlock(&priv->lock); +	/*-------------- spin unlock -----------------*/ +  	return IRQ_HANDLED;  } @@ -409,14 +420,21 @@ static int rcar_i2c_master_xfer(struct i2c_adapter *adap,  {  	struct rcar_i2c_priv *priv = i2c_get_adapdata(adap);  	struct device *dev = rcar_i2c_priv_to_dev(priv); +	unsigned long flags;  	int i, ret, timeout;  	pm_runtime_get_sync(dev); +	/*-------------- spin lock -----------------*/ +	spin_lock_irqsave(&priv->lock, flags); +  	rcar_i2c_init(priv);  	/* start clock */  	rcar_i2c_write(priv, ICCCR, priv->icccr); +	spin_unlock_irqrestore(&priv->lock, flags); +	/*-------------- spin unlock -----------------*/ +  	ret = rcar_i2c_bus_barrier(priv);  	if (ret < 0)  		goto out; @@ -428,6 +446,9 @@ static int rcar_i2c_master_xfer(struct i2c_adapter *adap,  			break;  		} +		/*-------------- spin lock -----------------*/ +		spin_lock_irqsave(&priv->lock, flags); +  		/* init each data */  		priv->msg	= &msgs[i];  		priv->pos	= 0; @@ -437,6 +458,9 @@ static int rcar_i2c_master_xfer(struct i2c_adapter *adap,  		ret = rcar_i2c_prepare_msg(priv); +		spin_unlock_irqrestore(&priv->lock, flags); +		/*-------------- spin unlock -----------------*/ +  		if (ret < 0)  			break; @@ -540,6 +564,7 @@ static int rcar_i2c_probe(struct platform_device *pdev)  	irq = platform_get_irq(pdev, 0);  	init_waitqueue_head(&priv->wait); +	spin_lock_init(&priv->lock);  	adap = &priv->adap;  	adap->nr = pdev->id; diff --git a/drivers/i2c/busses/i2c-rk3x.c b/drivers/i2c/busses/i2c-rk3x.c index 69e11853e8bf..e637c32ae517 100644 --- a/drivers/i2c/busses/i2c-rk3x.c +++ b/drivers/i2c/busses/i2c-rk3x.c @@ -323,6 +323,10 @@ static void rk3x_i2c_handle_read(struct rk3x_i2c *i2c, unsigned int ipd)  	/* ack interrupt */  	i2c_writel(i2c, REG_INT_MBRF, REG_IPD); +	/* Can only handle a maximum of 32 bytes at a time */ +	if (len > 32) +		len = 32; +  	/* read the data from receive buffer */  	for (i = 0; i < len; ++i) {  		if (i % 4 == 0) diff --git a/drivers/i2c/i2c-acpi.c b/drivers/i2c/i2c-acpi.c index e8b61967334b..0dbc18c15c43 100644 --- a/drivers/i2c/i2c-acpi.c +++ b/drivers/i2c/i2c-acpi.c @@ -126,6 +126,7 @@ void acpi_i2c_register_devices(struct i2c_adapter *adap)  		dev_warn(&adap->dev, "failed to enumerate I2C slaves\n");  } +#ifdef CONFIG_ACPI_I2C_OPREGION  static int acpi_gsb_i2c_read_bytes(struct i2c_client *client,  		u8 cmd, u8 *data, u8 data_len)  { @@ -360,3 +361,4 @@ void acpi_i2c_remove_space_handler(struct i2c_adapter *adapter)  	acpi_bus_detach_private_data(handle);  } +#endif | 
