diff options
author | Russell King <rmk+kernel@armlinux.org.uk> | 2016-12-17 07:10:56 -0500 |
---|---|---|
committer | Wolfram Sang <wsa@the-dreams.de> | 2016-12-17 13:29:44 -0500 |
commit | 7f638c1cb0a1112dbe0b682a42db30521646686b (patch) | |
tree | c745c75df0389c0c779c71d0906169469b1a26c2 /drivers/i2c | |
parent | 493ff7e2cdda9182fb709d3681315180d9165bd8 (diff) |
i2c: mux: pca954x: fix i2c mux selection caching
smbus functions return -ve on error, 0 on success. However,
__i2c_transfer() have a different return signature - -ve on error, or
number of buffers transferred (which may be zero or greater.)
The upshot of this is that the sense of the test is reversed when using
the mux on a bus supporting the master_xfer method: we cache the value
and never retry if we fail to transfer any buffers, but if we succeed,
we clear the cached value.
Fix this by making pca954x_reg_write() return a negative error code for
all failure cases.
Fixes: 463e8f845cbf ("i2c: mux: pca954x: retry updating the mux selection on failure")
Acked-by: Peter Rosin <peda@axentia.se>
Signed-off-by: Russell King <rmk+kernel@armlinux.org.uk>
Signed-off-by: Wolfram Sang <wsa@the-dreams.de>
Diffstat (limited to 'drivers/i2c')
-rw-r--r-- | drivers/i2c/muxes/i2c-mux-pca954x.c | 5 |
1 files changed, 4 insertions, 1 deletions
diff --git a/drivers/i2c/muxes/i2c-mux-pca954x.c b/drivers/i2c/muxes/i2c-mux-pca954x.c index 9a348ee4dc14..dd18b9ccb1f4 100644 --- a/drivers/i2c/muxes/i2c-mux-pca954x.c +++ b/drivers/i2c/muxes/i2c-mux-pca954x.c | |||
@@ -167,6 +167,9 @@ static int pca954x_reg_write(struct i2c_adapter *adap, | |||
167 | buf[0] = val; | 167 | buf[0] = val; |
168 | msg.buf = buf; | 168 | msg.buf = buf; |
169 | ret = __i2c_transfer(adap, &msg, 1); | 169 | ret = __i2c_transfer(adap, &msg, 1); |
170 | |||
171 | if (ret >= 0 && ret != 1) | ||
172 | ret = -EREMOTEIO; | ||
170 | } else { | 173 | } else { |
171 | union i2c_smbus_data data; | 174 | union i2c_smbus_data data; |
172 | ret = adap->algo->smbus_xfer(adap, client->addr, | 175 | ret = adap->algo->smbus_xfer(adap, client->addr, |
@@ -195,7 +198,7 @@ static int pca954x_select_chan(struct i2c_mux_core *muxc, u32 chan) | |||
195 | /* Only select the channel if its different from the last channel */ | 198 | /* Only select the channel if its different from the last channel */ |
196 | if (data->last_chan != regval) { | 199 | if (data->last_chan != regval) { |
197 | ret = pca954x_reg_write(muxc->parent, client, regval); | 200 | ret = pca954x_reg_write(muxc->parent, client, regval); |
198 | data->last_chan = ret ? 0 : regval; | 201 | data->last_chan = ret < 0 ? 0 : regval; |
199 | } | 202 | } |
200 | 203 | ||
201 | return ret; | 204 | return ret; |