[media] dvbdev: add support for media controller
authorMauro Carvalho Chehab <mchehab@osg.samsung.com>
Fri, 2 Jan 2015 15:19:51 +0000 (12:19 -0300)
committerMauro Carvalho Chehab <mchehab@osg.samsung.com>
Fri, 13 Feb 2015 23:10:12 +0000 (21:10 -0200)
Provide a way to register media controller device nodes
at the DVB core.

Please notice that the dvbdev callers also require changes
for the devices to be registered via the media controller.

Signed-off-by: Mauro Carvalho Chehab <mchehab@osg.samsung.com>
drivers/media/Kconfig
drivers/media/dvb-core/dvbdev.c
drivers/media/dvb-core/dvbdev.h

index 49cd30870e0d6ff6fb293fc33fd0af850cd6590c..3ef0f90b128fc5bdf6d5e5dff0d5bbfbd5190d7b 100644 (file)
@@ -87,13 +87,21 @@ config MEDIA_RC_SUPPORT
 
 config MEDIA_CONTROLLER
        bool "Media Controller API"
-       depends on MEDIA_CAMERA_SUPPORT
+       depends on MEDIA_CAMERA_SUPPORT || MEDIA_ANALOG_TV_SUPPORT || MEDIA_DIGITAL_TV_SUPPORT
        ---help---
          Enable the media controller API used to query media devices internal
          topology and configure it dynamically.
 
          This API is mostly used by camera interfaces in embedded platforms.
 
+config MEDIA_CONTROLLER_DVB
+       bool "Enable Media controller for DVB"
+       depends on MEDIA_CONTROLLER
+       ---help---
+         Enable the media controller API support for DVB.
+
+         This is currently experimental.
+
 #
 # Video4Linux support
 #      Only enables if one of the V4L2 types (ATV, webcam, radio) is selected
index 983db75de3503952545f88700858f7a24131a9f9..f98fd3b29afefbdadede9e0d6ff104fb28135e12 100644 (file)
@@ -180,6 +180,59 @@ skip:
        return -ENFILE;
 }
 
+static void dvb_register_media_device(struct dvb_device *dvbdev,
+                                     int type, int minor)
+{
+#if defined(CONFIG_MEDIA_CONTROLLER_DVB)
+       int ret;
+
+       if (!dvbdev->adapter->mdev)
+               return;
+
+       dvbdev->entity = kzalloc(sizeof(*dvbdev->entity), GFP_KERNEL);
+       if (!dvbdev->entity)
+               return;
+
+       dvbdev->entity->info.dev.major = DVB_MAJOR;
+       dvbdev->entity->info.dev.minor = minor;
+       dvbdev->entity->name = dvbdev->name;
+       switch (type) {
+       case DVB_DEVICE_FRONTEND:
+               dvbdev->entity->type = MEDIA_ENT_T_DEVNODE_DVB_FE;
+               break;
+       case DVB_DEVICE_DEMUX:
+               dvbdev->entity->type = MEDIA_ENT_T_DEVNODE_DVB_DEMUX;
+               break;
+       case DVB_DEVICE_DVR:
+               dvbdev->entity->type = MEDIA_ENT_T_DEVNODE_DVB_DVR;
+               break;
+       case DVB_DEVICE_CA:
+               dvbdev->entity->type = MEDIA_ENT_T_DEVNODE_DVB_CA;
+               break;
+       case DVB_DEVICE_NET:
+               dvbdev->entity->type = MEDIA_ENT_T_DEVNODE_DVB_NET;
+               break;
+       default:
+               kfree(dvbdev->entity);
+               dvbdev->entity = NULL;
+               return;
+       }
+
+       ret = media_device_register_entity(dvbdev->adapter->mdev,
+                                          dvbdev->entity);
+       if (ret < 0) {
+               printk(KERN_ERR
+                       "%s: media_device_register_entity failed for %s\n",
+                       __func__, dvbdev->entity->name);
+               kfree(dvbdev->entity);
+               dvbdev->entity = NULL;
+               return;
+       }
+
+       printk(KERN_DEBUG "%s: media device '%s' registered.\n",
+               __func__, dvbdev->entity->name);
+#endif
+}
 
 int dvb_register_device(struct dvb_adapter *adap, struct dvb_device **pdvbdev,
                        const struct dvb_device *template, void *priv, int type)
@@ -258,10 +311,11 @@ int dvb_register_device(struct dvb_adapter *adap, struct dvb_device **pdvbdev,
                       __func__, adap->num, dnames[type], id, PTR_ERR(clsdev));
                return PTR_ERR(clsdev);
        }
-
        dprintk(KERN_DEBUG "DVB: register adapter%d/%s%d @ minor: %i (0x%02x)\n",
                adap->num, dnames[type], id, minor, minor);
 
+       dvb_register_media_device(dvbdev, type, minor);
+
        return 0;
 }
 EXPORT_SYMBOL(dvb_register_device);
@@ -278,6 +332,13 @@ void dvb_unregister_device(struct dvb_device *dvbdev)
 
        device_destroy(dvb_class, MKDEV(DVB_MAJOR, dvbdev->minor));
 
+#if defined(CONFIG_MEDIA_CONTROLLER_DVB)
+       if (dvbdev->entity) {
+               media_device_unregister_entity(dvbdev->entity);
+               kfree(dvbdev->entity);
+       }
+#endif
+
        list_del (&dvbdev->list_head);
        kfree (dvbdev->fops);
        kfree (dvbdev);
index f96b28e7fc952f210000b39879e91fb29192d292..485d8e660aeabb5835f8f098f91446d2c01aee84 100644 (file)
@@ -27,6 +27,7 @@
 #include <linux/poll.h>
 #include <linux/fs.h>
 #include <linux/list.h>
+#include <media/media-device.h>
 
 #define DVB_MAJOR 212
 
@@ -71,6 +72,10 @@ struct dvb_adapter {
        int mfe_shared;                 /* indicates mutually exclusive frontends */
        struct dvb_device *mfe_dvbdev;  /* frontend device in use */
        struct mutex mfe_lock;          /* access lock for thread creation */
+
+#if defined(CONFIG_MEDIA_CONTROLLER_DVB)
+       struct media_device *mdev;
+#endif
 };
 
 
@@ -92,6 +97,14 @@ struct dvb_device {
        /* don't really need those !? -- FIXME: use video_usercopy  */
        int (*kernel_ioctl)(struct file *file, unsigned int cmd, void *arg);
 
+       /* Needed for media controller register/unregister */
+#if defined(CONFIG_MEDIA_CONTROLLER_DVB)
+       const char *name;
+
+       /* Filled inside dvbdev.c */
+       struct media_entity *entity;
+#endif
+
        void *priv;
 };