Re: [PATCH] gpio:gpio-pl2303: add gpio driver for GPIOs on PL2303

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

 



On Sun, Jul 20, 2014 at 08:01:31AM +0800, Wang YanQing wrote:
> PL2303HX has two GPIOs, this patch add driver for it.
> 
> Signed-off-by: Wang YanQing <udknight@xxxxxxxxx>
> ---
>  MAINTAINERS                 |   5 +
>  drivers/gpio/Kconfig        |   7 ++
>  drivers/gpio/Makefile       |   1 +
>  drivers/gpio/gpio-pl2303.c  | 238 ++++++++++++++++++++++++++++++++++++++++++++
>  drivers/usb/serial/pl2303.c |  19 ++++
>  5 files changed, 270 insertions(+)
>  create mode 100644 drivers/gpio/gpio-pl2303.c
> 
> diff --git a/MAINTAINERS b/MAINTAINERS
> index 53feaaf..4a9d764 100644
> --- a/MAINTAINERS
> +++ b/MAINTAINERS
> @@ -6903,6 +6903,11 @@ F:	drivers/i2c/busses/i2c-puv3.c
>  F:	drivers/video/fb-puv3.c
>  F:	drivers/rtc/rtc-puv3.c
>  
> +PL2303 GPIO DRIVER
> +M:	Wang YanQing <udknight@xxxxxxxxx>
> +S:	Maintained
> +F:	drivers/gpio/gpio-pl2303.c
> +
>  PMBUS HARDWARE MONITORING DRIVERS
>  M:	Guenter Roeck <linux@xxxxxxxxxxxx>
>  L:	lm-sensors@xxxxxxxxxxxxxx
> diff --git a/drivers/gpio/Kconfig b/drivers/gpio/Kconfig
> index 4a1b511..0f90950 100644
> --- a/drivers/gpio/Kconfig
> +++ b/drivers/gpio/Kconfig
> @@ -110,6 +110,13 @@ config GPIO_DA9055
>  config GPIO_MAX730X
>  	tristate
>  
> +config GPIO_PL2303
> +	tristate "USB Prolific 2303 gpio support"
> +	depends on USB_SERIAL_PL2303
> +	help
> +	  Enable support for GPIOs on USB Prolific 2303
> +	  It support two GPIOs on PL2303HX currently.
> +
>  comment "Memory mapped GPIO drivers:"
>  
>  config GPIO_CLPS711X
> diff --git a/drivers/gpio/Makefile b/drivers/gpio/Makefile
> index d10f6a9..4ff59f6 100644
> --- a/drivers/gpio/Makefile
> +++ b/drivers/gpio/Makefile
> @@ -101,3 +101,4 @@ obj-$(CONFIG_GPIO_WM8994)	+= gpio-wm8994.o
>  obj-$(CONFIG_GPIO_XILINX)	+= gpio-xilinx.o
>  obj-$(CONFIG_GPIO_XTENSA)	+= gpio-xtensa.o
>  obj-$(CONFIG_GPIO_ZEVIO)	+= gpio-zevio.o
> +obj-$(CONFIG_GPIO_PL2303)	+= gpio-pl2303.o
> diff --git a/drivers/gpio/gpio-pl2303.c b/drivers/gpio/gpio-pl2303.c
> new file mode 100644
> index 0000000..a703440
> --- /dev/null
> +++ b/drivers/gpio/gpio-pl2303.c
> @@ -0,0 +1,238 @@
> +/*
> + * PL2303 GPIO driver
> + *
> + * Copyright (C) 2014 Wang YanQing <udknight@xxxxxxxxx>
> + *
> + * This program is free software; you can redistribute it and/or modify
> + * it under the terms of the GNU General Public License as published by
> + * the Free Software Foundation; either version 2 of the License, or
> + * (at your option) any later version.
> + *
> + * Check pl2303.c for further details.
> + */
> +
> +#include <linux/module.h>
> +#include <linux/init.h>
> +#include <linux/platform_device.h>
> +#include <linux/slab.h>
> +#include <linux/usb.h>
> +#include <linux/gpio.h>
> +
> +#define VENDOR_READ_REQUEST_TYPE	0xc0
> +#define VENDOR_READ_REQUEST		0x01
> +
> +#define VENDOR_WRITE_REQUEST_TYPE	0x40
> +#define VENDOR_WRITE_REQUEST		0x01
> +
> +struct pl2303_gpio_data {
> +	struct usb_device *pl2303;
> +	/*
> +	 * 0..3: unknown (zero)
> +	 * 4: gp0 output enable (1: gp0 pin is output, 0: gp0 pin is input)
> +	 * 5: gp1 output enable (1: gp1 pin is output, 0: gp1 pin is input)
> +	 * 6: gp0 pin value
> +	 * 7: gp1 pin value
> +	 */
> +	u8 index;
> +	struct gpio_chip gpio_chip;
> +};
> +
> +static inline struct pl2303_gpio_data *to_pl2303_gpio(struct gpio_chip *chip)
> +{
> +	return container_of(chip, struct pl2303_gpio_data, gpio_chip);
> +}
> +
> +static int pl2303_gpio_read(struct gpio_chip *chip, unsigned char buf[1])
> +{
> +	struct pl2303_gpio_data *pl2303_gpio = to_pl2303_gpio(chip);
> +	struct usb_device *pl2303 = pl2303_gpio->pl2303;
> +	struct device *dev = chip->dev;
> +	int res;
> +
> +	res = usb_control_msg(pl2303, usb_rcvctrlpipe(pl2303, 0),
> +			VENDOR_READ_REQUEST, VENDOR_READ_REQUEST_TYPE,
> +			0x0081, 0, buf, 1, 100);
> +	if (res != 1) {
> +		dev_err(dev, "%s - failed to read [%04x]: %d\n", __func__,
> +			0x0081, res);
> +		if (res >= 0)
> +			res = -EIO;
> +
> +		return res;
> +	}
> +
> +	dev_dbg(dev, "%s - [%04x] = %02x\n", __func__, 0x0081, buf[0]);
> +
> +	return 0;
> +}
> +
> +static int pl2303_gpio_write(struct gpio_chip *chip)
> +{
> +	struct pl2303_gpio_data *pl2303_gpio = to_pl2303_gpio(chip);
> +	struct usb_device *pl2303 = pl2303_gpio->pl2303;
> +	struct device *dev = chip->dev;
> +	int res;
> +
> +	dev_dbg(dev, "%s - [%04x] = %02x\n", __func__, 1, pl2303_gpio->index);
> +
> +	res = usb_control_msg(pl2303, usb_sndctrlpipe(pl2303, 0),
> +			VENDOR_WRITE_REQUEST, VENDOR_WRITE_REQUEST_TYPE,
> +			1, pl2303_gpio->index, NULL, 0, 100);
> +	if (res) {
> +		dev_err(dev, "%s - failed to write [%04x] = %02x: %d\n", __func__,
> +			1, pl2303_gpio->index, res);
> +		return res;
> +	}
> +
> +	return 0;
> +}
> +
> +static int pl2303_gpio_direction_in(struct gpio_chip *chip, unsigned offset)
> +{
> +	struct pl2303_gpio_data *pl2303_gpio = to_pl2303_gpio(chip);
> +
> +	if (offset == 0)
> +		pl2303_gpio->index &= ~0x10;
> +	else if (offset == 1)
> +		pl2303_gpio->index &= ~0x20;
> +	else
> +		return -EINVAL;
> +
> +	pl2303_gpio_write(chip);
> +	return 0;
> +}
> +
> +static int pl2303_gpio_direction_out(struct gpio_chip *chip, unsigned offset, int value)
> +{
> +	struct pl2303_gpio_data *pl2303_gpio = to_pl2303_gpio(chip);
> +
> +	if (offset == 0) {
> +		pl2303_gpio->index |= 0x10;
> +		if (value)
> +			pl2303_gpio->index |= 0x40;
> +		else
> +			pl2303_gpio->index &= ~0x40;
> +	} else if (offset == 1) {
> +		pl2303_gpio->index |= 0x20;
> +		if (value)
> +			pl2303_gpio->index |= 0x80;
> +		else
> +			pl2303_gpio->index &= ~0x80;
> +	} else {
> +		return -EINVAL;
> +	}
> +
> +	pl2303_gpio_write(chip);
> +	return 0;
> +}
> +
> +static void pl2303_gpio_set(struct gpio_chip *chip, unsigned offset, int value)
> +{
> +	struct pl2303_gpio_data *pl2303_gpio = to_pl2303_gpio(chip);
> +
> +	if (offset == 0) {
> +		if (value)
> +			pl2303_gpio->index |= 0x40;
> +		else
> +			pl2303_gpio->index &= ~0x40;
> +	} else if (offset == 1) {
> +		if (value)
> +			pl2303_gpio->index |= 0x80;
> +		else
> +			pl2303_gpio->index &= ~0x80;
> +	} else {
> +		return;
> +	}
> +
> +	pl2303_gpio_write(chip);
> +}
> +
> +static int pl2303_gpio_get(struct gpio_chip *chip, unsigned offset)
> +{
> +	unsigned char buf[1];
> +	int value = 0;
> +
> +	if(pl2303_gpio_read(chip, buf) < 1)
> +		return -EIO;
> +	if (offset == 0)
> +		value = buf[0] & 0x40;
> +	else if (offset == 1)
> +		value = buf[0] & 0x80;
> +	else
> +		return -EINVAL;
> +	return value;
> +}
> +
> +static struct gpio_chip template_chip = {
> +	.label			= "pl2303-gpio",
> +	.owner			= THIS_MODULE,
> +	.direction_input	= pl2303_gpio_direction_in,
> +	.get			= pl2303_gpio_get,
> +	.direction_output	= pl2303_gpio_direction_out,
> +	.set			= pl2303_gpio_set,
> +	.can_sleep		= 1,
> +};
> +
> +static int pl2303_gpio_probe(struct platform_device *pdev)
> +{
> +	struct usb_device *pl2303 = platform_get_drvdata(pdev);
> +	struct pl2303_gpio_data *pl2303_gpio;
> +	int ret;
> +
> +	pl2303_gpio = kzalloc(sizeof(*pl2303_gpio), GFP_KERNEL);
> +	if (pl2303_gpio == NULL)
> +		return -ENOMEM;
> +
> +	/* initialize gpio0,gpio1 for input as default*/
> +	pl2303_gpio->index = 0x00;
> +	pl2303_gpio->pl2303 = pl2303;
> +	pl2303_gpio->gpio_chip = template_chip;
> +	pl2303_gpio->gpio_chip.ngpio = 2;
> +	pl2303_gpio->gpio_chip.base = -1;
> +	pl2303_gpio->gpio_chip.dev = &pdev->dev;
> +	pl2303_gpio_write(&pl2303_gpio->gpio_chip);
> +
> +	ret = gpiochip_add(&pl2303_gpio->gpio_chip);
> +	if (ret < 0) {
> +		dev_err(&pdev->dev, "Could not register gpiochip, %d\n", ret);
> +		return ret;
> +	}
> +	platform_set_drvdata(pdev, pl2303_gpio);
> +
> +	return ret;
> +}
> +
> +static int pl2303_gpio_remove(struct platform_device *pdev)
> +{
> +	struct pl2303_gpio_data *pl2303_gpio = platform_get_drvdata(pdev);
> +
> +	if (gpiochip_remove(&pl2303_gpio->gpio_chip))
> +		dev_err(&pdev->dev, "unable to remove gpio_chip?\n");
> +	platform_set_drvdata(pdev, NULL);
> +	kfree(pl2303_gpio);
> +	return 0;
> +}
> +
> +static struct platform_driver pl2303_gpio_driver = {
> +	.driver.name	= "pl2303-gpio",
> +	.driver.owner	= THIS_MODULE,
> +	.probe		= pl2303_gpio_probe,
> +	.remove		= pl2303_gpio_remove,
> +};
> +
> +static int __init pl2303_gpio_init(void)
> +{
> +	return platform_driver_register(&pl2303_gpio_driver);
> +}
> +module_init(pl2303_gpio_init);
> +
> +static void __exit pl2303_gpio_exit(void)
> +{
> +	platform_driver_unregister(&pl2303_gpio_driver);
> +}
> +module_exit(pl2303_gpio_exit);
> +
> +MODULE_AUTHOR("Wang YanQing <udknight@xxxxxxxxx>");
> +MODULE_DESCRIPTION("GPIO driver for PL2303");
> +MODULE_LICENSE("GPL");
> +MODULE_DESCRIPTION("platform:pl2303-gpio");
> diff --git a/drivers/usb/serial/pl2303.c b/drivers/usb/serial/pl2303.c
> index b3d5a35..1bb8928 100644
> --- a/drivers/usb/serial/pl2303.c
> +++ b/drivers/usb/serial/pl2303.c
> @@ -28,6 +28,7 @@
>  #include <linux/usb.h>
>  #include <linux/usb/serial.h>
>  #include <asm/unaligned.h>
> +#include <linux/platform_device.h>
>  #include "pl2303.h"
>  
>  
> @@ -146,6 +147,7 @@ struct pl2303_type_data {
>  struct pl2303_serial_private {
>  	const struct pl2303_type_data *type;
>  	unsigned long quirks;
> +	struct platform_device *pdev;
>  };
>  
>  struct pl2303_private {
> @@ -261,7 +263,22 @@ static int pl2303_startup(struct usb_serial *serial)
>  		pl2303_vendor_write(serial, 2, 0x44);
>  
>  	kfree(buf);
> +	if (type != TYPE_HX)
> +		return 0;
>  
> +	spriv->pdev = platform_device_alloc("pl2303-gpio", PLATFORM_DEVID_AUTO);

It's a USB device, not a "platform" device at all, so don't try to fit
it into the platform device model.

Just add the gpio code here to the pl2303.c driver, and hang the gpio
stuff off of the USB device, like it really is in the system, don't make
things up.

thanks,

greg k-h
--
To unsubscribe from this list: send the line "unsubscribe linux-usb" in
the body of a message to majordomo@xxxxxxxxxxxxxxx
More majordomo info at  http://vger.kernel.org/majordomo-info.html




[Index of Archives]     [Linux Media]     [Linux Input]     [Linux Audio Users]     [Yosemite News]     [Linux Kernel]     [Linux SCSI]     [Old Linux USB Devel Archive]

  Powered by Linux