The patch titled I2C-MPC: Fix up error handling has been removed from the -mm tree. Its filename is i2c-mpc-fix-up-error-handling.patch This patch was dropped because it was nacked by the maintainer ------------------------------------------------------ Subject: I2C-MPC: Fix up error handling From: Kumar Gala <galak@xxxxxxxxxxxxxxxxxxx> * If we have an Unfinished (MCF) or Arbitration Lost (MAL) error and the bus is still busy reset the controller. This prevents the controller from getting in a hung state for transactions for other devices. * Fixed up propogating the errors from i2c_wait. Signed-off-by: Kumar Gala <galak@xxxxxxxxxxxxxxxxxxx> Cc: Jean Delvare <khali@xxxxxxxxxxxx> Cc: Greg KH <greg@xxxxxxxxx> Signed-off-by: Andrew Morton <akpm@xxxxxxxx> --- drivers/i2c/busses/i2c-mpc.c | 43 +++++++++++++++++++++++---------- 1 file changed, 31 insertions(+), 12 deletions(-) diff -puN drivers/i2c/busses/i2c-mpc.c~i2c-mpc-fix-up-error-handling drivers/i2c/busses/i2c-mpc.c --- a/drivers/i2c/busses/i2c-mpc.c~i2c-mpc-fix-up-error-handling +++ a/drivers/i2c/busses/i2c-mpc.c @@ -114,11 +114,20 @@ static int i2c_wait(struct mpc_i2c *i2c, if (!(x & CSR_MCF)) { pr_debug("I2C: unfinished\n"); + + /* reset the controller if the bus is still busy */ + if (x & CSR_MBB) + writeccr(i2c, 0); + return -EIO; } if (x & CSR_MAL) { pr_debug("I2C: MAL\n"); + + /* reset the controller if the bus is still busy */ + if (x & CSR_MBB) + writeccr(i2c, 0); return -EIO; } @@ -159,7 +168,7 @@ static void mpc_i2c_stop(struct mpc_i2c static int mpc_write(struct mpc_i2c *i2c, int target, const u8 * data, int length, int restart) { - int i; + int i, ret; unsigned timeout = i2c->adap.timeout; u32 flags = restart ? CCR_RSTA : 0; @@ -171,15 +180,17 @@ static int mpc_write(struct mpc_i2c *i2c /* Write target byte */ writeb((target << 1), i2c->base + MPC_I2C_DR); - if (i2c_wait(i2c, timeout, 1) < 0) - return -1; + ret = i2c_wait(i2c, timeout, 1); + if (ret < 0) + return ret; for (i = 0; i < length; i++) { /* Write data byte */ writeb(data[i], i2c->base + MPC_I2C_DR); - if (i2c_wait(i2c, timeout, 1) < 0) - return -1; + ret = i2c_wait(i2c, timeout, 1); + if (ret < 0) + return ret; } return 0; @@ -189,7 +200,7 @@ static int mpc_read(struct mpc_i2c *i2c, u8 * data, int length, int restart) { unsigned timeout = i2c->adap.timeout; - int i; + int i, ret; u32 flags = restart ? CCR_RSTA : 0; /* Start with MEN */ @@ -200,8 +211,9 @@ static int mpc_read(struct mpc_i2c *i2c, /* Write target address byte - this time with the read flag set */ writeb((target << 1) | 1, i2c->base + MPC_I2C_DR); - if (i2c_wait(i2c, timeout, 1) < 0) - return -1; + ret = i2c_wait(i2c, timeout, 1); + if (ret < 0) + return ret; if (length) { if (length == 1) @@ -213,8 +225,9 @@ static int mpc_read(struct mpc_i2c *i2c, } for (i = 0; i < length; i++) { - if (i2c_wait(i2c, timeout, 0) < 0) - return -1; + ret = i2c_wait(i2c, timeout, 0); + if (ret < 0) + return ret; /* Generate txack on next to last byte */ if (i == length - 2) @@ -245,8 +258,13 @@ static int mpc_xfer(struct i2c_adapter * return -EINTR; } if (time_after(jiffies, orig_jiffies + HZ)) { - pr_debug("I2C: timeout\n"); - return -EIO; + writeccr(i2c, 0); + + /* try one more time before we error */ + if (readb(i2c->base + MPC_I2C_SR) & CSR_MBB) { + pr_debug("I2C: timeout\n"); + return -EIO; + } } schedule(); } @@ -324,6 +342,7 @@ static int fsl_i2c_probe(struct platform goto fail_irq; } + writeccr(i2c, 0); mpc_i2c_setclock(i2c); platform_set_drvdata(pdev, i2c); _ Patches currently in -mm which might be from galak@xxxxxxxxxxxxxxxxxxx are i2c-mpc-fix-up-error-handling.patch - To unsubscribe from this list: send the line "unsubscribe mm-commits" in the body of a message to majordomo@xxxxxxxxxxxxxxx More majordomo info at http://vger.kernel.org/majordomo-info.html