[alsa-devel] [RFC PATCH 3/4] media: Add refcount to keep track of media device registrations

Mauro Carvalho Chehab mchehab at osg.samsung.com
Mon Mar 28 20:28:41 CEST 2016


Em Fri, 25 Mar 2016 22:38:44 -0600
Shuah Khan <shuahkh at osg.samsung.com> escreveu:

> Add refcount to keep track of media device registrations to avoid release
> of media device when one of the drivers does unregister when media device
> belongs to more than one driver. Also add a new interfaces to unregister
> a media device allocated using Media Device Allocator and a hold register
> refcount. Change media_open() to get media device reference and put the
> reference in media_release().
> 
> Signed-off-by: Shuah Khan <shuahkh at osg.samsung.com>
> ---
>  drivers/media/media-device.c  | 53 +++++++++++++++++++++++++++++++++++++++++++
>  drivers/media/media-devnode.c |  3 +++
>  include/media/media-device.h  | 32 ++++++++++++++++++++++++++
>  3 files changed, 88 insertions(+)
> 
> diff --git a/drivers/media/media-device.c b/drivers/media/media-device.c
> index 93aff4e..3359235 100644
> --- a/drivers/media/media-device.c
> +++ b/drivers/media/media-device.c
> @@ -36,6 +36,7 @@
>  #include <media/media-device.h>
>  #include <media/media-devnode.h>
>  #include <media/media-entity.h>
> +#include <media/media-dev-allocator.h>
>  
>  #ifdef CONFIG_MEDIA_CONTROLLER
>  
> @@ -702,6 +703,7 @@ void media_device_init(struct media_device *mdev)
>  	INIT_LIST_HEAD(&mdev->entity_notify);
>  	mutex_init(&mdev->graph_mutex);
>  	ida_init(&mdev->entity_internal_idx);
> +	kref_init(&mdev->refcount);
>  
>  	dev_dbg(mdev->dev, "Media device initialized\n");
>  }
> @@ -730,6 +732,13 @@ printk("%s: mdev %p\n", __func__, mdev);
>  	/* Check if mdev was ever registered at all */
>  	mutex_lock(&mdev->graph_mutex);
>  
> +	/* if media device is already registered, bump the register refcount */
> +	if (media_devnode_is_registered(&mdev->devnode)) {
> +		kref_get(&mdev->refcount);
> +		mutex_unlock(&mdev->graph_mutex);
> +		return 0;
> +	}
> +
>  	/* Register the device node. */
>  	mdev->devnode.fops = &media_device_fops;
>  	mdev->devnode.parent = mdev->dev;
> @@ -756,6 +765,22 @@ err:
>  }
>  EXPORT_SYMBOL_GPL(__media_device_register);
>  
> +void media_device_register_ref(struct media_device *mdev)
> +{
> +	if (!mdev)
> +		return;
> +
> +	pr_info("%s: mdev %p\n", __func__, mdev);
> +	mutex_lock(&mdev->graph_mutex);
> +
> +	/* Check if mdev is registered - bump registered refcount */
> +	if (media_devnode_is_registered(&mdev->devnode))
> +		kref_get(&mdev->refcount);
> +
> +	mutex_unlock(&mdev->graph_mutex);
> +}
> +EXPORT_SYMBOL_GPL(media_device_register_ref);
> +
>  int __must_check media_device_register_entity_notify(struct media_device *mdev,
>  					struct media_entity_notify *nptr)
>  {
> @@ -829,6 +854,34 @@ printk("%s: mdev=%p\n", __func__, mdev);
>  }
>  EXPORT_SYMBOL_GPL(media_device_unregister);
>  
> +static void __media_device_unregister_kref(struct kref *kref)
> +{
> +	struct media_device *mdev;
> +
> +	mdev = container_of(kref, struct media_device, refcount);
> +	__media_device_unregister(mdev);
> +}
> +
> +void media_device_unregister_put(struct media_device *mdev)
> +{
> +	int ret;
> +
> +	if (mdev == NULL)
> +		return;
> +
> +	pr_info("%s: mdev=%p\n", __func__, mdev);
> +	ret = kref_put_mutex(&mdev->refcount, __media_device_unregister_kref,
> +			     &mdev->graph_mutex);
> +	if (ret) {
> +		/* __media_device_unregister() ran */
> +		__media_device_cleanup(mdev);
> +		mutex_unlock(&mdev->graph_mutex);
> +		mutex_destroy(&mdev->graph_mutex);
> +		media_device_set_to_delete_state(mdev->dev);

Where you're freeing the media_dev (or its container struct)?

You need to be sure that it will be freed only here.

> +	}
> +}
> +EXPORT_SYMBOL_GPL(media_device_unregister_put);
> +
>  static void media_device_release_devres(struct device *dev, void *res)
>  {
>  }
> diff --git a/drivers/media/media-devnode.c b/drivers/media/media-devnode.c
> index 29409f4..d1d1263 100644
> --- a/drivers/media/media-devnode.c
> +++ b/drivers/media/media-devnode.c
> @@ -44,6 +44,7 @@
>  #include <linux/uaccess.h>
>  
>  #include <media/media-devnode.h>
> +#include <media/media-dev-allocator.h>
>  
>  #define MEDIA_NUM_DEVICES	256
>  #define MEDIA_NAME		"media"
> @@ -186,6 +187,7 @@ static int media_open(struct inode *inode, struct file *filp)
>  		}
>  	}
>  
> +	media_device_get_ref(mdev->parent);
>  	return 0;
>  }
>  
> @@ -201,6 +203,7 @@ static int media_release(struct inode *inode, struct file *filp)
>  	   return value is ignored. */
>  	put_device(&mdev->dev);
>  	filp->private_data = NULL;
> +	media_device_put(mdev->parent);
>  	return 0;
>  }
>  
> diff --git a/include/media/media-device.h b/include/media/media-device.h
> index e59772e..64114ae 100644
> --- a/include/media/media-device.h
> +++ b/include/media/media-device.h
> @@ -284,6 +284,8 @@ struct media_entity_notify {
>   * struct media_device - Media device
>   * @dev:	Parent device
>   * @devnode:	Media device node
> + * @refcount:	Media device register reference count. Used when more
> + *		than one driver owns the device.
>   * @driver_name: Optional device driver name. If not set, calls to
>   *		%MEDIA_IOC_DEVICE_INFO will return dev->driver->name.
>   *		This is needed for USB drivers for example, as otherwise
> @@ -348,6 +350,7 @@ struct media_device {
>  	/* dev->driver_data points to this struct. */
>  	struct device *dev;
>  	struct media_devnode devnode;
> +	struct kref refcount;

You can't simply embed a kref at media_device. The problem is that
several conditions should be met for this approach to work:

1) The structs that have struct media_device should not have a kref
their own (I guess that's the current case, but it should be documented
somewhere);

2) the struct that embeds it can only be destroyed when kref refcount
reaches zero. That actually means that the core should either allocate
the struct itself or that a release callback for those structs should
do the kfree(). It means that all drivers should be changed for it to
happen.

