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

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

 



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);
+	if (spriv->pdev == NULL) {
+		dev_err(&serial->interface->dev, "Failed to allocate %s\n", "pl2303-gpio");
+	} else {
+		spriv->pdev->dev.parent = &serial->interface->dev;
+		platform_set_drvdata(spriv->pdev, serial->dev);
+		platform_device_add(spriv->pdev);
+		if (platform_device_add(spriv->pdev) != 0) {
+			dev_err(&serial->interface->dev, "Failed to register %s\n", "pl2303-gpio");
+			platform_device_put(spriv->pdev);
+			spriv->pdev = NULL;
+		}
+	}
 	return 0;
 }
 
@@ -269,6 +286,8 @@ static void pl2303_release(struct usb_serial *serial)
 {
 	struct pl2303_serial_private *spriv = usb_get_serial_data(serial);
 
+	if (spriv && spriv->pdev)
+		platform_device_unregister(spriv->pdev);
 	kfree(spriv);
 }
 
-- 
1.8.5.5.dirty
--
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