The musb struct is currently allocated along with the hcd, which makes
it difficult to build a driver that only acts as gadget device.

Fix this by allocation musb directly, and keep the hcd around as pointer.

struct hc_driver musb_hc_driver can now also be static to musb_host.c,
and the macro musb_to_hcd() is just a pointer dereferencer for now, and
will be elminiated later.

Signed-off-by: Daniel Mack <zon...@gmail.com>
---
 drivers/usb/musb/musb_core.c   | 60 ++++++++++++++++++--------------------
 drivers/usb/musb/musb_core.h   |  1 +
 drivers/usb/musb/musb_gadget.c | 10 -------
 drivers/usb/musb/musb_host.c   | 65 ++++++++++++++++++++++++++++++++++++++++--
 drivers/usb/musb/musb_host.h   | 21 +++++++-------
 5 files changed, 102 insertions(+), 55 deletions(-)

diff --git a/drivers/usb/musb/musb_core.c b/drivers/usb/musb/musb_core.c
index 37a261a..d5e9794 100644
--- a/drivers/usb/musb/musb_core.c
+++ b/drivers/usb/musb/musb_core.c
@@ -404,7 +404,8 @@ void musb_hnp_stop(struct musb *musb)
                break;
        case OTG_STATE_B_HOST:
                dev_dbg(musb->controller, "HNP: Disabling HR\n");
-               hcd->self.is_b_host = 0;
+               if (hcd)
+                       hcd->self.is_b_host = 0;
                musb->xceiv->state = OTG_STATE_B_PERIPHERAL;
                MUSB_DEV_MODE(musb);
                reg = musb_readb(mbase, MUSB_POWER);
