Commit ffc59c49 authored by Wolfram Sang's avatar Wolfram Sang Committed by Wolfram Sang

i2c: recovery: require either get_sda or set_sda

For bus recovery, we either need to bail out early if we can read SDA or
we need to send STOP after every pulse. Otherwise recovery might be
misinterpreted as an unwanted write. So, require one of those SDA
handling functions to avoid this problem.
Signed-off-by: default avatarWolfram Sang <wsa+renesas@sang-engineering.com>
Acked-by: default avatarPeter Rosin <peda@axentia.se>
Signed-off-by: default avatarWolfram Sang <wsa@the-dreams.de>
parent 08948b75
...@@ -202,7 +202,8 @@ int i2c_generic_scl_recovery(struct i2c_adapter *adap) ...@@ -202,7 +202,8 @@ int i2c_generic_scl_recovery(struct i2c_adapter *adap)
/* /*
* If we can set SDA, we will always create STOP here to ensure * If we can set SDA, we will always create STOP here to ensure
* the additional pulses will do no harm. This is achieved by * the additional pulses will do no harm. This is achieved by
* letting SDA follow SCL half a cycle later. * letting SDA follow SCL half a cycle later. Check the
* 'incomplete_write_byte' fault injector for details.
*/ */
ndelay(RECOVERY_NDELAY / 2); ndelay(RECOVERY_NDELAY / 2);
if (bri->set_sda) if (bri->set_sda)
...@@ -274,6 +275,10 @@ static void i2c_init_recovery(struct i2c_adapter *adap) ...@@ -274,6 +275,10 @@ static void i2c_init_recovery(struct i2c_adapter *adap)
err_str = "no {get|set}_scl() found"; err_str = "no {get|set}_scl() found";
goto err; goto err;
} }
if (!bri->set_sda && !bri->get_sda) {
err_str = "either get_sda() or set_sda() needed";
goto err;
}
} }
return; return;
......
...@@ -581,12 +581,12 @@ struct i2c_timings { ...@@ -581,12 +581,12 @@ struct i2c_timings {
* recovery. Populated internally for generic GPIO recovery. * recovery. Populated internally for generic GPIO recovery.
* @set_scl: This sets/clears the SCL line. Mandatory for generic SCL recovery. * @set_scl: This sets/clears the SCL line. Mandatory for generic SCL recovery.
* Populated internally for generic GPIO recovery. * Populated internally for generic GPIO recovery.
* @get_sda: This gets current value of SDA line. Optional for generic SCL * @get_sda: This gets current value of SDA line. This or set_sda() is mandatory
* recovery. Populated internally, if sda_gpio is a valid GPIO, for generic * for generic SCL recovery. Populated internally, if sda_gpio is a valid
* GPIO recovery. * GPIO, for generic GPIO recovery.
* @set_sda: This sets/clears the SDA line. Optional for generic SCL recovery. * @set_sda: This sets/clears the SDA line. This or get_sda() is mandatory for
* Populated internally, if sda_gpio is a valid GPIO, for generic GPIO * generic SCL recovery. Populated internally, if sda_gpio is a valid GPIO,
* recovery. * for generic GPIO recovery.
* @prepare_recovery: This will be called before starting recovery. Platform may * @prepare_recovery: This will be called before starting recovery. Platform may
* configure padmux here for SDA/SCL line or something else they want. * configure padmux here for SDA/SCL line or something else they want.
* @unprepare_recovery: This will be called after completing recovery. Platform * @unprepare_recovery: This will be called after completing recovery. Platform
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment