| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 1 | /* | 
 | 2 |  *  linux/include/linux/mmc/card.h | 
 | 3 |  * | 
 | 4 |  * This program is free software; you can redistribute it and/or modify | 
 | 5 |  * it under the terms of the GNU General Public License version 2 as | 
 | 6 |  * published by the Free Software Foundation. | 
 | 7 |  * | 
 | 8 |  *  Card driver specific definitions. | 
 | 9 |  */ | 
 | 10 | #ifndef LINUX_MMC_CARD_H | 
 | 11 | #define LINUX_MMC_CARD_H | 
 | 12 |  | 
| Pierre Ossman | aaac1b4 | 2007-02-28 15:33:10 +0100 | [diff] [blame] | 13 | #include <linux/mmc/core.h> | 
| Andrei Warkentin | 32780cd | 2011-04-11 17:02:15 -0500 | [diff] [blame] | 14 | #include <linux/mod_devicetable.h> | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 15 |  | 
 | 16 | struct mmc_cid { | 
 | 17 | 	unsigned int		manfid; | 
 | 18 | 	char			prod_name[8]; | 
 | 19 | 	unsigned int		serial; | 
 | 20 | 	unsigned short		oemid; | 
 | 21 | 	unsigned short		year; | 
 | 22 | 	unsigned char		hwrev; | 
 | 23 | 	unsigned char		fwrev; | 
 | 24 | 	unsigned char		month; | 
 | 25 | }; | 
 | 26 |  | 
 | 27 | struct mmc_csd { | 
| Kyungmin Park | 6da24b7 | 2010-08-10 18:01:36 -0700 | [diff] [blame] | 28 | 	unsigned char		structure; | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 29 | 	unsigned char		mmca_vsn; | 
 | 30 | 	unsigned short		cmdclass; | 
 | 31 | 	unsigned short		tacc_clks; | 
 | 32 | 	unsigned int		tacc_ns; | 
| Arindam Nath | 3a30351 | 2011-05-05 12:19:03 +0530 | [diff] [blame] | 33 | 	unsigned int		c_size; | 
| Russell King | 37be4e7 | 2006-05-02 17:24:59 +0100 | [diff] [blame] | 34 | 	unsigned int		r2w_factor; | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 35 | 	unsigned int		max_dtr; | 
| Adrian Hunter | dfe86cb | 2010-08-11 14:17:46 -0700 | [diff] [blame] | 36 | 	unsigned int		erase_size;		/* In sectors */ | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 37 | 	unsigned int		read_blkbits; | 
| Russell King | a6f6c96 | 2006-01-03 22:38:44 +0000 | [diff] [blame] | 38 | 	unsigned int		write_blkbits; | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 39 | 	unsigned int		capacity; | 
| Russell King | a6f6c96 | 2006-01-03 22:38:44 +0000 | [diff] [blame] | 40 | 	unsigned int		read_partial:1, | 
 | 41 | 				read_misalign:1, | 
| Russell King | ce11a16 | 2006-01-04 12:40:39 +0000 | [diff] [blame] | 42 | 				write_partial:1, | 
| Russell King | a6f6c96 | 2006-01-03 22:38:44 +0000 | [diff] [blame] | 43 | 				write_misalign:1; | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 44 | }; | 
 | 45 |  | 
