diff options
Diffstat (limited to 'drivers/input/input-polldev.c')
| -rw-r--r-- | drivers/input/input-polldev.c | 207 | 
1 files changed, 138 insertions, 69 deletions
diff --git a/drivers/input/input-polldev.c b/drivers/input/input-polldev.c index 10c9b0a845f..3664f81655c 100644 --- a/drivers/input/input-polldev.c +++ b/drivers/input/input-polldev.c @@ -8,9 +8,13 @@   * the Free Software Foundation.   */ +#define pr_fmt(fmt) KBUILD_MODNAME ": " fmt +  #include <linux/jiffies.h>  #include <linux/slab.h>  #include <linux/mutex.h> +#include <linux/workqueue.h> +#include <linux/module.h>  #include <linux/input-polldev.h>  MODULE_AUTHOR("Dmitry Torokhov <dtor@mail.ru>"); @@ -18,45 +22,6 @@ MODULE_DESCRIPTION("Generic implementation of a polled input device");  MODULE_LICENSE("GPL v2");  MODULE_VERSION("0.1"); -static DEFINE_MUTEX(polldev_mutex); -static int polldev_users; -static struct workqueue_struct *polldev_wq; - -static int input_polldev_start_workqueue(void) -{ -	int retval; - -	retval = mutex_lock_interruptible(&polldev_mutex); -	if (retval) -		return retval; - -	if (!polldev_users) { -		polldev_wq = create_singlethread_workqueue("ipolldevd"); -		if (!polldev_wq) { -			printk(KERN_ERR "input-polldev: failed to create " -				"ipolldevd workqueue\n"); -			retval = -ENOMEM; -			goto out; -		} -	} - -	polldev_users++; - - out: -	mutex_unlock(&polldev_mutex); -	return retval; -} - -static void input_polldev_stop_workqueue(void) -{ -	mutex_lock(&polldev_mutex); - -	if (!--polldev_users) -		destroy_workqueue(polldev_wq); - -	mutex_unlock(&polldev_mutex); -} -  static void input_polldev_queue_work(struct input_polled_dev *dev)  {  	unsigned long delay; @@ -65,7 +30,7 @@ static void input_polldev_queue_work(struct input_polled_dev *dev)  	if (delay >= HZ)  		delay = round_jiffies_relative(delay); -	queue_delayed_work(polldev_wq, &dev->work, delay); +	queue_delayed_work(system_freezable_wq, &dev->work, delay);  }  static void input_polled_device_work(struct work_struct *work) @@ -80,18 +45,15 @@ static void input_polled_device_work(struct work_struct *work)  static int input_open_polled_device(struct input_dev *input)  {  	struct input_polled_dev *dev = input_get_drvdata(input); -	int error; - -	error = input_polldev_start_workqueue(); -	if (error) -		return error;  	if (dev->open)  		dev->open(dev);  	/* Only start polling if polling is enabled */ -	if (dev->poll_interval > 0) -		queue_delayed_work(polldev_wq, &dev->work, 0); +	if (dev->poll_interval > 0) { +		dev->poll(dev); +		input_polldev_queue_work(dev); +	}  	return 0;  } @@ -101,13 +63,6 @@ static void input_close_polled_device(struct input_dev *input)  	struct input_polled_dev *dev = input_get_drvdata(input);  	cancel_delayed_work_sync(&dev->work); -	/* -	 * Clean up work struct to remove references to the workqueue. -	 * It may be destroyed by the next call. This causes problems -	 * at next device open-close in case of poll_interval == 0. -	 */ -	INIT_DELAYED_WORK(&dev->work, dev->work.work.func); -	input_polldev_stop_workqueue();  	if (dev->close)  		dev->close(dev); @@ -129,10 +84,12 @@ static ssize_t input_polldev_set_poll(struct device *dev,  {  	struct input_polled_dev *polldev = dev_get_drvdata(dev);  	struct input_dev *input = polldev->input; -	unsigned long interval; +	unsigned int interval; +	int err; -	if (strict_strtoul(buf, 0, &interval)) -		return -EINVAL; +	err = kstrtouint(buf, 0, &interval); +	if (err) +		return err;  	if (interval < polldev->poll_interval_min)  		return -EINVAL; @@ -190,8 +147,13 @@ static struct attribute_group input_polldev_attribute_group = {  	.attrs = sysfs_attrs  }; +static const struct attribute_group *input_polldev_attribute_groups[] = { +	&input_polldev_attribute_group, +	NULL +}; +  /** - * input_allocate_polled_device - allocated memory polled device + * input_allocate_polled_device - allocate memory for polled device   *   * The function allocates memory for a polled device and also   * for an input device associated with this polled device. @@ -214,6 +176,91 @@ struct input_polled_dev *input_allocate_polled_device(void)  }  EXPORT_SYMBOL(input_allocate_polled_device); +struct input_polled_devres { +	struct input_polled_dev *polldev; +}; + +static int devm_input_polldev_match(struct device *dev, void *res, void *data) +{ +	struct input_polled_devres *devres = res; + +	return devres->polldev == data; +} + +static void devm_input_polldev_release(struct device *dev, void *res) +{ +	struct input_polled_devres *devres = res; +	struct input_polled_dev *polldev = devres->polldev; + +	dev_dbg(dev, "%s: dropping reference/freeing %s\n", +		__func__, dev_name(&polldev->input->dev)); + +	input_put_device(polldev->input); +	kfree(polldev); +} + +static void devm_input_polldev_unregister(struct device *dev, void *res) +{ +	struct input_polled_devres *devres = res; +	struct input_polled_dev *polldev = devres->polldev; + +	dev_dbg(dev, "%s: unregistering device %s\n", +		__func__, dev_name(&polldev->input->dev)); +	input_unregister_device(polldev->input); + +	/* +	 * Note that we are still holding extra reference to the input +	 * device so it will stick around until devm_input_polldev_release() +	 * is called. +	 */ +} + +/** + * devm_input_allocate_polled_device - allocate managed polled device + * @dev: device owning the polled device being created + * + * Returns prepared &struct input_polled_dev or %NULL. + * + * Managed polled input devices do not need to be explicitly unregistered + * or freed as it will be done automatically when owner device unbinds + * from * its driver (or binding fails). Once such managed polled device + * is allocated, it is ready to be set up and registered in the same + * fashion as regular polled input devices (using + * input_register_polled_device() function). + * + * If you want to manually unregister and free such managed polled devices, + * it can be still done by calling input_unregister_polled_device() and + * input_free_polled_device(), although it is rarely needed. + * + * NOTE: the owner device is set up as parent of input device and users + * should not override it. + */ +struct input_polled_dev *devm_input_allocate_polled_device(struct device *dev) +{ +	struct input_polled_dev *polldev; +	struct input_polled_devres *devres; + +	devres = devres_alloc(devm_input_polldev_release, sizeof(*devres), +			      GFP_KERNEL); +	if (!devres) +		return NULL; + +	polldev = input_allocate_polled_device(); +	if (!polldev) { +		devres_free(devres); +		return NULL; +	} + +	polldev->input->dev.parent = dev; +	polldev->devres_managed = true; + +	devres->polldev = polldev; +	devres_add(dev, devres); + +	return polldev; +} +EXPORT_SYMBOL(devm_input_allocate_polled_device); +  /**   * input_free_polled_device - free memory allocated for polled device   * @dev: device to free @@ -224,7 +271,12 @@ EXPORT_SYMBOL(input_allocate_polled_device);  void input_free_polled_device(struct input_polled_dev *dev)  {  	if (dev) { -		input_free_device(dev->input); +		if (dev->devres_managed) +			WARN_ON(devres_destroy(dev->input->dev.parent, +						devm_input_polldev_release, +						devm_input_polldev_match, +						dev)); +		input_put_device(dev->input);  		kfree(dev);  	}  } @@ -238,30 +290,39 @@ EXPORT_SYMBOL(input_free_polled_device);   * with input layer. The device should be allocated with call to   * input_allocate_polled_device(). Callers should also set up poll()   * method and set up capabilities (id, name, phys, bits) of the - * corresponing input_dev structure. + * corresponding input_dev structure.   */  int input_register_polled_device(struct input_polled_dev *dev)  { +	struct input_polled_devres *devres = NULL;  	struct input_dev *input = dev->input;  	int error; +	if (dev->devres_managed) { +		devres = devres_alloc(devm_input_polldev_unregister, +				      sizeof(*devres), GFP_KERNEL); +		if (!devres) +			return -ENOMEM; + +		devres->polldev = dev; +	} +  	input_set_drvdata(input, dev);  	INIT_DELAYED_WORK(&dev->work, input_polled_device_work); +  	if (!dev->poll_interval)  		dev->poll_interval = 500;  	if (!dev->poll_interval_max)  		dev->poll_interval_max = dev->poll_interval; +  	input->open = input_open_polled_device;  	input->close = input_close_polled_device; -	error = input_register_device(input); -	if (error) -		return error; +	input->dev.groups = input_polldev_attribute_groups; -	error = sysfs_create_group(&input->dev.kobj, -				   &input_polldev_attribute_group); +	error = input_register_device(input);  	if (error) { -		input_unregister_device(input); +		devres_free(devres);  		return error;  	} @@ -274,6 +335,12 @@ int input_register_polled_device(struct input_polled_dev *dev)  	 */  	input_get_device(input); +	if (dev->devres_managed) { +		dev_dbg(input->dev.parent, "%s: registering %s with devres.\n", +			__func__, dev_name(&input->dev)); +		devres_add(input->dev.parent, devres); +	} +  	return 0;  }  EXPORT_SYMBOL(input_register_polled_device); @@ -288,10 +355,12 @@ EXPORT_SYMBOL(input_register_polled_device);   */  void input_unregister_polled_device(struct input_polled_dev *dev)  { -	sysfs_remove_group(&dev->input->dev.kobj, -			   &input_polldev_attribute_group); +	if (dev->devres_managed) +		WARN_ON(devres_destroy(dev->input->dev.parent, +					devm_input_polldev_unregister, +					devm_input_polldev_match, +					dev));  	input_unregister_device(dev->input);  }  EXPORT_SYMBOL(input_unregister_polled_device); -  | 
