[PATCH] drivers: usb: host: max3421: device tree support for gpout

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

 



Adding device tree support for gpout feature of the max3421 SPI-USB controller.
Intention is to supply vbus-gpout and vbus-active-level via device tree, e.g:

&peripherals_axi_quad_spi_2 {
	#address-cells = <1>;
	#size-cells = <0>;
	status = "okay";
	usb@0 {
		compatible = "max3421-hcd";
		reg = <0>;
		spi-max-frequency = <25000000>;
		interrupt-parent = <&peripherals_microblaze_0_axi_intc>;
		interrupts = <10 0>;
		vbus-gpout = <1>;
		vbus-active-level = <1>;
	};
};


kernel source base is 4.7

---
 drivers/usb/host/max3421-hcd.c |   45 +++++++++++++++++++++++++++++++++-------
 1 file changed, 37 insertions(+), 8 deletions(-)

diff --git a/drivers/usb/host/max3421-hcd.c b/drivers/usb/host/max3421-hcd.c
index c369c29..dfb9df0 100644
--- a/drivers/usb/host/max3421-hcd.c
+++ b/drivers/usb/host/max3421-hcd.c
@@ -60,6 +60,7 @@
 #include <linux/spi/spi.h>
 #include <linux/usb.h>
 #include <linux/usb/hcd.h>
+#include <linux/of_device.h>
 
 #include <linux/platform_data/max3421-hcd.h>
 
@@ -157,6 +158,7 @@ struct max3421_hcd {
 	u8 mode;
 	u8 iopins[2];
 	unsigned long todo;
+	struct max3421_hcd_platform_data *pdata;
 #ifdef DEBUG
 	unsigned long err_stat[16];
 #endif
@@ -1684,22 +1686,19 @@ max3421_gpout_set_value(struct usb_hcd *hcd, u8 pin_number, u8 value)
 		max3421_hcd->iopins[idx] &= ~mask;
 	set_bit(IOPIN_UPDATE, &max3421_hcd->todo);
 	wake_up_process(max3421_hcd->spi_thread);
+
 }
 
 static int
 max3421_hub_control(struct usb_hcd *hcd, u16 type_req, u16 value, u16 index,
 		    char *buf, u16 length)
 {
-	struct spi_device *spi = to_spi_device(hcd->self.controller);
 	struct max3421_hcd *max3421_hcd = hcd_to_max3421(hcd);
-	struct max3421_hcd_platform_data *pdata;
 	unsigned long flags;
 	int retval = 0;
 
 	spin_lock_irqsave(&max3421_hcd->lock, flags);
 
-	pdata = spi->dev.platform_data;
-
 	switch (type_req) {
 	case ClearHubFeature:
 		break;
@@ -1709,8 +1708,8 @@ max3421_hub_control(struct usb_hcd *hcd, u16 type_req, u16 value, u16 index,
 			break;
 		case USB_PORT_FEAT_POWER:
 			dev_dbg(hcd->self.controller, "power-off\n");
-			max3421_gpout_set_value(hcd, pdata->vbus_gpout,
-						!pdata->vbus_active_level);
+			max3421_gpout_set_value(hcd, max3421_hcd->pdata->vbus_gpout,
+				!max3421_hcd->pdata->vbus_active_level);
 			/* FALLS THROUGH */
 		default:
 			max3421_hcd->port_status &= ~(1 << value);
@@ -1759,8 +1758,9 @@ max3421_hub_control(struct usb_hcd *hcd, u16 type_req, u16 value, u16 index,
 		case USB_PORT_FEAT_POWER:
 			dev_dbg(hcd->self.controller, "power-on\n");
 			max3421_hcd->port_status |= USB_PORT_STAT_POWER;
-			max3421_gpout_set_value(hcd, pdata->vbus_gpout,
-						pdata->vbus_active_level);
+			max3421_gpout_set_value(hcd, max3421_hcd->pdata->vbus_gpout,
+						max3421_hcd->pdata->vbus_active_level);
+
 			break;
 		case USB_PORT_FEAT_RESET:
 			max3421_reset_port(hcd);
@@ -1831,6 +1831,23 @@ static struct hc_driver max3421_hcd_desc = {
 	.bus_resume =		max3421_bus_resume,
 };
 
+#ifdef CONFIG_OF
+int max3421_init_dt(struct spi_device *spi, struct max3421_hcd *max3421_hcd)
+{
+	struct device_node *np = spi->dev.of_node;
+	u32 val;
+
+	if (!of_property_read_u32(np, "vbus-gpout", &val))
+		max3421_hcd->pdata->vbus_gpout = (u8) val;
+
+	if (!of_property_read_u32(np, "vbus-active-level", &val))
+		max3421_hcd->pdata->vbus_active_level = (u8) val;
+
+
+	return 0;
+}
+#endif
+
 static int
 max3421_probe(struct spi_device *spi)
 {
@@ -1866,6 +1883,17 @@ max3421_probe(struct spi_device *spi)
 		goto error;
 	}
 
+#ifdef CONFIG_OF
+	max3421_hcd->pdata = kmalloc(sizeof(*max3421_hcd->pdata), GFP_KERNEL);
+	if (!max3421_hcd->pdata) {
+		dev_err(&spi->dev, "failed to kmalloc pdata buffer\n");
+		goto error;
+	}
+	max3421_init_dt(spi, max3421_hcd);
+#else
+	max3421_hcd->pdata = spi->dev.platform_data;
+#endif
+
 	max3421_hcd->spi_thread = kthread_run(max3421_spi_thread, hcd,
 					      "max3421_spi_thread");
 	if (max3421_hcd->spi_thread == ERR_PTR(-ENOMEM)) {
@@ -1892,6 +1920,7 @@ error:
 	if (hcd) {
 		kfree(max3421_hcd->tx);
 		kfree(max3421_hcd->rx);
+		kfree(max3421_hcd->pdata);
 		if (max3421_hcd->spi_thread)
 			kthread_stop(max3421_hcd->spi_thread);
 		usb_put_hcd(hcd);
-- 
1.7.10.4

--
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