| Philip Langdale | bce40a3 | 2006-10-21 12:35:02 +0200 | [diff] [blame] | 46 | struct mmc_ext_csd { | 
| Jarkko Lavinen | b1ebe38 | 2009-09-22 16:44:34 -0700 | [diff] [blame] | 47 | 	u8			rev; | 
| Adrian Hunter | dfe86cb | 2010-08-11 14:17:46 -0700 | [diff] [blame] | 48 | 	u8			erase_group_def; | 
 | 49 | 	u8			sec_feature_support; | 
| Andrei Warkentin | f4c5522 | 2011-03-31 18:40:00 -0500 | [diff] [blame] | 50 | 	u8			rel_sectors; | 
 | 51 | 	u8			rel_param; | 
| Andrei Warkentin | 371a689 | 2011-04-11 18:10:25 -0500 | [diff] [blame] | 52 | 	u8			part_config; | 
 | 53 | 	unsigned int		part_time;		/* Units: ms */ | 
| Jarkko Lavinen | b1ebe38 | 2009-09-22 16:44:34 -0700 | [diff] [blame] | 54 | 	unsigned int		sa_timeout;		/* Units: 100ns */ | 
| Philip Langdale | bce40a3 | 2006-10-21 12:35:02 +0200 | [diff] [blame] | 55 | 	unsigned int		hs_max_dtr; | 
| Pierre Ossman | 85a18ad | 2007-02-17 22:15:27 +0100 | [diff] [blame] | 56 | 	unsigned int		sectors; | 
| Hanumath Prasad | dfc13e8 | 2010-09-30 17:37:23 -0400 | [diff] [blame] | 57 | 	unsigned int		card_type; | 
| Adrian Hunter | dfe86cb | 2010-08-11 14:17:46 -0700 | [diff] [blame] | 58 | 	unsigned int		hc_erase_size;		/* In sectors */ | 
 | 59 | 	unsigned int		hc_erase_timeout;	/* In milliseconds */ | 
 | 60 | 	unsigned int		sec_trim_mult;	/* Secure trim multiplier  */ | 
 | 61 | 	unsigned int		sec_erase_mult;	/* Secure erase multiplier */ | 
 | 62 | 	unsigned int		trim_timeout;		/* In milliseconds */ | 
| Chuanxiao Dong | 709de99 | 2011-01-22 04:09:41 +0800 | [diff] [blame] | 63 | 	bool			enhanced_area_en;	/* enable bit */ | 
 | 64 | 	unsigned long long	enhanced_area_offset;	/* Units: Byte */ | 
 | 65 | 	unsigned int		enhanced_area_size;	/* Units: KB */ | 
| Andrei Warkentin | 371a689 | 2011-04-11 18:10:25 -0500 | [diff] [blame] | 66 | 	unsigned int		boot_size;		/* in bytes */ | 
| Philip Rakity | f39b2dd | 2011-07-07 09:04:55 -0700 | [diff] [blame] | 67 | 	u8			raw_partition_support;	/* 160 */ | 
 | 68 | 	u8			raw_erased_mem_count;	/* 181 */ | 
 | 69 | 	u8			raw_ext_csd_structure;	/* 194 */ | 
 | 70 | 	u8			raw_card_type;		/* 196 */ | 
 | 71 | 	u8			raw_s_a_timeout;		/* 217 */ | 
 | 72 | 	u8			raw_hc_erase_gap_size;	/* 221 */ | 
 | 73 | 	u8			raw_erase_timeout_mult;	/* 223 */ | 
 | 74 | 	u8			raw_hc_erase_grp_size;	/* 224 */ | 
 | 75 | 	u8			raw_sec_trim_mult;	/* 229 */ | 
 | 76 | 	u8			raw_sec_erase_mult;	/* 230 */ | 
 | 77 | 	u8			raw_sec_feature_support;/* 231 */ | 
 | 78 | 	u8			raw_trim_mult;		/* 232 */ | 
 | 79 | 	u8			raw_sectors[4];		/* 212 - 4 bytes */ | 
| Philip Langdale | bce40a3 | 2006-10-21 12:35:02 +0200 | [diff] [blame] | 80 | }; | 
 | 81 |  | 
| Pierre Ossman | b57c43a | 2005-09-06 15:18:53 -0700 | [diff] [blame] | 82 | struct sd_scr { | 
 | 83 | 	unsigned char		sda_vsn; | 
| Arindam Nath | 013909c | 2011-05-05 12:18:58 +0530 | [diff] [blame] | 84 | 	unsigned char		sda_spec3; | 
| Pierre Ossman | b57c43a | 2005-09-06 15:18:53 -0700 | [diff] [blame] | 85 | 	unsigned char		bus_widths; | 
 | 86 | #define SD_SCR_BUS_WIDTH_1	(1<<0) | 
 | 87 | #define SD_SCR_BUS_WIDTH_4	(1<<2) | 
| Andrei Warkentin | f0d8997 | 2011-05-23 15:06:38 -0500 | [diff] [blame] | 88 | 	unsigned char		cmds; | 
 | 89 | #define SD_SCR_CMD20_SUPPORT   (1<<0) | 
 | 90 | #define SD_SCR_CMD23_SUPPORT   (1<<1) | 
| Pierre Ossman | b57c43a | 2005-09-06 15:18:53 -0700 | [diff] [blame] | 91 | }; | 
 | 92 |  | 
| Adrian Hunter | dfe86cb | 2010-08-11 14:17:46 -0700 | [diff] [blame] | 93 | struct sd_ssr { | 
 | 94 | 	unsigned int		au;			/* In sectors */ | 
 | 95 | 	unsigned int		erase_timeout;		/* In milliseconds */ | 
 | 96 | 	unsigned int		erase_offset;		/* In milliseconds */ | 
 | 97 | }; | 
 | 98 |  | 
