1
0
Fork 0
mirror of https://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git synced 2025-01-22 16:06:04 -05:00

i2c: testunit: on errors, repeat NACK until STOP

This backend requests a NACK from the controller driver when it detects
an error. If that request gets ignored from some reason, subsequent
accesses will wrongly be handled OK. To fix this, an error now changes
the state machine, so the backend will report NACK until a STOP
condition has been detected. This make the driver more robust against
controllers which will sadly apply the NACK not to the current byte but
the next one.

Fixes: a8335c64c5 ("i2c: add slave testunit driver")
Signed-off-by: Wolfram Sang <wsa+renesas@sang-engineering.com>
This commit is contained in:
Wolfram Sang 2025-01-15 17:23:47 +01:00
parent 093f70c134
commit 6ad30f7890

View file

@ -38,6 +38,7 @@ enum testunit_regs {
enum testunit_flags { enum testunit_flags {
TU_FLAG_IN_PROCESS, TU_FLAG_IN_PROCESS,
TU_FLAG_NACK,
}; };
struct testunit_data { struct testunit_data {
@ -90,8 +91,10 @@ static int i2c_slave_testunit_slave_cb(struct i2c_client *client,
switch (event) { switch (event) {
case I2C_SLAVE_WRITE_REQUESTED: case I2C_SLAVE_WRITE_REQUESTED:
if (test_bit(TU_FLAG_IN_PROCESS, &tu->flags)) if (test_bit(TU_FLAG_IN_PROCESS | TU_FLAG_NACK, &tu->flags)) {
return -EBUSY; ret = -EBUSY;
break;
}
memset(tu->regs, 0, TU_NUM_REGS); memset(tu->regs, 0, TU_NUM_REGS);
tu->reg_idx = 0; tu->reg_idx = 0;
@ -99,8 +102,10 @@ static int i2c_slave_testunit_slave_cb(struct i2c_client *client,
break; break;
case I2C_SLAVE_WRITE_RECEIVED: case I2C_SLAVE_WRITE_RECEIVED:
if (test_bit(TU_FLAG_IN_PROCESS, &tu->flags)) if (test_bit(TU_FLAG_IN_PROCESS | TU_FLAG_NACK, &tu->flags)) {
return -EBUSY; ret = -EBUSY;
break;
}
if (tu->reg_idx < TU_NUM_REGS) if (tu->reg_idx < TU_NUM_REGS)
tu->regs[tu->reg_idx] = *val; tu->regs[tu->reg_idx] = *val;
@ -129,6 +134,8 @@ static int i2c_slave_testunit_slave_cb(struct i2c_client *client,
* here because we still need them in the workqueue! * here because we still need them in the workqueue!
*/ */
tu->reg_idx = 0; tu->reg_idx = 0;
clear_bit(TU_FLAG_NACK, &tu->flags);
break; break;
case I2C_SLAVE_READ_PROCESSED: case I2C_SLAVE_READ_PROCESSED:
@ -151,6 +158,10 @@ static int i2c_slave_testunit_slave_cb(struct i2c_client *client,
break; break;
} }
/* If an error occurred somewhen, we NACK everything until next STOP */
if (ret)
set_bit(TU_FLAG_NACK, &tu->flags);
return ret; return ret;
} }