V4L/DVB (6857): cx23885: correctly join I2C writes and reads from same address

When an I2C message specifies a write then a read from the same I2C address,
we need to tell the chip to not release the bus between the message parts.

Signed-off-by: Chris Pascoe <c.pascoe@itee.uq.edu.au>
Signed-off-by: Mauro Carvalho Chehab <mchehab@infradead.org>
This commit is contained in:
Chris Pascoe 2007-12-15 03:31:09 -03:00 committed by Mauro Carvalho Chehab
parent bc51471088
commit 5e3c5967da

View file

@ -37,8 +37,10 @@ static unsigned int i2c_scan = 0;
module_param(i2c_scan, int, 0444); module_param(i2c_scan, int, 0444);
MODULE_PARM_DESC(i2c_scan, "scan i2c bus at insmod time"); MODULE_PARM_DESC(i2c_scan, "scan i2c bus at insmod time");
#define dprintk(level,fmt, arg...) if (i2c_debug >= level) \ #define dprintk(level, fmt, arg...) do { \
printk(KERN_DEBUG "%s: " fmt, dev->name , ## arg) if (i2c_debug >= level) \
printk(KERN_DEBUG "%s: " fmt, dev->name , ## arg); \
} while (0)
#define I2C_WAIT_DELAY 32 #define I2C_WAIT_DELAY 32
#define I2C_WAIT_RETRY 64 #define I2C_WAIT_RETRY 64
@ -77,14 +79,18 @@ static int i2c_wait_done(struct i2c_adapter *i2c_adap)
} }
static int i2c_sendbytes(struct i2c_adapter *i2c_adap, static int i2c_sendbytes(struct i2c_adapter *i2c_adap,
const struct i2c_msg *msg, int last) const struct i2c_msg *msg, int joined_rlen)
{ {
struct cx23885_i2c *bus = i2c_adap->algo_data; struct cx23885_i2c *bus = i2c_adap->algo_data;
struct cx23885_dev *dev = bus->dev; struct cx23885_dev *dev = bus->dev;
u32 wdata, addr, ctrl; u32 wdata, addr, ctrl;
int retval, cnt; int retval, cnt;
dprintk(1, "%s(msg->len=%d, last=%d)\n", __FUNCTION__, msg->len, last); if (joined_rlen)
dprintk(1, "%s(msg->wlen=%d, nextmsg->rlen=%d)\n", __FUNCTION__,
msg->len, joined_rlen);
else
dprintk(1, "%s(msg->len=%d)\n", __FUNCTION__, msg->len);
/* Deal with i2c probe functions with zero payload */ /* Deal with i2c probe functions with zero payload */
if (msg->len == 0) { if (msg->len == 0) {
@ -107,6 +113,8 @@ static int i2c_sendbytes(struct i2c_adapter *i2c_adap,
if (msg->len > 1) if (msg->len > 1)
ctrl |= I2C_NOSTOP | I2C_EXTEND; ctrl |= I2C_NOSTOP | I2C_EXTEND;
else if (joined_rlen)
ctrl |= I2C_NOSTOP;
cx_write(bus->reg_addr, addr); cx_write(bus->reg_addr, addr);
cx_write(bus->reg_wdata, wdata); cx_write(bus->reg_wdata, wdata);
@ -130,6 +138,8 @@ static int i2c_sendbytes(struct i2c_adapter *i2c_adap,
if (cnt < msg->len - 1) if (cnt < msg->len - 1)
ctrl |= I2C_NOSTOP | I2C_EXTEND; ctrl |= I2C_NOSTOP | I2C_EXTEND;
else if (joined_rlen)
ctrl |= I2C_NOSTOP;
cx_write(bus->reg_addr, addr); cx_write(bus->reg_addr, addr);
cx_write(bus->reg_wdata, wdata); cx_write(bus->reg_wdata, wdata);
@ -151,19 +161,22 @@ static int i2c_sendbytes(struct i2c_adapter *i2c_adap,
eio: eio:
retval = -EIO; retval = -EIO;
err: err:
printk(" ERR: %d\n", retval); if (i2c_debug)
printk(" ERR: %d\n", retval);
return retval; return retval;
} }
static int i2c_readbytes(struct i2c_adapter *i2c_adap, static int i2c_readbytes(struct i2c_adapter *i2c_adap,
const struct i2c_msg *msg, int last) const struct i2c_msg *msg, int joined)
{ {
struct cx23885_i2c *bus = i2c_adap->algo_data; struct cx23885_i2c *bus = i2c_adap->algo_data;
struct cx23885_dev *dev = bus->dev; struct cx23885_dev *dev = bus->dev;
u32 ctrl, cnt; u32 ctrl, cnt;
int retval; int retval;
dprintk(1, "%s(msg->len=%d, last=%d)\n", __FUNCTION__, msg->len, last);
if (i2c_debug && !joined)
dprintk(1, "%s(msg->len=%d)\n", __FUNCTION__, msg->len);
/* Deal with i2c probe functions with zero payload */ /* Deal with i2c probe functions with zero payload */
if (msg->len == 0) { if (msg->len == 0) {
@ -179,8 +192,12 @@ static int i2c_readbytes(struct i2c_adapter *i2c_adap,
return 0; return 0;
} }
if (i2c_debug) if (i2c_debug) {
printk(" <R %02x", (msg->addr << 1) + 1); if (joined)
printk(" R");
else
printk(" <R %02x", (msg->addr << 1) + 1);
}
for(cnt = 0; cnt < msg->len; cnt++) { for(cnt = 0; cnt < msg->len; cnt++) {
@ -209,7 +226,8 @@ static int i2c_readbytes(struct i2c_adapter *i2c_adap,
eio: eio:
retval = -EIO; retval = -EIO;
err: err:
printk(" ERR: %d\n", retval); if (i2c_debug)
printk(" ERR: %d\n", retval);
return retval; return retval;
} }
@ -227,15 +245,22 @@ static int i2c_xfer(struct i2c_adapter *i2c_adap,
__FUNCTION__, num, msgs[i].addr, msgs[i].len); __FUNCTION__, num, msgs[i].addr, msgs[i].len);
if (msgs[i].flags & I2C_M_RD) { if (msgs[i].flags & I2C_M_RD) {
/* read */ /* read */
retval = i2c_readbytes(i2c_adap, &msgs[i], i+1 == num); retval = i2c_readbytes(i2c_adap, &msgs[i], 0);
} else if (i + 1 < num && (msgs[i + 1].flags & I2C_M_RD) &&
msgs[i].addr == msgs[i + 1].addr) {
/* write then read from same address */
retval = i2c_sendbytes(i2c_adap, &msgs[i],
msgs[i + 1].len);
if (retval < 0) if (retval < 0)
goto err; goto err;
i++;
retval = i2c_readbytes(i2c_adap, &msgs[i], 1);
} else { } else {
/* write */ /* write */
retval = i2c_sendbytes(i2c_adap, &msgs[i], i+1 == num); retval = i2c_sendbytes(i2c_adap, &msgs[i], 0);
if (retval < 0)
goto err;
} }
if (retval < 0)
goto err;
} }
return num; return num;