| Pierre Ossman | 7ccd266 | 2006-11-08 23:03:10 +0100 | [diff] [blame] | 99 | struct sd_switch_caps { | 
 | 100 | 	unsigned int		hs_max_dtr; | 
| Arindam Nath | 49c468f | 2011-05-05 12:19:01 +0530 | [diff] [blame] | 101 | 	unsigned int		uhs_max_dtr; | 
 | 102 | #define UHS_SDR104_MAX_DTR	208000000 | 
 | 103 | #define UHS_SDR50_MAX_DTR	100000000 | 
 | 104 | #define UHS_DDR50_MAX_DTR	50000000 | 
 | 105 | #define UHS_SDR25_MAX_DTR	UHS_DDR50_MAX_DTR | 
 | 106 | #define UHS_SDR12_MAX_DTR	25000000 | 
| Arindam Nath | 013909c | 2011-05-05 12:18:58 +0530 | [diff] [blame] | 107 | 	unsigned int		sd3_bus_mode; | 
| Arindam Nath | 49c468f | 2011-05-05 12:19:01 +0530 | [diff] [blame] | 108 | #define UHS_SDR12_BUS_SPEED	0 | 
 | 109 | #define UHS_SDR25_BUS_SPEED	1 | 
 | 110 | #define UHS_SDR50_BUS_SPEED	2 | 
 | 111 | #define UHS_SDR104_BUS_SPEED	3 | 
 | 112 | #define UHS_DDR50_BUS_SPEED	4 | 
 | 113 |  | 
 | 114 | #define SD_MODE_UHS_SDR12	(1 << UHS_SDR12_BUS_SPEED) | 
 | 115 | #define SD_MODE_UHS_SDR25	(1 << UHS_SDR25_BUS_SPEED) | 
 | 116 | #define SD_MODE_UHS_SDR50	(1 << UHS_SDR50_BUS_SPEED) | 
 | 117 | #define SD_MODE_UHS_SDR104	(1 << UHS_SDR104_BUS_SPEED) | 
 | 118 | #define SD_MODE_UHS_DDR50	(1 << UHS_DDR50_BUS_SPEED) | 
| Arindam Nath | 013909c | 2011-05-05 12:18:58 +0530 | [diff] [blame] | 119 | 	unsigned int		sd3_drv_type; | 
| Arindam Nath | d6d50a1 | 2011-05-05 12:18:59 +0530 | [diff] [blame] | 120 | #define SD_DRIVER_TYPE_B	0x01 | 
 | 121 | #define SD_DRIVER_TYPE_A	0x02 | 
 | 122 | #define SD_DRIVER_TYPE_C	0x04 | 
 | 123 | #define SD_DRIVER_TYPE_D	0x08 | 
| Arindam Nath | 013909c | 2011-05-05 12:18:58 +0530 | [diff] [blame] | 124 | 	unsigned int		sd3_curr_limit; | 
| Arindam Nath | 5371c92 | 2011-05-05 12:19:02 +0530 | [diff] [blame] | 125 | #define SD_SET_CURRENT_LIMIT_200	0 | 
 | 126 | #define SD_SET_CURRENT_LIMIT_400	1 | 
 | 127 | #define SD_SET_CURRENT_LIMIT_600	2 | 
 | 128 | #define SD_SET_CURRENT_LIMIT_800	3 | 
 | 129 |  | 
 | 130 | #define SD_MAX_CURRENT_200	(1 << SD_SET_CURRENT_LIMIT_200) | 
 | 131 | #define SD_MAX_CURRENT_400	(1 << SD_SET_CURRENT_LIMIT_400) | 
 | 132 | #define SD_MAX_CURRENT_600	(1 << SD_SET_CURRENT_LIMIT_600) | 
 | 133 | #define SD_MAX_CURRENT_800	(1 << SD_SET_CURRENT_LIMIT_800) | 
| Pierre Ossman | 7ccd266 | 2006-11-08 23:03:10 +0100 | [diff] [blame] | 134 | }; | 
 | 135 |  | 
| Pierre Ossman | 35c66c1 | 2007-06-11 20:25:43 +0200 | [diff] [blame] | 136 | struct sdio_cccr { | 
 | 137 | 	unsigned int		sdio_vsn; | 
 | 138 | 	unsigned int		sd_vsn; | 
 | 139 | 	unsigned int		multi_block:1, | 
 | 140 | 				low_speed:1, | 
 | 141 | 				wide_bus:1, | 
 | 142 | 				high_power:1, | 
| Ohad Ben-Cohen | 006ebd5 | 2009-09-22 16:45:07 -0700 | [diff] [blame] | 143 | 				high_speed:1, | 
 | 144 | 				disable_cd:1; | 
| Pierre Ossman | 35c66c1 | 2007-06-11 20:25:43 +0200 | [diff] [blame] | 145 | }; | 
 | 146 |  | 
| Pierre Ossman | 1a632f8 | 2007-07-30 15:15:30 +0200 | [diff] [blame] | 147 | struct sdio_cis { | 
 | 148 | 	unsigned short		vendor; | 
 | 149 | 	unsigned short		device; | 
 | 150 | 	unsigned short		blksize; | 
 | 151 | 	unsigned int		max_dtr; | 
 | 152 | }; | 
 | 153 |  | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 154 | struct mmc_host; | 
