Commit d950fd99 authored by Niklas Söderlund's avatar Niklas Söderlund Committed by Mauro Carvalho Chehab

media: rcar-vin: Fix lockdep warning at stream on

Changes to v4l2-fwnode in commit [1] triggered a lockdep warning in
rcar-vin. The first attempt to solve this warning in the rcar-vin driver
was incomplete and only pushed the warning to happen at stream on time
instead of at probe time.

This change reverts the incomplete fix and properly fixes the warning by
removing the need to hold the rcar-vin specific group lock when calling
v4l2_async_notifier_parse_fwnode_endpoints_by_port(). And instead takes
it in the callback where it's really needed.

[1] commit eae2aed1 ("media: v4l2-fwnode: Switch to
v4l2_async_notifier_add_subdev")

Fixes: 6458afc8 ("media: rcar-vin: remove unneeded locking in async callbacks")
Signed-off-by: default avatarNiklas Söderlund <niklas.soderlund+renesas@ragnatech.se>
Reviewed-by: default avatarKieran Bingham <kieran.bingham+renesas@ideasonboard.com>
Signed-off-by: default avatarHans Verkuil <hverkuil-cisco@xs4all.nl>
Signed-off-by: default avatarMauro Carvalho Chehab <mchehab+samsung@kernel.org>
parent 2b177f28
...@@ -546,7 +546,9 @@ static void rvin_parallel_notify_unbind(struct v4l2_async_notifier *notifier, ...@@ -546,7 +546,9 @@ static void rvin_parallel_notify_unbind(struct v4l2_async_notifier *notifier,
vin_dbg(vin, "unbind parallel subdev %s\n", subdev->name); vin_dbg(vin, "unbind parallel subdev %s\n", subdev->name);
mutex_lock(&vin->lock);
rvin_parallel_subdevice_detach(vin); rvin_parallel_subdevice_detach(vin);
mutex_unlock(&vin->lock);
} }
static int rvin_parallel_notify_bound(struct v4l2_async_notifier *notifier, static int rvin_parallel_notify_bound(struct v4l2_async_notifier *notifier,
...@@ -556,7 +558,9 @@ static int rvin_parallel_notify_bound(struct v4l2_async_notifier *notifier, ...@@ -556,7 +558,9 @@ static int rvin_parallel_notify_bound(struct v4l2_async_notifier *notifier,
struct rvin_dev *vin = v4l2_dev_to_vin(notifier->v4l2_dev); struct rvin_dev *vin = v4l2_dev_to_vin(notifier->v4l2_dev);
int ret; int ret;
mutex_lock(&vin->lock);
ret = rvin_parallel_subdevice_attach(vin, subdev); ret = rvin_parallel_subdevice_attach(vin, subdev);
mutex_unlock(&vin->lock);
if (ret) if (ret)
return ret; return ret;
...@@ -664,6 +668,7 @@ static int rvin_group_notify_complete(struct v4l2_async_notifier *notifier) ...@@ -664,6 +668,7 @@ static int rvin_group_notify_complete(struct v4l2_async_notifier *notifier)
} }
/* Create all media device links between VINs and CSI-2's. */ /* Create all media device links between VINs and CSI-2's. */
mutex_lock(&vin->group->lock);
for (route = vin->info->routes; route->mask; route++) { for (route = vin->info->routes; route->mask; route++) {
struct media_pad *source_pad, *sink_pad; struct media_pad *source_pad, *sink_pad;
struct media_entity *source, *sink; struct media_entity *source, *sink;
...@@ -699,6 +704,7 @@ static int rvin_group_notify_complete(struct v4l2_async_notifier *notifier) ...@@ -699,6 +704,7 @@ static int rvin_group_notify_complete(struct v4l2_async_notifier *notifier)
break; break;
} }
} }
mutex_unlock(&vin->group->lock);
return ret; return ret;
} }
...@@ -714,6 +720,8 @@ static void rvin_group_notify_unbind(struct v4l2_async_notifier *notifier, ...@@ -714,6 +720,8 @@ static void rvin_group_notify_unbind(struct v4l2_async_notifier *notifier,
if (vin->group->vin[i]) if (vin->group->vin[i])
rvin_v4l2_unregister(vin->group->vin[i]); rvin_v4l2_unregister(vin->group->vin[i]);
mutex_lock(&vin->group->lock);
for (i = 0; i < RVIN_CSI_MAX; i++) { for (i = 0; i < RVIN_CSI_MAX; i++) {
if (vin->group->csi[i].fwnode != asd->match.fwnode) if (vin->group->csi[i].fwnode != asd->match.fwnode)
continue; continue;
...@@ -721,6 +729,8 @@ static void rvin_group_notify_unbind(struct v4l2_async_notifier *notifier, ...@@ -721,6 +729,8 @@ static void rvin_group_notify_unbind(struct v4l2_async_notifier *notifier,
vin_dbg(vin, "Unbind CSI-2 %s from slot %u\n", subdev->name, i); vin_dbg(vin, "Unbind CSI-2 %s from slot %u\n", subdev->name, i);
break; break;
} }
mutex_unlock(&vin->group->lock);
} }
static int rvin_group_notify_bound(struct v4l2_async_notifier *notifier, static int rvin_group_notify_bound(struct v4l2_async_notifier *notifier,
...@@ -730,6 +740,8 @@ static int rvin_group_notify_bound(struct v4l2_async_notifier *notifier, ...@@ -730,6 +740,8 @@ static int rvin_group_notify_bound(struct v4l2_async_notifier *notifier,
struct rvin_dev *vin = v4l2_dev_to_vin(notifier->v4l2_dev); struct rvin_dev *vin = v4l2_dev_to_vin(notifier->v4l2_dev);
unsigned int i; unsigned int i;
mutex_lock(&vin->group->lock);
for (i = 0; i < RVIN_CSI_MAX; i++) { for (i = 0; i < RVIN_CSI_MAX; i++) {
if (vin->group->csi[i].fwnode != asd->match.fwnode) if (vin->group->csi[i].fwnode != asd->match.fwnode)
continue; continue;
...@@ -738,6 +750,8 @@ static int rvin_group_notify_bound(struct v4l2_async_notifier *notifier, ...@@ -738,6 +750,8 @@ static int rvin_group_notify_bound(struct v4l2_async_notifier *notifier,
break; break;
} }
mutex_unlock(&vin->group->lock);
return 0; return 0;
} }
...@@ -752,6 +766,7 @@ static int rvin_mc_parse_of_endpoint(struct device *dev, ...@@ -752,6 +766,7 @@ static int rvin_mc_parse_of_endpoint(struct device *dev,
struct v4l2_async_subdev *asd) struct v4l2_async_subdev *asd)
{ {
struct rvin_dev *vin = dev_get_drvdata(dev); struct rvin_dev *vin = dev_get_drvdata(dev);
int ret = 0;
if (vep->base.port != 1 || vep->base.id >= RVIN_CSI_MAX) if (vep->base.port != 1 || vep->base.id >= RVIN_CSI_MAX)
return -EINVAL; return -EINVAL;
...@@ -762,38 +777,48 @@ static int rvin_mc_parse_of_endpoint(struct device *dev, ...@@ -762,38 +777,48 @@ static int rvin_mc_parse_of_endpoint(struct device *dev,
return -ENOTCONN; return -ENOTCONN;
} }
mutex_lock(&vin->group->lock);
if (vin->group->csi[vep->base.id].fwnode) { if (vin->group->csi[vep->base.id].fwnode) {
vin_dbg(vin, "OF device %pOF already handled\n", vin_dbg(vin, "OF device %pOF already handled\n",
to_of_node(asd->match.fwnode)); to_of_node(asd->match.fwnode));
return -ENOTCONN; ret = -ENOTCONN;
goto out;
} }
vin->group->csi[vep->base.id].fwnode = asd->match.fwnode; vin->group->csi[vep->base.id].fwnode = asd->match.fwnode;
vin_dbg(vin, "Add group OF device %pOF to slot %u\n", vin_dbg(vin, "Add group OF device %pOF to slot %u\n",
to_of_node(asd->match.fwnode), vep->base.id); to_of_node(asd->match.fwnode), vep->base.id);
out:
mutex_unlock(&vin->group->lock);
return 0; return ret;
} }
static int rvin_mc_parse_of_graph(struct rvin_dev *vin) static int rvin_mc_parse_of_graph(struct rvin_dev *vin)
{ {
unsigned int count = 0; unsigned int count = 0, vin_mask = 0;
unsigned int i; unsigned int i;
int ret; int ret;
mutex_lock(&vin->group->lock); mutex_lock(&vin->group->lock);
/* If not all VIN's are registered don't register the notifier. */ /* If not all VIN's are registered don't register the notifier. */
for (i = 0; i < RCAR_VIN_NUM; i++) for (i = 0; i < RCAR_VIN_NUM; i++) {
if (vin->group->vin[i]) if (vin->group->vin[i]) {
count++; count++;
vin_mask |= BIT(i);
}
}
if (vin->group->count != count) { if (vin->group->count != count) {
mutex_unlock(&vin->group->lock); mutex_unlock(&vin->group->lock);
return 0; return 0;
} }
mutex_unlock(&vin->group->lock);
v4l2_async_notifier_init(&vin->group->notifier); v4l2_async_notifier_init(&vin->group->notifier);
/* /*
...@@ -802,20 +827,16 @@ static int rvin_mc_parse_of_graph(struct rvin_dev *vin) ...@@ -802,20 +827,16 @@ static int rvin_mc_parse_of_graph(struct rvin_dev *vin)
* will only be registered once with the group notifier. * will only be registered once with the group notifier.
*/ */
for (i = 0; i < RCAR_VIN_NUM; i++) { for (i = 0; i < RCAR_VIN_NUM; i++) {
if (!vin->group->vin[i]) if (!(vin_mask & BIT(i)))
continue; continue;
ret = v4l2_async_notifier_parse_fwnode_endpoints_by_port( ret = v4l2_async_notifier_parse_fwnode_endpoints_by_port(
vin->group->vin[i]->dev, &vin->group->notifier, vin->group->vin[i]->dev, &vin->group->notifier,
sizeof(struct v4l2_async_subdev), 1, sizeof(struct v4l2_async_subdev), 1,
rvin_mc_parse_of_endpoint); rvin_mc_parse_of_endpoint);
if (ret) { if (ret)
mutex_unlock(&vin->group->lock);
return ret; return ret;
} }
}
mutex_unlock(&vin->group->lock);
if (list_empty(&vin->group->notifier.asd_list)) if (list_empty(&vin->group->notifier.asd_list))
return 0; return 0;
......
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