Commit 54732a53 authored by Greg Kroah-Hartman's avatar Greg Kroah-Hartman

Merge tag 'iio-fixes-5.13b-take2' of...

Merge tag 'iio-fixes-5.13b-take2' of https://git.kernel.org/pub/scm/linux/kernel/git/jic23/iio into staging-linus

Jonathan writes:

Second set of IIO fixes for the 5.13 cycle

A mixed bag of fixes for various drivers.
Changes since take 1: Fixed a ; in a tag.

adi,ad7124
 - Fix miss balanced regulator enable / disable in error path
 - Fix potential overflow with non sequential channel numbers from dt.
adi,ad7192
 - Avoid disabling clock that was never enabled in error path + remove
 - Avoid nasty corner case if regulator voltage is 0 that would result
   in a good return half way through probe.
adi,ad7746
 - Avoid overwriting num_channels just after setting it correctly.
adi,ad7768
 - Buffer passed to iio_push_to_buffers_with_timestamp() too small and
   not aligned appropriately.
adi,ad7793
 - Missing return code setting in an error path.
adi,ad7923
 - Buffer too small after support for more channels added.
adi,ad5770r
 - Missing fwnode_handle_put in error paths.
fsl,fxa21002c
 - Missing runtime pm put in error path.

* tag 'iio-fixes-5.13b-take2' of https://git.kernel.org/pub/scm/linux/kernel/git/jic23/iio:
  iio: adc: ad7793: Add missing error code in ad7793_setup()
  iio: adc: ad7923: Fix undersized rx buffer.
  iio: adc: ad7768-1: Fix too small buffer passed to iio_push_to_buffers_with_timestamp()
  iio: dac: ad5770r: Put fwnode in error case during ->probe()
  iio: gyro: fxas21002c: balance runtime power in error path
  staging: iio: cdc: ad7746: avoid overwrite of num_channels
  iio: adc: ad7192: handle regulator voltage error first
  iio: adc: ad7192: Avoid disabling a clock that was never enabled.
  iio: adc: ad7124: Fix potential overflow due to non sequential channel numbers
  iio: adc: ad7124: Fix missbalanced regulator enable / disable on error.
