Commit 61f5db54 authored by Laurent Pinchart's avatar Laurent Pinchart Committed by Mauro Carvalho Chehab

[media] v4l: Make v4l2_subdev inherit from media_entity

V4L2 subdevices are media entities. As such they need to inherit from
(include) the media_entity structure.

When registering/unregistering the subdevice, the media entity is
automatically registered/unregistered. The entity is acquired on device
open and released on device close.
Signed-off-by: default avatarLaurent Pinchart <laurent.pinchart@ideasonboard.com>
Signed-off-by: default avatarSakari Ailus <sakari.ailus@iki.fi>
Acked-by: default avatarHans Verkuil <hverkuil@xs4all.nl>
Signed-off-by: default avatarMauro Carvalho Chehab <mchehab@redhat.com>
parent 2c0ab67b
...@@ -268,6 +268,26 @@ A sub-device driver initializes the v4l2_subdev struct using: ...@@ -268,6 +268,26 @@ A sub-device driver initializes the v4l2_subdev struct using:
Afterwards you need to initialize subdev->name with a unique name and set the Afterwards you need to initialize subdev->name with a unique name and set the
module owner. This is done for you if you use the i2c helper functions. module owner. This is done for you if you use the i2c helper functions.
If integration with the media framework is needed, you must initialize the
media_entity struct embedded in the v4l2_subdev struct (entity field) by
calling media_entity_init():
struct media_pad *pads = &my_sd->pads;
int err;
err = media_entity_init(&sd->entity, npads, pads, 0);
The pads array must have been previously initialized. There is no need to
manually set the struct media_entity type and name fields, but the revision
field must be initialized if needed.
A reference to the entity will be automatically acquired/released when the
subdev device node (if any) is opened/closed.
Don't forget to cleanup the media entity before the sub-device is destroyed:
media_entity_cleanup(&sd->entity);
A device (bridge) driver needs to register the v4l2_subdev with the A device (bridge) driver needs to register the v4l2_subdev with the
v4l2_device: v4l2_device:
...@@ -277,6 +297,9 @@ This can fail if the subdev module disappeared before it could be registered. ...@@ -277,6 +297,9 @@ This can fail if the subdev module disappeared before it could be registered.
After this function was called successfully the subdev->dev field points to After this function was called successfully the subdev->dev field points to
the v4l2_device. the v4l2_device.
If the v4l2_device parent device has a non-NULL mdev field, the sub-device
entity will be automatically registered with the media device.
You can unregister a sub-device using: You can unregister a sub-device using:
v4l2_device_unregister_subdev(sd); v4l2_device_unregister_subdev(sd);
......
...@@ -118,8 +118,11 @@ void v4l2_device_unregister(struct v4l2_device *v4l2_dev) ...@@ -118,8 +118,11 @@ void v4l2_device_unregister(struct v4l2_device *v4l2_dev)
EXPORT_SYMBOL_GPL(v4l2_device_unregister); EXPORT_SYMBOL_GPL(v4l2_device_unregister);
int v4l2_device_register_subdev(struct v4l2_device *v4l2_dev, int v4l2_device_register_subdev(struct v4l2_device *v4l2_dev,
struct v4l2_subdev *sd) struct v4l2_subdev *sd)
{ {
#if defined(CONFIG_MEDIA_CONTROLLER)
struct media_entity *entity = &sd->entity;
#endif
int err; int err;
/* Check for valid input */ /* Check for valid input */
...@@ -147,6 +150,19 @@ int v4l2_device_register_subdev(struct v4l2_device *v4l2_dev, ...@@ -147,6 +150,19 @@ int v4l2_device_register_subdev(struct v4l2_device *v4l2_dev,
return err; return err;
} }
#if defined(CONFIG_MEDIA_CONTROLLER)
/* Register the entity. */
if (v4l2_dev->mdev) {
err = media_device_register_entity(v4l2_dev->mdev, entity);
if (err < 0) {
if (sd->internal_ops && sd->internal_ops->unregistered)
sd->internal_ops->unregistered(sd);
module_put(sd->owner);
return err;
}
}
#endif
spin_lock(&v4l2_dev->lock); spin_lock(&v4l2_dev->lock);
list_add_tail(&sd->list, &v4l2_dev->subdevs); list_add_tail(&sd->list, &v4l2_dev->subdevs);
spin_unlock(&v4l2_dev->lock); spin_unlock(&v4l2_dev->lock);
...@@ -177,25 +193,37 @@ int v4l2_device_register_subdev_nodes(struct v4l2_device *v4l2_dev) ...@@ -177,25 +193,37 @@ int v4l2_device_register_subdev_nodes(struct v4l2_device *v4l2_dev)
sd->owner); sd->owner);
if (err < 0) if (err < 0)
return err; return err;
#if defined(CONFIG_MEDIA_CONTROLLER)
sd->entity.v4l.major = VIDEO_MAJOR;
sd->entity.v4l.minor = vdev->minor;
#endif
} }
return 0; return 0;
} }
EXPORT_SYMBOL_GPL(v4l2_device_register_subdev_nodes); EXPORT_SYMBOL_GPL(v4l2_device_register_subdev_nodes);
void v4l2_device_unregister_subdev(struct v4l2_subdev *sd) void v4l2_device_unregister_subdev(struct v4l2_subdev *sd)
{ {
struct v4l2_device *v4l2_dev;
/* return if it isn't registered */ /* return if it isn't registered */
if (sd == NULL || sd->v4l2_dev == NULL) if (sd == NULL || sd->v4l2_dev == NULL)
return; return;
spin_lock(&sd->v4l2_dev->lock); v4l2_dev = sd->v4l2_dev;
spin_lock(&v4l2_dev->lock);
list_del(&sd->list); list_del(&sd->list);
spin_unlock(&sd->v4l2_dev->lock); spin_unlock(&v4l2_dev->lock);
if (sd->internal_ops && sd->internal_ops->unregistered) if (sd->internal_ops && sd->internal_ops->unregistered)
sd->internal_ops->unregistered(sd); sd->internal_ops->unregistered(sd);
sd->v4l2_dev = NULL; sd->v4l2_dev = NULL;
#if defined(CONFIG_MEDIA_CONTROLLER)
if (v4l2_dev->mdev)
media_device_unregister_entity(&sd->entity);
#endif
video_unregister_device(&sd->devnode); video_unregister_device(&sd->devnode);
module_put(sd->owner); module_put(sd->owner);
} }
......
...@@ -35,7 +35,10 @@ static int subdev_open(struct file *file) ...@@ -35,7 +35,10 @@ static int subdev_open(struct file *file)
{ {
struct video_device *vdev = video_devdata(file); struct video_device *vdev = video_devdata(file);
struct v4l2_subdev *sd = vdev_to_v4l2_subdev(vdev); struct v4l2_subdev *sd = vdev_to_v4l2_subdev(vdev);
struct v4l2_fh *vfh; #if defined(CONFIG_MEDIA_CONTROLLER)
struct media_entity *entity;
#endif
struct v4l2_fh *vfh = NULL;
int ret; int ret;
if (sd->flags & V4L2_SUBDEV_FL_HAS_EVENTS) { if (sd->flags & V4L2_SUBDEV_FL_HAS_EVENTS) {
...@@ -58,11 +61,20 @@ static int subdev_open(struct file *file) ...@@ -58,11 +61,20 @@ static int subdev_open(struct file *file)
v4l2_fh_add(vfh); v4l2_fh_add(vfh);
file->private_data = vfh; file->private_data = vfh;
} }
#if defined(CONFIG_MEDIA_CONTROLLER)
if (sd->v4l2_dev->mdev) {
entity = media_entity_get(&sd->entity);
if (!entity) {
ret = -EBUSY;
goto err;
}
}
#endif
return 0; return 0;
err: err:
if (vfh != NULL) { if (vfh != NULL) {
v4l2_fh_del(vfh);
v4l2_fh_exit(vfh); v4l2_fh_exit(vfh);
kfree(vfh); kfree(vfh);
} }
...@@ -72,8 +84,16 @@ static int subdev_open(struct file *file) ...@@ -72,8 +84,16 @@ static int subdev_open(struct file *file)
static int subdev_close(struct file *file) static int subdev_close(struct file *file)
{ {
#if defined(CONFIG_MEDIA_CONTROLLER)
struct video_device *vdev = video_devdata(file);
struct v4l2_subdev *sd = vdev_to_v4l2_subdev(vdev);
#endif
struct v4l2_fh *vfh = file->private_data; struct v4l2_fh *vfh = file->private_data;
#if defined(CONFIG_MEDIA_CONTROLLER)
if (sd->v4l2_dev->mdev)
media_entity_put(&sd->entity);
#endif
if (vfh != NULL) { if (vfh != NULL) {
v4l2_fh_del(vfh); v4l2_fh_del(vfh);
v4l2_fh_exit(vfh); v4l2_fh_exit(vfh);
...@@ -172,5 +192,9 @@ void v4l2_subdev_init(struct v4l2_subdev *sd, const struct v4l2_subdev_ops *ops) ...@@ -172,5 +192,9 @@ void v4l2_subdev_init(struct v4l2_subdev *sd, const struct v4l2_subdev_ops *ops)
sd->grp_id = 0; sd->grp_id = 0;
sd->dev_priv = NULL; sd->dev_priv = NULL;
sd->host_priv = NULL; sd->host_priv = NULL;
#if defined(CONFIG_MEDIA_CONTROLLER)
sd->entity.name = sd->name;
sd->entity.type = MEDIA_ENT_T_V4L2_SUBDEV;
#endif
} }
EXPORT_SYMBOL(v4l2_subdev_init); EXPORT_SYMBOL(v4l2_subdev_init);
...@@ -21,6 +21,7 @@ ...@@ -21,6 +21,7 @@
#ifndef _V4L2_SUBDEV_H #ifndef _V4L2_SUBDEV_H
#define _V4L2_SUBDEV_H #define _V4L2_SUBDEV_H
#include <media/media-entity.h>
#include <media/v4l2-common.h> #include <media/v4l2-common.h>
#include <media/v4l2-dev.h> #include <media/v4l2-dev.h>
#include <media/v4l2-mediabus.h> #include <media/v4l2-mediabus.h>
...@@ -450,6 +451,9 @@ struct v4l2_subdev_internal_ops { ...@@ -450,6 +451,9 @@ struct v4l2_subdev_internal_ops {
stand-alone or embedded in a larger struct. stand-alone or embedded in a larger struct.
*/ */
struct v4l2_subdev { struct v4l2_subdev {
#if defined(CONFIG_MEDIA_CONTROLLER)
struct media_entity entity;
#endif
struct list_head list; struct list_head list;
struct module *owner; struct module *owner;
u32 flags; u32 flags;
...@@ -472,6 +476,8 @@ struct v4l2_subdev { ...@@ -472,6 +476,8 @@ struct v4l2_subdev {
unsigned int nevents; unsigned int nevents;
}; };
#define media_entity_to_v4l2_subdev(ent) \
container_of(ent, struct v4l2_subdev, entity)
#define vdev_to_v4l2_subdev(vdev) \ #define vdev_to_v4l2_subdev(vdev) \
container_of(vdev, struct v4l2_subdev, devnode) container_of(vdev, struct v4l2_subdev, devnode)
......
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