Hi Jacopo, Thanks for the thorough review. On Fri, Apr 14, 2023 at 10:52:25AM +0200, Jacopo Mondi wrote: > Hi Sakari > > On Thu, Mar 30, 2023 at 02:58:44PM +0300, Sakari Ailus wrote: > > When the v4l2-async framework was introduced, the use case for it was to > > connect a camera sensor with a parallel receiver. Both tended to be rather > > simple devices with a single connection between them. > > > > The framework has been since improved in multiple ways but there are > > limitations that have remained, for instance the assumption an async > > sub-device is connected towards a single notifier and via a single link > > only. > > > > This patch adds an object that represents the device while an earlier > > patch in the series re-purposed the old struct v4l2_async_subdev as the > > connection. > > > > Signed-off-by: Sakari Ailus <sakari.ailus@xxxxxxxxxxxxxxx> > > --- > > .../platform/rockchip/rkisp1/rkisp1-common.h | 2 +- > > .../platform/rockchip/rkisp1/rkisp1-dev.c | 8 +- > > drivers/media/platform/ti/omap3isp/isp.h | 2 +- > > drivers/media/v4l2-core/v4l2-async.c | 163 ++++++++++++++++-- > > include/media/v4l2-async.h | 32 +++- > > include/media/v4l2-subdev.h | 2 +- > > 6 files changed, 179 insertions(+), 30 deletions(-) > > > > diff --git a/drivers/media/platform/rockchip/rkisp1/rkisp1-common.h b/drivers/media/platform/rockchip/rkisp1/rkisp1-common.h > > index d30f0ecb1bfd..a1293c45aae1 100644 > > --- a/drivers/media/platform/rockchip/rkisp1/rkisp1-common.h > > +++ b/drivers/media/platform/rockchip/rkisp1/rkisp1-common.h > > @@ -148,7 +148,7 @@ struct rkisp1_info { > > * @port: port number (0: MIPI, 1: Parallel) > > */ > > struct rkisp1_sensor_async { > > - struct v4l2_async_connection asd; > > + struct v4l2_async_subdev asd; > > unsigned int index; > > struct fwnode_handle *source_ep; > > unsigned int lanes; > > diff --git a/drivers/media/platform/rockchip/rkisp1/rkisp1-dev.c b/drivers/media/platform/rockchip/rkisp1/rkisp1-dev.c > > index 39fa98e6dbbc..5bdb1ecedf6a 100644 > > --- a/drivers/media/platform/rockchip/rkisp1/rkisp1-dev.c > > +++ b/drivers/media/platform/rockchip/rkisp1/rkisp1-dev.c > > @@ -122,12 +122,12 @@ struct rkisp1_isr_data { > > > > static int rkisp1_subdev_notifier_bound(struct v4l2_async_notifier *notifier, > > struct v4l2_subdev *sd, > > - struct v4l2_async_connection *asd) > > + struct v4l2_async_connection *asc) > > { > > struct rkisp1_device *rkisp1 = > > container_of(notifier, struct rkisp1_device, notifier); > > struct rkisp1_sensor_async *s_asd = > > - container_of(asd, struct rkisp1_sensor_async, asd); > > + container_of(asc->asd, struct rkisp1_sensor_async, asd); > > int source_pad; > > int ret; > > > > @@ -165,10 +165,10 @@ static int rkisp1_subdev_notifier_complete(struct v4l2_async_notifier *notifier) > > return v4l2_device_register_subdev_nodes(&rkisp1->v4l2_dev); > > } > > > > -static void rkisp1_subdev_notifier_destroy(struct v4l2_async_connection *asd) > > +static void rkisp1_subdev_notifier_destroy(struct v4l2_async_connection *asc) > > { > > struct rkisp1_sensor_async *rk_asd = > > - container_of(asd, struct rkisp1_sensor_async, asd); > > + container_of(asc->asd, struct rkisp1_sensor_async, asd); > > > > fwnode_handle_put(rk_asd->source_ep); > > } > > diff --git a/drivers/media/platform/ti/omap3isp/isp.h b/drivers/media/platform/ti/omap3isp/isp.h > > index 32ea70c8d2f9..a9d760fbf349 100644 > > --- a/drivers/media/platform/ti/omap3isp/isp.h > > +++ b/drivers/media/platform/ti/omap3isp/isp.h > > @@ -220,7 +220,7 @@ struct isp_device { > > }; > > > > struct isp_async_subdev { > > - struct v4l2_async_connection asd; > > + struct v4l2_async_subdev asd; > > struct isp_bus_cfg bus; > > }; > > > > diff --git a/drivers/media/v4l2-core/v4l2-async.c b/drivers/media/v4l2-core/v4l2-async.c > > index 56ce40481ec4..4c3bd64d6a00 100644 > > --- a/drivers/media/v4l2-core/v4l2-async.c > > +++ b/drivers/media/v4l2-core/v4l2-async.c > > @@ -134,6 +134,7 @@ static bool match_fwnode(struct v4l2_async_notifier *notifier, > > } > > > > static LIST_HEAD(subdev_head); > > +static LIST_HEAD(asd_head); > > static LIST_HEAD(notifier_head); > > static DEFINE_MUTEX(list_lock); > > > > @@ -304,13 +305,20 @@ static int v4l2_async_match_notify(struct v4l2_async_notifier *notifier, > > struct v4l2_async_notifier *subdev_notifier; > > int ret; > > > > - ret = v4l2_device_register_subdev(v4l2_dev, sd); > > - if (ret < 0) > > - return ret; > > + if (!asc->asd->registered) { > > + ret = v4l2_device_register_subdev(v4l2_dev, sd); > > + if (ret < 0) > > + return ret; > > + } > > > > ret = v4l2_async_nf_call_bound(notifier, sd, asc); > > This is the part that puzzles me the most: are we going to receive > multiple bound() calls for the same subdevice when matched on multiple > connections ? If that's the case, is this desirable ? Yes, that is the intent: we're dealing with connections, not to much sub-devices as such. That is what the current bound callbacks generally do. Some also deal with sub-devices but that's fine: driver changes may be needed to add support for new functionality. > > > if (ret < 0) { > > - v4l2_device_unregister_subdev(sd); > > + if (asc->match.type == V4L2_ASYNC_MATCH_FWNODE) > > + dev_dbg(notifier_dev(notifier), > > + "failed binding %pfw (%d)\n", > > + asc->match.fwnode, ret); > > + if (!asc->asd->registered) > > This should probably be > if (asc->asd->registered) Oops! Well, almost. The sub-device should only be unregistered here if it was just registered. But I'll fix it for v2. > > + v4l2_device_unregister_subdev(sd); > > return ret; > > } > > > > @@ -322,14 +330,26 @@ static int v4l2_async_match_notify(struct v4l2_async_notifier *notifier, > > */ > > ret = v4l2_async_create_ancillary_links(notifier, sd); > > Also this part seems suspicious if called multiple times for the same > subdevice when matched on multiple connections > > Do we need to refcount the connections for each async sub-dev, > decrement at each match, and only when all of them are matched > operated on the subdevice ? > : The ancillary link can be created when the sub-device is registered so I don't think we'll need to refcount it. I'll address this in v2. > > > if (ret) { > > + if (asc->match.type == V4L2_ASYNC_MATCH_FWNODE) > > + dev_dbg(notifier_dev(notifier), > > + "failed creating links for %pfw (%d)\n", > > + asc->match.fwnode, ret); > > v4l2_async_nf_call_unbind(notifier, sd, asc); > > - v4l2_device_unregister_subdev(sd); > > + list_del(&asc->asc_subdev_list); > > + if (!asc->asd->registered) > > + v4l2_device_unregister_subdev(sd); > > return ret; > > } > > > > list_del(&asc->waiting_list); > > - sd->asd = asc; > > - sd->notifier = notifier; > > + if (!sd->asd) { > > + WARN_ON(asc->asd->registered); > > + sd->asd = asc->asd; > > + sd->notifier = notifier; > > + asc->asd->registered = true; > > + } else { > > + WARN_ON(sd->asd != asc->asd); > > + } > > > > /* Move from the global subdevice list to notifier's done */ > > list_move(&sd->async_list, ¬ifier->done_head); > > @@ -403,6 +423,21 @@ static void v4l2_async_cleanup(struct v4l2_subdev *sd) > > sd->asd = NULL; > > } > > > > +static void v4l2_async_unbind_subdev_one(struct v4l2_async_notifier *notifier, > > + struct v4l2_subdev *sd, bool readd) > > +{ > > + struct v4l2_async_connection *asc, *tmp; > > + > > + list_for_each_entry_safe(asc, tmp, &sd->asd->asc_head, > > + asc_subdev_list) { > > + v4l2_async_nf_call_unbind(notifier, sd, asc); > > + list_del(&asc->asc_subdev_list); > > + if (readd) > > + list_add_tail(&asc->waiting_list, > > + ¬ifier->waiting_head); > > + } > > +} > > + > > /* Unbind all sub-devices in the notifier tree. */ > > static void > > v4l2_async_nf_unbind_all_subdevs(struct v4l2_async_notifier *notifier, > > @@ -417,10 +452,8 @@ v4l2_async_nf_unbind_all_subdevs(struct v4l2_async_notifier *notifier, > > if (subdev_notifier) > > v4l2_async_nf_unbind_all_subdevs(subdev_notifier, true); > > > > - v4l2_async_nf_call_unbind(notifier, sd, sd->asd); > > - if (readd) > > - list_add_tail(&sd->asd->waiting_list, > > - ¬ifier->waiting_head); > > + v4l2_async_unbind_subdev_one(notifier, sd, readd); > > + > > v4l2_async_cleanup(sd); > > > > list_move(&sd->async_list, &subdev_head); > > @@ -445,8 +478,9 @@ __v4l2_async_nf_has_async_subdev(struct v4l2_async_notifier *notifier, > > if (WARN_ON(!sd->asd)) > > continue; > > > > - if (asc_equal(&sd->asd->match, match)) > > - return true; > > + list_for_each_entry(asc, &sd->asd->asc_head, asc_list) > > + if (asc_equal(&asc->match, match)) > > + return true; > > } > > > > return false; > > @@ -619,6 +653,18 @@ void v4l2_async_nf_unregister(struct v4l2_async_notifier *notifier) > > } > > EXPORT_SYMBOL(v4l2_async_nf_unregister); > > > > +static void release_async_subdev(struct kref *kref) > > +{ > > + struct v4l2_async_subdev *asd = > > + container_of_const(kref, struct v4l2_async_subdev, kref); > > + > > + list_del(&asd->asd_list); > > + > > + WARN_ON(!list_empty(&asd->asc_head)); > > + > > + kfree(asd); > > +} > > + > > static void __v4l2_async_nf_cleanup(struct v4l2_async_notifier *notifier) > > { > > struct v4l2_async_connection *asc, *tmp; > > @@ -627,16 +673,24 @@ static void __v4l2_async_nf_cleanup(struct v4l2_async_notifier *notifier) > > return; > > > > list_for_each_entry_safe(asc, tmp, ¬ifier->asc_head, asc_list) { > > + list_del(&asc->asc_list); > > + v4l2_async_nf_call_destroy(notifier, asc); > > + > > switch (asc->match.type) { > > case V4L2_ASYNC_MATCH_FWNODE: > > + pr_debug("release async connection for fwnode %pfw\n", > > + asc->match.fwnode); > > Why pr_debug ? Can't you use notifier_dev() ? The notifier is being cleaned up. We should have dev still around though, so I'll change this. > > > fwnode_handle_put(asc->match.fwnode); > > break; > > - default: > > + case V4L2_ASYNC_MATCH_I2C: > > + pr_debug("release I²C async connection\n"); > > break; > > + default: > > + pr_debug("release invalid async connection type %u\n", > > + asc->match.type); > > } > > > > - list_del(&asc->asc_list); > > - v4l2_async_nf_call_destroy(notifier, asc); > > + kref_put(&asc->asd->kref, release_async_subdev); > > kfree(asc); > > } > > } > > @@ -651,6 +705,71 @@ void v4l2_async_nf_cleanup(struct v4l2_async_notifier *notifier) > > } > > EXPORT_SYMBOL_GPL(v4l2_async_nf_cleanup); > > > > +static bool async_subdev_has_connection(struct v4l2_async_notifier *notifier, > > + struct v4l2_async_subdev *asd, > > + struct v4l2_async_connection *asc) > > +{ > > + struct v4l2_async_connection *__asc; > > + > > + list_for_each_entry(__asc, &asd->asc_head, asc_subdev_list) { > > + if (__asc->match.type != V4L2_ASYNC_MATCH_FWNODE) > > + continue; > > + > > + if (__asc->match.fwnode != asc->match.fwnode) > > + continue; > > + > > + dev_dbg(notifier_dev(notifier), "found!\n"); > > Such message without much context can quickly become noise I think I'll just drop it. > > > + > > + return true; > > + } > > + > > + return false; > > +} > > + > > +/* Find an async sub-device for the async connection. */ > > +static int v4l2_async_find_async_subdev(struct v4l2_async_notifier *notifier, > > + struct v4l2_async_connection *asc) > > +{ > > + struct v4l2_async_subdev *asd; > > + > > + lockdep_assert_held(&list_lock); > > + > > + if (asc->match.type == V4L2_ASYNC_MATCH_FWNODE) > > + dev_dbg(notifier_dev(notifier), > > + "async: looking up subdev for %pfw\n", > > + asc->match.fwnode); > > + > > + /* > > + * Matching by endpoint nodes may mean there are multiple connections to > > + * a single device. This is only possible with fwnode matching. > > + */ > > + if (asc->match.type == V4L2_ASYNC_MATCH_FWNODE && > > + fwnode_graph_is_endpoint(asc->match.fwnode)) { > > + list_for_each_entry(asd, &asd_head, asd_list) { > > + if (async_subdev_has_connection(notifier, asd, asc)) { > > + kref_get(&asd->kref); > > + goto found; > > + } > > + } > > + } > > + > > + dev_dbg(notifier_dev(notifier), "not found, allocating new one\n"); > > + > > + asd = kzalloc(sizeof(*asd), GFP_KERNEL); > > + if (!asd) > > + return -ENOMEM; > > + > > + kref_init(&asd->kref); > > + INIT_LIST_HEAD(&asd->asc_head); > > + list_add(&asd->asd_list, &asd_head); > > + > > +found: > > + list_add(&asc->asc_subdev_list, &asd->asc_head); > > + asc->asd = asd; > > + > > + return 0; > > +} > > + > > int __v4l2_async_nf_add_connection(struct v4l2_async_notifier *notifier, > > struct v4l2_async_connection *asc) > > { > > @@ -662,6 +781,10 @@ int __v4l2_async_nf_add_connection(struct v4l2_async_notifier *notifier, > > if (ret) > > goto unlock; > > > > + ret = v4l2_async_find_async_subdev(notifier, asc); > > + if (ret) > > + goto unlock; > > + > > list_add_tail(&asc->asc_list, ¬ifier->asc_head); > > > > unlock: > > @@ -809,7 +932,7 @@ int v4l2_async_register_subdev(struct v4l2_subdev *sd) > > v4l2_async_nf_unbind_all_subdevs(subdev_notifier, false); > > > > if (sd->asd) > > - v4l2_async_nf_call_unbind(notifier, sd, sd->asd); > > + v4l2_async_unbind_subdev_one(notifier, sd, true); > > v4l2_async_cleanup(sd); > > > > mutex_unlock(&list_lock); > > @@ -832,10 +955,12 @@ void v4l2_async_unregister_subdev(struct v4l2_subdev *sd) > > > > if (sd->asd) { > > struct v4l2_async_notifier *notifier = sd->notifier; > > + struct v4l2_async_connection *asc; > > > > - list_add(&sd->asd->waiting_list, ¬ifier->waiting_head); > > + list_for_each_entry(asc, &sd->asd->asc_head, asc_subdev_list) > > + list_add(&asc->waiting_list, ¬ifier->waiting_head); > > > > - v4l2_async_nf_call_unbind(notifier, sd, sd->asd); > > + v4l2_async_unbind_subdev_one(notifier, sd, true); > > } > > > > v4l2_async_cleanup(sd); > > diff --git a/include/media/v4l2-async.h b/include/media/v4l2-async.h > > index 9cf383e81a16..750bf4ddb267 100644 > > --- a/include/media/v4l2-async.h > > +++ b/include/media/v4l2-async.h > > @@ -8,6 +8,7 @@ > > #ifndef V4L2_ASYNC_H > > #define V4L2_ASYNC_H > > > > +#include <linux/kref.h> > > #include <linux/list.h> > > #include <linux/mutex.h> > > > > @@ -63,24 +64,47 @@ struct v4l2_async_match { > > }; > > > > /** > > - * struct v4l2_async_connection - sub-device descriptor, as known to a bridge > > + * struct v4l2_async_subdev - sub-device descriptor > > * > > + * @kref: kref for refcounting the subdev > > + * @asd_list: Entry in the list of async sub-devices > > + * @subdev_list: used to link struct v4l2_async_subdev objects, waiting to be > > + * probed, to a notifier->waiting_head list > > + * @asc_head: head for struct v4l2_async_connection.asd_list list > > + * @registered: whether the sub-device has been registered > > + */ > > +struct v4l2_async_subdev { > > + struct kref kref; > > + struct list_head asd_list; > > + struct list_head subdev_list; > > subdev_list is not used Thanks, I think I just forgot this here when splitting the two. > > > + struct list_head asc_head; > > + bool registered; > > +}; > > + > > +/** > > + * struct v4l2_async_connection - sub-device connection descriptor, as known to > > + * a bridge > > + * > > + * @asd: the async sub-device related to this connection > > * @match: struct of match type and per-bus type matching data sets > > * @asc_list: used to add struct v4l2_async_connection objects to the > > * master notifier @asc_list > > * @waiting_list: used to link struct v4l2_async_connection objects, waiting to > > * be probed, to a notifier->waiting list > > + * @asc_subdev_list: entry in struct v4l2_async_subdev.asc_head list > > * > > - * When this struct is used as a member in a driver specific struct, > > - * the driver specific struct shall contain the &struct > > - * v4l2_async_connection as its first member. > > + * When this struct is used as a member in a driver specific struct, the driver > > + * specific struct shall contain the &struct v4l2_async_connection as its first > > + * member. > > */ > > struct v4l2_async_connection { > > + struct v4l2_async_subdev *asd; > > struct v4l2_async_match match; > > > > /* v4l2-async core private: not to be used by drivers */ > > struct list_head asc_list; > > struct list_head waiting_list; > > + struct list_head asc_subdev_list; > > }; > > > > /** > > diff --git a/include/media/v4l2-subdev.h b/include/media/v4l2-subdev.h > > index a2cce11dda5c..d510fe6ea243 100644 > > --- a/include/media/v4l2-subdev.h > > +++ b/include/media/v4l2-subdev.h > > @@ -1063,7 +1063,7 @@ struct v4l2_subdev { > > struct device *dev; > > struct fwnode_handle *fwnode; > > struct list_head async_list; > > - struct v4l2_async_connection *asd; > > + struct v4l2_async_subdev *asd; > > struct v4l2_async_notifier *notifier; > > struct v4l2_async_notifier *subdev_notifier; > > struct v4l2_subdev_platform_data *pdata; -- Kind regards, Sakari Ailus