2 * EHCI HCD (Host Controller Driver) for USB.
4 * Bus Glue for AMD Alchemy Au1xxx
6 * Based on "ohci-au1xxx.c" by Matt Porter <mporter@kernel.crashing.org>
8 * Modified for AMD Alchemy Au1200 EHC
9 * by K.Boge <karsten.boge@amd.com>
11 * This file is licenced under the GPL.
14 #include <linux/platform_device.h>
15 #include <asm/mach-au1x00/au1000.h>
18 extern int usb_disabled(void);
20 static int au1xxx_ehci_setup(struct usb_hcd
*hcd
)
22 struct ehci_hcd
*ehci
= hcd_to_ehci(hcd
);
23 int ret
= ehci_init(hcd
);
25 ehci
->need_io_watchdog
= 0;
29 static const struct hc_driver ehci_au1xxx_hc_driver
= {
30 .description
= hcd_name
,
31 .product_desc
= "Au1xxx EHCI",
32 .hcd_priv_size
= sizeof(struct ehci_hcd
),
35 * generic hardware linkage
38 .flags
= HCD_MEMORY
| HCD_USB2
,
41 * basic lifecycle operations
43 * FIXME -- ehci_init() doesn't do enough here.
44 * See ehci-ppc-soc for a complete implementation.
46 .reset
= au1xxx_ehci_setup
,
49 .shutdown
= ehci_shutdown
,
52 * managing i/o requests and associated device resources
54 .urb_enqueue
= ehci_urb_enqueue
,
55 .urb_dequeue
= ehci_urb_dequeue
,
56 .endpoint_disable
= ehci_endpoint_disable
,
57 .endpoint_reset
= ehci_endpoint_reset
,
62 .get_frame_number
= ehci_get_frame
,
67 .hub_status_data
= ehci_hub_status_data
,
68 .hub_control
= ehci_hub_control
,
69 .bus_suspend
= ehci_bus_suspend
,
70 .bus_resume
= ehci_bus_resume
,
71 .relinquish_port
= ehci_relinquish_port
,
72 .port_handed_over
= ehci_port_handed_over
,
74 .clear_tt_buffer_complete
= ehci_clear_tt_buffer_complete
,
77 static int ehci_hcd_au1xxx_drv_probe(struct platform_device
*pdev
)
80 struct ehci_hcd
*ehci
;
87 if (pdev
->resource
[1].flags
!= IORESOURCE_IRQ
) {
88 pr_debug("resource[1] is not IORESOURCE_IRQ");
91 hcd
= usb_create_hcd(&ehci_au1xxx_hc_driver
, &pdev
->dev
, "Au1xxx");
95 res
= platform_get_resource(pdev
, IORESOURCE_MEM
, 0);
96 hcd
->rsrc_start
= res
->start
;
97 hcd
->rsrc_len
= resource_size(res
);
99 if (!request_mem_region(hcd
->rsrc_start
, hcd
->rsrc_len
, hcd_name
)) {
100 pr_debug("request_mem_region failed");
105 hcd
->regs
= ioremap(hcd
->rsrc_start
, hcd
->rsrc_len
);
107 pr_debug("ioremap failed");
112 if (alchemy_usb_control(ALCHEMY_USB_EHCI0
, 1)) {
113 printk(KERN_INFO
"%s: controller init failed!\n", pdev
->name
);
118 ehci
= hcd_to_ehci(hcd
);
119 ehci
->caps
= hcd
->regs
;
120 ehci
->regs
= hcd
->regs
+
121 HC_LENGTH(ehci
, readl(&ehci
->caps
->hc_capbase
));
122 /* cache this readonly data; minimize chip reads */
123 ehci
->hcs_params
= readl(&ehci
->caps
->hcs_params
);
125 ret
= usb_add_hcd(hcd
, pdev
->resource
[1].start
,
128 platform_set_drvdata(pdev
, hcd
);
132 alchemy_usb_control(ALCHEMY_USB_EHCI0
, 0);
136 release_mem_region(hcd
->rsrc_start
, hcd
->rsrc_len
);
142 static int ehci_hcd_au1xxx_drv_remove(struct platform_device
*pdev
)
144 struct usb_hcd
*hcd
= platform_get_drvdata(pdev
);
147 alchemy_usb_control(ALCHEMY_USB_EHCI0
, 0);
149 release_mem_region(hcd
->rsrc_start
, hcd
->rsrc_len
);
151 platform_set_drvdata(pdev
, NULL
);
157 static int ehci_hcd_au1xxx_drv_suspend(struct device
*dev
)
159 struct usb_hcd
*hcd
= dev_get_drvdata(dev
);
160 struct ehci_hcd
*ehci
= hcd_to_ehci(hcd
);
164 if (time_before(jiffies
, ehci
->next_statechange
))
167 /* Root hub was already suspended. Disable irq emission and
168 * mark HW unaccessible. The PM and USB cores make sure that
169 * the root hub is either suspended or stopped.
171 ehci_prepare_ports_for_controller_suspend(ehci
, device_may_wakeup(dev
));
172 spin_lock_irqsave(&ehci
->lock
, flags
);
173 ehci_writel(ehci
, 0, &ehci
->regs
->intr_enable
);
174 (void)ehci_readl(ehci
, &ehci
->regs
->intr_enable
);
176 clear_bit(HCD_FLAG_HW_ACCESSIBLE
, &hcd
->flags
);
177 spin_unlock_irqrestore(&ehci
->lock
, flags
);
179 // could save FLADJ in case of Vaux power loss
180 // ... we'd only use it to handle clock skew
182 alchemy_usb_control(ALCHEMY_USB_EHCI0
, 0);
187 static int ehci_hcd_au1xxx_drv_resume(struct device
*dev
)
189 struct usb_hcd
*hcd
= dev_get_drvdata(dev
);
190 struct ehci_hcd
*ehci
= hcd_to_ehci(hcd
);
192 alchemy_usb_control(ALCHEMY_USB_EHCI0
, 1);
194 // maybe restore FLADJ
196 if (time_before(jiffies
, ehci
->next_statechange
))
199 /* Mark hardware accessible again as we are out of D3 state by now */
200 set_bit(HCD_FLAG_HW_ACCESSIBLE
, &hcd
->flags
);
202 /* If CF is still set, we maintained PCI Vaux power.
203 * Just undo the effect of ehci_pci_suspend().
205 if (ehci_readl(ehci
, &ehci
->regs
->configured_flag
) == FLAG_CF
) {
206 int mask
= INTR_MASK
;
208 ehci_prepare_ports_for_controller_resume(ehci
);
209 if (!hcd
->self
.root_hub
->do_remote_wakeup
)
211 ehci_writel(ehci
, mask
, &ehci
->regs
->intr_enable
);
212 ehci_readl(ehci
, &ehci
->regs
->intr_enable
);
216 ehci_dbg(ehci
, "lost power, restarting\n");
217 usb_root_hub_lost_power(hcd
->self
.root_hub
);
219 /* Else reset, to cope with power loss or flush-to-storage
220 * style "resume" having let BIOS kick in during reboot.
222 (void) ehci_halt(ehci
);
223 (void) ehci_reset(ehci
);
225 /* emptying the schedule aborts any urbs */
226 spin_lock_irq(&ehci
->lock
);
228 end_unlink_async(ehci
);
230 spin_unlock_irq(&ehci
->lock
);
232 ehci_writel(ehci
, ehci
->command
, &ehci
->regs
->command
);
233 ehci_writel(ehci
, FLAG_CF
, &ehci
->regs
->configured_flag
);
234 ehci_readl(ehci
, &ehci
->regs
->command
); /* unblock posted writes */
236 /* here we "know" root ports should always stay powered */
237 ehci_port_power(ehci
, 1);
239 ehci
->rh_state
= EHCI_RH_SUSPENDED
;
244 static const struct dev_pm_ops au1xxx_ehci_pmops
= {
245 .suspend
= ehci_hcd_au1xxx_drv_suspend
,
246 .resume
= ehci_hcd_au1xxx_drv_resume
,
249 #define AU1XXX_EHCI_PMOPS &au1xxx_ehci_pmops
252 #define AU1XXX_EHCI_PMOPS NULL
255 static struct platform_driver ehci_hcd_au1xxx_driver
= {
256 .probe
= ehci_hcd_au1xxx_drv_probe
,
257 .remove
= ehci_hcd_au1xxx_drv_remove
,
258 .shutdown
= usb_hcd_platform_shutdown
,
260 .name
= "au1xxx-ehci",
261 .owner
= THIS_MODULE
,
262 .pm
= AU1XXX_EHCI_PMOPS
,
266 MODULE_ALIAS("platform:au1xxx-ehci");