Commit 1250186a authored by Lars-Peter Clausen's avatar Lars-Peter Clausen Committed by Jonathan Cameron

iio: __iio_update_buffers: Leave device in sane state on error

Currently when something goes wrong at some step when disabling the buffers
we immediately abort. This has the effect that the enable/disable calls are
no longer balanced. So make sure that even if one step in the disable
sequence fails the other steps are still executed.

The other issue is that when either enable or disable fails buffers that
were active at that time stay active while the device itself is disabled.
This leaves things in a inconsistent state and can cause unbalanced
enable/disable calls. Furthermore when enable fails we restore the old scan
mask, but still keeps things disabled.

Given that verification of the configuration was performed earlier and it
is valid at the point where we try to enable/disable the most likely reason
of failure is a communication failure with the device or maybe a
out-of-memory situation. There is not really a good recovery strategy in
such a case, so it makes sense to leave the device disabled, but we should
still leave it in a consistent state.

What the patch does if disable/enable fails is to deactivate all buffers
and make sure that the device will be in the same state as if all buffers
had been manually disabled.
Signed-off-by: default avatarLars-Peter Clausen <lars@metafoo.de>
Signed-off-by: default avatarJonathan Cameron <jic23@kernel.org>
parent 623d74e3
...@@ -539,6 +539,15 @@ static void iio_buffer_deactivate(struct iio_buffer *buffer) ...@@ -539,6 +539,15 @@ static void iio_buffer_deactivate(struct iio_buffer *buffer)
iio_buffer_put(buffer); iio_buffer_put(buffer);
} }
static void iio_buffer_deactivate_all(struct iio_dev *indio_dev)
{
struct iio_buffer *buffer, *_buffer;
list_for_each_entry_safe(buffer, _buffer,
&indio_dev->buffer_list, buffer_list)
iio_buffer_deactivate(buffer);
}
static void iio_buffer_update_bytes_per_datum(struct iio_dev *indio_dev, static void iio_buffer_update_bytes_per_datum(struct iio_dev *indio_dev,
struct iio_buffer *buffer) struct iio_buffer *buffer)
{ {
...@@ -719,36 +728,46 @@ static int iio_enable_buffers(struct iio_dev *indio_dev, ...@@ -719,36 +728,46 @@ static int iio_enable_buffers(struct iio_dev *indio_dev,
static int iio_disable_buffers(struct iio_dev *indio_dev) static int iio_disable_buffers(struct iio_dev *indio_dev)
{ {
int ret; int ret = 0;
int ret2;
/* Wind down existing buffers - iff there are any */ /* Wind down existing buffers - iff there are any */
if (list_empty(&indio_dev->buffer_list)) if (list_empty(&indio_dev->buffer_list))
return 0; return 0;
/*
* If things go wrong at some step in disable we still need to continue
* to perform the other steps, otherwise we leave the device in a
* inconsistent state. We return the error code for the first error we
* encountered.
*/
if (indio_dev->setup_ops->predisable) { if (indio_dev->setup_ops->predisable) {
ret = indio_dev->setup_ops->predisable(indio_dev); ret2 = indio_dev->setup_ops->predisable(indio_dev);
if (ret) if (ret2 && !ret)
return ret; ret = ret2;
} }
indio_dev->currentmode = INDIO_DIRECT_MODE; indio_dev->currentmode = INDIO_DIRECT_MODE;
if (indio_dev->setup_ops->postdisable) { if (indio_dev->setup_ops->postdisable) {
ret = indio_dev->setup_ops->postdisable(indio_dev); ret2 = indio_dev->setup_ops->postdisable(indio_dev);
if (ret) if (ret2 && !ret)
return ret; ret = ret2;
} }
return 0; iio_free_scan_mask(indio_dev, indio_dev->active_scan_mask);
indio_dev->active_scan_mask = NULL;
return ret;
} }
static int __iio_update_buffers(struct iio_dev *indio_dev, static int __iio_update_buffers(struct iio_dev *indio_dev,
struct iio_buffer *insert_buffer, struct iio_buffer *insert_buffer,
struct iio_buffer *remove_buffer) struct iio_buffer *remove_buffer)
{ {
int ret;
const unsigned long *old_mask;
struct iio_device_config new_config; struct iio_device_config new_config;
int ret;
ret = iio_verify_update(indio_dev, insert_buffer, remove_buffer, ret = iio_verify_update(indio_dev, insert_buffer, remove_buffer,
&new_config); &new_config);
...@@ -761,15 +780,9 @@ static int __iio_update_buffers(struct iio_dev *indio_dev, ...@@ -761,15 +780,9 @@ static int __iio_update_buffers(struct iio_dev *indio_dev,
goto err_free_config; goto err_free_config;
} }
/* Keep a copy of current setup to allow roll back */
old_mask = indio_dev->active_scan_mask;
indio_dev->active_scan_mask = NULL;
ret = iio_disable_buffers(indio_dev); ret = iio_disable_buffers(indio_dev);
if (ret) { if (ret)
iio_free_scan_mask(indio_dev, old_mask); goto err_deactivate_all;
goto err_free_config;
}
if (remove_buffer) if (remove_buffer)
iio_buffer_deactivate(remove_buffer); iio_buffer_deactivate(remove_buffer);
...@@ -777,22 +790,26 @@ static int __iio_update_buffers(struct iio_dev *indio_dev, ...@@ -777,22 +790,26 @@ static int __iio_update_buffers(struct iio_dev *indio_dev,
iio_buffer_activate(indio_dev, insert_buffer); iio_buffer_activate(indio_dev, insert_buffer);
/* If no buffers in list, we are done */ /* If no buffers in list, we are done */
if (list_empty(&indio_dev->buffer_list)) { if (list_empty(&indio_dev->buffer_list))
iio_free_scan_mask(indio_dev, old_mask);
return 0; return 0;
}
ret = iio_enable_buffers(indio_dev, &new_config); ret = iio_enable_buffers(indio_dev, &new_config);
if (ret) { if (ret)
if (insert_buffer) goto err_deactivate_all;
iio_buffer_deactivate(insert_buffer);
indio_dev->active_scan_mask = old_mask;
goto err_free_config;
}
iio_free_scan_mask(indio_dev, old_mask);
return 0; return 0;
err_deactivate_all:
/*
* We've already verified that the config is valid earlier. If things go
* wrong in either enable or disable the most likely reason is an IO
* error from the device. In this case there is no good recovery
* strategy. Just make sure to disable everything and leave the device
* in a sane state. With a bit of luck the device might come back to
* life again later and userspace can try again.
*/
iio_buffer_deactivate_all(indio_dev);
err_free_config: err_free_config:
iio_free_scan_mask(indio_dev, new_config.scan_mask); iio_free_scan_mask(indio_dev, new_config.scan_mask);
return ret; return ret;
...@@ -838,15 +855,8 @@ EXPORT_SYMBOL_GPL(iio_update_buffers); ...@@ -838,15 +855,8 @@ EXPORT_SYMBOL_GPL(iio_update_buffers);
void iio_disable_all_buffers(struct iio_dev *indio_dev) void iio_disable_all_buffers(struct iio_dev *indio_dev)
{ {
struct iio_buffer *buffer, *_buffer;
iio_disable_buffers(indio_dev); iio_disable_buffers(indio_dev);
iio_free_scan_mask(indio_dev, indio_dev->active_scan_mask); iio_buffer_deactivate_all(indio_dev);
indio_dev->active_scan_mask = NULL;
list_for_each_entry_safe(buffer, _buffer,
&indio_dev->buffer_list, buffer_list)
iio_buffer_deactivate(buffer);
} }
static ssize_t iio_buffer_store_enable(struct device *dev, static ssize_t iio_buffer_store_enable(struct device *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