1*10465441SEvalZero /*
2*10465441SEvalZero * Copyright (c) 2006-2018, RT-Thread Development Team
3*10465441SEvalZero *
4*10465441SEvalZero * SPDX-License-Identifier: Apache-2.0
5*10465441SEvalZero *
6*10465441SEvalZero * Change Logs:
7*10465441SEvalZero * Date Author Notes
8*10465441SEvalZero * 2011-07-25 weety first version
9*10465441SEvalZero */
10*10465441SEvalZero
11*10465441SEvalZero #include <drivers/mmcsd_core.h>
12*10465441SEvalZero #include <drivers/sd.h>
13*10465441SEvalZero
14*10465441SEvalZero #define DBG_ENABLE
15*10465441SEvalZero #define DBG_SECTION_NAME "SDIO"
16*10465441SEvalZero #ifdef RT_SDIO_DEBUG
17*10465441SEvalZero #define DBG_LEVEL DBG_LOG
18*10465441SEvalZero #else
19*10465441SEvalZero #define DBG_LEVEL DBG_INFO
20*10465441SEvalZero #endif /* RT_SDIO_DEBUG */
21*10465441SEvalZero #define DBG_COLOR
22*10465441SEvalZero #include <rtdbg.h>
23*10465441SEvalZero
24*10465441SEvalZero static const rt_uint32_t tran_unit[] =
25*10465441SEvalZero {
26*10465441SEvalZero 10000, 100000, 1000000, 10000000,
27*10465441SEvalZero 0, 0, 0, 0
28*10465441SEvalZero };
29*10465441SEvalZero
30*10465441SEvalZero static const rt_uint8_t tran_value[] =
31*10465441SEvalZero {
32*10465441SEvalZero 0, 10, 12, 13, 15, 20, 25, 30,
33*10465441SEvalZero 35, 40, 45, 50, 55, 60, 70, 80,
34*10465441SEvalZero };
35*10465441SEvalZero
36*10465441SEvalZero static const rt_uint32_t tacc_uint[] =
37*10465441SEvalZero {
38*10465441SEvalZero 1, 10, 100, 1000, 10000, 100000, 1000000, 10000000,
39*10465441SEvalZero };
40*10465441SEvalZero
41*10465441SEvalZero static const rt_uint8_t tacc_value[] =
42*10465441SEvalZero {
43*10465441SEvalZero 0, 10, 12, 13, 15, 20, 25, 30,
44*10465441SEvalZero 35, 40, 45, 50, 55, 60, 70, 80,
45*10465441SEvalZero };
46*10465441SEvalZero
GET_BITS(rt_uint32_t * resp,rt_uint32_t start,rt_uint32_t size)47*10465441SEvalZero rt_inline rt_uint32_t GET_BITS(rt_uint32_t *resp,
48*10465441SEvalZero rt_uint32_t start,
49*10465441SEvalZero rt_uint32_t size)
50*10465441SEvalZero {
51*10465441SEvalZero const rt_int32_t __size = size;
52*10465441SEvalZero const rt_uint32_t __mask = (__size < 32 ? 1 << __size : 0) - 1;
53*10465441SEvalZero const rt_int32_t __off = 3 - ((start) / 32);
54*10465441SEvalZero const rt_int32_t __shft = (start) & 31;
55*10465441SEvalZero rt_uint32_t __res;
56*10465441SEvalZero
57*10465441SEvalZero __res = resp[__off] >> __shft;
58*10465441SEvalZero if (__size + __shft > 32)
59*10465441SEvalZero __res |= resp[__off-1] << ((32 - __shft) % 32);
60*10465441SEvalZero
61*10465441SEvalZero return __res & __mask;
62*10465441SEvalZero }
63*10465441SEvalZero
mmcsd_parse_csd(struct rt_mmcsd_card * card)64*10465441SEvalZero static rt_int32_t mmcsd_parse_csd(struct rt_mmcsd_card *card)
65*10465441SEvalZero {
66*10465441SEvalZero struct rt_mmcsd_csd *csd = &card->csd;
67*10465441SEvalZero rt_uint32_t *resp = card->resp_csd;
68*10465441SEvalZero
69*10465441SEvalZero csd->csd_structure = GET_BITS(resp, 126, 2);
70*10465441SEvalZero
71*10465441SEvalZero switch (csd->csd_structure)
72*10465441SEvalZero {
73*10465441SEvalZero case 0:
74*10465441SEvalZero csd->taac = GET_BITS(resp, 112, 8);
75*10465441SEvalZero csd->nsac = GET_BITS(resp, 104, 8);
76*10465441SEvalZero csd->tran_speed = GET_BITS(resp, 96, 8);
77*10465441SEvalZero csd->card_cmd_class = GET_BITS(resp, 84, 12);
78*10465441SEvalZero csd->rd_blk_len = GET_BITS(resp, 80, 4);
79*10465441SEvalZero csd->rd_blk_part = GET_BITS(resp, 79, 1);
80*10465441SEvalZero csd->wr_blk_misalign = GET_BITS(resp, 78, 1);
81*10465441SEvalZero csd->rd_blk_misalign = GET_BITS(resp, 77, 1);
82*10465441SEvalZero csd->dsr_imp = GET_BITS(resp, 76, 1);
83*10465441SEvalZero csd->c_size = GET_BITS(resp, 62, 12);
84*10465441SEvalZero csd->c_size_mult = GET_BITS(resp, 47, 3);
85*10465441SEvalZero csd->r2w_factor = GET_BITS(resp, 26, 3);
86*10465441SEvalZero csd->wr_blk_len = GET_BITS(resp, 22, 4);
87*10465441SEvalZero csd->wr_blk_partial = GET_BITS(resp, 21, 1);
88*10465441SEvalZero csd->csd_crc = GET_BITS(resp, 1, 7);
89*10465441SEvalZero
90*10465441SEvalZero card->card_blksize = 1 << csd->rd_blk_len;
91*10465441SEvalZero card->card_capacity = (csd->c_size + 1) << (csd->c_size_mult + 2);
92*10465441SEvalZero card->card_capacity *= card->card_blksize;
93*10465441SEvalZero card->card_capacity >>= 10; /* unit:KB */
94*10465441SEvalZero card->tacc_clks = csd->nsac * 100;
95*10465441SEvalZero card->tacc_ns = (tacc_uint[csd->taac&0x07] * tacc_value[(csd->taac&0x78)>>3] + 9) / 10;
96*10465441SEvalZero card->max_data_rate = tran_unit[csd->tran_speed&0x07] * tran_value[(csd->tran_speed&0x78)>>3];
97*10465441SEvalZero
98*10465441SEvalZero #if 0
99*10465441SEvalZero val = GET_BITS(resp, 115, 4);
100*10465441SEvalZero unit = GET_BITS(resp, 112, 3);
101*10465441SEvalZero csd->tacc_ns = (tacc_uint[unit] * tacc_value[val] + 9) / 10;
102*10465441SEvalZero csd->tacc_clks = GET_BITS(resp, 104, 8) * 100;
103*10465441SEvalZero
104*10465441SEvalZero val = GET_BITS(resp, 99, 4);
105*10465441SEvalZero unit = GET_BITS(resp, 96, 3);
106*10465441SEvalZero csd->max_data_rate = tran_unit[unit] * tran_value[val];
107*10465441SEvalZero csd->ccc = GET_BITS(resp, 84, 12);
108*10465441SEvalZero
109*10465441SEvalZero unit = GET_BITS(resp, 47, 3);
110*10465441SEvalZero val = GET_BITS(resp, 62, 12);
111*10465441SEvalZero csd->device_size = (1 + val) << (unit + 2);
112*10465441SEvalZero
113*10465441SEvalZero csd->read_bl_len = GET_BITS(resp, 80, 4);
114*10465441SEvalZero csd->write_bl_len = GET_BITS(resp, 22, 4);
115*10465441SEvalZero csd->r2w_factor = GET_BITS(resp, 26, 3);
116*10465441SEvalZero #endif
117*10465441SEvalZero break;
118*10465441SEvalZero case 1:
119*10465441SEvalZero card->flags |= CARD_FLAG_SDHC;
120*10465441SEvalZero
121*10465441SEvalZero /*This field is fixed to 0Eh, which indicates 1 ms.
122*10465441SEvalZero The host should not use TAAC, NSAC, and R2W_FACTOR
123*10465441SEvalZero to calculate timeout and should uses fixed timeout
124*10465441SEvalZero values for read and write operations*/
125*10465441SEvalZero csd->taac = GET_BITS(resp, 112, 8);
126*10465441SEvalZero csd->nsac = GET_BITS(resp, 104, 8);
127*10465441SEvalZero csd->tran_speed = GET_BITS(resp, 96, 8);
128*10465441SEvalZero csd->card_cmd_class = GET_BITS(resp, 84, 12);
129*10465441SEvalZero csd->rd_blk_len = GET_BITS(resp, 80, 4);
130*10465441SEvalZero csd->rd_blk_part = GET_BITS(resp, 79, 1);
131*10465441SEvalZero csd->wr_blk_misalign = GET_BITS(resp, 78, 1);
132*10465441SEvalZero csd->rd_blk_misalign = GET_BITS(resp, 77, 1);
133*10465441SEvalZero csd->dsr_imp = GET_BITS(resp, 76, 1);
134*10465441SEvalZero csd->c_size = GET_BITS(resp, 48, 22);
135*10465441SEvalZero
136*10465441SEvalZero csd->r2w_factor = GET_BITS(resp, 26, 3);
137*10465441SEvalZero csd->wr_blk_len = GET_BITS(resp, 22, 4);
138*10465441SEvalZero csd->wr_blk_partial = GET_BITS(resp, 21, 1);
139*10465441SEvalZero csd->csd_crc = GET_BITS(resp, 1, 7);
140*10465441SEvalZero
141*10465441SEvalZero card->card_blksize = 512;
142*10465441SEvalZero card->card_capacity = (csd->c_size + 1) * 512; /* unit:KB */
143*10465441SEvalZero card->tacc_clks = 0;
144*10465441SEvalZero card->tacc_ns = 0;
145*10465441SEvalZero card->max_data_rate = tran_unit[csd->tran_speed&0x07] * tran_value[(csd->tran_speed&0x78)>>3];
146*10465441SEvalZero
147*10465441SEvalZero #if 0
148*10465441SEvalZero csd->tacc_ns = 0;
149*10465441SEvalZero csd->tacc_clks = 0;
150*10465441SEvalZero
151*10465441SEvalZero val = GET_BITS(resp, 99, 4);
152*10465441SEvalZero unit = GET_BITS(resp, 96, 3);
153*10465441SEvalZero csd->max_data_rate = tran_unit[unit] * tran_value[val];
154*10465441SEvalZero csd->ccc = GET_BITS(resp, 84, 12);
155*10465441SEvalZero
156*10465441SEvalZero val = GET_BITS(resp, 48, 22);
157*10465441SEvalZero csd->device_size = (1 + val) << 10;
158*10465441SEvalZero
159*10465441SEvalZero csd->read_bl_len = 9;
160*10465441SEvalZero csd->write_bl_len = 9;
161*10465441SEvalZero /* host should not use this factor and should use 250ms for write timeout */
162*10465441SEvalZero csd->r2w_factor = 2;
163*10465441SEvalZero #endif
164*10465441SEvalZero break;
165*10465441SEvalZero default:
166*10465441SEvalZero LOG_E("unrecognised CSD structure version %d!", csd->csd_structure);
167*10465441SEvalZero
168*10465441SEvalZero return -RT_ERROR;
169*10465441SEvalZero }
170*10465441SEvalZero LOG_I("SD card capacity %d KB.", card->card_capacity);
171*10465441SEvalZero
172*10465441SEvalZero return 0;
173*10465441SEvalZero }
174*10465441SEvalZero
mmcsd_parse_scr(struct rt_mmcsd_card * card)175*10465441SEvalZero static rt_int32_t mmcsd_parse_scr(struct rt_mmcsd_card *card)
176*10465441SEvalZero {
177*10465441SEvalZero struct rt_sd_scr *scr = &card->scr;
178*10465441SEvalZero rt_uint32_t resp[4];
179*10465441SEvalZero
180*10465441SEvalZero resp[3] = card->resp_scr[1];
181*10465441SEvalZero resp[2] = card->resp_scr[0];
182*10465441SEvalZero scr->sd_version = GET_BITS(resp, 56, 4);
183*10465441SEvalZero scr->sd_bus_widths = GET_BITS(resp, 48, 4);
184*10465441SEvalZero
185*10465441SEvalZero return 0;
186*10465441SEvalZero }
187*10465441SEvalZero
mmcsd_switch(struct rt_mmcsd_card * card)188*10465441SEvalZero static rt_int32_t mmcsd_switch(struct rt_mmcsd_card *card)
189*10465441SEvalZero {
190*10465441SEvalZero rt_int32_t err;
191*10465441SEvalZero struct rt_mmcsd_host *host = card->host;
192*10465441SEvalZero struct rt_mmcsd_req req;
193*10465441SEvalZero struct rt_mmcsd_cmd cmd;
194*10465441SEvalZero struct rt_mmcsd_data data;
195*10465441SEvalZero rt_uint8_t *buf;
196*10465441SEvalZero
197*10465441SEvalZero buf = (rt_uint8_t*)rt_malloc(64);
198*10465441SEvalZero if (!buf)
199*10465441SEvalZero {
200*10465441SEvalZero LOG_E("alloc memory failed!");
201*10465441SEvalZero
202*10465441SEvalZero return -RT_ENOMEM;
203*10465441SEvalZero }
204*10465441SEvalZero
205*10465441SEvalZero if (card->card_type != CARD_TYPE_SD)
206*10465441SEvalZero goto err;
207*10465441SEvalZero if (card->scr.sd_version < SCR_SPEC_VER_1)
208*10465441SEvalZero goto err;
209*10465441SEvalZero
210*10465441SEvalZero rt_memset(&cmd, 0, sizeof(struct rt_mmcsd_cmd));
211*10465441SEvalZero
212*10465441SEvalZero cmd.cmd_code = SD_SWITCH;
213*10465441SEvalZero cmd.arg = 0x00FFFFF1;
214*10465441SEvalZero cmd.flags = RESP_R1 | CMD_ADTC;
215*10465441SEvalZero
216*10465441SEvalZero rt_memset(&data, 0, sizeof(struct rt_mmcsd_data));
217*10465441SEvalZero
218*10465441SEvalZero mmcsd_set_data_timeout(&data, card);
219*10465441SEvalZero
220*10465441SEvalZero data.blksize = 64;
221*10465441SEvalZero data.blks = 1;
222*10465441SEvalZero data.flags = DATA_DIR_READ;
223*10465441SEvalZero data.buf = (rt_uint32_t *)buf;
224*10465441SEvalZero
225*10465441SEvalZero rt_memset(&req, 0, sizeof(struct rt_mmcsd_req));
226*10465441SEvalZero
227*10465441SEvalZero req.cmd = &cmd;
228*10465441SEvalZero req.data = &data;
229*10465441SEvalZero
230*10465441SEvalZero mmcsd_send_request(host, &req);
231*10465441SEvalZero
232*10465441SEvalZero if (cmd.err || data.err)
233*10465441SEvalZero {
234*10465441SEvalZero goto err1;
235*10465441SEvalZero }
236*10465441SEvalZero
237*10465441SEvalZero if (buf[13] & 0x02)
238*10465441SEvalZero card->hs_max_data_rate = 50000000;
239*10465441SEvalZero
240*10465441SEvalZero rt_memset(&cmd, 0, sizeof(struct rt_mmcsd_cmd));
241*10465441SEvalZero
242*10465441SEvalZero cmd.cmd_code = SD_SWITCH;
243*10465441SEvalZero cmd.arg = 0x80FFFFF1;
244*10465441SEvalZero cmd.flags = RESP_R1 | CMD_ADTC;
245*10465441SEvalZero
246*10465441SEvalZero rt_memset(&data, 0, sizeof(struct rt_mmcsd_data));
247*10465441SEvalZero
248*10465441SEvalZero mmcsd_set_data_timeout(&data, card);
249*10465441SEvalZero
250*10465441SEvalZero data.blksize = 64;
251*10465441SEvalZero data.blks = 1;
252*10465441SEvalZero data.flags = DATA_DIR_READ;
253*10465441SEvalZero data.buf = (rt_uint32_t *)buf;
254*10465441SEvalZero
255*10465441SEvalZero rt_memset(&req, 0, sizeof(struct rt_mmcsd_req));
256*10465441SEvalZero
257*10465441SEvalZero req.cmd = &cmd;
258*10465441SEvalZero req.data = &data;
259*10465441SEvalZero
260*10465441SEvalZero mmcsd_send_request(host, &req);
261*10465441SEvalZero
262*10465441SEvalZero if (cmd.err || data.err)
263*10465441SEvalZero {
264*10465441SEvalZero goto err1;
265*10465441SEvalZero }
266*10465441SEvalZero
267*10465441SEvalZero if ((buf[16] & 0xF) != 1)
268*10465441SEvalZero {
269*10465441SEvalZero LOG_E("switching card to high speed failed!");
270*10465441SEvalZero goto err;
271*10465441SEvalZero }
272*10465441SEvalZero
273*10465441SEvalZero card->flags |= CARD_FLAG_HIGHSPEED;
274*10465441SEvalZero
275*10465441SEvalZero err:
276*10465441SEvalZero rt_free(buf);
277*10465441SEvalZero return 0;
278*10465441SEvalZero
279*10465441SEvalZero err1:
280*10465441SEvalZero if (cmd.err)
281*10465441SEvalZero err = cmd.err;
282*10465441SEvalZero if (data.err)
283*10465441SEvalZero err = data.err;
284*10465441SEvalZero
285*10465441SEvalZero return err;
286*10465441SEvalZero }
287*10465441SEvalZero
mmcsd_app_cmd(struct rt_mmcsd_host * host,struct rt_mmcsd_card * card)288*10465441SEvalZero static rt_err_t mmcsd_app_cmd(struct rt_mmcsd_host *host,
289*10465441SEvalZero struct rt_mmcsd_card *card)
290*10465441SEvalZero {
291*10465441SEvalZero rt_err_t err;
292*10465441SEvalZero struct rt_mmcsd_cmd cmd = {0};
293*10465441SEvalZero
294*10465441SEvalZero cmd.cmd_code = APP_CMD;
295*10465441SEvalZero
296*10465441SEvalZero if (card)
297*10465441SEvalZero {
298*10465441SEvalZero cmd.arg = card->rca << 16;
299*10465441SEvalZero cmd.flags = RESP_R1 | CMD_AC;
300*10465441SEvalZero }
301*10465441SEvalZero else
302*10465441SEvalZero {
303*10465441SEvalZero cmd.arg = 0;
304*10465441SEvalZero cmd.flags = RESP_R1 | CMD_BCR;
305*10465441SEvalZero }
306*10465441SEvalZero
307*10465441SEvalZero err = mmcsd_send_cmd(host, &cmd, 0);
308*10465441SEvalZero if (err)
309*10465441SEvalZero return err;
310*10465441SEvalZero
311*10465441SEvalZero /* Check that card supported application commands */
312*10465441SEvalZero if (!controller_is_spi(host) && !(cmd.resp[0] & R1_APP_CMD))
313*10465441SEvalZero return -RT_ERROR;
314*10465441SEvalZero
315*10465441SEvalZero return RT_EOK;
316*10465441SEvalZero }
317*10465441SEvalZero
318*10465441SEvalZero
mmcsd_send_app_cmd(struct rt_mmcsd_host * host,struct rt_mmcsd_card * card,struct rt_mmcsd_cmd * cmd,int retry)319*10465441SEvalZero rt_err_t mmcsd_send_app_cmd(struct rt_mmcsd_host *host,
320*10465441SEvalZero struct rt_mmcsd_card *card,
321*10465441SEvalZero struct rt_mmcsd_cmd *cmd,
322*10465441SEvalZero int retry)
323*10465441SEvalZero {
324*10465441SEvalZero struct rt_mmcsd_req req;
325*10465441SEvalZero
326*10465441SEvalZero rt_uint32_t i;
327*10465441SEvalZero rt_err_t err;
328*10465441SEvalZero
329*10465441SEvalZero err = -RT_ERROR;
330*10465441SEvalZero
331*10465441SEvalZero /*
332*10465441SEvalZero * We have to resend MMC_APP_CMD for each attempt so
333*10465441SEvalZero * we cannot use the retries field in mmc_command.
334*10465441SEvalZero */
335*10465441SEvalZero for (i = 0;i <= retry;i++)
336*10465441SEvalZero {
337*10465441SEvalZero rt_memset(&req, 0, sizeof(struct rt_mmcsd_req));
338*10465441SEvalZero
339*10465441SEvalZero err = mmcsd_app_cmd(host, card);
340*10465441SEvalZero if (err)
341*10465441SEvalZero {
342*10465441SEvalZero /* no point in retrying; no APP commands allowed */
343*10465441SEvalZero if (controller_is_spi(host))
344*10465441SEvalZero {
345*10465441SEvalZero if (cmd->resp[0] & R1_SPI_ILLEGAL_COMMAND)
346*10465441SEvalZero break;
347*10465441SEvalZero }
348*10465441SEvalZero continue;
349*10465441SEvalZero }
350*10465441SEvalZero
351*10465441SEvalZero rt_memset(&req, 0, sizeof(struct rt_mmcsd_req));
352*10465441SEvalZero
353*10465441SEvalZero rt_memset(cmd->resp, 0, sizeof(cmd->resp));
354*10465441SEvalZero
355*10465441SEvalZero req.cmd = cmd;
356*10465441SEvalZero //cmd->data = NULL;
357*10465441SEvalZero
358*10465441SEvalZero mmcsd_send_request(host, &req);
359*10465441SEvalZero
360*10465441SEvalZero err = cmd->err;
361*10465441SEvalZero if (!cmd->err)
362*10465441SEvalZero break;
363*10465441SEvalZero
364*10465441SEvalZero /* no point in retrying illegal APP commands */
365*10465441SEvalZero if (controller_is_spi(host))
366*10465441SEvalZero {
367*10465441SEvalZero if (cmd->resp[0] & R1_SPI_ILLEGAL_COMMAND)
368*10465441SEvalZero break;
369*10465441SEvalZero }
370*10465441SEvalZero }
371*10465441SEvalZero
372*10465441SEvalZero return err;
373*10465441SEvalZero }
374*10465441SEvalZero
mmcsd_app_set_bus_width(struct rt_mmcsd_card * card,rt_int32_t width)375*10465441SEvalZero rt_err_t mmcsd_app_set_bus_width(struct rt_mmcsd_card *card, rt_int32_t width)
376*10465441SEvalZero {
377*10465441SEvalZero rt_err_t err;
378*10465441SEvalZero struct rt_mmcsd_cmd cmd;
379*10465441SEvalZero
380*10465441SEvalZero rt_memset(&cmd, 0, sizeof(struct rt_mmcsd_cmd));
381*10465441SEvalZero
382*10465441SEvalZero cmd.cmd_code = SD_APP_SET_BUS_WIDTH;
383*10465441SEvalZero cmd.flags = RESP_R1 | CMD_AC;
384*10465441SEvalZero
385*10465441SEvalZero switch (width)
386*10465441SEvalZero {
387*10465441SEvalZero case MMCSD_BUS_WIDTH_1:
388*10465441SEvalZero cmd.arg = MMCSD_BUS_WIDTH_1;
389*10465441SEvalZero break;
390*10465441SEvalZero case MMCSD_BUS_WIDTH_4:
391*10465441SEvalZero cmd.arg = MMCSD_BUS_WIDTH_4;
392*10465441SEvalZero break;
393*10465441SEvalZero default:
394*10465441SEvalZero return -RT_ERROR;
395*10465441SEvalZero }
396*10465441SEvalZero
397*10465441SEvalZero err = mmcsd_send_app_cmd(card->host, card, &cmd, 3);
398*10465441SEvalZero if (err)
399*10465441SEvalZero return err;
400*10465441SEvalZero
401*10465441SEvalZero return RT_EOK;
402*10465441SEvalZero }
403*10465441SEvalZero
mmcsd_send_app_op_cond(struct rt_mmcsd_host * host,rt_uint32_t ocr,rt_uint32_t * rocr)404*10465441SEvalZero rt_err_t mmcsd_send_app_op_cond(struct rt_mmcsd_host *host,
405*10465441SEvalZero rt_uint32_t ocr,
406*10465441SEvalZero rt_uint32_t *rocr)
407*10465441SEvalZero {
408*10465441SEvalZero struct rt_mmcsd_cmd cmd;
409*10465441SEvalZero rt_uint32_t i;
410*10465441SEvalZero rt_err_t err = RT_EOK;
411*10465441SEvalZero
412*10465441SEvalZero rt_memset(&cmd, 0, sizeof(struct rt_mmcsd_cmd));
413*10465441SEvalZero
414*10465441SEvalZero cmd.cmd_code = SD_APP_OP_COND;
415*10465441SEvalZero if (controller_is_spi(host))
416*10465441SEvalZero cmd.arg = ocr & (1 << 30); /* SPI only defines one bit */
417*10465441SEvalZero else
418*10465441SEvalZero cmd.arg = ocr;
419*10465441SEvalZero cmd.flags = RESP_SPI_R1 | RESP_R3 | CMD_BCR;
420*10465441SEvalZero
421*10465441SEvalZero for (i = 100; i; i--)
422*10465441SEvalZero {
423*10465441SEvalZero err = mmcsd_send_app_cmd(host, RT_NULL, &cmd, 3);
424*10465441SEvalZero if (err)
425*10465441SEvalZero break;
426*10465441SEvalZero
427*10465441SEvalZero /* if we're just probing, do a single pass */
428*10465441SEvalZero if (ocr == 0)
429*10465441SEvalZero break;
430*10465441SEvalZero
431*10465441SEvalZero /* otherwise wait until reset completes */
432*10465441SEvalZero if (controller_is_spi(host))
433*10465441SEvalZero {
434*10465441SEvalZero if (!(cmd.resp[0] & R1_SPI_IDLE))
435*10465441SEvalZero break;
436*10465441SEvalZero }
437*10465441SEvalZero else
438*10465441SEvalZero {
439*10465441SEvalZero if (cmd.resp[0] & CARD_BUSY)
440*10465441SEvalZero break;
441*10465441SEvalZero }
442*10465441SEvalZero
443*10465441SEvalZero err = -RT_ETIMEOUT;
444*10465441SEvalZero
445*10465441SEvalZero mmcsd_delay_ms(10); //delay 10ms
446*10465441SEvalZero }
447*10465441SEvalZero
448*10465441SEvalZero if (rocr && !controller_is_spi(host))
449*10465441SEvalZero *rocr = cmd.resp[0];
450*10465441SEvalZero
451*10465441SEvalZero return err;
452*10465441SEvalZero }
453*10465441SEvalZero
454*10465441SEvalZero /*
455*10465441SEvalZero * To support SD 2.0 cards, we must always invoke SD_SEND_IF_COND
456*10465441SEvalZero * before SD_APP_OP_COND. This command will harmlessly fail for
457*10465441SEvalZero * SD 1.0 cards.
458*10465441SEvalZero */
mmcsd_send_if_cond(struct rt_mmcsd_host * host,rt_uint32_t ocr)459*10465441SEvalZero rt_err_t mmcsd_send_if_cond(struct rt_mmcsd_host *host, rt_uint32_t ocr)
460*10465441SEvalZero {
461*10465441SEvalZero struct rt_mmcsd_cmd cmd;
462*10465441SEvalZero rt_err_t err;
463*10465441SEvalZero rt_uint8_t pattern;
464*10465441SEvalZero
465*10465441SEvalZero cmd.cmd_code = SD_SEND_IF_COND;
466*10465441SEvalZero cmd.arg = ((ocr & 0xFF8000) != 0) << 8 | 0xAA;
467*10465441SEvalZero cmd.flags = RESP_SPI_R7 | RESP_R7 | CMD_BCR;
468*10465441SEvalZero
469*10465441SEvalZero err = mmcsd_send_cmd(host, &cmd, 0);
470*10465441SEvalZero if (err)
471*10465441SEvalZero return err;
472*10465441SEvalZero
473*10465441SEvalZero if (controller_is_spi(host))
474*10465441SEvalZero pattern = cmd.resp[1] & 0xFF;
475*10465441SEvalZero else
476*10465441SEvalZero pattern = cmd.resp[0] & 0xFF;
477*10465441SEvalZero
478*10465441SEvalZero if (pattern != 0xAA)
479*10465441SEvalZero return -RT_ERROR;
480*10465441SEvalZero
481*10465441SEvalZero return RT_EOK;
482*10465441SEvalZero }
483*10465441SEvalZero
mmcsd_get_card_addr(struct rt_mmcsd_host * host,rt_uint32_t * rca)484*10465441SEvalZero rt_err_t mmcsd_get_card_addr(struct rt_mmcsd_host *host, rt_uint32_t *rca)
485*10465441SEvalZero {
486*10465441SEvalZero rt_err_t err;
487*10465441SEvalZero struct rt_mmcsd_cmd cmd;
488*10465441SEvalZero
489*10465441SEvalZero rt_memset(&cmd, 0, sizeof(struct rt_mmcsd_cmd));
490*10465441SEvalZero
491*10465441SEvalZero cmd.cmd_code = SD_SEND_RELATIVE_ADDR;
492*10465441SEvalZero cmd.arg = 0;
493*10465441SEvalZero cmd.flags = RESP_R6 | CMD_BCR;
494*10465441SEvalZero
495*10465441SEvalZero err = mmcsd_send_cmd(host, &cmd, 3);
496*10465441SEvalZero if (err)
497*10465441SEvalZero return err;
498*10465441SEvalZero
499*10465441SEvalZero *rca = cmd.resp[0] >> 16;
500*10465441SEvalZero
501*10465441SEvalZero return RT_EOK;
502*10465441SEvalZero }
503*10465441SEvalZero
504*10465441SEvalZero #define be32_to_cpu(x) ((rt_uint32_t)( \
505*10465441SEvalZero (((rt_uint32_t)(x) & (rt_uint32_t)0x000000ffUL) << 24) | \
506*10465441SEvalZero (((rt_uint32_t)(x) & (rt_uint32_t)0x0000ff00UL) << 8) | \
507*10465441SEvalZero (((rt_uint32_t)(x) & (rt_uint32_t)0x00ff0000UL) >> 8) | \
508*10465441SEvalZero (((rt_uint32_t)(x) & (rt_uint32_t)0xff000000UL) >> 24)))
509*10465441SEvalZero
mmcsd_get_scr(struct rt_mmcsd_card * card,rt_uint32_t * scr)510*10465441SEvalZero rt_int32_t mmcsd_get_scr(struct rt_mmcsd_card *card, rt_uint32_t *scr)
511*10465441SEvalZero {
512*10465441SEvalZero rt_int32_t err;
513*10465441SEvalZero struct rt_mmcsd_req req;
514*10465441SEvalZero struct rt_mmcsd_cmd cmd;
515*10465441SEvalZero struct rt_mmcsd_data data;
516*10465441SEvalZero
517*10465441SEvalZero err = mmcsd_app_cmd(card->host, card);
518*10465441SEvalZero if (err)
519*10465441SEvalZero return err;
520*10465441SEvalZero
521*10465441SEvalZero rt_memset(&req, 0, sizeof(struct rt_mmcsd_req));
522*10465441SEvalZero rt_memset(&cmd, 0, sizeof(struct rt_mmcsd_cmd));
523*10465441SEvalZero rt_memset(&data, 0, sizeof(struct rt_mmcsd_data));
524*10465441SEvalZero
525*10465441SEvalZero req.cmd = &cmd;
526*10465441SEvalZero req.data = &data;
527*10465441SEvalZero
528*10465441SEvalZero cmd.cmd_code = SD_APP_SEND_SCR;
529*10465441SEvalZero cmd.arg = 0;
530*10465441SEvalZero cmd.flags = RESP_SPI_R1 | RESP_R1 | CMD_ADTC;
531*10465441SEvalZero
532*10465441SEvalZero data.blksize = 8;
533*10465441SEvalZero data.blks = 1;
534*10465441SEvalZero data.flags = DATA_DIR_READ;
535*10465441SEvalZero data.buf = scr;
536*10465441SEvalZero
537*10465441SEvalZero mmcsd_set_data_timeout(&data, card);
538*10465441SEvalZero
539*10465441SEvalZero mmcsd_send_request(card->host, &req);
540*10465441SEvalZero
541*10465441SEvalZero if (cmd.err)
542*10465441SEvalZero return cmd.err;
543*10465441SEvalZero if (data.err)
544*10465441SEvalZero return data.err;
545*10465441SEvalZero
546*10465441SEvalZero scr[0] = be32_to_cpu(scr[0]);
547*10465441SEvalZero scr[1] = be32_to_cpu(scr[1]);
548*10465441SEvalZero
549*10465441SEvalZero return 0;
550*10465441SEvalZero }
551*10465441SEvalZero
552*10465441SEvalZero
mmcsd_sd_init_card(struct rt_mmcsd_host * host,rt_uint32_t ocr)553*10465441SEvalZero static rt_int32_t mmcsd_sd_init_card(struct rt_mmcsd_host *host,
554*10465441SEvalZero rt_uint32_t ocr)
555*10465441SEvalZero {
556*10465441SEvalZero struct rt_mmcsd_card *card;
557*10465441SEvalZero rt_int32_t err;
558*10465441SEvalZero rt_uint32_t resp[4];
559*10465441SEvalZero rt_uint32_t max_data_rate;
560*10465441SEvalZero
561*10465441SEvalZero mmcsd_go_idle(host);
562*10465441SEvalZero
563*10465441SEvalZero /*
564*10465441SEvalZero * If SD_SEND_IF_COND indicates an SD 2.0
565*10465441SEvalZero * compliant card and we should set bit 30
566*10465441SEvalZero * of the ocr to indicate that we can handle
567*10465441SEvalZero * block-addressed SDHC cards.
568*10465441SEvalZero */
569*10465441SEvalZero err = mmcsd_send_if_cond(host, ocr);
570*10465441SEvalZero if (!err)
571*10465441SEvalZero ocr |= 1 << 30;
572*10465441SEvalZero
573*10465441SEvalZero err = mmcsd_send_app_op_cond(host, ocr, RT_NULL);
574*10465441SEvalZero if (err)
575*10465441SEvalZero goto err;
576*10465441SEvalZero
577*10465441SEvalZero if (controller_is_spi(host))
578*10465441SEvalZero err = mmcsd_get_cid(host, resp);
579*10465441SEvalZero else
580*10465441SEvalZero err = mmcsd_all_get_cid(host, resp);
581*10465441SEvalZero if (err)
582*10465441SEvalZero goto err;
583*10465441SEvalZero
584*10465441SEvalZero card = rt_malloc(sizeof(struct rt_mmcsd_card));
585*10465441SEvalZero if (!card)
586*10465441SEvalZero {
587*10465441SEvalZero LOG_E("malloc card failed!");
588*10465441SEvalZero err = -RT_ENOMEM;
589*10465441SEvalZero goto err;
590*10465441SEvalZero }
591*10465441SEvalZero rt_memset(card, 0, sizeof(struct rt_mmcsd_card));
592*10465441SEvalZero
593*10465441SEvalZero card->card_type = CARD_TYPE_SD;
594*10465441SEvalZero card->host = host;
595*10465441SEvalZero rt_memcpy(card->resp_cid, resp, sizeof(card->resp_cid));
596*10465441SEvalZero
597*10465441SEvalZero /*
598*10465441SEvalZero * For native busses: get card RCA and quit open drain mode.
599*10465441SEvalZero */
600*10465441SEvalZero if (!controller_is_spi(host))
601*10465441SEvalZero {
602*10465441SEvalZero err = mmcsd_get_card_addr(host, &card->rca);
603*10465441SEvalZero if (err)
604*10465441SEvalZero goto err1;
605*10465441SEvalZero
606*10465441SEvalZero mmcsd_set_bus_mode(host, MMCSD_BUSMODE_PUSHPULL);
607*10465441SEvalZero }
608*10465441SEvalZero
609*10465441SEvalZero err = mmcsd_get_csd(card, card->resp_csd);
610*10465441SEvalZero if (err)
611*10465441SEvalZero goto err1;
612*10465441SEvalZero
613*10465441SEvalZero err = mmcsd_parse_csd(card);
614*10465441SEvalZero if (err)
615*10465441SEvalZero goto err1;
616*10465441SEvalZero
617*10465441SEvalZero if (!controller_is_spi(host))
618*10465441SEvalZero {
619*10465441SEvalZero err = mmcsd_select_card(card);
620*10465441SEvalZero if (err)
621*10465441SEvalZero goto err1;
622*10465441SEvalZero }
623*10465441SEvalZero
624*10465441SEvalZero err = mmcsd_get_scr(card, card->resp_scr);
625*10465441SEvalZero if (err)
626*10465441SEvalZero goto err1;
627*10465441SEvalZero
628*10465441SEvalZero mmcsd_parse_scr(card);
629*10465441SEvalZero
630*10465441SEvalZero if (controller_is_spi(host))
631*10465441SEvalZero {
632*10465441SEvalZero err = mmcsd_spi_use_crc(host, 1);
633*10465441SEvalZero if (err)
634*10465441SEvalZero goto err1;
635*10465441SEvalZero }
636*10465441SEvalZero
637*10465441SEvalZero /*
638*10465441SEvalZero * change SD card to high-speed, only SD2.0 spec
639*10465441SEvalZero */
640*10465441SEvalZero err = mmcsd_switch(card);
641*10465441SEvalZero if (err)
642*10465441SEvalZero goto err1;
643*10465441SEvalZero
644*10465441SEvalZero /* set bus speed */
645*10465441SEvalZero max_data_rate = (unsigned int)-1;
646*10465441SEvalZero
647*10465441SEvalZero if (card->flags & CARD_FLAG_HIGHSPEED)
648*10465441SEvalZero {
649*10465441SEvalZero if (max_data_rate > card->hs_max_data_rate)
650*10465441SEvalZero max_data_rate = card->hs_max_data_rate;
651*10465441SEvalZero }
652*10465441SEvalZero else if (max_data_rate > card->max_data_rate)
653*10465441SEvalZero {
654*10465441SEvalZero max_data_rate = card->max_data_rate;
655*10465441SEvalZero }
656*10465441SEvalZero
657*10465441SEvalZero mmcsd_set_clock(host, max_data_rate);
658*10465441SEvalZero
659*10465441SEvalZero /*switch bus width*/
660*10465441SEvalZero if ((host->flags & MMCSD_BUSWIDTH_4) &&
661*10465441SEvalZero (card->scr.sd_bus_widths & SD_SCR_BUS_WIDTH_4))
662*10465441SEvalZero {
663*10465441SEvalZero err = mmcsd_app_set_bus_width(card, MMCSD_BUS_WIDTH_4);
664*10465441SEvalZero if (err)
665*10465441SEvalZero goto err1;
666*10465441SEvalZero
667*10465441SEvalZero mmcsd_set_bus_width(host, MMCSD_BUS_WIDTH_4);
668*10465441SEvalZero }
669*10465441SEvalZero
670*10465441SEvalZero host->card = card;
671*10465441SEvalZero
672*10465441SEvalZero return 0;
673*10465441SEvalZero
674*10465441SEvalZero err1:
675*10465441SEvalZero rt_free(card);
676*10465441SEvalZero err:
677*10465441SEvalZero
678*10465441SEvalZero return err;
679*10465441SEvalZero }
680*10465441SEvalZero
681*10465441SEvalZero /*
682*10465441SEvalZero * Starting point for SD card init.
683*10465441SEvalZero */
init_sd(struct rt_mmcsd_host * host,rt_uint32_t ocr)684*10465441SEvalZero rt_int32_t init_sd(struct rt_mmcsd_host *host, rt_uint32_t ocr)
685*10465441SEvalZero {
686*10465441SEvalZero rt_int32_t err;
687*10465441SEvalZero rt_uint32_t current_ocr;
688*10465441SEvalZero /*
689*10465441SEvalZero * We need to get OCR a different way for SPI.
690*10465441SEvalZero */
691*10465441SEvalZero if (controller_is_spi(host))
692*10465441SEvalZero {
693*10465441SEvalZero mmcsd_go_idle(host);
694*10465441SEvalZero
695*10465441SEvalZero err = mmcsd_spi_read_ocr(host, 0, &ocr);
696*10465441SEvalZero if (err)
697*10465441SEvalZero goto err;
698*10465441SEvalZero }
699*10465441SEvalZero
700*10465441SEvalZero if (ocr & VDD_165_195)
701*10465441SEvalZero {
702*10465441SEvalZero LOG_I(" SD card claims to support the "
703*10465441SEvalZero "incompletely defined 'low voltage range'. This "
704*10465441SEvalZero "will be ignored.");
705*10465441SEvalZero ocr &= ~VDD_165_195;
706*10465441SEvalZero }
707*10465441SEvalZero
708*10465441SEvalZero current_ocr = mmcsd_select_voltage(host, ocr);
709*10465441SEvalZero
710*10465441SEvalZero /*
711*10465441SEvalZero * Can we support the voltage(s) of the card(s)?
712*10465441SEvalZero */
713*10465441SEvalZero if (!current_ocr)
714*10465441SEvalZero {
715*10465441SEvalZero err = -RT_ERROR;
716*10465441SEvalZero goto err;
717*10465441SEvalZero }
718*10465441SEvalZero
719*10465441SEvalZero /*
720*10465441SEvalZero * Detect and init the card.
721*10465441SEvalZero */
722*10465441SEvalZero err = mmcsd_sd_init_card(host, current_ocr);
723*10465441SEvalZero if (err)
724*10465441SEvalZero goto err;
725*10465441SEvalZero
726*10465441SEvalZero mmcsd_host_unlock(host);
727*10465441SEvalZero
728*10465441SEvalZero err = rt_mmcsd_blk_probe(host->card);
729*10465441SEvalZero if (err)
730*10465441SEvalZero goto remove_card;
731*10465441SEvalZero mmcsd_host_lock(host);
732*10465441SEvalZero
733*10465441SEvalZero return 0;
734*10465441SEvalZero
735*10465441SEvalZero remove_card:
736*10465441SEvalZero mmcsd_host_lock(host);
737*10465441SEvalZero rt_mmcsd_blk_remove(host->card);
738*10465441SEvalZero rt_free(host->card);
739*10465441SEvalZero host->card = RT_NULL;
740*10465441SEvalZero err:
741*10465441SEvalZero
742*10465441SEvalZero LOG_D("init SD card failed!");
743*10465441SEvalZero
744*10465441SEvalZero return err;
745*10465441SEvalZero }
746