Commit 18215da0 authored by Lukas Wunner's avatar Lukas Wunner Committed by Greg Kroah-Hartman

iio: adc: mcp320x: Fix readout of negative voltages

commit e6f47943 upstream.

Commit f686a36b ("iio: adc: mcp320x: Add support for mcp3301")
returns a signed voltage from mcp320x_adc_conversion() but neglects that
the caller interprets a negative return value as failure.  Only mcp3301
(and the upcoming mcp3550/1/3) is affected as the other chips are
incapable of measuring negative voltages.

Fix and while at it, add mcp3301 to the list of supported chips at the
top of the file.

Fixes: f686a36b ("iio: adc: mcp320x: Add support for mcp3301")
Cc: Andrea Galbusera <gizero@gmail.com>
Signed-off-by: default avatarLukas Wunner <lukas@wunner.de>
Signed-off-by: default avatarJonathan Cameron <Jonathan.Cameron@huawei.com>
Signed-off-by: default avatarGreg Kroah-Hartman <gregkh@linuxfoundation.org>
parent f2f68ec0
...@@ -17,6 +17,8 @@ ...@@ -17,6 +17,8 @@
* MCP3204 * MCP3204
* MCP3208 * MCP3208
* ------------ * ------------
* 13 bit converter
* MCP3301
* *
* Datasheet can be found here: * Datasheet can be found here:
* http://ww1.microchip.com/downloads/en/DeviceDoc/21293C.pdf mcp3001 * http://ww1.microchip.com/downloads/en/DeviceDoc/21293C.pdf mcp3001
...@@ -96,7 +98,7 @@ static int mcp320x_channel_to_tx_data(int device_index, ...@@ -96,7 +98,7 @@ static int mcp320x_channel_to_tx_data(int device_index,
} }
static int mcp320x_adc_conversion(struct mcp320x *adc, u8 channel, static int mcp320x_adc_conversion(struct mcp320x *adc, u8 channel,
bool differential, int device_index) bool differential, int device_index, int *val)
{ {
int ret; int ret;
...@@ -117,19 +119,25 @@ static int mcp320x_adc_conversion(struct mcp320x *adc, u8 channel, ...@@ -117,19 +119,25 @@ static int mcp320x_adc_conversion(struct mcp320x *adc, u8 channel,
switch (device_index) { switch (device_index) {
case mcp3001: case mcp3001:
return (adc->rx_buf[0] << 5 | adc->rx_buf[1] >> 3); *val = (adc->rx_buf[0] << 5 | adc->rx_buf[1] >> 3);
return 0;
case mcp3002: case mcp3002:
case mcp3004: case mcp3004:
case mcp3008: case mcp3008:
return (adc->rx_buf[0] << 2 | adc->rx_buf[1] >> 6); *val = (adc->rx_buf[0] << 2 | adc->rx_buf[1] >> 6);
return 0;
case mcp3201: case mcp3201:
return (adc->rx_buf[0] << 7 | adc->rx_buf[1] >> 1); *val = (adc->rx_buf[0] << 7 | adc->rx_buf[1] >> 1);
return 0;
case mcp3202: case mcp3202:
case mcp3204: case mcp3204:
case mcp3208: case mcp3208:
return (adc->rx_buf[0] << 4 | adc->rx_buf[1] >> 4); *val = (adc->rx_buf[0] << 4 | adc->rx_buf[1] >> 4);
return 0;
case mcp3301: case mcp3301:
return sign_extend32((adc->rx_buf[0] & 0x1f) << 8 | adc->rx_buf[1], 12); *val = sign_extend32((adc->rx_buf[0] & 0x1f) << 8
| adc->rx_buf[1], 12);
return 0;
default: default:
return -EINVAL; return -EINVAL;
} }
...@@ -150,12 +158,10 @@ static int mcp320x_read_raw(struct iio_dev *indio_dev, ...@@ -150,12 +158,10 @@ static int mcp320x_read_raw(struct iio_dev *indio_dev,
switch (mask) { switch (mask) {
case IIO_CHAN_INFO_RAW: case IIO_CHAN_INFO_RAW:
ret = mcp320x_adc_conversion(adc, channel->address, ret = mcp320x_adc_conversion(adc, channel->address,
channel->differential, device_index); channel->differential, device_index, val);
if (ret < 0) if (ret < 0)
goto out; goto out;
*val = ret;
ret = IIO_VAL_INT; ret = IIO_VAL_INT;
break; break;
......
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