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

[media] v4l: omap4iss: Reset the ISS when the pipeline can't be stopped

When a failure to stop a module in the pipeline is detected, the only
way to recover is to reset the ISS. However, as other users can be using
a different pipeline with other modules, the ISS can't be reset
synchronously with the error detection.
Keep track of modules that have failed to stop, and reset the ISS
accordingly when the last user releases the last reference to the ISS.
Refuse to start streaming on a pipeline that contains a crashed module,
as the hardware wouldn't work anyway.
Modify the omap4iss_pipeline_set_stream() function to record the new ISS
pipeline state only when no error occurs, except when stopping the
pipeline in which case the pipeline is still marked as stopped.
Signed-off-by: default avatarLaurent Pinchart <laurent.pinchart@ideasonboard.com>
Signed-off-by: default avatarMauro Carvalho Chehab <m.chehab@samsung.com>
parent af15d025
...@@ -573,12 +573,22 @@ static int iss_pipeline_link_notify(struct media_link *link, u32 flags, ...@@ -573,12 +573,22 @@ static int iss_pipeline_link_notify(struct media_link *link, u32 flags,
static int iss_pipeline_enable(struct iss_pipeline *pipe, static int iss_pipeline_enable(struct iss_pipeline *pipe,
enum iss_pipeline_stream_state mode) enum iss_pipeline_stream_state mode)
{ {
struct iss_device *iss = pipe->output->iss;
struct media_entity *entity; struct media_entity *entity;
struct media_pad *pad; struct media_pad *pad;
struct v4l2_subdev *subdev; struct v4l2_subdev *subdev;
unsigned long flags; unsigned long flags;
int ret; int ret;
/* If one of the entities in the pipeline has crashed it will not work
* properly. 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 & iss->crashed)
return -EIO;
spin_lock_irqsave(&pipe->lock, flags); spin_lock_irqsave(&pipe->lock, flags);
pipe->state &= ~(ISS_PIPELINE_IDLE_INPUT | ISS_PIPELINE_IDLE_OUTPUT); pipe->state &= ~(ISS_PIPELINE_IDLE_INPUT | ISS_PIPELINE_IDLE_OUTPUT);
spin_unlock_irqrestore(&pipe->lock, flags); spin_unlock_irqrestore(&pipe->lock, flags);
...@@ -617,6 +627,7 @@ static int iss_pipeline_enable(struct iss_pipeline *pipe, ...@@ -617,6 +627,7 @@ static int iss_pipeline_enable(struct iss_pipeline *pipe,
*/ */
static int iss_pipeline_disable(struct iss_pipeline *pipe) static int iss_pipeline_disable(struct iss_pipeline *pipe)
{ {
struct iss_device *iss = pipe->output->iss;
struct media_entity *entity; struct media_entity *entity;
struct media_pad *pad; struct media_pad *pad;
struct v4l2_subdev *subdev; struct v4l2_subdev *subdev;
...@@ -641,6 +652,11 @@ static int iss_pipeline_disable(struct iss_pipeline *pipe) ...@@ -641,6 +652,11 @@ static int iss_pipeline_disable(struct iss_pipeline *pipe)
if (ret < 0) { if (ret < 0) {
dev_dbg(iss->dev, "%s: module stop timeout.\n", dev_dbg(iss->dev, "%s: module stop timeout.\n",
subdev->name); subdev->name);
/* If the entity failed to stopped, assume it has
* crashed. Mark it as such, the ISS will be reset when
* applications will release it.
*/
iss->crashed |= 1U << subdev->entity.id;
failure = -ETIMEDOUT; failure = -ETIMEDOUT;
} }
} }
...@@ -715,6 +731,7 @@ static int iss_reset(struct iss_device *iss) ...@@ -715,6 +731,7 @@ static int iss_reset(struct iss_device *iss)
usleep_range(10, 10); usleep_range(10, 10);
} }
iss->crashed = 0;
return 0; return 0;
} }
...@@ -1058,6 +1075,13 @@ void omap4iss_put(struct iss_device *iss) ...@@ -1058,6 +1075,13 @@ void omap4iss_put(struct iss_device *iss)
BUG_ON(iss->ref_count == 0); BUG_ON(iss->ref_count == 0);
if (--iss->ref_count == 0) { if (--iss->ref_count == 0) {
iss_disable_interrupts(iss); iss_disable_interrupts(iss);
/* Reset the ISS if an entity has failed to stop. This is the
* only way to recover from such conditions, although it would
* be worth investigating whether resetting the ISP only can't
* fix the problem in some cases.
*/
if (iss->crashed)
iss_reset(iss);
iss_disable_clocks(iss); iss_disable_clocks(iss);
} }
mutex_unlock(&iss->iss_mutex); mutex_unlock(&iss->iss_mutex);
......
...@@ -77,6 +77,10 @@ struct iss_reg { ...@@ -77,6 +77,10 @@ struct iss_reg {
u32 val; u32 val;
}; };
/*
* struct iss_device - ISS device structure.
* @crashed: Bitmask of crashed entities (indexed by entity ID)
*/
struct iss_device { struct iss_device {
struct v4l2_device v4l2_dev; struct v4l2_device v4l2_dev;
struct media_device media_dev; struct media_device media_dev;
...@@ -93,6 +97,7 @@ struct iss_device { ...@@ -93,6 +97,7 @@ struct iss_device {
u64 raw_dmamask; u64 raw_dmamask;
struct mutex iss_mutex; /* For handling ref_count field */ struct mutex iss_mutex; /* For handling ref_count field */
bool crashed;
int has_context; int has_context;
int ref_count; int ref_count;
......
...@@ -772,6 +772,8 @@ iss_video_streamon(struct file *file, void *fh, enum v4l2_buf_type type) ...@@ -772,6 +772,8 @@ iss_video_streamon(struct file *file, void *fh, enum v4l2_buf_type type)
{ {
struct iss_video_fh *vfh = to_iss_video_fh(fh); struct iss_video_fh *vfh = to_iss_video_fh(fh);
struct iss_video *video = video_drvdata(file); struct iss_video *video = video_drvdata(file);
struct media_entity_graph graph;
struct media_entity *entity;
enum iss_pipeline_state state; enum iss_pipeline_state state;
struct iss_pipeline *pipe; struct iss_pipeline *pipe;
struct iss_video *far_end; struct iss_video *far_end;
...@@ -791,6 +793,7 @@ iss_video_streamon(struct file *file, void *fh, enum v4l2_buf_type type) ...@@ -791,6 +793,7 @@ iss_video_streamon(struct file *file, void *fh, enum v4l2_buf_type type)
pipe->external = NULL; pipe->external = NULL;
pipe->external_rate = 0; pipe->external_rate = 0;
pipe->external_bpp = 0; pipe->external_bpp = 0;
pipe->entities = 0;
if (video->iss->pdata->set_constraints) if (video->iss->pdata->set_constraints)
video->iss->pdata->set_constraints(video->iss, true); video->iss->pdata->set_constraints(video->iss, true);
...@@ -799,6 +802,11 @@ iss_video_streamon(struct file *file, void *fh, enum v4l2_buf_type type) ...@@ -799,6 +802,11 @@ iss_video_streamon(struct file *file, void *fh, enum v4l2_buf_type type)
if (ret < 0) if (ret < 0)
goto err_media_entity_pipeline_start; goto err_media_entity_pipeline_start;
entity = &video->video.entity;
media_entity_graph_walk_start(&graph, entity);
while ((entity = media_entity_graph_walk_next(&graph)))
pipe->entities |= 1 << entity->id;
/* Verify that the currently configured format matches the output of /* Verify that the currently configured format matches the output of
* the connected subdev. * the connected subdev.
*/ */
......
...@@ -77,6 +77,7 @@ enum iss_pipeline_state { ...@@ -77,6 +77,7 @@ enum iss_pipeline_state {
/* /*
* struct iss_pipeline - An OMAP4 ISS hardware pipeline * struct iss_pipeline - An OMAP4 ISS hardware pipeline
* @entities: Bitmask of entities in the pipeline (indexed by entity ID)
* @error: A hardware error occurred during capture * @error: A hardware error occurred during capture
*/ */
struct iss_pipeline { struct iss_pipeline {
...@@ -86,6 +87,7 @@ struct iss_pipeline { ...@@ -86,6 +87,7 @@ struct iss_pipeline {
enum iss_pipeline_stream_state stream_state; enum iss_pipeline_stream_state stream_state;
struct iss_video *input; struct iss_video *input;
struct iss_video *output; struct iss_video *output;
unsigned int entities;
atomic_t frame_number; atomic_t frame_number;
bool do_propagation; /* of frame number */ bool do_propagation; /* of frame number */
bool error; bool error;
......
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