| Pierre Ossman | e29a7d7 | 2007-05-26 13:48:18 +0200 | [diff] [blame] | 155 | struct sdio_func; | 
| Pierre Ossman | 1a632f8 | 2007-07-30 15:15:30 +0200 | [diff] [blame] | 156 | struct sdio_func_tuple; | 
| Pierre Ossman | e29a7d7 | 2007-05-26 13:48:18 +0200 | [diff] [blame] | 157 |  | 
 | 158 | #define SDIO_MAX_FUNCS		7 | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 159 |  | 
 | 160 | /* | 
 | 161 |  * MMC device | 
 | 162 |  */ | 
 | 163 | struct mmc_card { | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 164 | 	struct mmc_host		*host;		/* the host this device belongs to */ | 
 | 165 | 	struct device		dev;		/* the device */ | 
 | 166 | 	unsigned int		rca;		/* relative card address of device */ | 
| Pierre Ossman | 9c2c0af | 2006-12-26 15:25:58 +0100 | [diff] [blame] | 167 | 	unsigned int		type;		/* card type */ | 
 | 168 | #define MMC_TYPE_MMC		0		/* MMC card */ | 
 | 169 | #define MMC_TYPE_SD		1		/* SD card */ | 
| Pierre Ossman | 5c4e6f1 | 2007-05-21 20:23:20 +0200 | [diff] [blame] | 170 | #define MMC_TYPE_SDIO		2		/* SDIO card */ | 
| Michal Miroslaw | 7310ece | 2010-08-10 18:01:40 -0700 | [diff] [blame] | 171 | #define MMC_TYPE_SD_COMBO	3		/* SD combo (IO+mem) card */ | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 172 | 	unsigned int		state;		/* (our) card state */ | 
 | 173 | #define MMC_STATE_PRESENT	(1<<0)		/* present in sysfs */ | 
| Pierre Ossman | bd76631 | 2007-05-01 16:11:57 +0200 | [diff] [blame] | 174 | #define MMC_STATE_READONLY	(1<<1)		/* card is read-only */ | 
 | 175 | #define MMC_STATE_HIGHSPEED	(1<<2)		/* card is in high speed mode */ | 
 | 176 | #define MMC_STATE_BLOCKADDR	(1<<3)		/* card uses block-addressing */ | 
| Hanumath Prasad | dfc13e8 | 2010-09-30 17:37:23 -0400 | [diff] [blame] | 177 | #define MMC_STATE_HIGHSPEED_DDR (1<<4)		/* card is in high speed mode */ | 
| Arindam Nath | 3a30351 | 2011-05-05 12:19:03 +0530 | [diff] [blame] | 178 | #define MMC_STATE_ULTRAHIGHSPEED (1<<5)		/* card is in ultra high speed mode */ | 
 | 179 | #define MMC_CARD_SDXC		(1<<6)		/* card is SDXC */ | 
| Ohad Ben-Cohen | 7c979ec | 2009-09-22 16:45:18 -0700 | [diff] [blame] | 180 | 	unsigned int		quirks; 	/* card quirks */ | 
 | 181 | #define MMC_QUIRK_LENIENT_FN0	(1<<0)		/* allow SDIO FN0 writes outside of the VS CCCR range */ | 
| Bing Zhao | 3fb7fb4 | 2010-03-05 13:43:25 -0800 | [diff] [blame] | 182 | #define MMC_QUIRK_BLKSZ_FOR_BYTE_MODE (1<<1)	/* use func->cur_blksize */ | 
 | 183 | 						/* for byte mode */ | 
| Grazvydas Ignotas | 6f51be3 | 2010-08-10 18:01:50 -0700 | [diff] [blame] | 184 | #define MMC_QUIRK_NONSTD_SDIO	(1<<2)		/* non-standard SDIO card attached */ | 
 | 185 | 						/* (missing CIA registers) */ | 
| Pierre Tardy | db99350 | 2011-02-06 19:03:47 +0100 | [diff] [blame] | 186 | #define MMC_QUIRK_BROKEN_CLK_GATING (1<<3)	/* clock gating the sdio bus will make card fail */ | 
| Ohad Ben-Cohen | eab4068 | 2011-04-05 17:50:14 +0300 | [diff] [blame] | 187 | #define MMC_QUIRK_NONSTD_FUNC_IF (1<<4)		/* SDIO card has nonstd function interfaces */ | 
| Ohad Ben-Cohen | 2059a02 | 2011-04-05 18:02:25 +0300 | [diff] [blame] | 188 | #define MMC_QUIRK_DISABLE_CD	(1<<5)		/* disconnect CD/DAT[3] resistor */ | 
| Andrei Warkentin | 6a7a6b4 | 2011-04-12 15:06:53 -0500 | [diff] [blame] | 189 | #define MMC_QUIRK_INAND_CMD38	(1<<6)		/* iNAND devices have broken CMD38 */ | 
| Andrei Warkentin | d0c97cf | 2011-05-23 15:06:36 -0500 | [diff] [blame] | 190 | #define MMC_QUIRK_BLK_NO_CMD23	(1<<7)		/* Avoid CMD23 for regular multiblock */ | 
| Pierre Ossman | e29a7d7 | 2007-05-26 13:48:18 +0200 | [diff] [blame] | 191 |  | 
| Adrian Hunter | dfe86cb | 2010-08-11 14:17:46 -0700 | [diff] [blame] | 192 | 	unsigned int		erase_size;	/* erase size in sectors */ | 
 | 193 |  	unsigned int		erase_shift;	/* if erase unit is power 2 */ | 
 | 194 |  	unsigned int		pref_erase;	/* in sectors */ | 
 | 195 |  	u8			erased_byte;	/* value of erased bytes */ | 
 | 196 |  | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 197 | 	u32			raw_cid[4];	/* raw card CID */ | 
 | 198 | 	u32			raw_csd[4];	/* raw card CSD */ | 
| Pierre Ossman | b57c43a | 2005-09-06 15:18:53 -0700 | [diff] [blame] | 199 | 	u32			raw_scr[2];	/* raw card SCR */ | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 200 | 	struct mmc_cid		cid;		/* card identification */ | 
 | 201 | 	struct mmc_csd		csd;		/* card specific */ | 
| Philip Langdale | bce40a3 | 2006-10-21 12:35:02 +0200 | [diff] [blame] | 202 | 	struct mmc_ext_csd	ext_csd;	/* mmc v4 extended card specific */ | 
| Pierre Ossman | b57c43a | 2005-09-06 15:18:53 -0700 | [diff] [blame] | 203 | 	struct sd_scr		scr;		/* extra SD information */ | 
| Adrian Hunter | dfe86cb | 2010-08-11 14:17:46 -0700 | [diff] [blame] | 204 | 	struct sd_ssr		ssr;		/* yet more SD information */ | 
| Pierre Ossman | 7ccd266 | 2006-11-08 23:03:10 +0100 | [diff] [blame] | 205 | 	struct sd_switch_caps	sw_caps;	/* switch (CMD6) caps */ | 
| Pierre Ossman | e29a7d7 | 2007-05-26 13:48:18 +0200 | [diff] [blame] | 206 |  | 
 | 207 | 	unsigned int		sdio_funcs;	/* number of SDIO functions */ | 
| Pierre Ossman | 35c66c1 | 2007-06-11 20:25:43 +0200 | [diff] [blame] | 208 | 	struct sdio_cccr	cccr;		/* common card info */ | 
| Pierre Ossman | 1a632f8 | 2007-07-30 15:15:30 +0200 | [diff] [blame] | 209 | 	struct sdio_cis		cis;		/* common tuple info */ | 
| Pierre Ossman | e29a7d7 | 2007-05-26 13:48:18 +0200 | [diff] [blame] | 210 | 	struct sdio_func	*sdio_func[SDIO_MAX_FUNCS]; /* SDIO functions (devices) */ | 
| Stefan Nilsson XK | 06e8935 | 2011-05-11 17:48:05 +0200 | [diff] [blame] | 211 | 	struct sdio_func	*sdio_single_irq; /* SDIO function when only one IRQ active */ | 
| Pierre Ossman | 759bdc7 | 2007-09-19 18:42:16 +0200 | [diff] [blame] | 212 | 	unsigned		num_info;	/* number of info strings */ | 
 | 213 | 	const char		**info;		/* info strings */ | 
| Pierre Ossman | 1a632f8 | 2007-07-30 15:15:30 +0200 | [diff] [blame] | 214 | 	struct sdio_func_tuple	*tuples;	/* unknown common tuples */ | 
| Haavard Skinnemoen | f4b7f92 | 2008-07-24 14:18:58 +0200 | [diff] [blame] | 215 |  | 
| Arindam Nath | 49c468f | 2011-05-05 12:19:01 +0530 | [diff] [blame] | 216 | 	unsigned int		sd_bus_speed;	/* Bus Speed Mode set for the card */ | 
 | 217 |  | 
| Haavard Skinnemoen | f4b7f92 | 2008-07-24 14:18:58 +0200 | [diff] [blame] | 218 | 	struct dentry		*debugfs_root; | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 219 | }; | 
 | 220 |  | 
