From a6db46c4a08a03b4344cff7b6176fe4d1142f0dc Mon Sep 17 00:00:00 2001 From: Tony Lindgren Date: Thu, 7 Sep 2006 16:10:52 +0300 Subject: [PATCH] MUSB: Workarounds for TUSB PLL Add workarounds for various chip issues relating to the TUSB chip PLL getting properly initialized after resume. Signed-off-by: Tony Lindgren Signed-off-by: David Brownell --- drivers/usb/musb/tusb6010.c | 39 ++++++++++++++++++++++++++++++++++++- 1 file changed, 38 insertions(+), 1 deletion(-) diff --git a/drivers/usb/musb/tusb6010.c b/drivers/usb/musb/tusb6010.c index 3a1dad2cdea..2176bcb3bdd 100644 --- a/drivers/usb/musb/tusb6010.c +++ b/drivers/usb/musb/tusb6010.c @@ -162,6 +162,24 @@ static inline void tusb_enable_vbus_charge(struct musb *musb) musb_writel(base, TUSB_PRCM_MNGMT, reg); } +/* workaround for issue 13: change clock during chip idle + * (to be fixed in rev3 silicon) ... symptoms include disconnect + * or looping suspend/resume cycles + */ +void tusb_set_clock_source(struct musb *musb, int mode) +{ + void __iomem *base = musb->ctrl_base; + u32 reg; + + reg = musb_readl(base, TUSB_PRCM_CONF); + reg &= ~TUSB_PRCM_CONF_SYS_CLKSEL(0x3); + + if (mode > 0) + reg |= TUSB_PRCM_CONF_SYS_CLKSEL(mode & 0x3); + + musb_writel(base, TUSB_PRCM_CONF, reg); +} + /* * Idle TUSB6010 until next wake-up event; NOR access always wakes. * Other code ensures that we idle unless we're connected _and_ the @@ -173,6 +191,8 @@ static inline void tusb_allow_idle(struct musb *musb, u32 wakeup_enables) void __iomem *base = musb->ctrl_base; u32 reg; + tusb_set_clock_source(musb, 0); + wakeup_enables |= TUSB_PRCM_WNORCS; musb_writel(base, TUSB_PRCM_WAKEUP_MASK, ~wakeup_enables); @@ -351,8 +371,25 @@ irqreturn_t tusb_interrupt(int irq, void *__hci, struct pt_regs *r) /* Acknowledge wake-up source interrupts */ if (int_src & TUSB_INT_SRC_DEV_WAKEUP) { - u32 reg = musb_readl(base, TUSB_PRCM_WAKEUP_SOURCE); + u32 reg; + u32 i; + + /* there are issues re-locking the PLL on wakeup ... */ + + /* work around issue 8 */ + for (i = 0xf7f7f7; i > 0xf7f7f7 - 1000; i--) { + musb_writel(base, TUSB_SCRATCH_PAD, 0); + musb_writel(base, TUSB_SCRATCH_PAD, i); + reg = musb_readl(base, TUSB_SCRATCH_PAD); + if (reg == i) + break; + DBG(1, "TUSB NOR not ready\n"); + } + + /* work around issue 13 (2nd half) */ + tusb_set_clock_source(musb, 1); + reg = musb_readl(base, TUSB_PRCM_WAKEUP_SOURCE); musb_writel(base, TUSB_PRCM_WAKEUP_CLEAR, reg); if (reg & ~TUSB_PRCM_WNORCS) { musb->is_active = 1; -- 2.41.1