Also, if you're adding a kref here, you likely should not have a kref
at struct media_device_instance(), as there's no need for two kref
destroy logic.

>  
>  	char model[32];
>  	char driver_name[32];
> @@ -501,6 +504,17 @@ int __must_check __media_device_register(struct media_device *mdev,
>  #define media_device_register(mdev) __media_device_register(mdev, THIS_MODULE)
>  
>  /**
> + * media_device_register_ref() - Increments media device register refcount
> + *
> + * @mdev:	pointer to struct &media_device
> + *
> + * When more than one driver is associated with the media device, it is
> + * necessary to refcount the number of registrations to avoid unregister
> + * when it is still in use.
> + */
> +void media_device_register_ref(struct media_device *mdev);
> +
> +/**
>   * media_device_unregister() - Unregisters a media device element
>   *
>   * @mdev:	pointer to struct &media_device
> @@ -512,6 +526,18 @@ int __must_check __media_device_register(struct media_device *mdev,
>  void media_device_unregister(struct media_device *mdev);
>  
>  /**
> + * media_device_unregister_put() - Unregisters a media device element
> + *
> + * @mdev:	pointer to struct &media_device
> + *
> + * Should be called to unregister media device allocated with Media Device
> + * Allocator API media_device_get() interface.
> + * It is safe to call this function on an unregistered (but initialised)
> + * media device.
> + */
> +void media_device_unregister_put(struct media_device *mdev);
> +
> +/**
>   * media_device_register_entity() - registers a media entity inside a
>   *	previously registered media device.
>   *
> @@ -681,9 +707,15 @@ static inline int media_device_register(struct media_device *mdev)
>  {
>  	return 0;
>  }
> +static inline void media_device_register_ref(struct media_device *mdev)
> +{
> +}
>  static inline void media_device_unregister(struct media_device *mdev)
>  {
>  }
> +static inline void media_device_unregister_put(struct media_device *mdev)
> +{
> +}
>  static inline int media_device_register_entity(struct media_device *mdev,
>  						struct media_entity *entity)
>  {


-- 
Thanks,
Mauro


More information about the Alsa-devel mailing list