]> asedeno.scripts.mit.edu Git - linux.git/commitdiff
usb: musb: davinci: Convert to use GPIO descriptor
authorLinus Walleij <linus.walleij@linaro.org>
Wed, 15 Jan 2020 13:25:46 +0000 (07:25 -0600)
committerGreg Kroah-Hartman <gregkh@linuxfoundation.org>
Wed, 15 Jan 2020 13:47:00 +0000 (14:47 +0100)
The DaVinci MUSB glue contains an optional GPIO line to
control VBUS power, convert this to use a GPIO descriptor
and augment the EVM board file to provide this descriptor.

I can't get this driver to compile properly and it depends
on broken but when I didn get it to compile brokenly, it
did at least not complain about THIS code being broken so
I don't think I broke the driver any more than what it
already is.

I did away with the ifdefs that do not work with
multiplatform anyway so the day someone decides to
resurrect the code, the path to get it working should be
easier as well since DaVinci is now multiplatform.

Cc: Sekhar Nori <nsekhar@ti.com>
Cc: Bartosz Golaszewski <bgolaszewski@baylibre.com>
Cc: Tony Lindgren <tony@atomide.com>
Signed-off-by: Linus Walleij <linus.walleij@linaro.org>
[b-liu@ti.com: fixed one instance still ref to global variable vbus_state]
Signed-off-by: Bin Liu <b-liu@ti.com>
Link: https://lore.kernel.org/r/20200115132547.364-25-b-liu@ti.com
Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
arch/arm/mach-davinci/board-dm644x-evm.c
drivers/usb/musb/davinci.c

index 9d87d4e440eaa4a7f552cd6c55ae71989c5a3ca7..040c949414faeec8591acb715307c8bff6e5a00f 100644 (file)
@@ -823,6 +823,17 @@ static int davinci_phy_fixup(struct phy_device *phydev)
 
 #define HAS_NAND       IS_ENABLED(CONFIG_MTD_NAND_DAVINCI)
 
+#define GPIO_nVBUS_DRV         160
+
+static struct gpiod_lookup_table dm644evm_usb_gpio_table = {
+       .dev_id = "musb-davinci",
+       .table = {
+               GPIO_LOOKUP("davinci_gpio", GPIO_nVBUS_DRV, NULL,
+                           GPIO_ACTIVE_HIGH),
+               { }
+       },
+};
+
 static __init void davinci_evm_init(void)
 {
        int ret;
@@ -875,6 +886,7 @@ static __init void davinci_evm_init(void)
        dm644x_init_asp();
 
        /* irlml6401 switches over 1A, in under 8 msec */
+       gpiod_add_lookup_table(&dm644evm_usb_gpio_table);
        davinci_setup_usb(1000, 8);
 
        if (IS_BUILTIN(CONFIG_PHYLIB)) {
index fb6bbd254ab738afba0829d33438cb4ab4ccd368..704435526394383d1d73b84cfc642e85e589f440 100644 (file)
@@ -13,7 +13,7 @@
 #include <linux/clk.h>
 #include <linux/err.h>
 #include <linux/io.h>
-#include <linux/gpio.h>
+#include <linux/gpio/consumer.h>
 #include <linux/platform_device.h>
 #include <linux/dma-mapping.h>
 #include <linux/usb/usb_phy_generic.h>
 
 #include "musb_core.h"
 
-#ifdef CONFIG_MACH_DAVINCI_EVM
-#define GPIO_nVBUS_DRV         160
-#endif
-
 #include "davinci.h"
 #include "cppi_dma.h"
 
@@ -40,6 +36,9 @@ struct davinci_glue {
        struct device           *dev;
        struct platform_device  *musb;
        struct clk              *clk;
+       bool                    vbus_state;
+       struct gpio_desc        *vbus;
+       struct work_struct      vbus_work;
 };
 
 /* REVISIT (PM) we should be able to keep the PHY in low power mode most
@@ -135,43 +134,44 @@ static void davinci_musb_disable(struct musb *musb)
  * when J10 is out, and TI documents it as handling OTG.
  */
 
-#ifdef CONFIG_MACH_DAVINCI_EVM
-
-static int vbus_state = -1;
-
 /* I2C operations are always synchronous, and require a task context.
  * With unloaded systems, using the shared workqueue seems to suffice
  * to satisfy the 100msec A_WAIT_VRISE timeout...
  */
-static void evm_deferred_drvvbus(struct work_struct *ignored)
+static void evm_deferred_drvvbus(struct work_struct *work)
 {
-       gpio_set_value_cansleep(GPIO_nVBUS_DRV, vbus_state);
-       vbus_state = !vbus_state;
-}
+       struct davinci_glue *glue = container_of(work, struct davinci_glue,
+                                                vbus_work);
 
-#endif /* EVM */
+       gpiod_set_value_cansleep(glue->vbus, glue->vbus_state);
+       glue->vbus_state = !glue->vbus_state;
+}
 
-static void davinci_musb_source_power(struct musb *musb, int is_on, int immediate)
+static void davinci_musb_source_power(struct musb *musb, int is_on,
+                                     int immediate)
 {
-#ifdef CONFIG_MACH_DAVINCI_EVM
+       struct davinci_glue *glue = dev_get_drvdata(musb->controller->parent);
+
+       /* This GPIO handling is entirely optional */
+       if (!glue->vbus)
+               return;
+
        if (is_on)
                is_on = 1;
 
-       if (vbus_state == is_on)
+       if (glue->vbus_state == is_on)
                return;
-       vbus_state = !is_on;            /* 0/1 vs "-1 == unknown/init" */
+       /* 0/1 vs "-1 == unknown/init" */
+       glue->vbus_state = !is_on;
 
        if (machine_is_davinci_evm()) {
-               static DECLARE_WORK(evm_vbus_work, evm_deferred_drvvbus);
-
                if (immediate)
-                       gpio_set_value_cansleep(GPIO_nVBUS_DRV, vbus_state);
+                       gpiod_set_value_cansleep(glue->vbus, glue->vbus_state);
                else
-                       schedule_work(&evm_vbus_work);
+                       schedule_work(&glue->vbus_work);
        }
        if (immediate)
-               vbus_state = is_on;
-#endif
+               glue->vbus_state = is_on;
 }
 
 static void davinci_musb_set_vbus(struct musb *musb, int is_on)
@@ -524,6 +524,15 @@ static int davinci_probe(struct platform_device *pdev)
 
        pdata->platform_ops             = &davinci_ops;
 
+       glue->vbus = devm_gpiod_get_optional(&pdev->dev, NULL, GPIOD_OUT_LOW);
+       if (IS_ERR(glue->vbus)) {
+               ret = PTR_ERR(glue->vbus);
+               goto err0;
+       } else {
+               glue->vbus_state = -1;
+               INIT_WORK(&glue->vbus_work, evm_deferred_drvvbus);
+       }
+
        usb_phy_generic_register();
        platform_set_drvdata(pdev, glue);