Re: [PATCH] media: v4l2-async: fix binding async subdevs with multiple source ports

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

 



Hi Laurent,

On Tue, Jan 17, 2023 at 03:29:33PM +0200, Laurent Pinchart wrote:
> On Tue, Jan 17, 2023 at 01:16:30PM +0000, Sakari Ailus wrote:
> > Hi Philipp,
> > 
> > On Wed, Aug 10, 2022 at 12:48:48PM +0200, Philipp Zabel wrote:
> > > Asynchronous subdevice probing on imx-media with imx6-mipi-csi2 is
> > > broken since commit 1f391df44607 ("media: v4l2-async: Use endpoints in
> > > __v4l2_async_nf_add_fwnode_remote()").
> > > 
> > > This is a side effect of imx6-mipi-csi2 having a single subdevice with
> > > four separate source ports connected to different subdevices. Before,
> > > the imx-media-csi and video-mux devices registered async sub-devices
> > > via device fwnode that matched the imx6-mipi-csi2 device on their
> > > respective notifiers. This caused the first asd to be put on the
> > > notifier waiting list, and the other three to return -EEXIST and be
> > > ignored.
> > > 
> > > With async subdev registration via endpoint fwnode, all four asds are
> > > distinct and three of them keep dangling on their notifiers after the
> > > first one is bound.
> > > 
> > > This patch modifies __v4l2_async_nf_has_async_subdev() to consider
> > > asds matching different fwnode endpoints of the same sub-device equal
> > > if the latter is already bound and matches via device fwnode. Further,
> > > v4l2_async_register_subdev() is modified to remove dangling duplicate
> > > asds that were registered before the sub-device was available to check
> > > its fwnode.
> > > 
> > > Fixes: 1f391df44607 ("media: v4l2-async: Use endpoints in __v4l2_async_nf_add_fwnode_remote()")
> > > Signed-off-by: Philipp Zabel <p.zabel@xxxxxxxxxxxxxx>
> > > ---
> > >  drivers/media/v4l2-core/v4l2-async.c | 43 ++++++++++++++++++++++++++++
> > >  1 file changed, 43 insertions(+)
> > > 
> > > diff --git a/drivers/media/v4l2-core/v4l2-async.c b/drivers/media/v4l2-core/v4l2-async.c
> > > index 2f1b718a9189..b24220ed7077 100644
> > > --- a/drivers/media/v4l2-core/v4l2-async.c
> > > +++ b/drivers/media/v4l2-core/v4l2-async.c
> > > @@ -452,6 +452,22 @@ __v4l2_async_nf_has_async_subdev(struct v4l2_async_notifier *notifier,
> > >  
> > >  		if (asd_equal(asd, sd->asd))
> > >  			return true;
> > > +
> > > +		/*
> > > +		 * If the asd matches an endpoint fwnode, handle sub-devices
> > > +		 * with device fwnode that were already matched by another asd
> > > +		 * with a different endpoint fwnode on the same device.
> > > +		 */
> > > +		if (asd->match_type == V4L2_ASYNC_MATCH_FWNODE &&
> > > +		    fwnode_graph_is_endpoint(asd->match.fwnode) &&
> > > +		    sd->fwnode && !fwnode_graph_is_endpoint(sd->fwnode)) {
> > > +			struct fwnode_handle *devnode;
> > > +
> > > +			devnode = fwnode_graph_get_port_parent(asd->match.fwnode);
> > > +			fwnode_handle_put(devnode);
> > > +			if (devnode == sd->fwnode)
> > > +				return true;
> > > +		}
> > >  	}
> > >  
> > >  	return false;
> > > @@ -749,6 +765,24 @@ __v4l2_async_nf_add_i2c(struct v4l2_async_notifier *notifier, int adapter_id,
> > >  }
> > >  EXPORT_SYMBOL_GPL(__v4l2_async_nf_add_i2c);
> > >  
> > > +static void v4l2_async_remove_duplicate_matches(struct v4l2_subdev *sd)
> > > +{
> > > +	struct v4l2_async_notifier *notifier;
> > > +
> > > +	lockdep_assert_held(&list_lock);
> > > +
> > > +	list_for_each_entry(notifier, &notifier_list, list) {
> > > +		struct v4l2_async_subdev *asd;
> > > +
> > > +		asd = v4l2_async_find_match(notifier, sd);
> > > +		if (!asd)
> > > +			continue;
> > > +
> > > +		/* Remove from the waiting list */
> > > +		list_del(&asd->list);
> > 
> > This leaves asd->list pointers dangling.
> > 
> > > +	}
> > > +}
> > > +
> > >  int v4l2_async_register_subdev(struct v4l2_subdev *sd)
> > >  {
> > >  	struct v4l2_async_notifier *subdev_notifier;
> > > @@ -783,6 +817,15 @@ int v4l2_async_register_subdev(struct v4l2_subdev *sd)
> > >  		if (ret)
> > >  			goto err_unbind;
> > >  
> > > +		/*
> > > +		 * At this point the asd is removed from the notifier wait list.
> > > +		 * There might be other notifiers with asds matching different
> > > +		 * fwnode endpoints of the same sd remaining. If the sd matches
> > > +		 * by device fwnode, remove the dangling asds.
> > > +		 */
> > > +		if (sd->fwnode && !fwnode_graph_is_endpoint(sd->fwnode))
> > > +			v4l2_async_remove_duplicate_matches(sd);
> > > +
> > >  		ret = v4l2_async_nf_try_complete(notifier);
> > >  		if (ret)
> > >  			goto err_unbind;
> > 
> > This patch is essentially a workaround, not a fix.
> > 
> > The root of the problem is that registering async sub-devices (and thus
> > registering a sub-device later) and link creation (via bound callback) are
> > handled together.
> 
> I'm not sure to see how that's related to the problem at hand. The issue
> here is that there are four consumers for one subdevice, and the
> v4l2-async framework doesn't support that. Two of those consumers could
> be made to know about each other, but the other two (video-mux) are
> modelled completely independently. That doesn't seem tied to link
> creation to me.

It is, partly least. Also async sub-device registration as a sub-device
needs to be disconnected from binding a sub-device with a notifier. I don't
want to base this on opportunistically removing async sub-devices from the
notifier list.

That being said, it's been in my plans to improve V4L2 async in these
respects.

> 
> > They should be separated instead, at least in the
> > v4l2-async framework if not in the interfaces. We could at least have a
> > helper doing the both using existing API for devices that have a single
> > source pad.
> > 
> > While merging this might not break anything as such, it would certainly
> > make fixing the underlying problem later on much harder as you'd need to
> > take drivers depending on it into account --- for which I, for instance,
> > don't have hardware for.
> > 
> > I'm thus not overly enthusiastic of the idea of merging this.

-- 
Regards,

Sakari Ailus



[Index of Archives]     [Linux Input]     [Video for Linux]     [Gstreamer Embedded]     [Mplayer Users]     [Linux USB Devel]     [Linux Audio Users]     [Linux Kernel]     [Linux SCSI]     [Yosemite Backpacking]

  Powered by Linux