Commit 3056a8e9 authored by Laurent Pinchart's avatar Laurent Pinchart Committed by Sakari Ailus

media: media-entity: Simplify media_pipeline_start()

The media_pipeline_start() function has two purposes: it constructs a
pipeline by recording the entities that are part of it, gathered from a
graph walk, and validate the media links. The pipeline pointer is stored
in the media_entity structure as part of this process, and the entity's
stream count is increased, to record that the entity is streaming.

When multiple video nodes are present in a pipeline,
media_pipeline_start() is typically called on all of them, with the same
pipeline pointer. This is taken into account in media_pipeline_start()
by skipping validation for entities that are already part of the
pipeline, while returning an error if an entity is part of a different
pipeline.

It turns out that this process is overly complicated. When
media_pipeline_start() is called for the first time, it constructs the
full pipeline, adding all entities and validating all the links.
Subsequent calls to media_pipeline_start() are then nearly no-ops, they
only increase the stream count on the pipeline and on all entities.

The media_entity stream_count field is used for two purposes: checking
if the entity is streaming, and detecting when a call to
media_pipeline_stop() balances needs to reset the entity pipe pointer to
NULL. The former can easily be replaced by a check of the pipe pointer.

Simplify media_pipeline_start() by avoiding the pipeline walk on all
calls but the first one, and drop the media_entity stream_count field.
media_pipeline_stop() is updated accordingly.
Signed-off-by: default avatarLaurent Pinchart <laurent.pinchart@ideasonboard.com>
[Sakari Ailus: Drop redundant '!= NULL' as discussed]
Signed-off-by: default avatarSakari Ailus <sakari.ailus@linux.intel.com>
parent f17bc788
...@@ -396,20 +396,21 @@ __must_check int __media_pipeline_start(struct media_entity *entity, ...@@ -396,20 +396,21 @@ __must_check int __media_pipeline_start(struct media_entity *entity,
struct media_link *link; struct media_link *link;
int ret; int ret;
if (!pipe->streaming_count++) { if (pipe->streaming_count) {
ret = media_graph_walk_init(&pipe->graph, mdev); pipe->streaming_count++;
if (ret) return 0;
goto error_graph_walk_start;
} }
ret = media_graph_walk_init(&pipe->graph, mdev);
if (ret)
return ret;
media_graph_walk_start(&pipe->graph, entity); media_graph_walk_start(&pipe->graph, entity);
while ((entity = media_graph_walk_next(graph))) { while ((entity = media_graph_walk_next(graph))) {
DECLARE_BITMAP(active, MEDIA_ENTITY_MAX_PADS); DECLARE_BITMAP(active, MEDIA_ENTITY_MAX_PADS);
DECLARE_BITMAP(has_no_links, MEDIA_ENTITY_MAX_PADS); DECLARE_BITMAP(has_no_links, MEDIA_ENTITY_MAX_PADS);
entity->stream_count++;
if (entity->pipe && entity->pipe != pipe) { if (entity->pipe && entity->pipe != pipe) {
pr_err("Pipe active for %s. Can't start for %s\n", pr_err("Pipe active for %s. Can't start for %s\n",
entity->name, entity->name,
...@@ -418,12 +419,12 @@ __must_check int __media_pipeline_start(struct media_entity *entity, ...@@ -418,12 +419,12 @@ __must_check int __media_pipeline_start(struct media_entity *entity,
goto error; goto error;
} }
entity->pipe = pipe;
/* Already streaming --- no need to check. */ /* Already streaming --- no need to check. */
if (entity->stream_count > 1) if (entity->pipe)
continue; continue;
entity->pipe = pipe;
if (!entity->ops || !entity->ops->link_validate) if (!entity->ops || !entity->ops->link_validate)
continue; continue;
...@@ -479,6 +480,8 @@ __must_check int __media_pipeline_start(struct media_entity *entity, ...@@ -479,6 +480,8 @@ __must_check int __media_pipeline_start(struct media_entity *entity,
} }
} }
pipe->streaming_count++;
return 0; return 0;
error: error:
...@@ -489,24 +492,17 @@ __must_check int __media_pipeline_start(struct media_entity *entity, ...@@ -489,24 +492,17 @@ __must_check int __media_pipeline_start(struct media_entity *entity,
media_graph_walk_start(graph, entity_err); media_graph_walk_start(graph, entity_err);
while ((entity_err = media_graph_walk_next(graph))) { while ((entity_err = media_graph_walk_next(graph))) {
/* Sanity check for negative stream_count */ entity_err->pipe = NULL;
if (!WARN_ON_ONCE(entity_err->stream_count <= 0)) {
entity_err->stream_count--;
if (entity_err->stream_count == 0)
entity_err->pipe = NULL;
}
/* /*
* We haven't increased stream_count further than this * We haven't started entities further than this so we quit
* so we quit here. * here.
*/ */
if (entity_err == entity) if (entity_err == entity)
break; break;
} }
error_graph_walk_start: media_graph_walk_cleanup(graph);
if (!--pipe->streaming_count)
media_graph_walk_cleanup(graph);
return ret; return ret;
} }
...@@ -537,19 +533,15 @@ void __media_pipeline_stop(struct media_entity *entity) ...@@ -537,19 +533,15 @@ void __media_pipeline_stop(struct media_entity *entity)
if (WARN_ON(!pipe)) if (WARN_ON(!pipe))
return; return;
if (--pipe->streaming_count)
return;
media_graph_walk_start(graph, entity); media_graph_walk_start(graph, entity);
while ((entity = media_graph_walk_next(graph))) { while ((entity = media_graph_walk_next(graph)))
/* Sanity check for negative stream_count */ entity->pipe = NULL;
if (!WARN_ON_ONCE(entity->stream_count <= 0)) {
entity->stream_count--;
if (entity->stream_count == 0)
entity->pipe = NULL;
}
}
if (!--pipe->streaming_count) media_graph_walk_cleanup(graph);
media_graph_walk_cleanup(graph);
} }
EXPORT_SYMBOL_GPL(__media_pipeline_stop); EXPORT_SYMBOL_GPL(__media_pipeline_stop);
......
...@@ -268,7 +268,6 @@ enum media_entity_type { ...@@ -268,7 +268,6 @@ enum media_entity_type {
* @pads: Pads array with the size defined by @num_pads. * @pads: Pads array with the size defined by @num_pads.
* @links: List of data links. * @links: List of data links.
* @ops: Entity operations. * @ops: Entity operations.
* @stream_count: Stream count for the entity.
* @use_count: Use count for the entity. * @use_count: Use count for the entity.
* @pipe: Pipeline this entity belongs to. * @pipe: Pipeline this entity belongs to.
* @info: Union with devnode information. Kept just for backward * @info: Union with devnode information. Kept just for backward
...@@ -283,10 +282,9 @@ enum media_entity_type { ...@@ -283,10 +282,9 @@ enum media_entity_type {
* *
* .. note:: * .. note::
* *
* @stream_count and @use_count reference counts must never be * The @use_count reference count must never be negative, but is a signed
* negative, but are signed integers on purpose: a simple ``WARN_ON(<0)`` * integer on purpose: a simple ``WARN_ON(<0)`` check can be used to detect
* check can be used to detect reference count bugs that would make them * reference count bugs that would make it negative.
* negative.
*/ */
struct media_entity { struct media_entity {
struct media_gobj graph_obj; /* must be first field in struct */ struct media_gobj graph_obj; /* must be first field in struct */
...@@ -305,7 +303,6 @@ struct media_entity { ...@@ -305,7 +303,6 @@ struct media_entity {
const struct media_entity_operations *ops; const struct media_entity_operations *ops;
int stream_count;
int use_count; int use_count;
struct media_pipeline *pipe; struct media_pipeline *pipe;
...@@ -867,7 +864,7 @@ struct media_pad *media_entity_remote_pad(const struct media_pad *pad); ...@@ -867,7 +864,7 @@ struct media_pad *media_entity_remote_pad(const struct media_pad *pad);
*/ */
static inline bool media_entity_is_streaming(const struct media_entity *entity) static inline bool media_entity_is_streaming(const struct media_entity *entity)
{ {
return entity->stream_count > 0; return entity->pipe;
} }
/** /**
......
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