On 08/30/2015 05:07 AM, Mauro Carvalho Chehab wrote:
> V4L2 device (and subdevice) nodes should create an
> interface, if the Media Controller support is enabled.
> 
> Please notice that radio devices should not create an
> entity, as radio input/output is either via wires or
> via ALSA.

A general note: I think this patch (and any prerequisite patches) should come 
before
the patches adding G_TOPOLOGY support.

What the G_TOPOLOGY ioctl returns only makes sense IMHO if this code is present 
as well,
so it is a more logical order for this patch series.

In addition, since G_TOPOLOGY is a userspace API it is likely that that will 
create
more discussions, whereas this is internal to the kernel and could be merged 
before
G_TOPOLOGY.

> 
> Signed-off-by: Mauro Carvalho Chehab <mche...@osg.samsung.com>
> 
> diff --git a/drivers/media/v4l2-core/v4l2-dev.c 
> b/drivers/media/v4l2-core/v4l2-dev.c
> index 44b330589787..427a5a32b3de 100644
> --- a/drivers/media/v4l2-core/v4l2-dev.c
> +++ b/drivers/media/v4l2-core/v4l2-dev.c
> @@ -194,9 +194,12 @@ static void v4l2_device_release(struct device *cd)
>       mutex_unlock(&videodev_lock);
>  
>  #if defined(CONFIG_MEDIA_CONTROLLER)
> -     if (v4l2_dev->mdev &&
> -         vdev->vfl_type != VFL_TYPE_SUBDEV)
> -             media_device_unregister_entity(&vdev->entity);
> +     if (v4l2_dev->mdev) {
> +             /* Remove interfaces and interface links */
> +             media_devnode_remove(vdev->intf_devnode);
> +             if (vdev->vfl_type != VFL_TYPE_SUBDEV)
> +                     media_device_unregister_entity(&vdev->entity);

RADIO doesn't have an entity either, so this should probably check for both
SUBDEV and RADIO.

I think it is cleaner if video_register_media_controller() sets a new 
video_device
flag: V4L2_FL_CREATED_ENTITY, and if this release function would just check the
flag.

> +     }
>  #endif
>  
>       /* Do not call v4l2_device_put if there is no release callback set.
> @@ -713,6 +716,85 @@ static void determine_valid_ioctls(struct video_device 
> *vdev)
>                       BASE_VIDIOC_PRIVATE);
>  }
>  
> +
> +static int video_register_media_controller(struct video_device *vdev, int 
> type)
> +{
> +#if defined(CONFIG_MEDIA_CONTROLLER)
> +     u32 entity_type = MEDIA_ENT_T_UNKNOWN;
> +     u32 intf_type;
> +     int ret;
> +     bool create_entity = true;
> +
> +     if (!vdev->v4l2_dev->mdev)
> +             return 0;
> +
> +     switch (type) {
> +     case VFL_TYPE_GRABBER:
> +             intf_type = MEDIA_INTF_T_V4L_VIDEO;
> +             entity_type = MEDIA_ENT_T_V4L2_VIDEO;
> +             break;
> +     case VFL_TYPE_VBI:
> +             intf_type = MEDIA_INTF_T_V4L_VBI;
> +             entity_type = MEDIA_ENT_T_V4L2_VBI;
> +             break;
> +     case VFL_TYPE_SDR:
> +             intf_type = MEDIA_INTF_T_V4L_SWRADIO;
> +             entity_type = MEDIA_ENT_T_V4L2_SWRADIO;
> +             break;
> +     case VFL_TYPE_RADIO:
> +             intf_type = MEDIA_INTF_T_V4L_RADIO;
> +             /*
> +              * Radio doesn't have an entity at the V4L2 side to represent
> +              * radio input or output. Instead, the audio input/output goes
> +              * via either physical wires or ALSA.
> +              */
> +             create_entity = false;
> +             break;
> +     case VFL_TYPE_SUBDEV:
> +             intf_type = MEDIA_INTF_T_V4L_SUBDEV;
> +             /* Entity will be created via v4l2_device_register_subdev() */
> +             create_entity = false;
> +             break;
> +     default:
> +             return 0;
> +     }
> +
> +     if (create_entity) {
> +             vdev->entity.type = entity_type;
> +             vdev->entity.name = vdev->name;
> +
> +             /* Needed just for backward compatibility with legacy MC API */
> +             vdev->entity.info.dev.major = VIDEO_MAJOR;
> +             vdev->entity.info.dev.minor = vdev->minor;
> +
> +             ret = media_device_register_entity(vdev->v4l2_dev->mdev,
> +                                                &vdev->entity);
> +             if (ret < 0) {
> +                     printk(KERN_WARNING
> +                             "%s: media_device_register_entity failed\n",
> +                             __func__);
> +                     return ret;
> +             }
> +     }
> +
> +     vdev->intf_devnode = media_devnode_create(vdev->v4l2_dev->mdev,
> +                                               intf_type,
> +                                               0, VIDEO_MAJOR,
> +                                               vdev->minor,
> +                                               GFP_KERNEL);
> +     if (!vdev->intf_devnode)
> +             return -ENOMEM;
> +
> +     if (create_entity)
> +             media_create_intf_link(&vdev->entity,
> +                                    &vdev->intf_devnode->intf, 0);

