2 * Copyright (c) 2010 Broadcom Corporation
4 * Permission to use, copy, modify, and/or distribute this software for any
5 * purpose with or without fee is hereby granted, provided that the above
6 * copyright notice and this permission notice appear in all copies.
8 * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
9 * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
10 * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY
11 * SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
12 * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION
13 * OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN
14 * CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
18 * @file bcmsdh_linux.c
21 #define __UNDEF_NO_VERSION__
26 #include <linux/pci.h>
27 #include <linux/completion.h>
34 #if defined(OOB_INTR_ONLY)
35 #include <linux/irq.h>
36 extern void dhdsdio_isr(void *args
);
38 #include <dngl_stats.h>
40 #endif /* defined(OOB_INTR_ONLY) */
41 #if defined(CONFIG_MACH_SANDGATE2G) || defined(CONFIG_MACH_LOGICPD_PXA270)
42 #if !defined(BCMPLATFORM_BUS)
43 #define BCMPLATFORM_BUS
44 #endif /* !defined(BCMPLATFORM_BUS) */
46 #include <linux/platform_device.h>
47 #endif /* CONFIG_MACH_SANDGATE2G */
50 * SDIO Host Controller info
52 typedef struct bcmsdh_hc bcmsdh_hc_t
;
56 #ifdef BCMPLATFORM_BUS
57 struct device
*dev
; /* platform device handle */
59 struct pci_dev
*dev
; /* pci device handle */
60 #endif /* BCMPLATFORM_BUS */
62 void *regs
; /* SDIO Host Controller address */
63 bcmsdh_info_t
*sdh
; /* SDIO Host Controller handle */
66 unsigned long oob_flags
; /* OOB Host specifiction
68 bool oob_irq_registered
;
69 #if defined(OOB_INTR_ONLY)
73 static bcmsdh_hc_t
*sdhcinfo
= NULL
;
75 /* driver info, initialized when bcmsdh_register is called */
76 static bcmsdh_driver_t drvinfo
= { NULL
, NULL
};
78 /* debugging macros */
82 * Checks to see if vendor and device IDs match a supported SDIO Host Controller.
84 bool bcmsdh_chipmatch(uint16 vendor
, uint16 device
)
86 /* Add other vendors and devices as required */
89 /* Check for Arasan host controller */
90 if (vendor
== VENDOR_SI_IMAGE
)
93 /* Check for BRCM 27XX Standard host controller */
94 if (device
== BCM27XX_SDIOH_ID
&& vendor
== VENDOR_BROADCOM
)
97 /* Check for BRCM Standard host controller */
98 if (device
== SDIOH_FPGA_ID
&& vendor
== VENDOR_BROADCOM
)
101 /* Check for TI PCIxx21 Standard host controller */
102 if (device
== PCIXX21_SDIOH_ID
&& vendor
== VENDOR_TI
)
105 if (device
== PCIXX21_SDIOH0_ID
&& vendor
== VENDOR_TI
)
108 /* Ricoh R5C822 Standard SDIO Host */
109 if (device
== R5C822_SDIOH_ID
&& vendor
== VENDOR_RICOH
)
112 /* JMicron Standard SDIO Host */
113 if (device
== JMICRON_SDIOH_ID
&& vendor
== VENDOR_JMICRON
)
115 #endif /* BCMSDIOH_STD */
117 /* This is the PciSpiHost. */
118 if (device
== SPIH_FPGA_ID
&& vendor
== VENDOR_BROADCOM
) {
119 printf("Found PCI SPI Host Controller\n");
122 #endif /* BCMSDIOH_SPI */
127 #if defined(BCMPLATFORM_BUS)
128 #if defined(BCMLXSDMMC)
129 /* forward declarations */
130 int bcmsdh_probe(struct device
*dev
);
131 EXPORT_SYMBOL(bcmsdh_probe
);
133 int bcmsdh_remove(struct device
*dev
);
134 EXPORT_SYMBOL(bcmsdh_remove
);
137 /* forward declarations */
138 static int __devinit
bcmsdh_probe(struct device
*dev
);
139 static int __devexit
bcmsdh_remove(struct device
*dev
);
140 #endif /* BCMLXSDMMC */
143 static struct device_driver bcmsdh_driver
= {
144 .name
= "pxa2xx-mci",
145 .bus
= &platform_bus_type
,
146 .probe
= bcmsdh_probe
,
147 .remove
= bcmsdh_remove
,
151 #endif /* BCMLXSDMMC */
155 #endif /* BCMLXSDMMC */
156 int bcmsdh_probe(struct device
*dev
)
159 bcmsdh_hc_t
*sdhc
= NULL
;
160 unsigned long regs
= 0;
161 bcmsdh_info_t
*sdh
= NULL
;
162 #if !defined(BCMLXSDMMC) && defined(BCMPLATFORM_BUS)
163 struct platform_device
*pdev
;
165 #endif /* BCMLXSDMMC */
168 unsigned long irq_flags
= 0;
170 #if !defined(BCMLXSDMMC) && defined(BCMPLATFORM_BUS)
171 pdev
= to_platform_device(dev
);
172 r
= platform_get_resource(pdev
, IORESOURCE_MEM
, 0);
173 irq
= platform_get_irq(pdev
, 0);
174 if (!r
|| irq
== NO_IRQ
)
176 #endif /* BCMLXSDMMC */
178 #if defined(OOB_INTR_ONLY)
181 IORESOURCE_IRQ
| IORESOURCE_IRQ_HIGHLEVEL
|
182 IORESOURCE_IRQ_SHAREABLE
;
184 irq_flags
= IRQF_TRIGGER_FALLING
;
186 irq
= dhd_customer_oob_irq_map(&irq_flags
);
188 SDLX_MSG(("%s: Host irq is not defined\n", __func__
));
191 #endif /* defined(OOB_INTR_ONLY) */
192 /* allocate SDIO Host Controller state info */
193 osh
= osl_attach(dev
, PCI_BUS
, FALSE
);
195 SDLX_MSG(("%s: osl_attach failed\n", __func__
));
198 sdhc
= MALLOC(osh
, sizeof(bcmsdh_hc_t
));
200 SDLX_MSG(("%s: out of memory, allocated %d bytes\n",
201 __func__
, MALLOCED(osh
)));
204 bzero(sdhc
, sizeof(bcmsdh_hc_t
));
207 sdhc
->dev
= (void *)dev
;
210 sdh
= bcmsdh_attach(osh
, (void *)0, (void **)®s
, irq
);
212 SDLX_MSG(("%s: bcmsdh_attach failed\n", __func__
));
216 sdh
= bcmsdh_attach(osh
, (void *)r
->start
, (void **)®s
, irq
);
218 SDLX_MSG(("%s: bcmsdh_attach failed\n", __func__
));
221 #endif /* BCMLXSDMMC */
224 sdhc
->oob_flags
= irq_flags
;
225 sdhc
->oob_irq_registered
= FALSE
; /* to make sure.. */
226 #if defined(OOB_INTR_ONLY)
227 spin_lock_init(&sdhc
->irq_lock
);
230 /* chain SDIO Host Controller info together */
231 sdhc
->next
= sdhcinfo
;
233 /* Read the vendor/device ID from the CIS */
234 vendevid
= bcmsdh_query_device(sdh
);
236 /* try to attach to the target device */
237 sdhc
->ch
= drvinfo
.attach((vendevid
>> 16), (vendevid
& 0xFFFF),
238 0, 0, 0, 0, (void *)regs
, NULL
, sdh
);
240 SDLX_MSG(("%s: device attach failed\n", __func__
));
250 bcmsdh_detach(sdhc
->osh
, sdhc
->sdh
);
251 MFREE(osh
, sdhc
, sizeof(bcmsdh_hc_t
));
260 #endif /* BCMLXSDMMC */
261 int bcmsdh_remove(struct device
*dev
)
263 bcmsdh_hc_t
*sdhc
, *prev
;
267 drvinfo
.detach(sdhc
->ch
);
268 bcmsdh_detach(sdhc
->osh
, sdhc
->sdh
);
269 /* find the SDIO Host Controller state for this pdev
270 and take it out from the list */
271 for (sdhc
= sdhcinfo
, prev
= NULL
; sdhc
; sdhc
= sdhc
->next
) {
272 if (sdhc
->dev
== (void *)dev
) {
274 prev
->next
= sdhc
->next
;
282 SDLX_MSG(("%s: failed\n", __func__
));
286 /* release SDIO Host Controller info */
288 MFREE(osh
, sdhc
, sizeof(bcmsdh_hc_t
));
291 #if !defined(BCMLXSDMMC)
292 dev_set_drvdata(dev
, NULL
);
293 #endif /* !defined(BCMLXSDMMC) */
298 #else /* BCMPLATFORM_BUS */
300 #if !defined(BCMLXSDMMC)
301 /* forward declarations for PCI probe and remove functions. */
302 static int __devinit
bcmsdh_pci_probe(struct pci_dev
*pdev
,
303 const struct pci_device_id
*ent
);
304 static void __devexit
bcmsdh_pci_remove(struct pci_dev
*pdev
);
309 static struct pci_device_id bcmsdh_pci_devid
[] __devinitdata
= {
311 .vendor
= PCI_ANY_ID
,
312 .device
= PCI_ANY_ID
,
313 .subvendor
= PCI_ANY_ID
,
314 .subdevice
= PCI_ANY_ID
,
322 MODULE_DEVICE_TABLE(pci
, bcmsdh_pci_devid
);
325 * SDIO Host Controller pci driver info
327 static struct pci_driver bcmsdh_pci_driver
= {
330 .id_table
= bcmsdh_pci_devid
,
331 .probe
= bcmsdh_pci_probe
,
332 .remove
= bcmsdh_pci_remove
,
337 extern uint sd_pci_slot
; /* Force detection to a particular PCI */
338 /* slot only . Allows for having multiple */
339 /* WL devices at once in a PC */
340 /* Only one instance of dhd will be */
341 /* usable at a time */
342 /* Upper word is bus number, */
343 /* lower word is slot number */
344 /* Default value of 0xFFFFffff turns this */
346 module_param(sd_pci_slot
, uint
, 0);
349 * Detect supported SDIO Host Controller and attach if found.
351 * Determine if the device described by pdev is a supported SDIO Host
352 * Controller. If so, attach to it and attach to the target device.
355 bcmsdh_pci_probe(struct pci_dev
*pdev
, const struct pci_device_id
*ent
)
358 bcmsdh_hc_t
*sdhc
= NULL
;
360 bcmsdh_info_t
*sdh
= NULL
;
363 if (sd_pci_slot
!= 0xFFFFffff) {
364 if (pdev
->bus
->number
!= (sd_pci_slot
>> 16) ||
365 PCI_SLOT(pdev
->devfn
) != (sd_pci_slot
& 0xffff)) {
366 SDLX_MSG(("%s: %s: bus %X, slot %X, vend %X, dev %X\n",
368 bcmsdh_chipmatch(pdev
->vendor
, pdev
->device
) ?
369 "Found compatible SDIOHC" :
370 "Probing unknown device",
371 pdev
->bus
->number
, PCI_SLOT(pdev
->devfn
),
372 pdev
->vendor
, pdev
->device
));
375 SDLX_MSG(("%s: %s: bus %X, slot %X, vendor %X, device %X "
376 "(good PCI location)\n", __func__
,
377 bcmsdh_chipmatch(pdev
->vendor
, pdev
->device
) ?
378 "Using compatible SDIOHC" : "WARNING, forced use "
380 pdev
->bus
->number
, PCI_SLOT(pdev
->devfn
), pdev
->vendor
,
384 if ((pdev
->vendor
== VENDOR_TI
)
385 && ((pdev
->device
== PCIXX21_FLASHMEDIA_ID
)
386 || (pdev
->device
== PCIXX21_FLASHMEDIA0_ID
))) {
389 SDLX_MSG(("%s: Disabling TI FlashMedia Controller.\n",
391 osh
= osl_attach(pdev
, PCI_BUS
, FALSE
);
393 SDLX_MSG(("%s: osl_attach failed\n", __func__
));
397 config_reg
= OSL_PCI_READ_CONFIG(osh
, 0x4c, 4);
400 * Set MMC_SD_DIS bit in FlashMedia Controller.
401 * Disbling the SD/MMC Controller in the FlashMedia Controller
402 * allows the Standard SD Host Controller to take over control
406 OSL_PCI_WRITE_CONFIG(osh
, 0x4c, 4, config_reg
);
409 /* match this pci device with what we support */
410 /* we can't solely rely on this to believe it is
411 our SDIO Host Controller! */
412 if (!bcmsdh_chipmatch(pdev
->vendor
, pdev
->device
))
415 /* this is a pci device we might support */
416 SDLX_MSG(("%s: Found possible SDIO Host Controller: "
417 "bus %d slot %d func %d irq %d\n", __func__
,
418 pdev
->bus
->number
, PCI_SLOT(pdev
->devfn
),
419 PCI_FUNC(pdev
->devfn
), pdev
->irq
));
421 /* use bcmsdh_query_device() to get the vendor ID of the target device
422 * so it will eventually appear in the Broadcom string on the console
425 /* allocate SDIO Host Controller state info */
426 osh
= osl_attach(pdev
, PCI_BUS
, FALSE
);
428 SDLX_MSG(("%s: osl_attach failed\n", __func__
));
431 sdhc
= MALLOC(osh
, sizeof(bcmsdh_hc_t
));
433 SDLX_MSG(("%s: out of memory, allocated %d bytes\n",
434 __func__
, MALLOCED(osh
)));
437 bzero(sdhc
, sizeof(bcmsdh_hc_t
));
442 /* map to address where host can access */
443 pci_set_master(pdev
);
444 rc
= pci_enable_device(pdev
);
446 SDLX_MSG(("%s: Cannot enable PCI device\n", __func__
));
449 sdh
= bcmsdh_attach(osh
, (void *)(uintptr
) pci_resource_start(pdev
, 0),
450 (void **)®s
, pdev
->irq
);
452 SDLX_MSG(("%s: bcmsdh_attach failed\n", __func__
));
458 /* try to attach to the target device */
459 sdhc
->ch
= drvinfo
.attach(VENDOR_BROADCOM
, /* pdev->vendor, */
460 bcmsdh_query_device(sdh
) & 0xFFFF, 0, 0, 0, 0,
461 (void *)regs
, NULL
, sdh
);
463 SDLX_MSG(("%s: device attach failed\n", __func__
));
467 /* chain SDIO Host Controller info together */
468 sdhc
->next
= sdhcinfo
;
476 bcmsdh_detach(sdhc
->osh
, sdhc
->sdh
);
478 MFREE(osh
, sdhc
, sizeof(bcmsdh_hc_t
));
485 * Detach from target devices and SDIO Host Controller
487 static void __devexit
bcmsdh_pci_remove(struct pci_dev
*pdev
)
489 bcmsdh_hc_t
*sdhc
, *prev
;
492 /* find the SDIO Host Controller state for this
493 pdev and take it out from the list */
494 for (sdhc
= sdhcinfo
, prev
= NULL
; sdhc
; sdhc
= sdhc
->next
) {
495 if (sdhc
->dev
== pdev
) {
497 prev
->next
= sdhc
->next
;
507 drvinfo
.detach(sdhc
->ch
);
509 bcmsdh_detach(sdhc
->osh
, sdhc
->sdh
);
511 /* release SDIO Host Controller info */
513 MFREE(osh
, sdhc
, sizeof(bcmsdh_hc_t
));
516 #endif /* BCMLXSDMMC */
517 #endif /* BCMPLATFORM_BUS */
519 extern int sdio_function_init(void);
521 int bcmsdh_register(bcmsdh_driver_t
*driver
)
527 #if defined(BCMPLATFORM_BUS)
528 #if defined(BCMLXSDMMC)
529 SDLX_MSG(("Linux Kernel SDIO/MMC Driver\n"));
530 error
= sdio_function_init();
532 SDLX_MSG(("Intel PXA270 SDIO Driver\n"));
533 error
= driver_register(&bcmsdh_driver
);
534 #endif /* defined(BCMLXSDMMC) */
536 #endif /* defined(BCMPLATFORM_BUS) */
538 #if !defined(BCMPLATFORM_BUS) && !defined(BCMLXSDMMC)
539 error
= pci_register_driver(&bcmsdh_pci_driver
);
543 SDLX_MSG(("%s: pci_module_init failed 0x%x\n", __func__
, error
));
544 #endif /* BCMPLATFORM_BUS */
549 extern void sdio_function_cleanup(void);
551 void bcmsdh_unregister(void)
553 #if defined(BCMPLATFORM_BUS) && !defined(BCMLXSDMMC)
554 driver_unregister(&bcmsdh_driver
);
556 #if defined(BCMLXSDMMC)
557 sdio_function_cleanup();
558 #endif /* BCMLXSDMMC */
559 #if !defined(BCMPLATFORM_BUS) && !defined(BCMLXSDMMC)
560 pci_unregister_driver(&bcmsdh_pci_driver
);
561 #endif /* BCMPLATFORM_BUS */
564 #if defined(OOB_INTR_ONLY)
565 void bcmsdh_oob_intr_set(bool enable
)
567 static bool curstate
= 1;
570 spin_lock_irqsave(&sdhcinfo
->irq_lock
, flags
);
571 if (curstate
!= enable
) {
573 enable_irq(sdhcinfo
->oob_irq
);
575 disable_irq_nosync(sdhcinfo
->oob_irq
);
578 spin_unlock_irqrestore(&sdhcinfo
->irq_lock
, flags
);
581 static irqreturn_t
wlan_oob_irq(int irq
, void *dev_id
)
585 dhdp
= (dhd_pub_t
*) dev_get_drvdata(sdhcinfo
->dev
);
587 bcmsdh_oob_intr_set(0);
590 SDLX_MSG(("Out of band GPIO interrupt fired way too early\n"));
594 WAKE_LOCK_TIMEOUT(dhdp
, WAKE_LOCK_TMOUT
, 25);
596 dhdsdio_isr((void *)dhdp
->bus
);
601 int bcmsdh_register_oob_intr(void *dhdp
)
605 SDLX_MSG(("%s Enter\n", __func__
));
607 sdhcinfo
->oob_flags
=
608 IORESOURCE_IRQ
| IORESOURCE_IRQ_HIGHLEVEL
|
609 IORESOURCE_IRQ_SHAREABLE
;
610 dev_set_drvdata(sdhcinfo
->dev
, dhdp
);
612 if (!sdhcinfo
->oob_irq_registered
) {
613 SDLX_MSG(("%s IRQ=%d Type=%X\n", __func__
,
614 (int)sdhcinfo
->oob_irq
, (int)sdhcinfo
->oob_flags
));
615 /* Refer to customer Host IRQ docs about
616 proper irqflags definition */
618 request_irq(sdhcinfo
->oob_irq
, wlan_oob_irq
,
619 sdhcinfo
->oob_flags
, "bcmsdh_sdmmc", NULL
);
623 set_irq_wake(sdhcinfo
->oob_irq
, 1);
624 sdhcinfo
->oob_irq_registered
= TRUE
;
630 void bcmsdh_unregister_oob_intr(void)
632 SDLX_MSG(("%s: Enter\n", __func__
));
634 set_irq_wake(sdhcinfo
->oob_irq
, 0);
635 disable_irq(sdhcinfo
->oob_irq
); /* just in case.. */
636 free_irq(sdhcinfo
->oob_irq
, NULL
);
637 sdhcinfo
->oob_irq_registered
= FALSE
;
639 #endif /* defined(OOB_INTR_ONLY) */
640 /* Module parameters specific to each host-controller driver */
642 extern uint sd_msglevel
; /* Debug message level */
643 module_param(sd_msglevel
, uint
, 0);
645 extern uint sd_power
; /* 0 = SD Power OFF,
647 module_param(sd_power
, uint
, 0);
649 extern uint sd_clock
; /* SD Clock Control, 0 = SD Clock OFF,
651 module_param(sd_clock
, uint
, 0);
653 extern uint sd_divisor
; /* Divisor (-1 means external clock) */
654 module_param(sd_divisor
, uint
, 0);
656 extern uint sd_sdmode
; /* Default is SD4, 0=SPI, 1=SD1, 2=SD4 */
657 module_param(sd_sdmode
, uint
, 0);
659 extern uint sd_hiok
; /* Ok to use hi-speed mode */
660 module_param(sd_hiok
, uint
, 0);
662 extern uint sd_f2_blocksize
;
663 module_param(sd_f2_blocksize
, int, 0);