parents e0112a7c 4ed243b1
...@@ -771,6 +771,13 @@ static int ad7124_of_parse_channel_config(struct iio_dev *indio_dev, ...@@ -771,6 +771,13 @@ static int ad7124_of_parse_channel_config(struct iio_dev *indio_dev,
if (ret) if (ret)
goto err; goto err;
if (channel >= indio_dev->num_channels) {
dev_err(indio_dev->dev.parent,
"Channel index >= number of channels\n");
ret = -EINVAL;
goto err;
}
ret = of_property_read_u32_array(child, "diff-channels", ret = of_property_read_u32_array(child, "diff-channels",
ain, 2); ain, 2);
if (ret) if (ret)
...@@ -850,6 +857,11 @@ static int ad7124_setup(struct ad7124_state *st) ...@@ -850,6 +857,11 @@ static int ad7124_setup(struct ad7124_state *st)
return ret; return ret;
} }
static void ad7124_reg_disable(void *r)
{
regulator_disable(r);
}
static int ad7124_probe(struct spi_device *spi) static int ad7124_probe(struct spi_device *spi)
{ {
const struct ad7124_chip_info *info; const struct ad7124_chip_info *info;
...@@ -895,17 +907,20 @@ static int ad7124_probe(struct spi_device *spi) ...@@ -895,17 +907,20 @@ static int ad7124_probe(struct spi_device *spi)
ret = regulator_enable(st->vref[i]); ret = regulator_enable(st->vref[i]);
if (ret) if (ret)
return ret; return ret;
ret = devm_add_action_or_reset(&spi->dev, ad7124_reg_disable,
st->vref[i]);
if (ret)
return ret;
} }
st->mclk = devm_clk_get(&spi->dev, "mclk"); st->mclk = devm_clk_get(&spi->dev, "mclk");
if (IS_ERR(st->mclk)) { if (IS_ERR(st->mclk))
ret = PTR_ERR(st->mclk); return PTR_ERR(st->mclk);
goto error_regulator_disable;
}
ret = clk_prepare_enable(st->mclk); ret = clk_prepare_enable(st->mclk);
if (ret < 0) if (ret < 0)
goto error_regulator_disable; return ret;
ret = ad7124_soft_reset(st); ret = ad7124_soft_reset(st);
if (ret < 0) if (ret < 0)
...@@ -935,11 +950,6 @@ static int ad7124_probe(struct spi_device *spi) ...@@ -935,11 +950,6 @@ static int ad7124_probe(struct spi_device *spi)
ad_sd_cleanup_buffer_and_trigger(indio_dev); ad_sd_cleanup_buffer_and_trigger(indio_dev);
error_clk_disable_unprepare: error_clk_disable_unprepare:
clk_disable_unprepare(st->mclk); clk_disable_unprepare(st->mclk);
error_regulator_disable:
for (i = ARRAY_SIZE(st->vref) - 1; i >= 0; i--) {
if (!IS_ERR_OR_NULL(st->vref[i]))
regulator_disable(st->vref[i]);
}
return ret; return ret;
} }
...@@ -948,17 +958,11 @@ static int ad7124_remove(struct spi_device *spi) ...@@ -948,17 +958,11 @@ static int ad7124_remove(struct spi_device *spi)
{ {
struct iio_dev *indio_dev = spi_get_drvdata(spi); struct iio_dev *indio_dev = spi_get_drvdata(spi);
struct ad7124_state *st = iio_priv(indio_dev); struct ad7124_state *st = iio_priv(indio_dev);
int i;
iio_device_unregister(indio_dev); iio_device_unregister(indio_dev);
ad_sd_cleanup_buffer_and_trigger(indio_dev); ad_sd_cleanup_buffer_and_trigger(indio_dev);
clk_disable_unprepare(st->mclk); clk_disable_unprepare(st->mclk);
for (i = ARRAY_SIZE(st->vref) - 1; i >= 0; i--) {
if (!IS_ERR_OR_NULL(st->vref[i]))
regulator_disable(st->vref[i]);
}
return 0; return 0;
} }
......
...@@ -912,7 +912,7 @@ static int ad7192_probe(struct spi_device *spi) ...@@ -912,7 +912,7 @@ static int ad7192_probe(struct spi_device *spi)
{ {
struct ad7192_state *st; struct ad7192_state *st;
struct iio_dev *indio_dev; struct iio_dev *indio_dev;
int ret, voltage_uv = 0; int ret;
if (!spi->irq) { if (!spi->irq) {
dev_err(&spi->dev, "no IRQ?\n"); dev_err(&spi->dev, "no IRQ?\n");
...@@ -949,15 +949,12 @@ static int ad7192_probe(struct spi_device *spi) ...@@ -949,15 +949,12 @@ static int ad7192_probe(struct spi_device *spi)
goto error_disable_avdd; goto error_disable_avdd;
} }
voltage_uv = regulator_get_voltage(st->avdd); ret = regulator_get_voltage(st->avdd);
if (ret < 0) {
if (voltage_uv > 0) {
st->int_vref_mv = voltage_uv / 1000;
} else {
ret = voltage_uv;
dev_err(&spi->dev, "Device tree error, reference voltage undefined\n"); dev_err(&spi->dev, "Device tree error, reference voltage undefined\n");
goto error_disable_avdd; goto error_disable_avdd;
} }
st->int_vref_mv = ret / 1000;
spi_set_drvdata(spi, indio_dev); spi_set_drvdata(spi, indio_dev);
st->chip_info = of_device_get_match_data(&spi->dev); st->chip_info = of_device_get_match_data(&spi->dev);
...@@ -1014,7 +1011,9 @@ static int ad7192_probe(struct spi_device *spi) ...@@ -1014,7 +1011,9 @@ static int ad7192_probe(struct spi_device *spi)
return 0; return 0;
error_disable_clk: error_disable_clk:
clk_disable_unprepare(st->mclk); if (st->clock_sel == AD7192_CLK_EXT_MCLK1_2 ||
st->clock_sel == AD7192_CLK_EXT_MCLK2)
clk_disable_unprepare(st->mclk);
error_remove_trigger: error_remove_trigger:
ad_sd_cleanup_buffer_and_trigger(indio_dev); ad_sd_cleanup_buffer_and_trigger(indio_dev);
error_disable_dvdd: error_disable_dvdd:
...@@ -1031,7 +1030,9 @@ static int ad7192_remove(struct spi_device *spi) ...@@ -1031,7 +1030,9 @@ static int ad7192_remove(struct spi_device *spi)
struct ad7192_state *st = iio_priv(indio_dev); struct ad7192_state *st = iio_priv(indio_dev);
iio_device_unregister(indio_dev); iio_device_unregister(indio_dev);
clk_disable_unprepare(st->mclk); if (st->clock_sel == AD7192_CLK_EXT_MCLK1_2 ||
st->clock_sel == AD7192_CLK_EXT_MCLK2)
clk_disable_unprepare(st->mclk);
ad_sd_cleanup_buffer_and_trigger(indio_dev); ad_sd_cleanup_buffer_and_trigger(indio_dev);
regulator_disable(st->dvdd); regulator_disable(st->dvdd);
......
...@@ -167,6 +167,10 @@ struct ad7768_state { ...@@ -167,6 +167,10 @@ struct ad7768_state {
* transfer buffers to live in their own cache lines. * transfer buffers to live in their own cache lines.
*/ */
union { union {
struct {
__be32 chan;
s64 timestamp;
} scan;
__be32 d32; __be32 d32;
u8 d8[2]; u8 d8[2];
} data ____cacheline_aligned; } data ____cacheline_aligned;
...@@ -469,11 +473,11 @@ static irqreturn_t ad7768_trigger_handler(int irq, void *p) ...@@ -469,11 +473,11 @@ static irqreturn_t ad7768_trigger_handler(int irq, void *p)
mutex_lock(&st->lock); mutex_lock(&st->lock);
ret = spi_read(st->spi, &st->data.d32, 3); ret = spi_read(st->spi, &st->data.scan.chan, 3);
if (ret < 0) if (ret < 0)
goto err_unlock; goto err_unlock;
iio_push_to_buffers_with_timestamp(indio_dev, &st->data.d32, iio_push_to_buffers_with_timestamp(indio_dev, &st->data.scan,
iio_get_time_ns(indio_dev)); iio_get_time_ns(indio_dev));
iio_trigger_notify_done(indio_dev->trig); iio_trigger_notify_done(indio_dev->trig);
......
...@@ -279,6 +279,7 @@ static int ad7793_setup(struct iio_dev *indio_dev, ...@@ -279,6 +279,7 @@ static int ad7793_setup(struct iio_dev *indio_dev,
id &= AD7793_ID_MASK; id &= AD7793_ID_MASK;
if (id != st->chip_info->id) { if (id != st->chip_info->id) {
ret = -ENODEV;
dev_err(&st->sd.spi->dev, "device ID query failed\n"); dev_err(&st->sd.spi->dev, "device ID query failed\n");
goto out; goto out;
} }
......
...@@ -59,8 +59,10 @@ struct ad7923_state { ...@@ -59,8 +59,10 @@ struct ad7923_state {
/* /*
* DMA (thus cache coherency maintenance) requires the * DMA (thus cache coherency maintenance) requires the
* transfer buffers to live in their own cache lines. * transfer buffers to live in their own cache lines.
* Ensure rx_buf can be directly used in iio_push_to_buffers_with_timetamp
* Length = 8 channels + 4 extra for 8 byte timestamp
*/ */
__be16 rx_buf[4] ____cacheline_aligned; __be16 rx_buf[12] ____cacheline_aligned;
__be16 tx_buf[4]; __be16 tx_buf[4];
}; };
......
...@@ -524,23 +524,29 @@ static int ad5770r_channel_config(struct ad5770r_state *st) ...@@ -524,23 +524,29 @@ static int ad5770r_channel_config(struct ad5770r_state *st)
device_for_each_child_node(&st->spi->dev, child) { device_for_each_child_node(&st->spi->dev, child) {
ret = fwnode_property_read_u32(child, "num", &num); ret = fwnode_property_read_u32(child, "num", &num);
if (ret) if (ret)
return ret; goto err_child_out;
if (num >= AD5770R_MAX_CHANNELS) if (num >= AD5770R_MAX_CHANNELS) {
return -EINVAL; ret = -EINVAL;
goto err_child_out;
}
ret = fwnode_property_read_u32_array(child, ret = fwnode_property_read_u32_array(child,
"adi,range-microamp", "adi,range-microamp",
tmp, 2); tmp, 2);
if (ret) if (ret)
return ret; goto err_child_out;
min = tmp[0] / 1000; min = tmp[0] / 1000;
max = tmp[1] / 1000; max = tmp[1] / 1000;
ret = ad5770r_store_output_range(st, min, max, num); ret = ad5770r_store_output_range(st, min, max, num);
if (ret) if (ret)
return ret; goto err_child_out;
} }
return 0;
err_child_out:
fwnode_handle_put(child);
return ret; return ret;
} }
......
...@@ -399,6 +399,7 @@ static int fxas21002c_temp_get(struct fxas21002c_data *data, int *val) ...@@ -399,6 +399,7 @@ static int fxas21002c_temp_get(struct fxas21002c_data *data, int *val)
ret = regmap_field_read(data->regmap_fields[F_TEMP], &temp); ret = regmap_field_read(data->regmap_fields[F_TEMP], &temp);
if (ret < 0) { if (ret < 0) {
dev_err(dev, "failed to read temp: %d\n", ret); dev_err(dev, "failed to read temp: %d\n", ret);
fxas21002c_pm_put(data);
goto data_unlock; goto data_unlock;
} }
...@@ -432,6 +433,7 @@ static int fxas21002c_axis_get(struct fxas21002c_data *data, ...@@ -432,6 +433,7 @@ static int fxas21002c_axis_get(struct fxas21002c_data *data,
&axis_be, sizeof(axis_be)); &axis_be, sizeof(axis_be));
if (ret < 0) { if (ret < 0) {
dev_err(dev, "failed to read axis: %d: %d\n", index, ret); dev_err(dev, "failed to read axis: %d: %d\n", index, ret);
fxas21002c_pm_put(data);
goto data_unlock; goto data_unlock;
} }
......
...@@ -700,7 +700,6 @@ static int ad7746_probe(struct i2c_client *client, ...@@ -700,7 +700,6 @@ static int ad7746_probe(struct i2c_client *client,
indio_dev->num_channels = ARRAY_SIZE(ad7746_channels); indio_dev->num_channels = ARRAY_SIZE(ad7746_channels);
else else
indio_dev->num_channels = ARRAY_SIZE(ad7746_channels) - 2; indio_dev->num_channels = ARRAY_SIZE(ad7746_channels) - 2;
indio_dev->num_channels = ARRAY_SIZE(ad7746_channels);
indio_dev->modes = INDIO_DIRECT_MODE; indio_dev->modes = INDIO_DIRECT_MODE;
if (pdata) { if (pdata) {
......
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