summaryrefslogtreecommitdiffstats
path: root/drivers/i2c
diff options
context:
space:
mode:
authorRussell King <rmk+kernel@armlinux.org.uk>2016-12-17 07:10:56 -0500
committerWolfram Sang <wsa@the-dreams.de>2016-12-17 13:29:44 -0500
commit7f638c1cb0a1112dbe0b682a42db30521646686b (patch)
treec745c75df0389c0c779c71d0906169469b1a26c2 /drivers/i2c
parent493ff7e2cdda9182fb709d3681315180d9165bd8 (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.c5
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;