aboutsummaryrefslogtreecommitdiffstats
path: root/drivers/usb
diff options
context:
space:
mode:
authorAjay Kumar Gupta <ajay.gupta@ti.com>2010-01-21 08:33:52 -0500
committerGreg Kroah-Hartman <gregkh@suse.de>2010-03-02 17:54:38 -0500
commit4f712e010b2da1cc01c178922f2eb5aaeae461b6 (patch)
tree4341a8c4f1e8184b0a727a9746be74267d20f68f /drivers/usb
parent8af6096caf8b3fb7ee33e636c44a29f373d27df5 (diff)
usb: musb: Add context save and restore support
Adding support for MUSB register save and restore during system suspend and resume. Changes: - Added musb_save/restore_context() functions - Added platform specific musb_platform_save/restore_context() to handle platform specific jobs. - Maintaining BlackFin compatibility by adding read/write functions for registers which are not available in BlackFin Tested system suspend and resume on OMAP3EVM board. Signed-off-by: Anand Gadiyar <gadiyar@ti.com> Signed-off-by: Ajay Kumar Gupta <ajay.gupta@ti.com> Signed-off-by: Felipe Balbi <felipe.balbi@nokia.com> Signed-off-by: Greg Kroah-Hartman <gregkh@suse.de>
Diffstat (limited to 'drivers/usb')
-rw-r--r--drivers/usb/musb/musb_core.c146
-rw-r--r--drivers/usb/musb/musb_core.h39
-rw-r--r--drivers/usb/musb/musb_regs.h90
-rw-r--r--drivers/usb/musb/omap2430.c16
4 files changed, 291 insertions, 0 deletions
diff --git a/drivers/usb/musb/musb_core.c b/drivers/usb/musb/musb_core.c
index 074d380bf883..2c53da771318 100644
--- a/drivers/usb/musb/musb_core.c
+++ b/drivers/usb/musb/musb_core.c
@@ -2167,6 +2167,148 @@ static int __exit musb_remove(struct platform_device *pdev)
2167 2167
2168#ifdef CONFIG_PM 2168#ifdef CONFIG_PM
2169 2169
2170static struct musb_context_registers musb_context;
2171
2172void musb_save_context(struct musb *musb)
2173{
2174 int i;
2175 void __iomem *musb_base = musb->mregs;
2176
2177 if (is_host_enabled(musb)) {
2178 musb_context.frame = musb_readw(musb_base, MUSB_FRAME);
2179 musb_context.testmode = musb_readb(musb_base, MUSB_TESTMODE);
2180 }
2181 musb_context.power = musb_readb(musb_base, MUSB_POWER);
2182 musb_context.intrtxe = musb_readw(musb_base, MUSB_INTRTXE);
2183 musb_context.intrrxe = musb_readw(musb_base, MUSB_INTRRXE);
2184 musb_context.intrusbe = musb_readb(musb_base, MUSB_INTRUSBE);
2185 musb_context.index = musb_readb(musb_base, MUSB_INDEX);
2186 musb_context.devctl = musb_readb(musb_base, MUSB_DEVCTL);
2187
2188 for (i = 0; i < MUSB_C_NUM_EPS; ++i) {
2189 musb_writeb(musb_base, MUSB_INDEX, i);
2190 musb_context.index_regs[i].txmaxp =
2191 musb_readw(musb_base, 0x10 + MUSB_TXMAXP);
2192 musb_context.index_regs[i].txcsr =
2193 musb_readw(musb_base, 0x10 + MUSB_TXCSR);
2194 musb_context.index_regs[i].rxmaxp =
2195 musb_readw(musb_base, 0x10 + MUSB_RXMAXP);
2196 musb_context.index_regs[i].rxcsr =
2197 musb_readw(musb_base, 0x10 + MUSB_RXCSR);
2198
2199 if (musb->dyn_fifo) {
2200 musb_context.index_regs[i].txfifoadd =
2201 musb_read_txfifoadd(musb_base);
2202 musb_context.index_regs[i].rxfifoadd =
2203 musb_read_rxfifoadd(musb_base);
2204 musb_context.index_regs[i].txfifosz =
2205 musb_read_txfifosz(musb_base);
2206 musb_context.index_regs[i].rxfifosz =
2207 musb_read_rxfifosz(musb_base);
2208 }
2209 if (is_host_enabled(musb)) {
2210 musb_context.index_regs[i].txtype =
2211 musb_readb(musb_base, 0x10 + MUSB_TXTYPE);
2212 musb_context.index_regs[i].txinterval =
2213 musb_readb(musb_base, 0x10 + MUSB_TXINTERVAL);
2214 musb_context.index_regs[i].rxtype =
2215 musb_readb(musb_base, 0x10 + MUSB_RXTYPE);
2216 musb_context.index_regs[i].rxinterval =
2217 musb_readb(musb_base, 0x10 + MUSB_RXINTERVAL);
2218
2219 musb_context.index_regs[i].txfunaddr =
2220 musb_read_txfunaddr(musb_base, i);
2221 musb_context.index_regs[i].txhubaddr =
2222 musb_read_txhubaddr(musb_base, i);
2223 musb_context.index_regs[i].txhubport =
2224 musb_read_txhubport(musb_base, i);
2225
2226 musb_context.index_regs[i].rxfunaddr =
2227 musb_read_rxfunaddr(musb_base, i);
2228 musb_context.index_regs[i].rxhubaddr =
2229 musb_read_rxhubaddr(musb_base, i);
2230 musb_context.index_regs[i].rxhubport =
2231 musb_read_rxhubport(musb_base, i);
2232 }
2233 }
2234
2235 musb_writeb(musb_base, MUSB_INDEX, musb_context.index);
2236
2237 musb_platform_save_context(&musb_context);
2238}
2239
2240void musb_restore_context(struct musb *musb)
2241{
2242 int i;
2243 void __iomem *musb_base = musb->mregs;
2244 void __iomem *ep_target_regs;
2245
2246 musb_platform_restore_context(&musb_context);
2247
2248 if (is_host_enabled(musb)) {
2249 musb_writew(musb_base, MUSB_FRAME, musb_context.frame);
2250 musb_writeb(musb_base, MUSB_TESTMODE, musb_context.testmode);
2251 }
2252 musb_writeb(musb_base, MUSB_POWER, musb_context.power);
2253 musb_writew(musb_base, MUSB_INTRTXE, musb_context.intrtxe);
2254 musb_writew(musb_base, MUSB_INTRRXE, musb_context.intrrxe);
2255 musb_writeb(musb_base, MUSB_INTRUSBE, musb_context.intrusbe);
2256 musb_writeb(musb_base, MUSB_DEVCTL, musb_context.devctl);
2257
2258 for (i = 0; i < MUSB_C_NUM_EPS; ++i) {
2259 musb_writeb(musb_base, MUSB_INDEX, i);
2260 musb_writew(musb_base, 0x10 + MUSB_TXMAXP,
2261 musb_context.index_regs[i].txmaxp);
2262 musb_writew(musb_base, 0x10 + MUSB_TXCSR,
2263 musb_context.index_regs[i].txcsr);
2264 musb_writew(musb_base, 0x10 + MUSB_RXMAXP,
2265 musb_context.index_regs[i].rxmaxp);
2266 musb_writew(musb_base, 0x10 + MUSB_RXCSR,
2267 musb_context.index_regs[i].rxcsr);
2268
2269 if (musb->dyn_fifo) {
2270 musb_write_txfifosz(musb_base,
2271 musb_context.index_regs[i].txfifosz);
2272 musb_write_rxfifosz(musb_base,
2273 musb_context.index_regs[i].rxfifosz);
2274 musb_write_txfifoadd(musb_base,
2275 musb_context.index_regs[i].txfifoadd);
2276 musb_write_rxfifoadd(musb_base,
2277 musb_context.index_regs[i].rxfifoadd);
2278 }
2279
2280 if (is_host_enabled(musb)) {
2281 musb_writeb(musb_base, 0x10 + MUSB_TXTYPE,
2282 musb_context.index_regs[i].txtype);
2283 musb_writeb(musb_base, 0x10 + MUSB_TXINTERVAL,
2284 musb_context.index_regs[i].txinterval);
2285 musb_writeb(musb_base, 0x10 + MUSB_RXTYPE,
2286 musb_context.index_regs[i].rxtype);
2287 musb_writeb(musb_base, 0x10 + MUSB_RXINTERVAL,
2288
2289 musb_context.index_regs[i].rxinterval);
2290 musb_write_txfunaddr(musb_base, i,
2291 musb_context.index_regs[i].txfunaddr);
2292 musb_write_txhubaddr(musb_base, i,
2293 musb_context.index_regs[i].txhubaddr);
2294 musb_write_txhubport(musb_base, i,
2295 musb_context.index_regs[i].txhubport);
2296
2297 ep_target_regs =
2298 musb_read_target_reg_base(i, musb_base);
2299
2300 musb_write_rxfunaddr(ep_target_regs,
2301 musb_context.index_regs[i].rxfunaddr);
2302 musb_write_rxhubaddr(ep_target_regs,
2303 musb_context.index_regs[i].rxhubaddr);
2304 musb_write_rxhubport(ep_target_regs,
2305 musb_context.index_regs[i].rxhubport);
2306 }
2307 }
2308
2309 musb_writeb(musb_base, MUSB_INDEX, musb_context.index);
2310}
2311
2170static int musb_suspend(struct device *dev) 2312static int musb_suspend(struct device *dev)
2171{ 2313{
2172 struct platform_device *pdev = to_platform_device(dev); 2314 struct platform_device *pdev = to_platform_device(dev);
@@ -2188,6 +2330,8 @@ static int musb_suspend(struct device *dev)
2188 */ 2330 */
2189 } 2331 }
2190 2332
2333 musb_save_context(musb);
2334
2191 if (musb->set_clock) 2335 if (musb->set_clock)
2192 musb->set_clock(musb->clock, 0); 2336 musb->set_clock(musb->clock, 0);
2193 else 2337 else
@@ -2209,6 +2353,8 @@ static int musb_resume_noirq(struct device *dev)
2209 else 2353 else
2210 clk_enable(musb->clock); 2354 clk_enable(musb->clock);
2211 2355
2356 musb_restore_context(musb);
2357
2212 /* for static cmos like DaVinci, register values were preserved 2358 /* for static cmos like DaVinci, register values were preserved
2213 * unless for some reason the whole soc powered down or the USB 2359 * unless for some reason the whole soc powered down or the USB
2214 * module got reset through the PSC (vs just being disabled). 2360 * module got reset through the PSC (vs just being disabled).
diff --git a/drivers/usb/musb/musb_core.h b/drivers/usb/musb/musb_core.h
index eaa01140183e..3d66c3e01dbd 100644
--- a/drivers/usb/musb/musb_core.h
+++ b/drivers/usb/musb/musb_core.h
@@ -454,6 +454,45 @@ struct musb {
454#endif 454#endif
455}; 455};
456 456
457#ifdef CONFIG_PM
458struct musb_csr_regs {
459 /* FIFO registers */
460 u16 txmaxp, txcsr, rxmaxp, rxcsr;
461 u16 rxfifoadd, txfifoadd;
462 u8 txtype, txinterval, rxtype, rxinterval;
463 u8 rxfifosz, txfifosz;
464 u8 txfunaddr, txhubaddr, txhubport;
465 u8 rxfunaddr, rxhubaddr, rxhubport;
466};
467
468struct musb_context_registers {
469
470#if defined(CONFIG_ARCH_OMAP34XX) || defined(CONFIG_ARCH_OMAP2430)
471 u32 otg_sysconfig, otg_forcestandby;
472#endif
473 u8 power;
474 u16 intrtxe, intrrxe;
475 u8 intrusbe;
476 u16 frame;
477 u8 index, testmode;
478
479 u8 devctl, misc;
480
481 struct musb_csr_regs index_regs[MUSB_C_NUM_EPS];
482};
483
484#if defined(CONFIG_ARCH_OMAP34XX) || defined(CONFIG_ARCH_OMAP2430)
485extern void musb_platform_save_context(struct musb_context_registers
486 *musb_context);
487extern void musb_platform_restore_context(struct musb_context_registers
488 *musb_context);
489#else
490#define musb_platform_save_context(x) do {} while (0)
491#define musb_platform_restore_context(x) do {} while (0)
492#endif
493
494#endif
495
457static inline void musb_set_vbus(struct musb *musb, int is_on) 496static inline void musb_set_vbus(struct musb *musb, int is_on)
458{ 497{
459 musb->board_set_vbus(musb, is_on); 498 musb->board_set_vbus(musb, is_on);
diff --git a/drivers/usb/musb/musb_regs.h b/drivers/usb/musb/musb_regs.h
index 9a8621ac5ac2..895fb057e443 100644
--- a/drivers/usb/musb/musb_regs.h
+++ b/drivers/usb/musb/musb_regs.h
@@ -326,6 +326,26 @@ static inline void musb_write_rxfifoadd(void __iomem *mbase, u16 c_off)
326 musb_writew(mbase, MUSB_RXFIFOADD, c_off); 326 musb_writew(mbase, MUSB_RXFIFOADD, c_off);
327} 327}
328 328
329static inline u8 musb_read_txfifosz(void __iomem *mbase)
330{
331 return musb_readb(mbase, MUSB_TXFIFOSZ);
332}
333
334static inline u16 musb_read_txfifoadd(void __iomem *mbase)
335{
336 return musb_readw(mbase, MUSB_TXFIFOADD);
337}
338
339static inline u8 musb_read_rxfifosz(void __iomem *mbase)
340{
341 return musb_readb(mbase, MUSB_RXFIFOSZ);
342}
343
344static inline u16 musb_read_rxfifoadd(void __iomem *mbase)
345{
346 return musb_readw(mbase, MUSB_RXFIFOADD);
347}
348
329static inline u8 musb_read_configdata(void __iomem *mbase) 349static inline u8 musb_read_configdata(void __iomem *mbase)
330{ 350{
331 musb_writeb(mbase, MUSB_INDEX, 0); 351 musb_writeb(mbase, MUSB_INDEX, 0);
@@ -381,6 +401,36 @@ static inline void musb_write_txhubport(void __iomem *mbase, u8 epnum,
381 qh_h_port_reg); 401 qh_h_port_reg);
382} 402}
383 403
404static inline u8 musb_read_rxfunaddr(void __iomem *mbase, u8 epnum)
405{
406 return musb_readb(mbase, MUSB_BUSCTL_OFFSET(epnum, MUSB_RXFUNCADDR));
407}
408
409static inline u8 musb_read_rxhubaddr(void __iomem *mbase, u8 epnum)
410{
411 return musb_readb(mbase, MUSB_BUSCTL_OFFSET(epnum, MUSB_RXHUBADDR));
412}
413
414static inline u8 musb_read_rxhubport(void __iomem *mbase, u8 epnum)
415{
416 return musb_readb(mbase, MUSB_BUSCTL_OFFSET(epnum, MUSB_RXHUBPORT));
417}
418
419static inline u8 musb_read_txfunaddr(void __iomem *mbase, u8 epnum)
420{
421 return musb_readb(mbase, MUSB_BUSCTL_OFFSET(epnum, MUSB_TXFUNCADDR));
422}
423
424static inline u8 musb_read_txhubaddr(void __iomem *mbase, u8 epnum)
425{
426 return musb_readb(mbase, MUSB_BUSCTL_OFFSET(epnum, MUSB_TXHUBADDR));
427}
428
429static inline u8 musb_read_txhubport(void __iomem *mbase, u8 epnum)
430{
431 return musb_readb(mbase, MUSB_BUSCTL_OFFSET(epnum, MUSB_TXHUBPORT));
432}
433
384#else /* CONFIG_BLACKFIN */ 434#else /* CONFIG_BLACKFIN */
385 435
386#define USB_BASE USB_FADDR 436#define USB_BASE USB_FADDR
@@ -460,6 +510,22 @@ static inline void musb_write_rxfifoadd(void __iomem *mbase, u16 c_off)
460{ 510{
461} 511}
462 512
513static inline u8 musb_read_txfifosz(void __iomem *mbase)
514{
515}
516
517static inline u16 musb_read_txfifoadd(void __iomem *mbase)
518{
519}
520
521static inline u8 musb_read_rxfifosz(void __iomem *mbase)
522{
523}
524
525static inline u16 musb_read_rxfifoadd(void __iomem *mbase)
526{
527}
528
463static inline u8 musb_read_configdata(void __iomem *mbase) 529static inline u8 musb_read_configdata(void __iomem *mbase)
464{ 530{
465 return 0; 531 return 0;
@@ -505,6 +571,30 @@ static inline void musb_write_txhubport(void __iomem *mbase, u8 epnum,
505{ 571{
506} 572}
507 573
574static inline u8 musb_read_rxfunaddr(void __iomem *mbase, u8 epnum)
575{
576}
577
578static inline u8 musb_read_rxhubaddr(void __iomem *mbase, u8 epnum)
579{
580}
581
582static inline u8 musb_read_rxhubport(void __iomem *mbase, u8 epnum)
583{
584}
585
586static inline u8 musb_read_txfunaddr(void __iomem *mbase, u8 epnum)
587{
588}
589
590static inline u8 musb_read_txhubaddr(void __iomem *mbase, u8 epnum)
591{
592}
593
594static inline void musb_read_txhubport(void __iomem *mbase, u8 epnum)
595{
596}
597
508#endif /* CONFIG_BLACKFIN */ 598#endif /* CONFIG_BLACKFIN */
509 599
510#endif /* __MUSB_REGS_H__ */ 600#endif /* __MUSB_REGS_H__ */
diff --git a/drivers/usb/musb/omap2430.c b/drivers/usb/musb/omap2430.c
index 83beeac5e7bf..15a3f27b5747 100644
--- a/drivers/usb/musb/omap2430.c
+++ b/drivers/usb/musb/omap2430.c
@@ -255,6 +255,22 @@ int __init musb_platform_init(struct musb *musb)
255 return 0; 255 return 0;
256} 256}
257 257
258#ifdef CONFIG_PM
259void musb_platform_save_context(struct musb_context_registers
260 *musb_context)
261{
262 musb_context->otg_sysconfig = omap_readl(OTG_SYSCONFIG);
263 musb_context->otg_forcestandby = omap_readl(OTG_FORCESTDBY);
264}
265
266void musb_platform_restore_context(struct musb_context_registers
267 *musb_context)
268{
269 omap_writel(musb_context->otg_sysconfig, OTG_SYSCONFIG);
270 omap_writel(musb_context->otg_forcestandby, OTG_FORCESTDBY);
271}
272#endif
273
258int musb_platform_suspend(struct musb *musb) 274int musb_platform_suspend(struct musb *musb)
259{ 275{
260 u32 l; 276 u32 l;