| Andrei Warkentin | 32780cd | 2011-04-11 17:02:15 -0500 | [diff] [blame] | 221 | /* | 
 | 222 |  *  The world is not perfect and supplies us with broken mmc/sdio devices. | 
 | 223 |  *  For at least some of these bugs we need a work-around. | 
 | 224 |  */ | 
 | 225 |  | 
 | 226 | struct mmc_fixup { | 
 | 227 | 	/* CID-specific fields. */ | 
 | 228 | 	const char *name; | 
 | 229 |  | 
 | 230 | 	/* Valid revision range */ | 
 | 231 | 	u64 rev_start, rev_end; | 
 | 232 |  | 
 | 233 | 	unsigned int manfid; | 
 | 234 | 	unsigned short oemid; | 
 | 235 |  | 
 | 236 | 	/* SDIO-specfic fields. You can use SDIO_ANY_ID here of course */ | 
 | 237 | 	u16 cis_vendor, cis_device; | 
 | 238 |  | 
 | 239 | 	void (*vendor_fixup)(struct mmc_card *card, int data); | 
 | 240 | 	int data; | 
 | 241 | }; | 
 | 242 |  | 
| Randy Dunlap | 853c6ca | 2011-04-12 12:59:09 -0400 | [diff] [blame] | 243 | #define CID_MANFID_ANY (-1u) | 
| Andrei Warkentin | 32780cd | 2011-04-11 17:02:15 -0500 | [diff] [blame] | 244 | #define CID_OEMID_ANY ((unsigned short) -1) | 
 | 245 | #define CID_NAME_ANY (NULL) | 
 | 246 |  | 
 | 247 | #define END_FIXUP { 0 } | 
 | 248 |  | 
 | 249 | #define _FIXUP_EXT(_name, _manfid, _oemid, _rev_start, _rev_end,	\ | 
 | 250 | 		   _cis_vendor, _cis_device,				\ | 
 | 251 | 		   _fixup, _data)					\ | 
 | 252 | 	{						   \ | 
 | 253 | 		.name = (_name),			   \ | 
 | 254 | 		.manfid = (_manfid),			   \ | 
 | 255 | 		.oemid = (_oemid),			   \ | 
 | 256 | 		.rev_start = (_rev_start),		   \ | 
 | 257 | 		.rev_end = (_rev_end),			   \ | 
 | 258 | 		.cis_vendor = (_cis_vendor),		   \ | 
 | 259 | 		.cis_device = (_cis_device),		   \ | 
 | 260 | 		.vendor_fixup = (_fixup),		   \ | 
 | 261 | 		.data = (_data),			   \ | 
 | 262 | 	 } | 
 | 263 |  | 
 | 264 | #define MMC_FIXUP_REV(_name, _manfid, _oemid, _rev_start, _rev_end,	\ | 
 | 265 | 		      _fixup, _data)					\ | 
 | 266 | 	_FIXUP_EXT(_name, _manfid,					\ | 
 | 267 | 		   _oemid, _rev_start, _rev_end,			\ | 
 | 268 | 		   SDIO_ANY_ID, SDIO_ANY_ID,				\ | 
 | 269 | 		   _fixup, _data)					\ | 
 | 270 |  | 
 | 271 | #define MMC_FIXUP(_name, _manfid, _oemid, _fixup, _data) \ | 
 | 272 | 	MMC_FIXUP_REV(_name, _manfid, _oemid, 0, -1ull, _fixup, _data) | 
 | 273 |  | 
 | 274 | #define SDIO_FIXUP(_vendor, _device, _fixup, _data)			\ | 
 | 275 | 	_FIXUP_EXT(CID_NAME_ANY, CID_MANFID_ANY,			\ | 
 | 276 | 		    CID_OEMID_ANY, 0, -1ull,				\ | 
 | 277 | 		   _vendor, _device,					\ | 
 | 278 | 		   _fixup, _data)					\ | 
 | 279 |  | 
 | 280 | #define cid_rev(hwrev, fwrev, year, month)	\ | 
 | 281 | 	(((u64) hwrev) << 40 |                  \ | 
 | 282 | 	 ((u64) fwrev) << 32 |                  \ | 
 | 283 | 	 ((u64) year) << 16 |                   \ | 
 | 284 | 	 ((u64) month)) | 
 | 285 |  | 
 | 286 | #define cid_rev_card(card)		  \ | 
 | 287 | 	cid_rev(card->cid.hwrev,	  \ | 
 | 288 | 		    card->cid.fwrev,      \ | 
 | 289 | 		    card->cid.year,	  \ | 
 | 290 | 		    card->cid.month) | 
 | 291 |  | 
 | 292 | /* | 
| Andrei Warkentin | c59de92 | 2011-05-23 15:06:35 -0500 | [diff] [blame] | 293 |  * Unconditionally quirk add/remove. | 
| Andrei Warkentin | 32780cd | 2011-04-11 17:02:15 -0500 | [diff] [blame] | 294 |  */ | 
| Andrei Warkentin | c59de92 | 2011-05-23 15:06:35 -0500 | [diff] [blame] | 295 |  | 
| Andrei Warkentin | 32780cd | 2011-04-11 17:02:15 -0500 | [diff] [blame] | 296 | static inline void __maybe_unused add_quirk(struct mmc_card *card, int data) | 
 | 297 | { | 
 | 298 | 	card->quirks |= data; | 
 | 299 | } | 
 | 300 |  | 
