xref: /OK3568_Linux_fs/kernel/drivers/mmc/core/quirks.h (revision 4882a59341e53eb6f0b4789bf948001014eff981)
1*4882a593Smuzhiyun /* SPDX-License-Identifier: GPL-2.0 */
2*4882a593Smuzhiyun /*
3*4882a593Smuzhiyun  *  This file contains work-arounds for many known SD/MMC
4*4882a593Smuzhiyun  *  and SDIO hardware bugs.
5*4882a593Smuzhiyun  *
6*4882a593Smuzhiyun  *  Copyright (c) 2011 Andrei Warkentin <andreiw@motorola.com>
7*4882a593Smuzhiyun  *  Copyright (c) 2011 Pierre Tardy <tardyp@gmail.com>
8*4882a593Smuzhiyun  *  Inspired from pci fixup code:
9*4882a593Smuzhiyun  *  Copyright (c) 1999 Martin Mares <mj@ucw.cz>
10*4882a593Smuzhiyun  *
11*4882a593Smuzhiyun  */
12*4882a593Smuzhiyun 
13*4882a593Smuzhiyun #include <linux/mmc/sdio_ids.h>
14*4882a593Smuzhiyun 
15*4882a593Smuzhiyun #include "card.h"
16*4882a593Smuzhiyun 
17*4882a593Smuzhiyun static const struct mmc_fixup __maybe_unused mmc_blk_fixups[] = {
18*4882a593Smuzhiyun #define INAND_CMD38_ARG_EXT_CSD  113
19*4882a593Smuzhiyun #define INAND_CMD38_ARG_ERASE    0x00
20*4882a593Smuzhiyun #define INAND_CMD38_ARG_TRIM     0x01
21*4882a593Smuzhiyun #define INAND_CMD38_ARG_SECERASE 0x80
22*4882a593Smuzhiyun #define INAND_CMD38_ARG_SECTRIM1 0x81
23*4882a593Smuzhiyun #define INAND_CMD38_ARG_SECTRIM2 0x88
24*4882a593Smuzhiyun 	/* CMD38 argument is passed through EXT_CSD[113] */
25*4882a593Smuzhiyun 	MMC_FIXUP("SEM02G", CID_MANFID_SANDISK, 0x100, add_quirk,
26*4882a593Smuzhiyun 		  MMC_QUIRK_INAND_CMD38),
27*4882a593Smuzhiyun 	MMC_FIXUP("SEM04G", CID_MANFID_SANDISK, 0x100, add_quirk,
28*4882a593Smuzhiyun 		  MMC_QUIRK_INAND_CMD38),
29*4882a593Smuzhiyun 	MMC_FIXUP("SEM08G", CID_MANFID_SANDISK, 0x100, add_quirk,
30*4882a593Smuzhiyun 		  MMC_QUIRK_INAND_CMD38),
31*4882a593Smuzhiyun 	MMC_FIXUP("SEM16G", CID_MANFID_SANDISK, 0x100, add_quirk,
32*4882a593Smuzhiyun 		  MMC_QUIRK_INAND_CMD38),
33*4882a593Smuzhiyun 	MMC_FIXUP("SEM32G", CID_MANFID_SANDISK, 0x100, add_quirk,
34*4882a593Smuzhiyun 		  MMC_QUIRK_INAND_CMD38),
35*4882a593Smuzhiyun 
36*4882a593Smuzhiyun 	/*
37*4882a593Smuzhiyun 	 * Some MMC cards experience performance degradation with CMD23
38*4882a593Smuzhiyun 	 * instead of CMD12-bounded multiblock transfers. For now we'll
39*4882a593Smuzhiyun 	 * black list what's bad...
40*4882a593Smuzhiyun 	 * - Certain Toshiba cards.
41*4882a593Smuzhiyun 	 *
42*4882a593Smuzhiyun 	 * N.B. This doesn't affect SD cards.
43*4882a593Smuzhiyun 	 */
44*4882a593Smuzhiyun 	MMC_FIXUP("SDMB-32", CID_MANFID_SANDISK, CID_OEMID_ANY, add_quirk_mmc,
45*4882a593Smuzhiyun 		  MMC_QUIRK_BLK_NO_CMD23),
46*4882a593Smuzhiyun 	MMC_FIXUP("SDM032", CID_MANFID_SANDISK, CID_OEMID_ANY, add_quirk_mmc,
47*4882a593Smuzhiyun 		  MMC_QUIRK_BLK_NO_CMD23),
48*4882a593Smuzhiyun 	MMC_FIXUP("MMC08G", CID_MANFID_TOSHIBA, CID_OEMID_ANY, add_quirk_mmc,
49*4882a593Smuzhiyun 		  MMC_QUIRK_BLK_NO_CMD23),
50*4882a593Smuzhiyun 	MMC_FIXUP("MMC16G", CID_MANFID_TOSHIBA, CID_OEMID_ANY, add_quirk_mmc,
51*4882a593Smuzhiyun 		  MMC_QUIRK_BLK_NO_CMD23),
52*4882a593Smuzhiyun 	MMC_FIXUP("MMC32G", CID_MANFID_TOSHIBA, CID_OEMID_ANY, add_quirk_mmc,
53*4882a593Smuzhiyun 		  MMC_QUIRK_BLK_NO_CMD23),
54*4882a593Smuzhiyun 
55*4882a593Smuzhiyun 	/*
56*4882a593Smuzhiyun 	 * Some SD cards lockup while using CMD23 multiblock transfers.
57*4882a593Smuzhiyun 	 */
58*4882a593Smuzhiyun 	MMC_FIXUP("AF SD", CID_MANFID_ATP, CID_OEMID_ANY, add_quirk_sd,
59*4882a593Smuzhiyun 		  MMC_QUIRK_BLK_NO_CMD23),
60*4882a593Smuzhiyun 	MMC_FIXUP("APUSD", CID_MANFID_APACER, 0x5048, add_quirk_sd,
61*4882a593Smuzhiyun 		  MMC_QUIRK_BLK_NO_CMD23),
62*4882a593Smuzhiyun 	MMC_FIXUP(CID_NAME_ANY, CID_MANFID_DH, CID_OEMID_ANY, add_quirk_sd,
63*4882a593Smuzhiyun 		  MMC_QUIRK_BLK_NO_CMD23),
64*4882a593Smuzhiyun 
65*4882a593Smuzhiyun 	/*
66*4882a593Smuzhiyun 	 * Some MMC cards need longer data read timeout than indicated in CSD.
67*4882a593Smuzhiyun 	 */
68*4882a593Smuzhiyun 	MMC_FIXUP(CID_NAME_ANY, CID_MANFID_MICRON, 0x200, add_quirk_mmc,
69*4882a593Smuzhiyun 		  MMC_QUIRK_LONG_READ_TIME),
70*4882a593Smuzhiyun 	MMC_FIXUP("008GE0", CID_MANFID_TOSHIBA, CID_OEMID_ANY, add_quirk_mmc,
71*4882a593Smuzhiyun 		  MMC_QUIRK_LONG_READ_TIME),
72*4882a593Smuzhiyun 
73*4882a593Smuzhiyun 	/*
74*4882a593Smuzhiyun 	 * On these Samsung MoviNAND parts, performing secure erase or
75*4882a593Smuzhiyun 	 * secure trim can result in unrecoverable corruption due to a
76*4882a593Smuzhiyun 	 * firmware bug.
77*4882a593Smuzhiyun 	 */
78*4882a593Smuzhiyun 	MMC_FIXUP("M8G2FA", CID_MANFID_SAMSUNG, CID_OEMID_ANY, add_quirk_mmc,
79*4882a593Smuzhiyun 		  MMC_QUIRK_SEC_ERASE_TRIM_BROKEN),
80*4882a593Smuzhiyun 	MMC_FIXUP("MAG4FA", CID_MANFID_SAMSUNG, CID_OEMID_ANY, add_quirk_mmc,
81*4882a593Smuzhiyun 		  MMC_QUIRK_SEC_ERASE_TRIM_BROKEN),
82*4882a593Smuzhiyun 	MMC_FIXUP("MBG8FA", CID_MANFID_SAMSUNG, CID_OEMID_ANY, add_quirk_mmc,
83*4882a593Smuzhiyun 		  MMC_QUIRK_SEC_ERASE_TRIM_BROKEN),
84*4882a593Smuzhiyun 	MMC_FIXUP("MCGAFA", CID_MANFID_SAMSUNG, CID_OEMID_ANY, add_quirk_mmc,
85*4882a593Smuzhiyun 		  MMC_QUIRK_SEC_ERASE_TRIM_BROKEN),
86*4882a593Smuzhiyun 	MMC_FIXUP("VAL00M", CID_MANFID_SAMSUNG, CID_OEMID_ANY, add_quirk_mmc,
87*4882a593Smuzhiyun 		  MMC_QUIRK_SEC_ERASE_TRIM_BROKEN),
88*4882a593Smuzhiyun 	MMC_FIXUP("VYL00M", CID_MANFID_SAMSUNG, CID_OEMID_ANY, add_quirk_mmc,
89*4882a593Smuzhiyun 		  MMC_QUIRK_SEC_ERASE_TRIM_BROKEN),
90*4882a593Smuzhiyun 	MMC_FIXUP("KYL00M", CID_MANFID_SAMSUNG, CID_OEMID_ANY, add_quirk_mmc,
91*4882a593Smuzhiyun 		  MMC_QUIRK_SEC_ERASE_TRIM_BROKEN),
92*4882a593Smuzhiyun 	MMC_FIXUP("VZL00M", CID_MANFID_SAMSUNG, CID_OEMID_ANY, add_quirk_mmc,
93*4882a593Smuzhiyun 		  MMC_QUIRK_SEC_ERASE_TRIM_BROKEN),
94*4882a593Smuzhiyun 
95*4882a593Smuzhiyun 	/*
96*4882a593Smuzhiyun 	 *  On Some Kingston eMMCs, performing trim can result in
97*4882a593Smuzhiyun 	 *  unrecoverable data conrruption occasionally due to a firmware bug.
98*4882a593Smuzhiyun 	 */
99*4882a593Smuzhiyun 	MMC_FIXUP("V10008", CID_MANFID_KINGSTON, CID_OEMID_ANY, add_quirk_mmc,
100*4882a593Smuzhiyun 		  MMC_QUIRK_TRIM_BROKEN),
101*4882a593Smuzhiyun 	MMC_FIXUP("V10016", CID_MANFID_KINGSTON, CID_OEMID_ANY, add_quirk_mmc,
102*4882a593Smuzhiyun 		  MMC_QUIRK_TRIM_BROKEN),
103*4882a593Smuzhiyun 
104*4882a593Smuzhiyun 	/*
105*4882a593Smuzhiyun 	 * Some SD cards reports discard support while they don't
106*4882a593Smuzhiyun 	 */
107*4882a593Smuzhiyun 	MMC_FIXUP(CID_NAME_ANY, CID_MANFID_SANDISK_SD, 0x5344, add_quirk_sd,
108*4882a593Smuzhiyun 		  MMC_QUIRK_BROKEN_SD_DISCARD),
109*4882a593Smuzhiyun 
110*4882a593Smuzhiyun 	END_FIXUP
111*4882a593Smuzhiyun };
112*4882a593Smuzhiyun 
113*4882a593Smuzhiyun static const struct mmc_fixup __maybe_unused mmc_ext_csd_fixups[] = {
114*4882a593Smuzhiyun 	/*
115*4882a593Smuzhiyun 	 * Certain Hynix eMMC 4.41 cards might get broken when HPI feature
116*4882a593Smuzhiyun 	 * is used so disable the HPI feature for such buggy cards.
117*4882a593Smuzhiyun 	 */
118*4882a593Smuzhiyun 	MMC_FIXUP_EXT_CSD_REV(CID_NAME_ANY, CID_MANFID_HYNIX,
119*4882a593Smuzhiyun 			      0x014a, add_quirk, MMC_QUIRK_BROKEN_HPI, 5),
120*4882a593Smuzhiyun 	/*
121*4882a593Smuzhiyun 	 * Certain Micron (Numonyx) eMMC 4.5 cards might get broken when HPI
122*4882a593Smuzhiyun 	 * feature is used so disable the HPI feature for such buggy cards.
123*4882a593Smuzhiyun 	 */
124*4882a593Smuzhiyun 	MMC_FIXUP_EXT_CSD_REV(CID_NAME_ANY, CID_MANFID_NUMONYX,
125*4882a593Smuzhiyun 			      0x014e, add_quirk, MMC_QUIRK_BROKEN_HPI, 6),
126*4882a593Smuzhiyun 
127*4882a593Smuzhiyun 	END_FIXUP
128*4882a593Smuzhiyun };
129*4882a593Smuzhiyun 
130*4882a593Smuzhiyun 
131*4882a593Smuzhiyun static const struct mmc_fixup __maybe_unused sdio_fixup_methods[] = {
132*4882a593Smuzhiyun 	SDIO_FIXUP(SDIO_VENDOR_ID_TI_WL1251, SDIO_DEVICE_ID_TI_WL1251,
133*4882a593Smuzhiyun 		   add_quirk, MMC_QUIRK_NONSTD_FUNC_IF),
134*4882a593Smuzhiyun 
135*4882a593Smuzhiyun 	SDIO_FIXUP(SDIO_VENDOR_ID_TI_WL1251, SDIO_DEVICE_ID_TI_WL1251,
136*4882a593Smuzhiyun 		   add_quirk, MMC_QUIRK_DISABLE_CD),
137*4882a593Smuzhiyun 
138*4882a593Smuzhiyun 	SDIO_FIXUP(SDIO_VENDOR_ID_TI, SDIO_DEVICE_ID_TI_WL1271,
139*4882a593Smuzhiyun 		   add_quirk, MMC_QUIRK_NONSTD_FUNC_IF),
140*4882a593Smuzhiyun 
141*4882a593Smuzhiyun 	SDIO_FIXUP(SDIO_VENDOR_ID_TI, SDIO_DEVICE_ID_TI_WL1271,
142*4882a593Smuzhiyun 		   add_quirk, MMC_QUIRK_DISABLE_CD),
143*4882a593Smuzhiyun 
144*4882a593Smuzhiyun 	SDIO_FIXUP(SDIO_VENDOR_ID_STE, SDIO_DEVICE_ID_STE_CW1200,
145*4882a593Smuzhiyun 		   add_quirk, MMC_QUIRK_BROKEN_BYTE_MODE_512),
146*4882a593Smuzhiyun 
147*4882a593Smuzhiyun 	SDIO_FIXUP(SDIO_VENDOR_ID_MARVELL, SDIO_DEVICE_ID_MARVELL_8797_F0,
148*4882a593Smuzhiyun 		   add_quirk, MMC_QUIRK_BROKEN_IRQ_POLLING),
149*4882a593Smuzhiyun 
150*4882a593Smuzhiyun 	SDIO_FIXUP(SDIO_VENDOR_ID_MARVELL, SDIO_DEVICE_ID_MARVELL_8887_F0,
151*4882a593Smuzhiyun 		   add_limit_rate_quirk, 150000000),
152*4882a593Smuzhiyun 
153*4882a593Smuzhiyun 	END_FIXUP
154*4882a593Smuzhiyun };
155*4882a593Smuzhiyun 
mmc_fixup_device(struct mmc_card * card,const struct mmc_fixup * table)156*4882a593Smuzhiyun static inline void mmc_fixup_device(struct mmc_card *card,
157*4882a593Smuzhiyun 				    const struct mmc_fixup *table)
158*4882a593Smuzhiyun {
159*4882a593Smuzhiyun 	const struct mmc_fixup *f;
160*4882a593Smuzhiyun 	u64 rev = cid_rev_card(card);
161*4882a593Smuzhiyun 
162*4882a593Smuzhiyun 	for (f = table; f->vendor_fixup; f++) {
163*4882a593Smuzhiyun 		if ((f->manfid == CID_MANFID_ANY ||
164*4882a593Smuzhiyun 		     f->manfid == card->cid.manfid) &&
165*4882a593Smuzhiyun 		    (f->oemid == CID_OEMID_ANY ||
166*4882a593Smuzhiyun 		     f->oemid == card->cid.oemid) &&
167*4882a593Smuzhiyun 		    (f->name == CID_NAME_ANY ||
168*4882a593Smuzhiyun 		     !strncmp(f->name, card->cid.prod_name,
169*4882a593Smuzhiyun 			      sizeof(card->cid.prod_name))) &&
170*4882a593Smuzhiyun 		    (f->cis_vendor == card->cis.vendor ||
171*4882a593Smuzhiyun 		     f->cis_vendor == (u16) SDIO_ANY_ID) &&
172*4882a593Smuzhiyun 		    (f->cis_device == card->cis.device ||
173*4882a593Smuzhiyun 		     f->cis_device == (u16) SDIO_ANY_ID) &&
174*4882a593Smuzhiyun 		    (f->ext_csd_rev == EXT_CSD_REV_ANY ||
175*4882a593Smuzhiyun 		     f->ext_csd_rev == card->ext_csd.rev) &&
176*4882a593Smuzhiyun 		    rev >= f->rev_start && rev <= f->rev_end) {
177*4882a593Smuzhiyun 			dev_dbg(&card->dev, "calling %ps\n", f->vendor_fixup);
178*4882a593Smuzhiyun 			f->vendor_fixup(card, f->data);
179*4882a593Smuzhiyun 		}
180*4882a593Smuzhiyun 	}
181*4882a593Smuzhiyun }
182