Commit 1567bb7d authored by Laurent Pinchart's avatar Laurent Pinchart Committed by Mauro Carvalho Chehab

[media] omap3isp: Prevent pipelines that contain a crashed entity from starting

The OMAP3 ISP preview engine will violate the L4 bus protocol if we try
to write some of its internal registers after it failed to stop
properly. This generates an external abort on non-linefetch fault,
triggering a fatal kernel oops.

We can't always prevent preview engine stop failures (they can for
instance be caused by a sensor crash), but we can improve the system
reliability by refusing to start streaming on a pipeline that contains
the preview engine if it failed to stop. The driver will then eventually
reset the ISP (when all applications will have closed their file handles
related to OMAP3 ISP device nodes), making the ISP usable again.

Fixes: NB#291334 - camera: Recover gracefully from ISP crash instead of oopsing
Signed-off-by: default avatarLaurent Pinchart <laurent.pinchart@ideasonboard.com>
Acked-by: default avatarSakari Ailus <sakari.ailus@maxwell.research.nokia.com>
Reviewed-by: default avatarPhil Carmody <ext-phil.2.carmody@nokia.com>
Signed-off-by: default avatarMauro Carvalho Chehab <mchehab@redhat.com>
parent 4742c82e
...@@ -739,6 +739,17 @@ static int isp_pipeline_enable(struct isp_pipeline *pipe, ...@@ -739,6 +739,17 @@ static int isp_pipeline_enable(struct isp_pipeline *pipe,
unsigned long flags; unsigned long flags;
int ret; int ret;
/* If the preview engine crashed it might not respond to read/write
* operations on the L4 bus. This would result in a bus fault and a
* kernel oops. Refuse to start streaming in that case. This check must
* be performed before the loop below to avoid starting entities if the
* pipeline won't start anyway (those entities would then likely fail to
* stop, making the problem worse).
*/
if ((pipe->entities & isp->crashed) &
(1U << isp->isp_prev.subdev.entity.id))
return -EIO;
spin_lock_irqsave(&pipe->lock, flags); spin_lock_irqsave(&pipe->lock, flags);
pipe->state &= ~(ISP_PIPELINE_IDLE_INPUT | ISP_PIPELINE_IDLE_OUTPUT); pipe->state &= ~(ISP_PIPELINE_IDLE_INPUT | ISP_PIPELINE_IDLE_OUTPUT);
spin_unlock_irqrestore(&pipe->lock, flags); spin_unlock_irqrestore(&pipe->lock, flags);
...@@ -879,13 +890,15 @@ static int isp_pipeline_disable(struct isp_pipeline *pipe) ...@@ -879,13 +890,15 @@ static int isp_pipeline_disable(struct isp_pipeline *pipe)
if (ret) { if (ret) {
dev_info(isp->dev, "Unable to stop %s\n", subdev->name); dev_info(isp->dev, "Unable to stop %s\n", subdev->name);
/* If the entity failed to stopped, assume it has
* crashed. Mark it as such, the ISP will be reset when
* applications will release it.
*/
isp->crashed |= 1U << subdev->entity.id;
failure = -ETIMEDOUT; failure = -ETIMEDOUT;
} }
} }
if (failure < 0)
isp->needs_reset = true;
return failure; return failure;
} }
...@@ -1069,6 +1082,7 @@ static int isp_reset(struct isp_device *isp) ...@@ -1069,6 +1082,7 @@ static int isp_reset(struct isp_device *isp)
udelay(1); udelay(1);
} }
isp->crashed = 0;
return 0; return 0;
} }
...@@ -1496,10 +1510,11 @@ void omap3isp_put(struct isp_device *isp) ...@@ -1496,10 +1510,11 @@ void omap3isp_put(struct isp_device *isp)
if (--isp->ref_count == 0) { if (--isp->ref_count == 0) {
isp_disable_interrupts(isp); isp_disable_interrupts(isp);
isp_save_ctx(isp); isp_save_ctx(isp);
if (isp->needs_reset) { /* Reset the ISP if an entity has failed to stop. This is the
* only way to recover from such conditions.
*/
if (isp->crashed)
isp_reset(isp); isp_reset(isp);
isp->needs_reset = false;
}
isp_disable_clocks(isp); isp_disable_clocks(isp);
} }
mutex_unlock(&isp->isp_mutex); mutex_unlock(&isp->isp_mutex);
......
...@@ -145,6 +145,7 @@ struct isp_platform_callback { ...@@ -145,6 +145,7 @@ struct isp_platform_callback {
* @raw_dmamask: Raw DMA mask * @raw_dmamask: Raw DMA mask
* @stat_lock: Spinlock for handling statistics * @stat_lock: Spinlock for handling statistics
* @isp_mutex: Mutex for serializing requests to ISP. * @isp_mutex: Mutex for serializing requests to ISP.
* @crashed: Bitmask of crashed entities (indexed by entity ID)
* @has_context: Context has been saved at least once and can be restored. * @has_context: Context has been saved at least once and can be restored.
* @ref_count: Reference count for handling multiple ISP requests. * @ref_count: Reference count for handling multiple ISP requests.
* @cam_ick: Pointer to camera interface clock structure. * @cam_ick: Pointer to camera interface clock structure.
...@@ -184,7 +185,7 @@ struct isp_device { ...@@ -184,7 +185,7 @@ struct isp_device {
/* ISP Obj */ /* ISP Obj */
spinlock_t stat_lock; /* common lock for statistic drivers */ spinlock_t stat_lock; /* common lock for statistic drivers */
struct mutex isp_mutex; /* For handling ref_count field */ struct mutex isp_mutex; /* For handling ref_count field */
bool needs_reset; u32 crashed;
int has_context; int has_context;
int ref_count; int ref_count;
unsigned int autoidle; unsigned int autoidle;
......
...@@ -292,6 +292,7 @@ static int isp_video_validate_pipeline(struct isp_pipeline *pipe) ...@@ -292,6 +292,7 @@ static int isp_video_validate_pipeline(struct isp_pipeline *pipe)
int ret; int ret;
pipe->max_rate = pipe->l3_ick; pipe->max_rate = pipe->l3_ick;
pipe->entities = 0;
subdev = isp_video_remote_subdev(pipe->output, NULL); subdev = isp_video_remote_subdev(pipe->output, NULL);
if (subdev == NULL) if (subdev == NULL)
...@@ -299,6 +300,9 @@ static int isp_video_validate_pipeline(struct isp_pipeline *pipe) ...@@ -299,6 +300,9 @@ static int isp_video_validate_pipeline(struct isp_pipeline *pipe)
while (1) { while (1) {
unsigned int shifter_link; unsigned int shifter_link;
pipe->entities |= 1U << subdev->entity.id;
/* Retrieve the sink format */ /* Retrieve the sink format */
pad = &subdev->entity.pads[0]; pad = &subdev->entity.pads[0];
if (!(pad->flags & MEDIA_PAD_FL_SINK)) if (!(pad->flags & MEDIA_PAD_FL_SINK))
......
...@@ -88,6 +88,7 @@ enum isp_pipeline_state { ...@@ -88,6 +88,7 @@ enum isp_pipeline_state {
/* /*
* struct isp_pipeline - An ISP hardware pipeline * struct isp_pipeline - An ISP hardware pipeline
* @error: A hardware error occurred during capture * @error: A hardware error occurred during capture
* @entities: Bitmask of entities in the pipeline (indexed by entity ID)
*/ */
struct isp_pipeline { struct isp_pipeline {
struct media_pipeline pipe; struct media_pipeline pipe;
...@@ -96,6 +97,7 @@ struct isp_pipeline { ...@@ -96,6 +97,7 @@ struct isp_pipeline {
enum isp_pipeline_stream_state stream_state; enum isp_pipeline_stream_state stream_state;
struct isp_video *input; struct isp_video *input;
struct isp_video *output; struct isp_video *output;
u32 entities;
unsigned long l3_ick; unsigned long l3_ick;
unsigned int max_rate; unsigned int max_rate;
atomic_t frame_number; atomic_t frame_number;
......
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