forked from Minki/linux
media: v4l: async: Move async subdev notifier operations to a separate structure
The async subdev notifier .bound(), .unbind() and .complete() operations are function pointers stored directly in the v4l2_async_subdev structure. As the structure isn't immutable, this creates a potential security risk as the function pointers are mutable. To fix this, move the function pointers to a new v4l2_async_subdev_operations structure that can be made const in drivers. Signed-off-by: Laurent Pinchart <laurent.pinchart@ideasonboard.com> Acked-by: Hans Verkuil <hans.verkuil@cisco.com> Acked-by: Niklas Söderlund <niklas.soderlund+renesas@ragnatech.se> Reviewed-by: Sebastian Reichel <sebastian.reichel@collabora.co.uk> Signed-off-by: Mauro Carvalho Chehab <mchehab@s-opensource.com>
This commit is contained in:
parent
bce9e317ce
commit
b6ee3f0dcf
@ -2417,6 +2417,11 @@ static int vpfe_async_complete(struct v4l2_async_notifier *notifier)
|
|||||||
return vpfe_probe_complete(vpfe);
|
return vpfe_probe_complete(vpfe);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static const struct v4l2_async_notifier_operations vpfe_async_ops = {
|
||||||
|
.bound = vpfe_async_bound,
|
||||||
|
.complete = vpfe_async_complete,
|
||||||
|
};
|
||||||
|
|
||||||
static struct vpfe_config *
|
static struct vpfe_config *
|
||||||
vpfe_get_pdata(struct platform_device *pdev)
|
vpfe_get_pdata(struct platform_device *pdev)
|
||||||
{
|
{
|
||||||
@ -2590,8 +2595,7 @@ static int vpfe_probe(struct platform_device *pdev)
|
|||||||
|
|
||||||
vpfe->notifier.subdevs = vpfe->cfg->asd;
|
vpfe->notifier.subdevs = vpfe->cfg->asd;
|
||||||
vpfe->notifier.num_subdevs = ARRAY_SIZE(vpfe->cfg->asd);
|
vpfe->notifier.num_subdevs = ARRAY_SIZE(vpfe->cfg->asd);
|
||||||
vpfe->notifier.bound = vpfe_async_bound;
|
vpfe->notifier.ops = &vpfe_async_ops;
|
||||||
vpfe->notifier.complete = vpfe_async_complete;
|
|
||||||
ret = v4l2_async_notifier_register(&vpfe->v4l2_dev,
|
ret = v4l2_async_notifier_register(&vpfe->v4l2_dev,
|
||||||
&vpfe->notifier);
|
&vpfe->notifier);
|
||||||
if (ret) {
|
if (ret) {
|
||||||
|
@ -1979,6 +1979,12 @@ static int isc_async_complete(struct v4l2_async_notifier *notifier)
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static const struct v4l2_async_notifier_operations isc_async_ops = {
|
||||||
|
.bound = isc_async_bound,
|
||||||
|
.unbind = isc_async_unbind,
|
||||||
|
.complete = isc_async_complete,
|
||||||
|
};
|
||||||
|
|
||||||
static void isc_subdev_cleanup(struct isc_device *isc)
|
static void isc_subdev_cleanup(struct isc_device *isc)
|
||||||
{
|
{
|
||||||
struct isc_subdev_entity *subdev_entity;
|
struct isc_subdev_entity *subdev_entity;
|
||||||
@ -2203,9 +2209,7 @@ static int atmel_isc_probe(struct platform_device *pdev)
|
|||||||
list_for_each_entry(subdev_entity, &isc->subdev_entities, list) {
|
list_for_each_entry(subdev_entity, &isc->subdev_entities, list) {
|
||||||
subdev_entity->notifier.subdevs = &subdev_entity->asd;
|
subdev_entity->notifier.subdevs = &subdev_entity->asd;
|
||||||
subdev_entity->notifier.num_subdevs = 1;
|
subdev_entity->notifier.num_subdevs = 1;
|
||||||
subdev_entity->notifier.bound = isc_async_bound;
|
subdev_entity->notifier.ops = &isc_async_ops;
|
||||||
subdev_entity->notifier.unbind = isc_async_unbind;
|
|
||||||
subdev_entity->notifier.complete = isc_async_complete;
|
|
||||||
|
|
||||||
ret = v4l2_async_notifier_register(&isc->v4l2_dev,
|
ret = v4l2_async_notifier_register(&isc->v4l2_dev,
|
||||||
&subdev_entity->notifier);
|
&subdev_entity->notifier);
|
||||||
|
@ -1103,6 +1103,12 @@ static int isi_graph_notify_bound(struct v4l2_async_notifier *notifier,
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static const struct v4l2_async_notifier_operations isi_graph_notify_ops = {
|
||||||
|
.bound = isi_graph_notify_bound,
|
||||||
|
.unbind = isi_graph_notify_unbind,
|
||||||
|
.complete = isi_graph_notify_complete,
|
||||||
|
};
|
||||||
|
|
||||||
static int isi_graph_parse(struct atmel_isi *isi, struct device_node *node)
|
static int isi_graph_parse(struct atmel_isi *isi, struct device_node *node)
|
||||||
{
|
{
|
||||||
struct device_node *ep = NULL;
|
struct device_node *ep = NULL;
|
||||||
@ -1150,9 +1156,7 @@ static int isi_graph_init(struct atmel_isi *isi)
|
|||||||
|
|
||||||
isi->notifier.subdevs = subdevs;
|
isi->notifier.subdevs = subdevs;
|
||||||
isi->notifier.num_subdevs = 1;
|
isi->notifier.num_subdevs = 1;
|
||||||
isi->notifier.bound = isi_graph_notify_bound;
|
isi->notifier.ops = &isi_graph_notify_ops;
|
||||||
isi->notifier.unbind = isi_graph_notify_unbind;
|
|
||||||
isi->notifier.complete = isi_graph_notify_complete;
|
|
||||||
|
|
||||||
ret = v4l2_async_notifier_register(&isi->v4l2_dev, &isi->notifier);
|
ret = v4l2_async_notifier_register(&isi->v4l2_dev, &isi->notifier);
|
||||||
if (ret < 0) {
|
if (ret < 0) {
|
||||||
|
@ -1500,6 +1500,11 @@ static int vpif_async_complete(struct v4l2_async_notifier *notifier)
|
|||||||
return vpif_probe_complete();
|
return vpif_probe_complete();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static const struct v4l2_async_notifier_operations vpif_async_ops = {
|
||||||
|
.bound = vpif_async_bound,
|
||||||
|
.complete = vpif_async_complete,
|
||||||
|
};
|
||||||
|
|
||||||
static struct vpif_capture_config *
|
static struct vpif_capture_config *
|
||||||
vpif_capture_get_pdata(struct platform_device *pdev)
|
vpif_capture_get_pdata(struct platform_device *pdev)
|
||||||
{
|
{
|
||||||
@ -1691,8 +1696,7 @@ static __init int vpif_probe(struct platform_device *pdev)
|
|||||||
} else {
|
} else {
|
||||||
vpif_obj.notifier.subdevs = vpif_obj.config->asd;
|
vpif_obj.notifier.subdevs = vpif_obj.config->asd;
|
||||||
vpif_obj.notifier.num_subdevs = vpif_obj.config->asd_sizes[0];
|
vpif_obj.notifier.num_subdevs = vpif_obj.config->asd_sizes[0];
|
||||||
vpif_obj.notifier.bound = vpif_async_bound;
|
vpif_obj.notifier.ops = &vpif_async_ops;
|
||||||
vpif_obj.notifier.complete = vpif_async_complete;
|
|
||||||
err = v4l2_async_notifier_register(&vpif_obj.v4l2_dev,
|
err = v4l2_async_notifier_register(&vpif_obj.v4l2_dev,
|
||||||
&vpif_obj.notifier);
|
&vpif_obj.notifier);
|
||||||
if (err) {
|
if (err) {
|
||||||
|
@ -1232,6 +1232,11 @@ static int vpif_async_complete(struct v4l2_async_notifier *notifier)
|
|||||||
return vpif_probe_complete();
|
return vpif_probe_complete();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static const struct v4l2_async_notifier_operations vpif_async_ops = {
|
||||||
|
.bound = vpif_async_bound,
|
||||||
|
.complete = vpif_async_complete,
|
||||||
|
};
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* vpif_probe: This function creates device entries by register itself to the
|
* vpif_probe: This function creates device entries by register itself to the
|
||||||
* V4L2 driver and initializes fields of each channel objects
|
* V4L2 driver and initializes fields of each channel objects
|
||||||
@ -1313,8 +1318,7 @@ static __init int vpif_probe(struct platform_device *pdev)
|
|||||||
} else {
|
} else {
|
||||||
vpif_obj.notifier.subdevs = vpif_obj.config->asd;
|
vpif_obj.notifier.subdevs = vpif_obj.config->asd;
|
||||||
vpif_obj.notifier.num_subdevs = vpif_obj.config->asd_sizes[0];
|
vpif_obj.notifier.num_subdevs = vpif_obj.config->asd_sizes[0];
|
||||||
vpif_obj.notifier.bound = vpif_async_bound;
|
vpif_obj.notifier.ops = &vpif_async_ops;
|
||||||
vpif_obj.notifier.complete = vpif_async_complete;
|
|
||||||
err = v4l2_async_notifier_register(&vpif_obj.v4l2_dev,
|
err = v4l2_async_notifier_register(&vpif_obj.v4l2_dev,
|
||||||
&vpif_obj.notifier);
|
&vpif_obj.notifier);
|
||||||
if (err) {
|
if (err) {
|
||||||
|
@ -1405,6 +1405,11 @@ unlock:
|
|||||||
return media_device_register(&fmd->media_dev);
|
return media_device_register(&fmd->media_dev);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static const struct v4l2_async_notifier_operations subdev_notifier_ops = {
|
||||||
|
.bound = subdev_notifier_bound,
|
||||||
|
.complete = subdev_notifier_complete,
|
||||||
|
};
|
||||||
|
|
||||||
static int fimc_md_probe(struct platform_device *pdev)
|
static int fimc_md_probe(struct platform_device *pdev)
|
||||||
{
|
{
|
||||||
struct device *dev = &pdev->dev;
|
struct device *dev = &pdev->dev;
|
||||||
@ -1479,8 +1484,7 @@ static int fimc_md_probe(struct platform_device *pdev)
|
|||||||
if (fmd->num_sensors > 0) {
|
if (fmd->num_sensors > 0) {
|
||||||
fmd->subdev_notifier.subdevs = fmd->async_subdevs;
|
fmd->subdev_notifier.subdevs = fmd->async_subdevs;
|
||||||
fmd->subdev_notifier.num_subdevs = fmd->num_sensors;
|
fmd->subdev_notifier.num_subdevs = fmd->num_sensors;
|
||||||
fmd->subdev_notifier.bound = subdev_notifier_bound;
|
fmd->subdev_notifier.ops = &subdev_notifier_ops;
|
||||||
fmd->subdev_notifier.complete = subdev_notifier_complete;
|
|
||||||
fmd->num_sensors = 0;
|
fmd->num_sensors = 0;
|
||||||
|
|
||||||
ret = v4l2_async_notifier_register(&fmd->v4l2_dev,
|
ret = v4l2_async_notifier_register(&fmd->v4l2_dev,
|
||||||
|
@ -2171,6 +2171,10 @@ static int isp_subdev_notifier_complete(struct v4l2_async_notifier *async)
|
|||||||
return media_device_register(&isp->media_dev);
|
return media_device_register(&isp->media_dev);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static const struct v4l2_async_notifier_operations isp_subdev_notifier_ops = {
|
||||||
|
.complete = isp_subdev_notifier_complete,
|
||||||
|
};
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* isp_probe - Probe ISP platform device
|
* isp_probe - Probe ISP platform device
|
||||||
* @pdev: Pointer to ISP platform device
|
* @pdev: Pointer to ISP platform device
|
||||||
@ -2341,7 +2345,7 @@ static int isp_probe(struct platform_device *pdev)
|
|||||||
if (ret < 0)
|
if (ret < 0)
|
||||||
goto error_register_entities;
|
goto error_register_entities;
|
||||||
|
|
||||||
isp->notifier.complete = isp_subdev_notifier_complete;
|
isp->notifier.ops = &isp_subdev_notifier_ops;
|
||||||
|
|
||||||
ret = v4l2_async_notifier_register(&isp->v4l2_dev, &isp->notifier);
|
ret = v4l2_async_notifier_register(&isp->v4l2_dev, &isp->notifier);
|
||||||
if (ret)
|
if (ret)
|
||||||
|
@ -2221,6 +2221,11 @@ static void pxa_camera_sensor_unbind(struct v4l2_async_notifier *notifier,
|
|||||||
mutex_unlock(&pcdev->mlock);
|
mutex_unlock(&pcdev->mlock);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static const struct v4l2_async_notifier_operations pxa_camera_sensor_ops = {
|
||||||
|
.bound = pxa_camera_sensor_bound,
|
||||||
|
.unbind = pxa_camera_sensor_unbind,
|
||||||
|
};
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Driver probe, remove, suspend and resume operations
|
* Driver probe, remove, suspend and resume operations
|
||||||
*/
|
*/
|
||||||
@ -2489,8 +2494,7 @@ static int pxa_camera_probe(struct platform_device *pdev)
|
|||||||
pcdev->asds[0] = &pcdev->asd;
|
pcdev->asds[0] = &pcdev->asd;
|
||||||
pcdev->notifier.subdevs = pcdev->asds;
|
pcdev->notifier.subdevs = pcdev->asds;
|
||||||
pcdev->notifier.num_subdevs = 1;
|
pcdev->notifier.num_subdevs = 1;
|
||||||
pcdev->notifier.bound = pxa_camera_sensor_bound;
|
pcdev->notifier.ops = &pxa_camera_sensor_ops;
|
||||||
pcdev->notifier.unbind = pxa_camera_sensor_unbind;
|
|
||||||
|
|
||||||
if (!of_have_populated_dt())
|
if (!of_have_populated_dt())
|
||||||
pcdev->asd.match_type = V4L2_ASYNC_MATCH_I2C;
|
pcdev->asd.match_type = V4L2_ASYNC_MATCH_I2C;
|
||||||
|
@ -601,6 +601,11 @@ static int camss_subdev_notifier_complete(struct v4l2_async_notifier *async)
|
|||||||
return media_device_register(&camss->media_dev);
|
return media_device_register(&camss->media_dev);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static const struct v4l2_async_notifier_operations camss_subdev_notifier_ops = {
|
||||||
|
.bound = camss_subdev_notifier_bound,
|
||||||
|
.complete = camss_subdev_notifier_complete,
|
||||||
|
};
|
||||||
|
|
||||||
static const struct media_device_ops camss_media_ops = {
|
static const struct media_device_ops camss_media_ops = {
|
||||||
.link_notify = v4l2_pipeline_link_notify,
|
.link_notify = v4l2_pipeline_link_notify,
|
||||||
};
|
};
|
||||||
@ -655,8 +660,7 @@ static int camss_probe(struct platform_device *pdev)
|
|||||||
goto err_register_entities;
|
goto err_register_entities;
|
||||||
|
|
||||||
if (camss->notifier.num_subdevs) {
|
if (camss->notifier.num_subdevs) {
|
||||||
camss->notifier.bound = camss_subdev_notifier_bound;
|
camss->notifier.ops = &camss_subdev_notifier_ops;
|
||||||
camss->notifier.complete = camss_subdev_notifier_complete;
|
|
||||||
|
|
||||||
ret = v4l2_async_notifier_register(&camss->v4l2_dev,
|
ret = v4l2_async_notifier_register(&camss->v4l2_dev,
|
||||||
&camss->notifier);
|
&camss->notifier);
|
||||||
|
@ -134,6 +134,12 @@ static int rvin_digital_notify_bound(struct v4l2_async_notifier *notifier,
|
|||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
static const struct v4l2_async_notifier_operations rvin_digital_notify_ops = {
|
||||||
|
.bound = rvin_digital_notify_bound,
|
||||||
|
.unbind = rvin_digital_notify_unbind,
|
||||||
|
.complete = rvin_digital_notify_complete,
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
static int rvin_digital_parse_v4l2(struct device *dev,
|
static int rvin_digital_parse_v4l2(struct device *dev,
|
||||||
struct v4l2_fwnode_endpoint *vep,
|
struct v4l2_fwnode_endpoint *vep,
|
||||||
@ -183,9 +189,7 @@ static int rvin_digital_graph_init(struct rvin_dev *vin)
|
|||||||
vin_dbg(vin, "Found digital subdevice %pOF\n",
|
vin_dbg(vin, "Found digital subdevice %pOF\n",
|
||||||
to_of_node(vin->digital->asd.match.fwnode.fwnode));
|
to_of_node(vin->digital->asd.match.fwnode.fwnode));
|
||||||
|
|
||||||
vin->notifier.bound = rvin_digital_notify_bound;
|
vin->notifier.ops = &rvin_digital_notify_ops;
|
||||||
vin->notifier.unbind = rvin_digital_notify_unbind;
|
|
||||||
vin->notifier.complete = rvin_digital_notify_complete;
|
|
||||||
ret = v4l2_async_notifier_register(&vin->v4l2_dev, &vin->notifier);
|
ret = v4l2_async_notifier_register(&vin->v4l2_dev, &vin->notifier);
|
||||||
if (ret < 0) {
|
if (ret < 0) {
|
||||||
vin_err(vin, "Notifier registration failed\n");
|
vin_err(vin, "Notifier registration failed\n");
|
||||||
|
@ -1185,6 +1185,12 @@ error:
|
|||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static const struct v4l2_async_notifier_operations rcar_drif_notify_ops = {
|
||||||
|
.bound = rcar_drif_notify_bound,
|
||||||
|
.unbind = rcar_drif_notify_unbind,
|
||||||
|
.complete = rcar_drif_notify_complete,
|
||||||
|
};
|
||||||
|
|
||||||
/* Read endpoint properties */
|
/* Read endpoint properties */
|
||||||
static void rcar_drif_get_ep_properties(struct rcar_drif_sdr *sdr,
|
static void rcar_drif_get_ep_properties(struct rcar_drif_sdr *sdr,
|
||||||
struct fwnode_handle *fwnode)
|
struct fwnode_handle *fwnode)
|
||||||
@ -1347,9 +1353,7 @@ static int rcar_drif_sdr_probe(struct rcar_drif_sdr *sdr)
|
|||||||
if (ret)
|
if (ret)
|
||||||
goto error;
|
goto error;
|
||||||
|
|
||||||
sdr->notifier.bound = rcar_drif_notify_bound;
|
sdr->notifier.ops = &rcar_drif_notify_ops;
|
||||||
sdr->notifier.unbind = rcar_drif_notify_unbind;
|
|
||||||
sdr->notifier.complete = rcar_drif_notify_complete;
|
|
||||||
|
|
||||||
/* Register notifier */
|
/* Register notifier */
|
||||||
ret = v4l2_async_notifier_register(&sdr->v4l2_dev, &sdr->notifier);
|
ret = v4l2_async_notifier_register(&sdr->v4l2_dev, &sdr->notifier);
|
||||||
|
@ -1391,6 +1391,12 @@ static int soc_camera_async_complete(struct v4l2_async_notifier *notifier)
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static const struct v4l2_async_notifier_operations soc_camera_async_ops = {
|
||||||
|
.bound = soc_camera_async_bound,
|
||||||
|
.unbind = soc_camera_async_unbind,
|
||||||
|
.complete = soc_camera_async_complete,
|
||||||
|
};
|
||||||
|
|
||||||
static int scan_async_group(struct soc_camera_host *ici,
|
static int scan_async_group(struct soc_camera_host *ici,
|
||||||
struct v4l2_async_subdev **asd, unsigned int size)
|
struct v4l2_async_subdev **asd, unsigned int size)
|
||||||
{
|
{
|
||||||
@ -1437,9 +1443,7 @@ static int scan_async_group(struct soc_camera_host *ici,
|
|||||||
|
|
||||||
sasc->notifier.subdevs = asd;
|
sasc->notifier.subdevs = asd;
|
||||||
sasc->notifier.num_subdevs = size;
|
sasc->notifier.num_subdevs = size;
|
||||||
sasc->notifier.bound = soc_camera_async_bound;
|
sasc->notifier.ops = &soc_camera_async_ops;
|
||||||
sasc->notifier.unbind = soc_camera_async_unbind;
|
|
||||||
sasc->notifier.complete = soc_camera_async_complete;
|
|
||||||
|
|
||||||
icd->sasc = sasc;
|
icd->sasc = sasc;
|
||||||
icd->parent = ici->v4l2_dev.dev;
|
icd->parent = ici->v4l2_dev.dev;
|
||||||
@ -1537,9 +1541,7 @@ static int soc_of_bind(struct soc_camera_host *ici,
|
|||||||
|
|
||||||
sasc->notifier.subdevs = &info->subdev;
|
sasc->notifier.subdevs = &info->subdev;
|
||||||
sasc->notifier.num_subdevs = 1;
|
sasc->notifier.num_subdevs = 1;
|
||||||
sasc->notifier.bound = soc_camera_async_bound;
|
sasc->notifier.ops = &soc_camera_async_ops;
|
||||||
sasc->notifier.unbind = soc_camera_async_unbind;
|
|
||||||
sasc->notifier.complete = soc_camera_async_complete;
|
|
||||||
|
|
||||||
icd->sasc = sasc;
|
icd->sasc = sasc;
|
||||||
icd->parent = ici->v4l2_dev.dev;
|
icd->parent = ici->v4l2_dev.dev;
|
||||||
|
@ -1495,6 +1495,12 @@ static int dcmi_graph_notify_bound(struct v4l2_async_notifier *notifier,
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static const struct v4l2_async_notifier_operations dcmi_graph_notify_ops = {
|
||||||
|
.bound = dcmi_graph_notify_bound,
|
||||||
|
.unbind = dcmi_graph_notify_unbind,
|
||||||
|
.complete = dcmi_graph_notify_complete,
|
||||||
|
};
|
||||||
|
|
||||||
static int dcmi_graph_parse(struct stm32_dcmi *dcmi, struct device_node *node)
|
static int dcmi_graph_parse(struct stm32_dcmi *dcmi, struct device_node *node)
|
||||||
{
|
{
|
||||||
struct device_node *ep = NULL;
|
struct device_node *ep = NULL;
|
||||||
@ -1542,9 +1548,7 @@ static int dcmi_graph_init(struct stm32_dcmi *dcmi)
|
|||||||
|
|
||||||
dcmi->notifier.subdevs = subdevs;
|
dcmi->notifier.subdevs = subdevs;
|
||||||
dcmi->notifier.num_subdevs = 1;
|
dcmi->notifier.num_subdevs = 1;
|
||||||
dcmi->notifier.bound = dcmi_graph_notify_bound;
|
dcmi->notifier.ops = &dcmi_graph_notify_ops;
|
||||||
dcmi->notifier.unbind = dcmi_graph_notify_unbind;
|
|
||||||
dcmi->notifier.complete = dcmi_graph_notify_complete;
|
|
||||||
|
|
||||||
ret = v4l2_async_notifier_register(&dcmi->v4l2_dev, &dcmi->notifier);
|
ret = v4l2_async_notifier_register(&dcmi->v4l2_dev, &dcmi->notifier);
|
||||||
if (ret < 0) {
|
if (ret < 0) {
|
||||||
|
@ -1522,6 +1522,11 @@ static int cal_async_complete(struct v4l2_async_notifier *notifier)
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static const struct v4l2_async_notifier_operations cal_async_ops = {
|
||||||
|
.bound = cal_async_bound,
|
||||||
|
.complete = cal_async_complete,
|
||||||
|
};
|
||||||
|
|
||||||
static int cal_complete_ctx(struct cal_ctx *ctx)
|
static int cal_complete_ctx(struct cal_ctx *ctx)
|
||||||
{
|
{
|
||||||
struct video_device *vfd;
|
struct video_device *vfd;
|
||||||
@ -1736,8 +1741,7 @@ static int of_cal_create_instance(struct cal_ctx *ctx, int inst)
|
|||||||
ctx->asd_list[0] = asd;
|
ctx->asd_list[0] = asd;
|
||||||
ctx->notifier.subdevs = ctx->asd_list;
|
ctx->notifier.subdevs = ctx->asd_list;
|
||||||
ctx->notifier.num_subdevs = 1;
|
ctx->notifier.num_subdevs = 1;
|
||||||
ctx->notifier.bound = cal_async_bound;
|
ctx->notifier.ops = &cal_async_ops;
|
||||||
ctx->notifier.complete = cal_async_complete;
|
|
||||||
ret = v4l2_async_notifier_register(&ctx->v4l2_dev,
|
ret = v4l2_async_notifier_register(&ctx->v4l2_dev,
|
||||||
&ctx->notifier);
|
&ctx->notifier);
|
||||||
if (ret) {
|
if (ret) {
|
||||||
|
@ -351,6 +351,11 @@ static int xvip_graph_notify_bound(struct v4l2_async_notifier *notifier,
|
|||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static const struct v4l2_async_notifier_operations xvip_graph_notify_ops = {
|
||||||
|
.bound = xvip_graph_notify_bound,
|
||||||
|
.complete = xvip_graph_notify_complete,
|
||||||
|
};
|
||||||
|
|
||||||
static int xvip_graph_parse_one(struct xvip_composite_device *xdev,
|
static int xvip_graph_parse_one(struct xvip_composite_device *xdev,
|
||||||
struct device_node *node)
|
struct device_node *node)
|
||||||
{
|
{
|
||||||
@ -548,8 +553,7 @@ static int xvip_graph_init(struct xvip_composite_device *xdev)
|
|||||||
|
|
||||||
xdev->notifier.subdevs = subdevs;
|
xdev->notifier.subdevs = subdevs;
|
||||||
xdev->notifier.num_subdevs = num_subdevs;
|
xdev->notifier.num_subdevs = num_subdevs;
|
||||||
xdev->notifier.bound = xvip_graph_notify_bound;
|
xdev->notifier.ops = &xvip_graph_notify_ops;
|
||||||
xdev->notifier.complete = xvip_graph_notify_complete;
|
|
||||||
|
|
||||||
ret = v4l2_async_notifier_register(&xdev->v4l2_dev, &xdev->notifier);
|
ret = v4l2_async_notifier_register(&xdev->v4l2_dev, &xdev->notifier);
|
||||||
if (ret < 0) {
|
if (ret < 0) {
|
||||||
|
@ -102,16 +102,16 @@ static int v4l2_async_match_notify(struct v4l2_async_notifier *notifier,
|
|||||||
{
|
{
|
||||||
int ret;
|
int ret;
|
||||||
|
|
||||||
if (notifier->bound) {
|
if (notifier->ops->bound) {
|
||||||
ret = notifier->bound(notifier, sd, asd);
|
ret = notifier->ops->bound(notifier, sd, asd);
|
||||||
if (ret < 0)
|
if (ret < 0)
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
ret = v4l2_device_register_subdev(notifier->v4l2_dev, sd);
|
ret = v4l2_device_register_subdev(notifier->v4l2_dev, sd);
|
||||||
if (ret < 0) {
|
if (ret < 0) {
|
||||||
if (notifier->unbind)
|
if (notifier->ops->unbind)
|
||||||
notifier->unbind(notifier, sd, asd);
|
notifier->ops->unbind(notifier, sd, asd);
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -140,9 +140,8 @@ static void v4l2_async_notifier_unbind_all_subdevs(
|
|||||||
struct v4l2_subdev *sd, *tmp;
|
struct v4l2_subdev *sd, *tmp;
|
||||||
|
|
||||||
list_for_each_entry_safe(sd, tmp, ¬ifier->done, async_list) {
|
list_for_each_entry_safe(sd, tmp, ¬ifier->done, async_list) {
|
||||||
if (notifier->unbind)
|
if (notifier->ops->unbind)
|
||||||
notifier->unbind(notifier, sd, sd->asd);
|
notifier->ops->unbind(notifier, sd, sd->asd);
|
||||||
|
|
||||||
v4l2_async_cleanup(sd);
|
v4l2_async_cleanup(sd);
|
||||||
|
|
||||||
list_move(&sd->async_list, &subdev_list);
|
list_move(&sd->async_list, &subdev_list);
|
||||||
@ -199,8 +198,8 @@ int v4l2_async_notifier_register(struct v4l2_device *v4l2_dev,
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (list_empty(¬ifier->waiting) && notifier->complete) {
|
if (list_empty(¬ifier->waiting) && notifier->ops->complete) {
|
||||||
ret = notifier->complete(notifier);
|
ret = notifier->ops->complete(notifier);
|
||||||
if (ret)
|
if (ret)
|
||||||
goto err_complete;
|
goto err_complete;
|
||||||
}
|
}
|
||||||
@ -297,10 +296,10 @@ int v4l2_async_register_subdev(struct v4l2_subdev *sd)
|
|||||||
if (ret)
|
if (ret)
|
||||||
goto err_unlock;
|
goto err_unlock;
|
||||||
|
|
||||||
if (!list_empty(¬ifier->waiting) || !notifier->complete)
|
if (!list_empty(¬ifier->waiting) || !notifier->ops->complete)
|
||||||
goto out_unlock;
|
goto out_unlock;
|
||||||
|
|
||||||
ret = notifier->complete(notifier);
|
ret = notifier->ops->complete(notifier);
|
||||||
if (ret)
|
if (ret)
|
||||||
goto err_cleanup;
|
goto err_cleanup;
|
||||||
|
|
||||||
@ -316,9 +315,8 @@ out_unlock:
|
|||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
err_cleanup:
|
err_cleanup:
|
||||||
if (notifier->unbind)
|
if (notifier->ops->unbind)
|
||||||
notifier->unbind(notifier, sd, sd->asd);
|
notifier->ops->unbind(notifier, sd, sd->asd);
|
||||||
|
|
||||||
v4l2_async_cleanup(sd);
|
v4l2_async_cleanup(sd);
|
||||||
|
|
||||||
err_unlock:
|
err_unlock:
|
||||||
@ -337,8 +335,8 @@ void v4l2_async_unregister_subdev(struct v4l2_subdev *sd)
|
|||||||
|
|
||||||
list_add(&sd->asd->list, ¬ifier->waiting);
|
list_add(&sd->asd->list, ¬ifier->waiting);
|
||||||
|
|
||||||
if (notifier->unbind)
|
if (notifier->ops->unbind)
|
||||||
notifier->unbind(notifier, sd, sd->asd);
|
notifier->ops->unbind(notifier, sd, sd->asd);
|
||||||
}
|
}
|
||||||
|
|
||||||
v4l2_async_cleanup(sd);
|
v4l2_async_cleanup(sd);
|
||||||
|
@ -440,6 +440,11 @@ unlock:
|
|||||||
return media_device_register(&imxmd->md);
|
return media_device_register(&imxmd->md);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static const struct v4l2_async_notifier_operations imx_media_subdev_ops = {
|
||||||
|
.bound = imx_media_subdev_bound,
|
||||||
|
.complete = imx_media_probe_complete,
|
||||||
|
};
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* adds controls to a video device from an entity subdevice.
|
* adds controls to a video device from an entity subdevice.
|
||||||
* Continues upstream from the entity's sink pads.
|
* Continues upstream from the entity's sink pads.
|
||||||
@ -608,8 +613,7 @@ static int imx_media_probe(struct platform_device *pdev)
|
|||||||
|
|
||||||
/* prepare the async subdev notifier and register it */
|
/* prepare the async subdev notifier and register it */
|
||||||
imxmd->subdev_notifier.subdevs = imxmd->async_ptrs;
|
imxmd->subdev_notifier.subdevs = imxmd->async_ptrs;
|
||||||
imxmd->subdev_notifier.bound = imx_media_subdev_bound;
|
imxmd->subdev_notifier.ops = &imx_media_subdev_ops;
|
||||||
imxmd->subdev_notifier.complete = imx_media_probe_complete;
|
|
||||||
ret = v4l2_async_notifier_register(&imxmd->v4l2_dev,
|
ret = v4l2_async_notifier_register(&imxmd->v4l2_dev,
|
||||||
&imxmd->subdev_notifier);
|
&imxmd->subdev_notifier);
|
||||||
if (ret) {
|
if (ret) {
|
||||||
|
@ -18,6 +18,7 @@ struct device;
|
|||||||
struct device_node;
|
struct device_node;
|
||||||
struct v4l2_device;
|
struct v4l2_device;
|
||||||
struct v4l2_subdev;
|
struct v4l2_subdev;
|
||||||
|
struct v4l2_async_notifier;
|
||||||
|
|
||||||
/* A random max subdevice number, used to allocate an array on stack */
|
/* A random max subdevice number, used to allocate an array on stack */
|
||||||
#define V4L2_MAX_SUBDEVS 128U
|
#define V4L2_MAX_SUBDEVS 128U
|
||||||
@ -79,27 +80,12 @@ struct v4l2_async_subdev {
|
|||||||
};
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* struct v4l2_async_notifier - v4l2_device notifier data
|
* struct v4l2_async_notifier_operations - Asynchronous V4L2 notifier operations
|
||||||
*
|
* @bound: a subdevice driver has successfully probed one of the subdevices
|
||||||
* @num_subdevs: number of subdevices used in the subdevs array
|
|
||||||
* @max_subdevs: number of subdevices allocated in the subdevs array
|
|
||||||
* @subdevs: array of pointers to subdevice descriptors
|
|
||||||
* @v4l2_dev: pointer to struct v4l2_device
|
|
||||||
* @waiting: list of struct v4l2_async_subdev, waiting for their drivers
|
|
||||||
* @done: list of struct v4l2_subdev, already probed
|
|
||||||
* @list: member in a global list of notifiers
|
|
||||||
* @bound: a subdevice driver has successfully probed one of subdevices
|
|
||||||
* @complete: all subdevices have been probed successfully
|
* @complete: all subdevices have been probed successfully
|
||||||
* @unbind: a subdevice is leaving
|
* @unbind: a subdevice is leaving
|
||||||
*/
|
*/
|
||||||
struct v4l2_async_notifier {
|
struct v4l2_async_notifier_operations {
|
||||||
unsigned int num_subdevs;
|
|
||||||
unsigned int max_subdevs;
|
|
||||||
struct v4l2_async_subdev **subdevs;
|
|
||||||
struct v4l2_device *v4l2_dev;
|
|
||||||
struct list_head waiting;
|
|
||||||
struct list_head done;
|
|
||||||
struct list_head list;
|
|
||||||
int (*bound)(struct v4l2_async_notifier *notifier,
|
int (*bound)(struct v4l2_async_notifier *notifier,
|
||||||
struct v4l2_subdev *subdev,
|
struct v4l2_subdev *subdev,
|
||||||
struct v4l2_async_subdev *asd);
|
struct v4l2_async_subdev *asd);
|
||||||
@ -109,6 +95,29 @@ struct v4l2_async_notifier {
|
|||||||
struct v4l2_async_subdev *asd);
|
struct v4l2_async_subdev *asd);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* struct v4l2_async_notifier - v4l2_device notifier data
|
||||||
|
*
|
||||||
|
* @ops: notifier operations
|
||||||
|
* @num_subdevs: number of subdevices used in the subdevs array
|
||||||
|
* @max_subdevs: number of subdevices allocated in the subdevs array
|
||||||
|
* @subdevs: array of pointers to subdevice descriptors
|
||||||
|
* @v4l2_dev: pointer to struct v4l2_device
|
||||||
|
* @waiting: list of struct v4l2_async_subdev, waiting for their drivers
|
||||||
|
* @done: list of struct v4l2_subdev, already probed
|
||||||
|
* @list: member in a global list of notifiers
|
||||||
|
*/
|
||||||
|
struct v4l2_async_notifier {
|
||||||
|
const struct v4l2_async_notifier_operations *ops;
|
||||||
|
unsigned int num_subdevs;
|
||||||
|
unsigned int max_subdevs;
|
||||||
|
struct v4l2_async_subdev **subdevs;
|
||||||
|
struct v4l2_device *v4l2_dev;
|
||||||
|
struct list_head waiting;
|
||||||
|
struct list_head done;
|
||||||
|
struct list_head list;
|
||||||
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* v4l2_async_notifier_register - registers a subdevice asynchronous notifier
|
* v4l2_async_notifier_register - registers a subdevice asynchronous notifier
|
||||||
*
|
*
|
||||||
|
Loading…
Reference in New Issue
Block a user