| Andrei Warkentin | 32780cd | 2011-04-11 17:02:15 -0500 | [diff] [blame] | 301 | static inline void __maybe_unused remove_quirk(struct mmc_card *card, int data) | 
 | 302 | { | 
 | 303 | 	card->quirks &= ~data; | 
 | 304 | } | 
| Pierre Tardy | 57f0adc | 2011-02-06 19:03:46 +0100 | [diff] [blame] | 305 |  | 
| Pierre Ossman | 9c2c0af | 2006-12-26 15:25:58 +0100 | [diff] [blame] | 306 | #define mmc_card_mmc(c)		((c)->type == MMC_TYPE_MMC) | 
 | 307 | #define mmc_card_sd(c)		((c)->type == MMC_TYPE_SD) | 
| Pierre Ossman | 5c4e6f1 | 2007-05-21 20:23:20 +0200 | [diff] [blame] | 308 | #define mmc_card_sdio(c)	((c)->type == MMC_TYPE_SDIO) | 
| Pierre Ossman | 9c2c0af | 2006-12-26 15:25:58 +0100 | [diff] [blame] | 309 |  | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 310 | #define mmc_card_present(c)	((c)->state & MMC_STATE_PRESENT) | 
| Pierre Ossman | a00fc09 | 2005-09-06 15:18:52 -0700 | [diff] [blame] | 311 | #define mmc_card_readonly(c)	((c)->state & MMC_STATE_READONLY) | 
| Philip Langdale | bce40a3 | 2006-10-21 12:35:02 +0200 | [diff] [blame] | 312 | #define mmc_card_highspeed(c)	((c)->state & MMC_STATE_HIGHSPEED) | 
| Philip Langdale | fba68bd | 2007-01-04 06:57:32 -0800 | [diff] [blame] | 313 | #define mmc_card_blockaddr(c)	((c)->state & MMC_STATE_BLOCKADDR) | 
| Hanumath Prasad | dfc13e8 | 2010-09-30 17:37:23 -0400 | [diff] [blame] | 314 | #define mmc_card_ddr_mode(c)	((c)->state & MMC_STATE_HIGHSPEED_DDR) | 
| Arindam Nath | 3a30351 | 2011-05-05 12:19:03 +0530 | [diff] [blame] | 315 | #define mmc_sd_card_uhs(c) ((c)->state & MMC_STATE_ULTRAHIGHSPEED) | 
 | 316 | #define mmc_card_ext_capacity(c) ((c)->state & MMC_CARD_SDXC) | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 317 |  | 
 | 318 | #define mmc_card_set_present(c)	((c)->state |= MMC_STATE_PRESENT) | 