@@ -484,7 +485,7 @@ static irqreturn_t musb_stage0_irq(struct musb *musb, u8 
int_usb,
 
                                musb->xceiv->state = OTG_STATE_A_HOST;
                                musb->is_active = 1;
-                               usb_hcd_resume_root_hub(musb_to_hcd(musb));
+                               musb_host_resume_root_hub(musb);
                                break;
                        case OTG_STATE_B_WAIT_ACON:
                                musb->xceiv->state = OTG_STATE_B_PERIPHERAL;
@@ -501,7 +502,7 @@ static irqreturn_t musb_stage0_irq(struct musb *musb, u8 
int_usb,
                        case OTG_STATE_A_SUSPEND:
                                /* possibly DISCONNECT is upcoming */
                                musb->xceiv->state = OTG_STATE_A_HOST;
-                               usb_hcd_resume_root_hub(musb_to_hcd(musb));
+                               musb_host_resume_root_hub(musb);
                                break;
                        case OTG_STATE_B_WAIT_ACON:
                        case OTG_STATE_B_PERIPHERAL:
@@ -643,7 +644,7 @@ static irqreturn_t musb_stage0_irq(struct musb *musb, u8 
int_usb,
                         * undesired detour through A_WAIT_BCON.
                         */
                        musb_hnp_stop(musb);
-                       usb_hcd_resume_root_hub(musb_to_hcd(musb));
+                       musb_host_resume_root_hub(musb);
                        musb_root_disconnect(musb);
                        musb_platform_try_idle(musb, jiffies
                                        + msecs_to_jiffies(musb->a_wait_bcon
@@ -726,7 +727,8 @@ static irqreturn_t musb_stage0_irq(struct musb *musb, u8 
int_usb,
                        dev_dbg(musb->controller, "HNP: CONNECT, now b_host\n");
 b_host:
                        musb->xceiv->state = OTG_STATE_B_HOST;
-                       hcd->self.is_b_host = 1;
+                       if (hcd)
+                               hcd->self.is_b_host = 1;
                        musb->ignore_disconnect = 0;
                        del_timer(&musb->otg_timer);
                        break;
@@ -734,17 +736,13 @@ b_host:
                        if ((devctl & MUSB_DEVCTL_VBUS)
                                        == (3 << MUSB_DEVCTL_VBUS_SHIFT)) {
                                musb->xceiv->state = OTG_STATE_A_HOST;
-                               hcd->self.is_b_host = 0;
+                               if (hcd)
+                                       hcd->self.is_b_host = 0;
                        }
                        break;
                }
 
-               /* poke the root hub */
-               MUSB_HST_MODE(musb);
-               if (hcd->status_urb)
-                       usb_hcd_poll_rh_status(hcd);
-               else
-                       usb_hcd_resume_root_hub(hcd);
+               musb_host_poke_root_hub(musb);
 
                dev_dbg(musb->controller, "CONNECT (%s) devctl %02x\n",
                                usb_otg_state_string(musb->xceiv->state), 
devctl);
@@ -759,7 +757,7 @@ b_host:
                switch (musb->xceiv->state) {
                case OTG_STATE_A_HOST:
                case OTG_STATE_A_SUSPEND:
-                       usb_hcd_resume_root_hub(musb_to_hcd(musb));
+                       musb_host_resume_root_hub(musb);
                        musb_root_disconnect(musb);
                        if (musb->a_wait_bcon != 0)
                                musb_platform_try_idle(musb, jiffies
@@ -772,7 +770,8 @@ b_host:
                         * in hnp_stop() is currently not used...
                         */
                        musb_root_disconnect(musb);
-                       musb_to_hcd(musb)->self.is_b_host = 0;
+                       if (musb->hcd)
+                               musb->hcd->self.is_b_host = 0;
                        musb->xceiv->state = OTG_STATE_B_PERIPHERAL;
                        MUSB_DEV_MODE(musb);
                        musb_g_disconnect(musb);
@@ -1763,24 +1762,18 @@ static struct musb *allocate_instance(struct device 
*dev,
        struct musb             *musb;
        struct musb_hw_ep       *ep;
        int                     epnum;
-       struct usb_hcd  *hcd;
+       int                     ret;
 
-       hcd = usb_create_hcd(&musb_hc_driver, dev, dev_name(dev));
-       if (!hcd)
+       musb = kzalloc(sizeof(*musb), GFP_KERNEL);
+       if (!musb)
                return NULL;
-       /* usbcore sets dev->driver_data to hcd, and sometimes uses that... */
 
-       musb = hcd_to_musb(hcd);
        INIT_LIST_HEAD(&musb->control);
        INIT_LIST_HEAD(&musb->in_bulk);
        INIT_LIST_HEAD(&musb->out_bulk);
 
-       hcd->uses_new_polling = 1;
-       hcd->has_tt = 1;
-
        musb->vbuserr_retry = VBUSERR_RETRY_COUNT;
        musb->a_wait_bcon = OTG_TIME_A_WAIT_BCON;
-       dev_set_drvdata(dev, musb);
        musb->mregs = mbase;
        musb->ctrl_base = mbase;
        musb->nIrq = -ENODEV;
@@ -1795,7 +1788,17 @@ static struct musb *allocate_instance(struct device *dev,
 
        musb->controller = dev;
 
+       ret = musb_host_alloc(musb);
+       if (ret < 0)
+               goto err_free;
+
+       dev_set_drvdata(dev, musb);
+
        return musb;
+
+err_free:
+       kfree(musb);
+       return NULL;
 }
 
 static void musb_free(struct musb *musb)
@@ -1821,7 +1824,8 @@ static void musb_free(struct musb *musb)
                dma_controller_destroy(c);
        }
 
-       usb_put_hcd(musb_to_hcd(musb));
+       musb_host_free(musb);
+       kfree(musb);
 }
 
 /*
@@ -1838,7 +1842,6 @@ musb_init_controller(struct device *dev, int nIrq, void 
__iomem *ctrl)
        int                     status;
        struct musb             *musb;
        struct musb_hdrc_platform_data *plat = dev->platform_data;
-       struct usb_hcd          *hcd;
 
        /* The driver might handle more features than the board; OK.
         * Fail when the board needs a feature that's not enabled.
@@ -1939,13 +1942,6 @@ musb_init_controller(struct device *dev, int nIrq, void 
__iomem *ctrl)
                musb->irq_wake = 0;
        }
 
-       /* host side needs more setup */
-       hcd = musb_to_hcd(musb);
-       otg_set_host(musb->xceiv->otg, &hcd->self);
-       hcd->self.otg_port = 1;
-       musb->xceiv->otg->host = &hcd->self;
-       hcd->power_budget = 2 * (plat->power ? : 250);
-
        /* program PHY to use external vBus if required */
        if (plat->extvbus) {
                u8 busctl = musb_read_ulpi_buscontrol(musb->mregs);
diff --git a/drivers/usb/musb/musb_core.h b/drivers/usb/musb/musb_core.h
index 04d8974..ef5b4e6 100644
--- a/drivers/usb/musb/musb_core.h
+++ b/drivers/usb/musb/musb_core.h
@@ -402,6 +402,7 @@ struct musb {
        enum musb_g_ep0_state   ep0_state;
        struct usb_gadget       g;                      /* the gadget */
        struct usb_gadget_driver *gadget_driver;        /* its driver */
+       struct usb_hcd          *hcd;                   /* the usb hcd */
 
        /*
         * FIXME: Remove this flag.
diff --git a/drivers/usb/musb/musb_gadget.c b/drivers/usb/musb/musb_gadget.c
index ba70923..0414bc1 100644
--- a/drivers/usb/musb/musb_gadget.c
+++ b/drivers/usb/musb/musb_gadget.c
@@ -1820,7 +1820,6 @@ static int musb_gadget_start(struct usb_gadget *g,
 {
        struct musb             *musb = gadget_to_musb(g);
        struct usb_otg          *otg = musb->xceiv->otg;
-       struct usb_hcd          *hcd = musb_to_hcd(musb);
        unsigned long           flags;
        int                     retval = 0;
 
@@ -1847,17 +1846,9 @@ static int musb_gadget_start(struct usb_gadget *g,
         * handles power budgeting ... this way also
         * ensures HdrcStart is indirectly called.
         */
-       retval = usb_add_hcd(hcd, 0, 0);
-       if (retval < 0) {
-               dev_dbg(musb->controller, "add_hcd failed, %d\n", retval);
-               goto err;
-       }
-
        if (musb->xceiv->last_event == USB_EVENT_ID)
                musb_platform_set_vbus(musb, 1);
 
-       hcd->self.uses_pio_for_control = 1;
-
        if (musb->xceiv->last_event == USB_EVENT_NONE)
                pm_runtime_put(musb->controller);
 
@@ -1942,7 +1933,6 @@ static int musb_gadget_stop(struct usb_gadget *g,
        musb_platform_try_idle(musb, 0);
        spin_unlock_irqrestore(&musb->lock, flags);
 
-       usb_remove_hcd(musb_to_hcd(musb));
        /*
         * FIXME we need to be able to register another
         * gadget driver here and have everything work;
diff --git a/drivers/usb/musb/musb_host.c b/drivers/usb/musb/musb_host.c
index 8914dec..d564580 100644
--- a/drivers/usb/musb/musb_host.c
+++ b/drivers/usb/musb/musb_host.c
@@ -96,6 +96,16 @@
  * of transfers between endpoints, or anything clever.
  */
 
+struct musb_hcd_link {
+       struct musb *musb;
+};
+
+struct musb *hcd_to_musb(struct usb_hcd *hcd)
+{
+       struct musb_hcd_link *link = (struct musb_hcd_link *) hcd->hcd_priv;
+       return link->musb;
+}
+
 
 static void musb_ep_program(struct musb *musb, u8 epnum,
                        struct urb *urb, int is_out,
@@ -2465,7 +2475,6 @@ static int musb_bus_resume(struct usb_hcd *hcd)
        return 0;
 }
 
-
 #ifndef CONFIG_MUSB_PIO_ONLY
 
 #define MUSB_USB_DMA_ALIGN 4
@@ -2577,10 +2586,10 @@ static void musb_unmap_urb_for_dma(struct usb_hcd *hcd, 
struct urb *urb)
 }
 #endif /* !CONFIG_MUSB_PIO_ONLY */
 
-const struct hc_driver musb_hc_driver = {
+static const struct hc_driver musb_hc_driver = {
        .description            = "musb-hcd",
        .product_desc           = "MUSB HDRC host driver",
-       .hcd_priv_size          = sizeof(struct musb),
+       .hcd_priv_size          = sizeof(struct musb_hcd_link),
        .flags                  = HCD_USB2 | HCD_MEMORY,
 
        /* not using irq handler or reset hooks from usbcore, since
@@ -2608,3 +2617,53 @@ const struct hc_driver musb_hc_driver = {
        /* .start_port_reset    = NULL, */
        /* .hub_irq_enable      = NULL, */
 };
+
+int musb_host_alloc(struct musb *musb)
+{
+       struct device           *dev = musb->controller;
+       struct musb_hcd_link    *link;
+
+       /* usbcore sets dev->driver_data to hcd, and sometimes uses that... */
+       musb->hcd = usb_create_hcd(&musb_hc_driver, dev, dev_name(dev));
+       if (!musb->hcd)
+               return -EINVAL;
+
+       link = (struct musb_hcd_link *) musb->hcd->hcd_priv;
+       link->musb = musb;
+
+       musb->hcd->self.uses_pio_for_control = 1;
+       musb->hcd->uses_new_polling = 1;
+       musb->hcd->has_tt = 1;
+
+       return 0;
+}
+
+void musb_host_cleanup(struct musb *musb)
+{
+       usb_remove_hcd(musb->hcd);
+       musb->hcd = NULL;
+}
+
+void musb_host_free(struct musb *musb)
+{
+       usb_put_hcd(musb->hcd);
+}
+
+void musb_host_resume_root_hub(struct musb *musb)
+{
+       usb_hcd_resume_root_hub(musb->hcd);
+}
+
+void musb_host_poll_rh_status(struct musb *musb)
+{
+       usb_hcd_poll_rh_status(musb->hcd);
+}
+
+void musb_host_poke_root_hub(struct musb *musb)
+{
+       MUSB_HST_MODE(musb);
+       if (musb->hcd->status_urb)
+               usb_hcd_poll_rh_status(musb->hcd);
+       else
+               usb_hcd_resume_root_hub(musb->hcd);
+}
diff --git a/drivers/usb/musb/musb_host.h b/drivers/usb/musb/musb_host.h
index 9670269..fb24422 100644
--- a/drivers/usb/musb/musb_host.h
+++ b/drivers/usb/musb/musb_host.h
@@ -37,15 +37,9 @@
 
 #include <linux/scatterlist.h>
 
-static inline struct usb_hcd *musb_to_hcd(struct musb *musb)
-{
-       return container_of((void *) musb, struct usb_hcd, hcd_priv);
-}
+#define musb_to_hcd(MUSB) ((MUSB)->hcd)
 
-static inline struct musb *hcd_to_musb(struct usb_hcd *hcd)
-{
-       return (struct musb *) (hcd->hcd_priv);
-}
+extern struct musb *hcd_to_musb(struct usb_hcd *);
 
 /* stored in "usb_host_endpoint.hcpriv" for scheduled endpoints */
 struct musb_qh {
@@ -85,9 +79,18 @@ static inline struct musb_qh *first_qh(struct list_head *q)
 }
 
 extern irqreturn_t musb_h_ep0_irq(struct musb *);
+extern int musb_host_alloc(struct musb *);
 extern void musb_host_tx(struct musb *, u8);
 extern void musb_host_rx(struct musb *, u8);
 extern void musb_root_disconnect(struct musb *musb);
+extern void musb_host_free(struct musb *);
+extern void musb_host_cleanup(struct musb *);
+extern void musb_host_tx(struct musb *, u8);
+extern void musb_host_rx(struct musb *, u8);
+extern void musb_root_disconnect(struct musb *musb);
+extern void musb_host_resume_root_hub(struct musb *musb);
+extern void musb_host_poll_rh_status(struct musb *musb);
+extern void musb_host_poke_root_hub(struct musb *musb);
 
 struct usb_hcd;
 
@@ -96,8 +99,6 @@ extern int musb_hub_control(struct usb_hcd *hcd,
                        u16 typeReq, u16 wValue, u16 wIndex,
                        char *buf, u16 wLength);
 
-extern const struct hc_driver musb_hc_driver;
-
 static inline struct urb *next_urb(struct musb_qh *qh)
 {
        struct list_head        *queue;
-- 
1.8.1.4

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

Reply via email to