diff options
Diffstat (limited to 'drivers/usb/host')
-rw-r--r-- | drivers/usb/host/isp1760-hcd.c | 53 | ||||
-rw-r--r-- | drivers/usb/host/isp1760-hcd.h | 5 | ||||
-rw-r--r-- | drivers/usb/host/ohci-hcd.c | 23 | ||||
-rw-r--r-- | drivers/usb/host/ohci-hub.c | 11 | ||||
-rw-r--r-- | drivers/usb/host/ohci-omap.c | 3 | ||||
-rw-r--r-- | drivers/usb/host/ohci-pci.c | 132 | ||||
-rw-r--r-- | drivers/usb/host/ohci-q.c | 6 | ||||
-rw-r--r-- | drivers/usb/host/ohci.h | 11 | ||||
-rw-r--r-- | drivers/usb/host/r8a66597-hcd.c | 49 |
9 files changed, 247 insertions, 46 deletions
diff --git a/drivers/usb/host/isp1760-hcd.c b/drivers/usb/host/isp1760-hcd.c index c858f2adb929..d22a84f86a33 100644 --- a/drivers/usb/host/isp1760-hcd.c +++ b/drivers/usb/host/isp1760-hcd.c | |||
@@ -126,9 +126,8 @@ static void isp1760_writel(const unsigned int val, __u32 __iomem *regs) | |||
126 | * doesn't quite work because some people have to enforce 32-bit access | 126 | * doesn't quite work because some people have to enforce 32-bit access |
127 | */ | 127 | */ |
128 | static void priv_read_copy(struct isp1760_hcd *priv, u32 *src, | 128 | static void priv_read_copy(struct isp1760_hcd *priv, u32 *src, |
129 | __u32 __iomem *dst, u32 offset, u32 len) | 129 | __u32 __iomem *dst, u32 len) |
130 | { | 130 | { |
131 | struct usb_hcd *hcd = priv_to_hcd(priv); | ||
132 | u32 val; | 131 | u32 val; |
133 | u8 *buff8; | 132 | u8 *buff8; |
134 | 133 | ||
@@ -136,11 +135,6 @@ static void priv_read_copy(struct isp1760_hcd *priv, u32 *src, | |||
136 | printk(KERN_ERR "ERROR: buffer: %p len: %d\n", src, len); | 135 | printk(KERN_ERR "ERROR: buffer: %p len: %d\n", src, len); |
137 | return; | 136 | return; |
138 | } | 137 | } |
139 | isp1760_writel(offset, hcd->regs + HC_MEMORY_REG); | ||
140 | /* XXX | ||
141 | * 90nsec delay, the spec says something how this could be avoided. | ||
142 | */ | ||
143 | mdelay(1); | ||
144 | 138 | ||
145 | while (len >= 4) { | 139 | while (len >= 4) { |
146 | *src = __raw_readl(dst); | 140 | *src = __raw_readl(dst); |
@@ -987,8 +981,20 @@ static void do_atl_int(struct usb_hcd *usb_hcd) | |||
987 | printk(KERN_ERR "qh is 0\n"); | 981 | printk(KERN_ERR "qh is 0\n"); |
988 | continue; | 982 | continue; |
989 | } | 983 | } |
990 | priv_read_copy(priv, (u32 *)&ptd, usb_hcd->regs + atl_regs, | 984 | isp1760_writel(atl_regs + ISP_BANK(0), usb_hcd->regs + |
991 | atl_regs, sizeof(ptd)); | 985 | HC_MEMORY_REG); |
986 | isp1760_writel(payload + ISP_BANK(1), usb_hcd->regs + | ||
987 | HC_MEMORY_REG); | ||
988 | /* | ||
989 | * write bank1 address twice to ensure the 90ns delay (time | ||
990 | * between BANK0 write and the priv_read_copy() call is at | ||
991 | * least 3*t_WHWL + 2*t_w11 = 3*25ns + 2*17ns = 92ns) | ||
992 | */ | ||
993 | isp1760_writel(payload + ISP_BANK(1), usb_hcd->regs + | ||
994 | HC_MEMORY_REG); | ||
995 | |||
996 | priv_read_copy(priv, (u32 *)&ptd, usb_hcd->regs + atl_regs + | ||
997 | ISP_BANK(0), sizeof(ptd)); | ||
992 | 998 | ||
993 | dw1 = le32_to_cpu(ptd.dw1); | 999 | dw1 = le32_to_cpu(ptd.dw1); |
994 | dw2 = le32_to_cpu(ptd.dw2); | 1000 | dw2 = le32_to_cpu(ptd.dw2); |
@@ -1091,7 +1097,7 @@ static void do_atl_int(struct usb_hcd *usb_hcd) | |||
1091 | case IN_PID: | 1097 | case IN_PID: |
1092 | priv_read_copy(priv, | 1098 | priv_read_copy(priv, |
1093 | priv->atl_ints[queue_entry].data_buffer, | 1099 | priv->atl_ints[queue_entry].data_buffer, |
1094 | usb_hcd->regs + payload, payload, | 1100 | usb_hcd->regs + payload + ISP_BANK(1), |
1095 | length); | 1101 | length); |
1096 | 1102 | ||
1097 | case OUT_PID: | 1103 | case OUT_PID: |
@@ -1122,11 +1128,11 @@ static void do_atl_int(struct usb_hcd *usb_hcd) | |||
1122 | } else if (usb_pipebulk(urb->pipe) && (length < qtd->length)) { | 1128 | } else if (usb_pipebulk(urb->pipe) && (length < qtd->length)) { |
1123 | /* short BULK received */ | 1129 | /* short BULK received */ |
1124 | 1130 | ||
1125 | printk(KERN_ERR "short bulk, %d instead %zu\n", length, | ||
1126 | qtd->length); | ||
1127 | if (urb->transfer_flags & URB_SHORT_NOT_OK) { | 1131 | if (urb->transfer_flags & URB_SHORT_NOT_OK) { |
1128 | urb->status = -EREMOTEIO; | 1132 | urb->status = -EREMOTEIO; |
1129 | printk(KERN_ERR "not okey\n"); | 1133 | isp1760_dbg(priv, "short bulk, %d instead %zu " |
1134 | "with URB_SHORT_NOT_OK flag.\n", | ||
1135 | length, qtd->length); | ||
1130 | } | 1136 | } |
1131 | 1137 | ||
1132 | if (urb->status == -EINPROGRESS) | 1138 | if (urb->status == -EINPROGRESS) |
@@ -1206,8 +1212,20 @@ static void do_intl_int(struct usb_hcd *usb_hcd) | |||
1206 | continue; | 1212 | continue; |
1207 | } | 1213 | } |
1208 | 1214 | ||
1209 | priv_read_copy(priv, (u32 *)&ptd, usb_hcd->regs + int_regs, | 1215 | isp1760_writel(int_regs + ISP_BANK(0), usb_hcd->regs + |
1210 | int_regs, sizeof(ptd)); | 1216 | HC_MEMORY_REG); |
1217 | isp1760_writel(payload + ISP_BANK(1), usb_hcd->regs + | ||
1218 | HC_MEMORY_REG); | ||
1219 | /* | ||
1220 | * write bank1 address twice to ensure the 90ns delay (time | ||
1221 | * between BANK0 write and the priv_read_copy() call is at | ||
1222 | * least 3*t_WHWL + 2*t_w11 = 3*25ns + 2*17ns = 92ns) | ||
1223 | */ | ||
1224 | isp1760_writel(payload + ISP_BANK(1), usb_hcd->regs + | ||
1225 | HC_MEMORY_REG); | ||
1226 | |||
1227 | priv_read_copy(priv, (u32 *)&ptd, usb_hcd->regs + int_regs + | ||
1228 | ISP_BANK(0), sizeof(ptd)); | ||
1211 | dw1 = le32_to_cpu(ptd.dw1); | 1229 | dw1 = le32_to_cpu(ptd.dw1); |
1212 | dw3 = le32_to_cpu(ptd.dw3); | 1230 | dw3 = le32_to_cpu(ptd.dw3); |
1213 | check_int_err_status(le32_to_cpu(ptd.dw4)); | 1231 | check_int_err_status(le32_to_cpu(ptd.dw4)); |
@@ -1242,7 +1260,7 @@ static void do_intl_int(struct usb_hcd *usb_hcd) | |||
1242 | case IN_PID: | 1260 | case IN_PID: |
1243 | priv_read_copy(priv, | 1261 | priv_read_copy(priv, |
1244 | priv->int_ints[queue_entry].data_buffer, | 1262 | priv->int_ints[queue_entry].data_buffer, |
1245 | usb_hcd->regs + payload , payload, | 1263 | usb_hcd->regs + payload + ISP_BANK(1), |
1246 | length); | 1264 | length); |
1247 | case OUT_PID: | 1265 | case OUT_PID: |
1248 | 1266 | ||
@@ -1615,8 +1633,7 @@ static int isp1760_urb_enqueue(struct usb_hcd *hcd, struct urb *urb, | |||
1615 | return -EPIPE; | 1633 | return -EPIPE; |
1616 | } | 1634 | } |
1617 | 1635 | ||
1618 | isp1760_prepare_enqueue(priv, urb, &qtd_list, mem_flags, pe); | 1636 | return isp1760_prepare_enqueue(priv, urb, &qtd_list, mem_flags, pe); |
1619 | return 0; | ||
1620 | } | 1637 | } |
1621 | 1638 | ||
1622 | static int isp1760_urb_dequeue(struct usb_hcd *hcd, struct urb *urb, | 1639 | static int isp1760_urb_dequeue(struct usb_hcd *hcd, struct urb *urb, |
diff --git a/drivers/usb/host/isp1760-hcd.h b/drivers/usb/host/isp1760-hcd.h index 6473dd86993c..4377277667d9 100644 --- a/drivers/usb/host/isp1760-hcd.h +++ b/drivers/usb/host/isp1760-hcd.h | |||
@@ -54,6 +54,8 @@ void deinit_kmem_cache(void); | |||
54 | #define BUFFER_MAP 0x7 | 54 | #define BUFFER_MAP 0x7 |
55 | 55 | ||
56 | #define HC_MEMORY_REG 0x33c | 56 | #define HC_MEMORY_REG 0x33c |
57 | #define ISP_BANK(x) ((x) << 16) | ||
58 | |||
57 | #define HC_PORT1_CTRL 0x374 | 59 | #define HC_PORT1_CTRL 0x374 |
58 | #define PORT1_POWER (3 << 3) | 60 | #define PORT1_POWER (3 << 3) |
59 | #define PORT1_INIT1 (1 << 7) | 61 | #define PORT1_INIT1 (1 << 7) |
@@ -119,6 +121,9 @@ struct inter_packet_info { | |||
119 | typedef void (packet_enqueue)(struct usb_hcd *hcd, struct isp1760_qh *qh, | 121 | typedef void (packet_enqueue)(struct usb_hcd *hcd, struct isp1760_qh *qh, |
120 | struct isp1760_qtd *qtd); | 122 | struct isp1760_qtd *qtd); |
121 | 123 | ||
124 | #define isp1760_dbg(priv, fmt, args...) \ | ||
125 | dev_dbg(priv_to_hcd(priv)->self.controller, fmt, ##args) | ||
126 | |||
122 | #define isp1760_info(priv, fmt, args...) \ | 127 | #define isp1760_info(priv, fmt, args...) \ |
123 | dev_info(priv_to_hcd(priv)->self.controller, fmt, ##args) | 128 | dev_info(priv_to_hcd(priv)->self.controller, fmt, ##args) |
124 | 129 | ||
diff --git a/drivers/usb/host/ohci-hcd.c b/drivers/usb/host/ohci-hcd.c index 26bc47941d01..89901962cbfd 100644 --- a/drivers/usb/host/ohci-hcd.c +++ b/drivers/usb/host/ohci-hcd.c | |||
@@ -86,6 +86,21 @@ static void ohci_stop (struct usb_hcd *hcd); | |||
86 | static int ohci_restart (struct ohci_hcd *ohci); | 86 | static int ohci_restart (struct ohci_hcd *ohci); |
87 | #endif | 87 | #endif |
88 | 88 | ||
89 | #ifdef CONFIG_PCI | ||
90 | static void quirk_amd_pll(int state); | ||
91 | static void amd_iso_dev_put(void); | ||
92 | #else | ||
93 | static inline void quirk_amd_pll(int state) | ||
94 | { | ||
95 | return; | ||
96 | } | ||
97 | static inline void amd_iso_dev_put(void) | ||
98 | { | ||
99 | return; | ||
100 | } | ||
101 | #endif | ||
102 | |||
103 | |||
89 | #include "ohci-hub.c" | 104 | #include "ohci-hub.c" |
90 | #include "ohci-dbg.c" | 105 | #include "ohci-dbg.c" |
91 | #include "ohci-mem.c" | 106 | #include "ohci-mem.c" |
@@ -483,6 +498,9 @@ static int ohci_init (struct ohci_hcd *ohci) | |||
483 | int ret; | 498 | int ret; |
484 | struct usb_hcd *hcd = ohci_to_hcd(ohci); | 499 | struct usb_hcd *hcd = ohci_to_hcd(ohci); |
485 | 500 | ||
501 | if (distrust_firmware) | ||
502 | ohci->flags |= OHCI_QUIRK_HUB_POWER; | ||
503 | |||
486 | disable (ohci); | 504 | disable (ohci); |
487 | ohci->regs = hcd->regs; | 505 | ohci->regs = hcd->regs; |
488 | 506 | ||
@@ -689,7 +707,8 @@ retry: | |||
689 | temp |= RH_A_NOCP; | 707 | temp |= RH_A_NOCP; |
690 | temp &= ~(RH_A_POTPGT | RH_A_NPS); | 708 | temp &= ~(RH_A_POTPGT | RH_A_NPS); |
691 | ohci_writel (ohci, temp, &ohci->regs->roothub.a); | 709 | ohci_writel (ohci, temp, &ohci->regs->roothub.a); |
692 | } else if ((ohci->flags & OHCI_QUIRK_AMD756) || distrust_firmware) { | 710 | } else if ((ohci->flags & OHCI_QUIRK_AMD756) || |
711 | (ohci->flags & OHCI_QUIRK_HUB_POWER)) { | ||
693 | /* hub power always on; required for AMD-756 and some | 712 | /* hub power always on; required for AMD-756 and some |
694 | * Mac platforms. ganged overcurrent reporting, if any. | 713 | * Mac platforms. ganged overcurrent reporting, if any. |
695 | */ | 714 | */ |
@@ -882,6 +901,8 @@ static void ohci_stop (struct usb_hcd *hcd) | |||
882 | 901 | ||
883 | if (quirk_zfmicro(ohci)) | 902 | if (quirk_zfmicro(ohci)) |
884 | del_timer(&ohci->unlink_watchdog); | 903 | del_timer(&ohci->unlink_watchdog); |
904 | if (quirk_amdiso(ohci)) | ||
905 | amd_iso_dev_put(); | ||
885 | 906 | ||
886 | remove_debug_files (ohci); | 907 | remove_debug_files (ohci); |
887 | ohci_mem_cleanup (ohci); | 908 | ohci_mem_cleanup (ohci); |
diff --git a/drivers/usb/host/ohci-hub.c b/drivers/usb/host/ohci-hub.c index b56739221d11..439beb784f3e 100644 --- a/drivers/usb/host/ohci-hub.c +++ b/drivers/usb/host/ohci-hub.c | |||
@@ -483,6 +483,13 @@ ohci_hub_status_data (struct usb_hcd *hcd, char *buf) | |||
483 | length++; | 483 | length++; |
484 | } | 484 | } |
485 | 485 | ||
486 | /* Some broken controllers never turn off RHCS in the interrupt | ||
487 | * status register. For their sake we won't re-enable RHSC | ||
488 | * interrupts if the flag is already set. | ||
489 | */ | ||
490 | if (ohci_readl(ohci, &ohci->regs->intrstatus) & OHCI_INTR_RHSC) | ||
491 | changed = 1; | ||
492 | |||
486 | /* look at each port */ | 493 | /* look at each port */ |
487 | for (i = 0; i < ohci->num_ports; i++) { | 494 | for (i = 0; i < ohci->num_ports; i++) { |
488 | u32 status = roothub_portstatus (ohci, i); | 495 | u32 status = roothub_portstatus (ohci, i); |
@@ -572,8 +579,6 @@ static int ohci_start_port_reset (struct usb_hcd *hcd, unsigned port) | |||
572 | return 0; | 579 | return 0; |
573 | } | 580 | } |
574 | 581 | ||
575 | static void start_hnp(struct ohci_hcd *ohci); | ||
576 | |||
577 | #else | 582 | #else |
578 | 583 | ||
579 | #define ohci_start_port_reset NULL | 584 | #define ohci_start_port_reset NULL |
@@ -760,7 +765,7 @@ static int ohci_hub_control ( | |||
760 | #ifdef CONFIG_USB_OTG | 765 | #ifdef CONFIG_USB_OTG |
761 | if (hcd->self.otg_port == (wIndex + 1) | 766 | if (hcd->self.otg_port == (wIndex + 1) |
762 | && hcd->self.b_hnp_enable) | 767 | && hcd->self.b_hnp_enable) |
763 | start_hnp(ohci); | 768 | ohci->start_hnp(ohci); |
764 | else | 769 | else |
765 | #endif | 770 | #endif |
766 | ohci_writel (ohci, RH_PS_PSS, | 771 | ohci_writel (ohci, RH_PS_PSS, |
diff --git a/drivers/usb/host/ohci-omap.c b/drivers/usb/host/ohci-omap.c index 94dfca02f7e1..3d532b709670 100644 --- a/drivers/usb/host/ohci-omap.c +++ b/drivers/usb/host/ohci-omap.c | |||
@@ -225,6 +225,7 @@ static int ohci_omap_init(struct usb_hcd *hcd) | |||
225 | dev_err(hcd->self.controller, "can't find transceiver\n"); | 225 | dev_err(hcd->self.controller, "can't find transceiver\n"); |
226 | return -ENODEV; | 226 | return -ENODEV; |
227 | } | 227 | } |
228 | ohci->start_hnp = start_hnp; | ||
228 | } | 229 | } |
229 | #endif | 230 | #endif |
230 | 231 | ||
@@ -260,7 +261,7 @@ static int ohci_omap_init(struct usb_hcd *hcd) | |||
260 | omap_cfg_reg(W4_USB_HIGHZ); | 261 | omap_cfg_reg(W4_USB_HIGHZ); |
261 | } | 262 | } |
262 | ohci_writel(ohci, rh, &ohci->regs->roothub.a); | 263 | ohci_writel(ohci, rh, &ohci->regs->roothub.a); |
263 | distrust_firmware = 0; | 264 | ohci->flags &= ~OHCI_QUIRK_HUB_POWER; |
264 | } else if (machine_is_nokia770()) { | 265 | } else if (machine_is_nokia770()) { |
265 | /* We require a self-powered hub, which should have | 266 | /* We require a self-powered hub, which should have |
266 | * plenty of power. */ | 267 | * plenty of power. */ |
diff --git a/drivers/usb/host/ohci-pci.c b/drivers/usb/host/ohci-pci.c index 4696cc912e16..083e8df0a817 100644 --- a/drivers/usb/host/ohci-pci.c +++ b/drivers/usb/host/ohci-pci.c | |||
@@ -18,6 +18,28 @@ | |||
18 | #error "This file is PCI bus glue. CONFIG_PCI must be defined." | 18 | #error "This file is PCI bus glue. CONFIG_PCI must be defined." |
19 | #endif | 19 | #endif |
20 | 20 | ||
21 | #include <linux/pci.h> | ||
22 | #include <linux/io.h> | ||
23 | |||
24 | |||
25 | /* constants used to work around PM-related transfer | ||
26 | * glitches in some AMD 700 series southbridges | ||
27 | */ | ||
28 | #define AB_REG_BAR 0xf0 | ||
29 | #define AB_INDX(addr) ((addr) + 0x00) | ||
30 | #define AB_DATA(addr) ((addr) + 0x04) | ||
31 | #define AX_INDXC 0X30 | ||
32 | #define AX_DATAC 0x34 | ||
33 | |||
34 | #define NB_PCIE_INDX_ADDR 0xe0 | ||
35 | #define NB_PCIE_INDX_DATA 0xe4 | ||
36 | #define PCIE_P_CNTL 0x10040 | ||
37 | #define BIF_NB 0x10002 | ||
38 | |||
39 | static struct pci_dev *amd_smbus_dev; | ||
40 | static struct pci_dev *amd_hb_dev; | ||
41 | static int amd_ohci_iso_count; | ||
42 | |||
21 | /*-------------------------------------------------------------------------*/ | 43 | /*-------------------------------------------------------------------------*/ |
22 | 44 | ||
23 | static int broken_suspend(struct usb_hcd *hcd) | 45 | static int broken_suspend(struct usb_hcd *hcd) |
@@ -143,6 +165,103 @@ static int ohci_quirk_nec(struct usb_hcd *hcd) | |||
143 | return 0; | 165 | return 0; |
144 | } | 166 | } |
145 | 167 | ||
168 | static int ohci_quirk_amd700(struct usb_hcd *hcd) | ||
169 | { | ||
170 | struct ohci_hcd *ohci = hcd_to_ohci(hcd); | ||
171 | u8 rev = 0; | ||
172 | |||
173 | if (!amd_smbus_dev) | ||
174 | amd_smbus_dev = pci_get_device(PCI_VENDOR_ID_ATI, | ||
175 | PCI_DEVICE_ID_ATI_SBX00_SMBUS, NULL); | ||
176 | if (!amd_smbus_dev) | ||
177 | return 0; | ||
178 | |||
179 | pci_read_config_byte(amd_smbus_dev, PCI_REVISION_ID, &rev); | ||
180 | if ((rev > 0x3b) || (rev < 0x30)) { | ||
181 | pci_dev_put(amd_smbus_dev); | ||
182 | amd_smbus_dev = NULL; | ||
183 | return 0; | ||
184 | } | ||
185 | |||
186 | amd_ohci_iso_count++; | ||
187 | |||
188 | if (!amd_hb_dev) | ||
189 | amd_hb_dev = pci_get_device(PCI_VENDOR_ID_AMD, 0x9600, NULL); | ||
190 | |||
191 | ohci->flags |= OHCI_QUIRK_AMD_ISO; | ||
192 | ohci_dbg(ohci, "enabled AMD ISO transfers quirk\n"); | ||
193 | |||
194 | return 0; | ||
195 | } | ||
196 | |||
197 | /* | ||
198 | * The hardware normally enables the A-link power management feature, which | ||
199 | * lets the system lower the power consumption in idle states. | ||
200 | * | ||
201 | * Assume the system is configured to have USB 1.1 ISO transfers going | ||
202 | * to or from a USB device. Without this quirk, that stream may stutter | ||
203 | * or have breaks occasionally. For transfers going to speakers, this | ||
204 | * makes a very audible mess... | ||
205 | * | ||
206 | * That audio playback corruption is due to the audio stream getting | ||
207 | * interrupted occasionally when the link goes in lower power state | ||
208 | * This USB quirk prevents the link going into that lower power state | ||
209 | * during audio playback or other ISO operations. | ||
210 | */ | ||
211 | static void quirk_amd_pll(int on) | ||
212 | { | ||
213 | u32 addr; | ||
214 | u32 val; | ||
215 | u32 bit = (on > 0) ? 1 : 0; | ||
216 | |||
217 | pci_read_config_dword(amd_smbus_dev, AB_REG_BAR, &addr); | ||
218 | |||
219 | /* BIT names/meanings are NDA-protected, sorry ... */ | ||
220 | |||
221 | outl(AX_INDXC, AB_INDX(addr)); | ||
222 | outl(0x40, AB_DATA(addr)); | ||
223 | outl(AX_DATAC, AB_INDX(addr)); | ||
224 | val = inl(AB_DATA(addr)); | ||
225 | val &= ~((1 << 3) | (1 << 4) | (1 << 9)); | ||
226 | val |= (bit << 3) | ((!bit) << 4) | ((!bit) << 9); | ||
227 | outl(val, AB_DATA(addr)); | ||
228 | |||
229 | if (amd_hb_dev) { | ||
230 | addr = PCIE_P_CNTL; | ||
231 | pci_write_config_dword(amd_hb_dev, NB_PCIE_INDX_ADDR, addr); | ||
232 | |||
233 | pci_read_config_dword(amd_hb_dev, NB_PCIE_INDX_DATA, &val); | ||
234 | val &= ~(1 | (1 << 3) | (1 << 4) | (1 << 9) | (1 << 12)); | ||
235 | val |= bit | (bit << 3) | (bit << 12); | ||
236 | val |= ((!bit) << 4) | ((!bit) << 9); | ||
237 | pci_write_config_dword(amd_hb_dev, NB_PCIE_INDX_DATA, val); | ||
238 | |||
239 | addr = BIF_NB; | ||
240 | pci_write_config_dword(amd_hb_dev, NB_PCIE_INDX_ADDR, addr); | ||
241 | |||
242 | pci_read_config_dword(amd_hb_dev, NB_PCIE_INDX_DATA, &val); | ||
243 | val &= ~(1 << 8); | ||
244 | val |= bit << 8; | ||
245 | pci_write_config_dword(amd_hb_dev, NB_PCIE_INDX_DATA, val); | ||
246 | } | ||
247 | } | ||
248 | |||
249 | static void amd_iso_dev_put(void) | ||
250 | { | ||
251 | amd_ohci_iso_count--; | ||
252 | if (amd_ohci_iso_count == 0) { | ||
253 | if (amd_smbus_dev) { | ||
254 | pci_dev_put(amd_smbus_dev); | ||
255 | amd_smbus_dev = NULL; | ||
256 | } | ||
257 | if (amd_hb_dev) { | ||
258 | pci_dev_put(amd_hb_dev); | ||
259 | amd_hb_dev = NULL; | ||
260 | } | ||
261 | } | ||
262 | |||
263 | } | ||
264 | |||
146 | /* List of quirks for OHCI */ | 265 | /* List of quirks for OHCI */ |
147 | static const struct pci_device_id ohci_pci_quirks[] = { | 266 | static const struct pci_device_id ohci_pci_quirks[] = { |
148 | { | 267 | { |
@@ -181,6 +300,19 @@ static const struct pci_device_id ohci_pci_quirks[] = { | |||
181 | PCI_DEVICE(PCI_VENDOR_ID_ITE, 0x8152), | 300 | PCI_DEVICE(PCI_VENDOR_ID_ITE, 0x8152), |
182 | .driver_data = (unsigned long) broken_suspend, | 301 | .driver_data = (unsigned long) broken_suspend, |
183 | }, | 302 | }, |
303 | { | ||
304 | PCI_DEVICE(PCI_VENDOR_ID_ATI, 0x4397), | ||
305 | .driver_data = (unsigned long)ohci_quirk_amd700, | ||
306 | }, | ||
307 | { | ||
308 | PCI_DEVICE(PCI_VENDOR_ID_ATI, 0x4398), | ||
309 | .driver_data = (unsigned long)ohci_quirk_amd700, | ||
310 | }, | ||
311 | { | ||
312 | PCI_DEVICE(PCI_VENDOR_ID_ATI, 0x4399), | ||
313 | .driver_data = (unsigned long)ohci_quirk_amd700, | ||
314 | }, | ||
315 | |||
184 | /* FIXME for some of the early AMD 760 southbridges, OHCI | 316 | /* FIXME for some of the early AMD 760 southbridges, OHCI |
185 | * won't work at all. blacklist them. | 317 | * won't work at all. blacklist them. |
186 | */ | 318 | */ |
diff --git a/drivers/usb/host/ohci-q.c b/drivers/usb/host/ohci-q.c index 6a9b4c557953..c2d80f80448b 100644 --- a/drivers/usb/host/ohci-q.c +++ b/drivers/usb/host/ohci-q.c | |||
@@ -49,6 +49,9 @@ __acquires(ohci->lock) | |||
49 | switch (usb_pipetype (urb->pipe)) { | 49 | switch (usb_pipetype (urb->pipe)) { |
50 | case PIPE_ISOCHRONOUS: | 50 | case PIPE_ISOCHRONOUS: |
51 | ohci_to_hcd(ohci)->self.bandwidth_isoc_reqs--; | 51 | ohci_to_hcd(ohci)->self.bandwidth_isoc_reqs--; |
52 | if (ohci_to_hcd(ohci)->self.bandwidth_isoc_reqs == 0 | ||
53 | && quirk_amdiso(ohci)) | ||
54 | quirk_amd_pll(1); | ||
52 | break; | 55 | break; |
53 | case PIPE_INTERRUPT: | 56 | case PIPE_INTERRUPT: |
54 | ohci_to_hcd(ohci)->self.bandwidth_int_reqs--; | 57 | ohci_to_hcd(ohci)->self.bandwidth_int_reqs--; |
@@ -677,6 +680,9 @@ static void td_submit_urb ( | |||
677 | data + urb->iso_frame_desc [cnt].offset, | 680 | data + urb->iso_frame_desc [cnt].offset, |
678 | urb->iso_frame_desc [cnt].length, urb, cnt); | 681 | urb->iso_frame_desc [cnt].length, urb, cnt); |
679 | } | 682 | } |
683 | if (ohci_to_hcd(ohci)->self.bandwidth_isoc_reqs == 0 | ||
684 | && quirk_amdiso(ohci)) | ||
685 | quirk_amd_pll(0); | ||
680 | periodic = ohci_to_hcd(ohci)->self.bandwidth_isoc_reqs++ == 0 | 686 | periodic = ohci_to_hcd(ohci)->self.bandwidth_isoc_reqs++ == 0 |
681 | && ohci_to_hcd(ohci)->self.bandwidth_int_reqs == 0; | 687 | && ohci_to_hcd(ohci)->self.bandwidth_int_reqs == 0; |
682 | break; | 688 | break; |
diff --git a/drivers/usb/host/ohci.h b/drivers/usb/host/ohci.h index dc544ddc7849..faf622eafce7 100644 --- a/drivers/usb/host/ohci.h +++ b/drivers/usb/host/ohci.h | |||
@@ -371,6 +371,7 @@ struct ohci_hcd { | |||
371 | * other external transceivers should be software-transparent | 371 | * other external transceivers should be software-transparent |
372 | */ | 372 | */ |
373 | struct otg_transceiver *transceiver; | 373 | struct otg_transceiver *transceiver; |
374 | void (*start_hnp)(struct ohci_hcd *ohci); | ||
374 | 375 | ||
375 | /* | 376 | /* |
376 | * memory management for queue data structures | 377 | * memory management for queue data structures |
@@ -399,6 +400,8 @@ struct ohci_hcd { | |||
399 | #define OHCI_QUIRK_ZFMICRO 0x20 /* Compaq ZFMicro chipset*/ | 400 | #define OHCI_QUIRK_ZFMICRO 0x20 /* Compaq ZFMicro chipset*/ |
400 | #define OHCI_QUIRK_NEC 0x40 /* lost interrupts */ | 401 | #define OHCI_QUIRK_NEC 0x40 /* lost interrupts */ |
401 | #define OHCI_QUIRK_FRAME_NO 0x80 /* no big endian frame_no shift */ | 402 | #define OHCI_QUIRK_FRAME_NO 0x80 /* no big endian frame_no shift */ |
403 | #define OHCI_QUIRK_HUB_POWER 0x100 /* distrust firmware power/oc setup */ | ||
404 | #define OHCI_QUIRK_AMD_ISO 0x200 /* ISO transfers*/ | ||
402 | // there are also chip quirks/bugs in init logic | 405 | // there are also chip quirks/bugs in init logic |
403 | 406 | ||
404 | struct work_struct nec_work; /* Worker for NEC quirk */ | 407 | struct work_struct nec_work; /* Worker for NEC quirk */ |
@@ -426,6 +429,10 @@ static inline int quirk_zfmicro(struct ohci_hcd *ohci) | |||
426 | { | 429 | { |
427 | return ohci->flags & OHCI_QUIRK_ZFMICRO; | 430 | return ohci->flags & OHCI_QUIRK_ZFMICRO; |
428 | } | 431 | } |
432 | static inline int quirk_amdiso(struct ohci_hcd *ohci) | ||
433 | { | ||
434 | return ohci->flags & OHCI_QUIRK_AMD_ISO; | ||
435 | } | ||
429 | #else | 436 | #else |
430 | static inline int quirk_nec(struct ohci_hcd *ohci) | 437 | static inline int quirk_nec(struct ohci_hcd *ohci) |
431 | { | 438 | { |
@@ -435,6 +442,10 @@ static inline int quirk_zfmicro(struct ohci_hcd *ohci) | |||
435 | { | 442 | { |
436 | return 0; | 443 | return 0; |
437 | } | 444 | } |
445 | static inline int quirk_amdiso(struct ohci_hcd *ohci) | ||
446 | { | ||
447 | return 0; | ||
448 | } | ||
438 | #endif | 449 | #endif |
439 | 450 | ||
440 | /* convert between an hcd pointer and the corresponding ohci_hcd */ | 451 | /* convert between an hcd pointer and the corresponding ohci_hcd */ |
diff --git a/drivers/usb/host/r8a66597-hcd.c b/drivers/usb/host/r8a66597-hcd.c index d5f02dddb120..ea7126f99cab 100644 --- a/drivers/usb/host/r8a66597-hcd.c +++ b/drivers/usb/host/r8a66597-hcd.c | |||
@@ -964,11 +964,34 @@ static void pipe_irq_disable(struct r8a66597 *r8a66597, u16 pipenum) | |||
964 | disable_irq_nrdy(r8a66597, pipenum); | 964 | disable_irq_nrdy(r8a66597, pipenum); |
965 | } | 965 | } |
966 | 966 | ||
967 | static void r8a66597_root_hub_start_polling(struct r8a66597 *r8a66597) | ||
968 | { | ||
969 | mod_timer(&r8a66597->rh_timer, | ||
970 | jiffies + msecs_to_jiffies(R8A66597_RH_POLL_TIME)); | ||
971 | } | ||
972 | |||
973 | static void start_root_hub_sampling(struct r8a66597 *r8a66597, int port, | ||
974 | int connect) | ||
975 | { | ||
976 | struct r8a66597_root_hub *rh = &r8a66597->root_hub[port]; | ||
977 | |||
978 | rh->old_syssts = r8a66597_read(r8a66597, get_syssts_reg(port)) & LNST; | ||
979 | rh->scount = R8A66597_MAX_SAMPLING; | ||
980 | if (connect) | ||
981 | rh->port |= 1 << USB_PORT_FEAT_CONNECTION; | ||
982 | else | ||
983 | rh->port &= ~(1 << USB_PORT_FEAT_CONNECTION); | ||
984 | rh->port |= 1 << USB_PORT_FEAT_C_CONNECTION; | ||
985 | |||
986 | r8a66597_root_hub_start_polling(r8a66597); | ||
987 | } | ||
988 | |||
967 | /* this function must be called with interrupt disabled */ | 989 | /* this function must be called with interrupt disabled */ |
968 | static void r8a66597_check_syssts(struct r8a66597 *r8a66597, int port, | 990 | static void r8a66597_check_syssts(struct r8a66597 *r8a66597, int port, |
969 | u16 syssts) | 991 | u16 syssts) |
970 | { | 992 | { |
971 | if (syssts == SE0) { | 993 | if (syssts == SE0) { |
994 | r8a66597_write(r8a66597, ~ATTCH, get_intsts_reg(port)); | ||
972 | r8a66597_bset(r8a66597, ATTCHE, get_intenb_reg(port)); | 995 | r8a66597_bset(r8a66597, ATTCHE, get_intenb_reg(port)); |
973 | return; | 996 | return; |
974 | } | 997 | } |
@@ -1002,13 +1025,10 @@ static void r8a66597_usb_disconnect(struct r8a66597 *r8a66597, int port) | |||
1002 | { | 1025 | { |
1003 | struct r8a66597_device *dev = r8a66597->root_hub[port].dev; | 1026 | struct r8a66597_device *dev = r8a66597->root_hub[port].dev; |
1004 | 1027 | ||
1005 | r8a66597->root_hub[port].port &= ~(1 << USB_PORT_FEAT_CONNECTION); | ||
1006 | r8a66597->root_hub[port].port |= (1 << USB_PORT_FEAT_C_CONNECTION); | ||
1007 | |||
1008 | disable_r8a66597_pipe_all(r8a66597, dev); | 1028 | disable_r8a66597_pipe_all(r8a66597, dev); |
1009 | free_usb_address(r8a66597, dev); | 1029 | free_usb_address(r8a66597, dev); |
1010 | 1030 | ||
1011 | r8a66597_bset(r8a66597, ATTCHE, get_intenb_reg(port)); | 1031 | start_root_hub_sampling(r8a66597, port, 0); |
1012 | } | 1032 | } |
1013 | 1033 | ||
1014 | /* this function must be called with interrupt disabled */ | 1034 | /* this function must be called with interrupt disabled */ |
@@ -1551,23 +1571,6 @@ static void irq_pipe_nrdy(struct r8a66597 *r8a66597) | |||
1551 | } | 1571 | } |
1552 | } | 1572 | } |
1553 | 1573 | ||
1554 | static void r8a66597_root_hub_start_polling(struct r8a66597 *r8a66597) | ||
1555 | { | ||
1556 | mod_timer(&r8a66597->rh_timer, | ||
1557 | jiffies + msecs_to_jiffies(R8A66597_RH_POLL_TIME)); | ||
1558 | } | ||
1559 | |||
1560 | static void start_root_hub_sampling(struct r8a66597 *r8a66597, int port) | ||
1561 | { | ||
1562 | struct r8a66597_root_hub *rh = &r8a66597->root_hub[port]; | ||
1563 | |||
1564 | rh->old_syssts = r8a66597_read(r8a66597, get_syssts_reg(port)) & LNST; | ||
1565 | rh->scount = R8A66597_MAX_SAMPLING; | ||
1566 | r8a66597->root_hub[port].port |= (1 << USB_PORT_FEAT_CONNECTION) | ||
1567 | | (1 << USB_PORT_FEAT_C_CONNECTION); | ||
1568 | r8a66597_root_hub_start_polling(r8a66597); | ||
1569 | } | ||
1570 | |||
1571 | static irqreturn_t r8a66597_irq(struct usb_hcd *hcd) | 1574 | static irqreturn_t r8a66597_irq(struct usb_hcd *hcd) |
1572 | { | 1575 | { |
1573 | struct r8a66597 *r8a66597 = hcd_to_r8a66597(hcd); | 1576 | struct r8a66597 *r8a66597 = hcd_to_r8a66597(hcd); |
@@ -1594,7 +1597,7 @@ static irqreturn_t r8a66597_irq(struct usb_hcd *hcd) | |||
1594 | r8a66597_bclr(r8a66597, ATTCHE, INTENB2); | 1597 | r8a66597_bclr(r8a66597, ATTCHE, INTENB2); |
1595 | 1598 | ||
1596 | /* start usb bus sampling */ | 1599 | /* start usb bus sampling */ |
1597 | start_root_hub_sampling(r8a66597, 1); | 1600 | start_root_hub_sampling(r8a66597, 1, 1); |
1598 | } | 1601 | } |
1599 | if (mask2 & DTCH) { | 1602 | if (mask2 & DTCH) { |
1600 | r8a66597_write(r8a66597, ~DTCH, INTSTS2); | 1603 | r8a66597_write(r8a66597, ~DTCH, INTSTS2); |
@@ -1609,7 +1612,7 @@ static irqreturn_t r8a66597_irq(struct usb_hcd *hcd) | |||
1609 | r8a66597_bclr(r8a66597, ATTCHE, INTENB1); | 1612 | r8a66597_bclr(r8a66597, ATTCHE, INTENB1); |
1610 | 1613 | ||
1611 | /* start usb bus sampling */ | 1614 | /* start usb bus sampling */ |
1612 | start_root_hub_sampling(r8a66597, 0); | 1615 | start_root_hub_sampling(r8a66597, 0, 1); |
1613 | } | 1616 | } |
1614 | if (mask1 & DTCH) { | 1617 | if (mask1 & DTCH) { |
1615 | r8a66597_write(r8a66597, ~DTCH, INTSTS1); | 1618 | r8a66597_write(r8a66597, ~DTCH, INTSTS1); |