| Pierre Ossman | a00fc09 | 2005-09-06 15:18:52 -0700 | [diff] [blame] | 319 | #define mmc_card_set_readonly(c) ((c)->state |= MMC_STATE_READONLY) | 
| Philip Langdale | bce40a3 | 2006-10-21 12:35:02 +0200 | [diff] [blame] | 320 | #define mmc_card_set_highspeed(c) ((c)->state |= MMC_STATE_HIGHSPEED) | 
| Philip Langdale | fba68bd | 2007-01-04 06:57:32 -0800 | [diff] [blame] | 321 | #define mmc_card_set_blockaddr(c) ((c)->state |= MMC_STATE_BLOCKADDR) | 
| Hanumath Prasad | dfc13e8 | 2010-09-30 17:37:23 -0400 | [diff] [blame] | 322 | #define mmc_card_set_ddr_mode(c) ((c)->state |= MMC_STATE_HIGHSPEED_DDR) | 
| Arindam Nath | 3a30351 | 2011-05-05 12:19:03 +0530 | [diff] [blame] | 323 | #define mmc_sd_card_set_uhs(c) ((c)->state |= MMC_STATE_ULTRAHIGHSPEED) | 
 | 324 | #define mmc_card_set_ext_capacity(c) ((c)->state |= MMC_CARD_SDXC) | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 325 |  | 
