11
12
#include <linux/types.h>
12
13
#include <linux/kernel.h>
13
14
#include <linux/mmc/card.h>
14
#include <linux/mod_devicetable.h>
17
* The world is not perfect and supplies us with broken mmc/sdio devices.
18
* For at least a part of these bugs we need a work-around
22
u16 vendor, device; /* You can use SDIO_ANY_ID here of course */
23
void (*vendor_fixup)(struct mmc_card *card, int data);
28
* This hook just adds a quirk unconditionnally
30
static void __maybe_unused add_quirk(struct mmc_card *card, int data)
36
* This hook just removes a quirk unconditionnally
38
static void __maybe_unused remove_quirk(struct mmc_card *card, int data)
40
card->quirks &= ~data;
16
#ifndef SDIO_VENDOR_ID_TI
17
#define SDIO_VENDOR_ID_TI 0x0097
20
#ifndef SDIO_DEVICE_ID_TI_WL1271
21
#define SDIO_DEVICE_ID_TI_WL1271 0x4076
44
25
* This hook just adds a quirk for all sdio devices
49
30
card->quirks |= data;
52
#ifndef SDIO_VENDOR_ID_TI
53
#define SDIO_VENDOR_ID_TI 0x0097
56
#ifndef SDIO_DEVICE_ID_TI_WL1271
57
#define SDIO_DEVICE_ID_TI_WL1271 0x4076
60
33
static const struct mmc_fixup mmc_fixup_methods[] = {
61
34
/* by default sdio devices are considered CLK_GATING broken */
62
35
/* good cards will be whitelisted as they are tested */
63
{ SDIO_ANY_ID, SDIO_ANY_ID,
64
add_quirk_for_sdio_devices, MMC_QUIRK_BROKEN_CLK_GATING },
65
{ SDIO_VENDOR_ID_TI, SDIO_DEVICE_ID_TI_WL1271,
66
remove_quirk, MMC_QUIRK_BROKEN_CLK_GATING },
36
SDIO_FIXUP(SDIO_ANY_ID, SDIO_ANY_ID,
37
add_quirk_for_sdio_devices,
38
MMC_QUIRK_BROKEN_CLK_GATING),
40
SDIO_FIXUP(SDIO_VENDOR_ID_TI, SDIO_DEVICE_ID_TI_WL1271,
41
remove_quirk, MMC_QUIRK_BROKEN_CLK_GATING),
43
SDIO_FIXUP(SDIO_VENDOR_ID_TI, SDIO_DEVICE_ID_TI_WL1271,
44
add_quirk, MMC_QUIRK_NONSTD_FUNC_IF),
46
SDIO_FIXUP(SDIO_VENDOR_ID_TI, SDIO_DEVICE_ID_TI_WL1271,
47
add_quirk, MMC_QUIRK_DISABLE_CD),
70
void mmc_fixup_device(struct mmc_card *card)
52
void mmc_fixup_device(struct mmc_card *card, const struct mmc_fixup *table)
72
54
const struct mmc_fixup *f;
74
for (f = mmc_fixup_methods; f->vendor_fixup; f++) {
75
if ((f->vendor == card->cis.vendor
76
|| f->vendor == (u16) SDIO_ANY_ID) &&
77
(f->device == card->cis.device
78
|| f->device == (u16) SDIO_ANY_ID)) {
55
u64 rev = cid_rev_card(card);
57
/* Non-core specific workarounds. */
59
table = mmc_fixup_methods;
61
for (f = table; f->vendor_fixup; f++) {
62
if ((f->manfid == CID_MANFID_ANY ||
63
f->manfid == card->cid.manfid) &&
64
(f->oemid == CID_OEMID_ANY ||
65
f->oemid == card->cid.oemid) &&
66
(f->name == CID_NAME_ANY ||
67
!strncmp(f->name, card->cid.prod_name,
68
sizeof(card->cid.prod_name))) &&
69
(f->cis_vendor == card->cis.vendor ||
70
f->cis_vendor == (u16) SDIO_ANY_ID) &&
71
(f->cis_device == card->cis.device ||
72
f->cis_device == (u16) SDIO_ANY_ID) &&
73
rev >= f->rev_start && rev <= f->rev_end) {
79
74
dev_dbg(&card->dev, "calling %pF\n", f->vendor_fixup);
80
75
f->vendor_fixup(card, f->data);