Commit 55ecaf17 authored by Linus Walleij's avatar Linus Walleij Committed by Jonathan Cameron

iio: magnetometer: ak8974: Break out measurement

This breaks out the measurement code to its own function
so we can handle this without swirling it up with the
big switch() statement inside ak8974_read_raw().

Keep a local s16 helper variable for the signed value
coming out of the measurement before assigning it to the
integer *val. The local variable makes the code easier
to read and the compiler will optimize it if possible.

Cc: Nick Reitemeyer <nick.reitemeyer@web.de>
Cc: Stephan Gerhold <stephan@gerhold.net>
Reviewed-by: default avatarMichał Mirosław <mirq-linux@rere.qmqm.pl>
Signed-off-by: default avatarLinus Walleij <linus.walleij@linaro.org>
Signed-off-by: default avatarJonathan Cameron <Jonathan.Cameron@huawei.com>
parent b67959eb
...@@ -554,42 +554,33 @@ static int ak8974_detect(struct ak8974 *ak8974) ...@@ -554,42 +554,33 @@ static int ak8974_detect(struct ak8974 *ak8974)
return 0; return 0;
} }
static int ak8974_read_raw(struct iio_dev *indio_dev, static int ak8974_measure_channel(struct ak8974 *ak8974, unsigned long address,
struct iio_chan_spec const *chan, int *val)
int *val, int *val2,
long mask)
{ {
struct ak8974 *ak8974 = iio_priv(indio_dev);
__le16 hw_values[3]; __le16 hw_values[3];
int ret = -EINVAL; int ret;
pm_runtime_get_sync(&ak8974->i2c->dev); pm_runtime_get_sync(&ak8974->i2c->dev);
mutex_lock(&ak8974->lock); mutex_lock(&ak8974->lock);
switch (mask) { /*
case IIO_CHAN_INFO_RAW: * We read all axes and discard all but one, for optimized
if (chan->address > 2) { * reading, use the triggered buffer.
dev_err(&ak8974->i2c->dev, "faulty channel address\n"); */
ret = -EIO;
goto out_unlock;
}
ret = ak8974_trigmeas(ak8974); ret = ak8974_trigmeas(ak8974);
if (ret) if (ret)
goto out_unlock; goto out_unlock;
ret = ak8974_getresult(ak8974, hw_values); ret = ak8974_getresult(ak8974, hw_values);
if (ret) if (ret)
goto out_unlock; goto out_unlock;
/* /*
* We read all axes and discard all but one, for optimized * This explicit cast to (s16) is necessary as the measurement
* reading, use the triggered buffer. * is done in 2's complement with positive and negative values.
* The follwing assignment to *val will then convert the signed
* s16 value to a signed int value.
*/ */
*val = (s16)le16_to_cpu(hw_values[chan->address]); *val = (s16)le16_to_cpu(hw_values[address]);
out_unlock:
ret = IIO_VAL_INT;
}
out_unlock:
mutex_unlock(&ak8974->lock); mutex_unlock(&ak8974->lock);
pm_runtime_mark_last_busy(&ak8974->i2c->dev); pm_runtime_mark_last_busy(&ak8974->i2c->dev);
pm_runtime_put_autosuspend(&ak8974->i2c->dev); pm_runtime_put_autosuspend(&ak8974->i2c->dev);
...@@ -597,6 +588,29 @@ static int ak8974_read_raw(struct iio_dev *indio_dev, ...@@ -597,6 +588,29 @@ static int ak8974_read_raw(struct iio_dev *indio_dev,
return ret; return ret;
} }
static int ak8974_read_raw(struct iio_dev *indio_dev,
struct iio_chan_spec const *chan,
int *val, int *val2,
long mask)
{
struct ak8974 *ak8974 = iio_priv(indio_dev);
int ret;
switch (mask) {
case IIO_CHAN_INFO_RAW:
if (chan->address > 2) {
dev_err(&ak8974->i2c->dev, "faulty channel address\n");
return -EIO;
}
ret = ak8974_measure_channel(ak8974, chan->address, val);
if (ret)
return ret;
return IIO_VAL_INT;
}
return -EINVAL;
}
static void ak8974_fill_buffer(struct iio_dev *indio_dev) static void ak8974_fill_buffer(struct iio_dev *indio_dev)
{ {
struct ak8974 *ak8974 = iio_priv(indio_dev); struct ak8974 *ak8974 = iio_priv(indio_dev);
......
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