| Andrei Warkentin | c59de92 | 2011-05-23 15:06:35 -0500 | [diff] [blame] | 326 | /* | 
 | 327 |  * Quirk add/remove for MMC products. | 
 | 328 |  */ | 
 | 329 |  | 
 | 330 | static inline void __maybe_unused add_quirk_mmc(struct mmc_card *card, int data) | 
 | 331 | { | 
 | 332 | 	if (mmc_card_mmc(card)) | 
 | 333 | 		card->quirks |= data; | 
 | 334 | } | 
 | 335 |  | 
 | 336 | static inline void __maybe_unused remove_quirk_mmc(struct mmc_card *card, | 
 | 337 | 						   int data) | 
 | 338 | { | 
 | 339 | 	if (mmc_card_mmc(card)) | 
 | 340 | 		card->quirks &= ~data; | 
 | 341 | } | 
 | 342 |  | 
 | 343 | /* | 
 | 344 |  * Quirk add/remove for SD products. | 
 | 345 |  */ | 
 | 346 |  | 
 | 347 | static inline void __maybe_unused add_quirk_sd(struct mmc_card *card, int data) | 
 | 348 | { | 
 | 349 | 	if (mmc_card_sd(card)) | 
 | 350 | 		card->quirks |= data; | 
 | 351 | } | 
 | 352 |  | 
 | 353 | static inline void __maybe_unused remove_quirk_sd(struct mmc_card *card, | 
 | 354 | 						   int data) | 
 | 355 | { | 
 | 356 | 	if (mmc_card_sd(card)) | 
 | 357 | 		card->quirks &= ~data; | 
 | 358 | } | 
 | 359 |  | 
| Ohad Ben-Cohen | 7c979ec | 2009-09-22 16:45:18 -0700 | [diff] [blame] | 360 | static inline int mmc_card_lenient_fn0(const struct mmc_card *c) | 
 | 361 | { | 
 | 362 | 	return c->quirks & MMC_QUIRK_LENIENT_FN0; | 
 | 363 | } | 
 | 364 |  | 
| Bing Zhao | 3fb7fb4 | 2010-03-05 13:43:25 -0800 | [diff] [blame] | 365 | static inline int mmc_blksz_for_byte_mode(const struct mmc_card *c) | 
 | 366 | { | 
 | 367 | 	return c->quirks & MMC_QUIRK_BLKSZ_FOR_BYTE_MODE; | 
 | 368 | } | 
 | 369 |  | 
| Ohad Ben-Cohen | 2059a02 | 2011-04-05 18:02:25 +0300 | [diff] [blame] | 370 | static inline int mmc_card_disable_cd(const struct mmc_card *c) | 
 | 371 | { | 
 | 372 | 	return c->quirks & MMC_QUIRK_DISABLE_CD; | 
 | 373 | } | 
 | 374 |  | 
| Ohad Ben-Cohen | eab4068 | 2011-04-05 17:50:14 +0300 | [diff] [blame] | 375 | static inline int mmc_card_nonstd_func_interface(const struct mmc_card *c) | 
 | 376 | { | 
 | 377 | 	return c->quirks & MMC_QUIRK_NONSTD_FUNC_IF; | 
 | 378 | } | 
 | 379 |  | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 380 | #define mmc_card_name(c)	((c)->cid.prod_name) | 
| Kay Sievers | d1b2686 | 2008-11-08 21:37:46 +0100 | [diff] [blame] | 381 | #define mmc_card_id(c)		(dev_name(&(c)->dev)) | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 382 |  | 
| Andy Shevchenko | 265cdc9 | 2010-09-17 20:32:25 -0400 | [diff] [blame] | 383 | #define mmc_dev_to_card(d)	container_of(d, struct mmc_card, dev) | 
| Andy Shevchenko | 453722b | 2010-08-20 10:46:46 +0300 | [diff] [blame] | 384 |  | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 385 | #define mmc_list_to_card(l)	container_of(l, struct mmc_card, node) | 
 | 386 | #define mmc_get_drvdata(c)	dev_get_drvdata(&(c)->dev) | 
 | 387 | #define mmc_set_drvdata(c,d)	dev_set_drvdata(&(c)->dev, d) | 
 | 388 |  | 
 | 389 | /* | 
 | 390 |  * MMC device driver (e.g., Flash card, I/O card...) | 
 | 391 |  */ | 
 | 392 | struct mmc_driver { | 
 | 393 | 	struct device_driver drv; | 
 | 394 | 	int (*probe)(struct mmc_card *); | 
 | 395 | 	void (*remove)(struct mmc_card *); | 
| Chuanxiao Dong | 72407e9 | 2011-08-24 14:00:41 +0530 | [diff] [blame] | 396 | 	int (*suspend)(struct mmc_card *); | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 397 | 	int (*resume)(struct mmc_card *); | 
 | 398 | }; | 
 | 399 |  | 
 | 400 | extern int mmc_register_driver(struct mmc_driver *); | 
 | 401 | extern void mmc_unregister_driver(struct mmc_driver *); | 
 | 402 |  | 
| Andrei Warkentin | 32780cd | 2011-04-11 17:02:15 -0500 | [diff] [blame] | 403 | extern void mmc_fixup_device(struct mmc_card *card, | 
 | 404 | 			     const struct mmc_fixup *table); | 
 | 405 |  | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 406 | #endif |