You need a NULL pointer check here as well. It might be a good idea to add
__must_check to the media_create_intf_link() prototype. I think there are
more places where this isn't checked.

> +
> +     /* FIXME: how to create the other interface links? */
> +
> +#endif
> +     return 0;
> +}
> +
>  /**
>   *   __video_register_device - register video4linux devices
>   *   @vdev: video device structure we want to register
> @@ -908,22 +990,9 @@ int __video_register_device(struct video_device *vdev, 
> int type, int nr,
>       /* Increase v4l2_device refcount */
>       v4l2_device_get(vdev->v4l2_dev);
>  
> -#if defined(CONFIG_MEDIA_CONTROLLER)
>       /* Part 5: Register the entity. */
> -     if (vdev->v4l2_dev->mdev &&
> -         vdev->vfl_type != VFL_TYPE_SUBDEV) {
> -             vdev->entity.type = MEDIA_ENT_T_V4L2_VIDEO;
> -             vdev->entity.name = vdev->name;
> -             vdev->entity.info.dev.major = VIDEO_MAJOR;
> -             vdev->entity.info.dev.minor = vdev->minor;
> -             ret = media_device_register_entity(vdev->v4l2_dev->mdev,
> -                     &vdev->entity);
> -             if (ret < 0)
> -                     printk(KERN_WARNING
> -                            "%s: media_device_register_entity failed\n",
> -                            __func__);
> -     }
> -#endif
> +     ret = video_register_media_controller(vdev, type);
> +
>       /* Part 6: Activate this minor. The char device can now be used. */
>       set_bit(V4L2_FL_REGISTERED, &vdev->flags);
>  
> diff --git a/drivers/media/v4l2-core/v4l2-device.c 
> b/drivers/media/v4l2-core/v4l2-device.c
> index 5b0a30b9252b..17ec73b1796e 100644
> --- a/drivers/media/v4l2-core/v4l2-device.c
> +++ b/drivers/media/v4l2-core/v4l2-device.c
> @@ -249,6 +249,11 @@ int v4l2_device_register_subdev_nodes(struct v4l2_device 
> *v4l2_dev)
>  #if defined(CONFIG_MEDIA_CONTROLLER)
>               sd->entity.info.dev.major = VIDEO_MAJOR;
>               sd->entity.info.dev.minor = vdev->minor;
> +
> +             /* Interface is created by __video_register_device() */
> +             if (vdev->v4l2_dev->mdev)
> +                     media_create_intf_link(&sd->entity,
> +                                            &vdev->intf_devnode->intf, 0);
>  #endif
>               sd->devnode = vdev;
>       }
> @@ -285,7 +290,10 @@ void v4l2_device_unregister_subdev(struct v4l2_subdev 
> *sd)
>  
>  #if defined(CONFIG_MEDIA_CONTROLLER)
>       if (v4l2_dev->mdev) {
> -             media_entity_remove_links(&sd->entity);
> +             /*
> +              * No need to explicitly remove links, as both pads and
> +              * links are removed by the function below, at the right order

s/, at/ in/

> +              */
>               media_device_unregister_entity(&sd->entity);
>       }
>  #endif
> diff --git a/include/media/v4l2-dev.h b/include/media/v4l2-dev.h
> index acbcd2f5fe7f..eeabf20e87a6 100644
> --- a/include/media/v4l2-dev.h
> +++ b/include/media/v4l2-dev.h
> @@ -86,6 +86,7 @@ struct video_device
>  {
>  #if defined(CONFIG_MEDIA_CONTROLLER)
>       struct media_entity entity;
> +     struct media_intf_devnode *intf_devnode;
>  #endif
>       /* device ops */
>       const struct v4l2_file_operations *fops;
> 

Regards,

        Hans
--
To unsubscribe from this list: send the line "unsubscribe linux-media" in
the body of a message to majord...@vger.kernel.org
More majordomo info at  http://vger.kernel.org/majordomo-info.html

Reply via email to