SWM341_sdio.c 23 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604
  1. /******************************************************************************************************************************************
  2. * 文件名称: SWM341_sdio.c
  3. * 功能说明: SWM341单片机的SDIO接口驱动库
  4. * 技术支持: http://www.synwit.com.cn/e/tool/gbook/?bid=1
  5. * 注意事项: 为了通用性、兼容性、易用性,只支持以512字节为单位的读写
  6. * 版本日期: V1.1.0 2017年10月25日
  7. * 升级记录:
  8. *
  9. *
  10. *******************************************************************************************************************************************
  11. * @attention
  12. *
  13. * THE PRESENT FIRMWARE WHICH IS FOR GUIDANCE ONLY AIMS AT PROVIDING CUSTOMERS WITH CODING INFORMATION
  14. * REGARDING THEIR PRODUCTS IN ORDER FOR THEM TO SAVE TIME. AS A RESULT, SYNWIT SHALL NOT BE HELD LIABLE
  15. * FOR ANY DIRECT, INDIRECT OR CONSEQUENTIAL DAMAGES WITH RESPECT TO ANY CLAIMS ARISING FROM THE CONTENT
  16. * OF SUCH FIRMWARE AND/OR THE USE MADE BY CUSTOMERS OF THE CODING INFORMATION CONTAINED HEREIN IN CONN-
  17. * -ECTION WITH THEIR PRODUCTS.
  18. *
  19. * COPYRIGHT 2012 Synwit Technology
  20. *******************************************************************************************************************************************/
  21. #include "SWM341.h"
  22. #include "SWM341_sdio.h"
  23. SD_CardInfo SD_cardInfo;
  24. /******************************************************************************************************************************************
  25. * 函数名称: SDIO_Init()
  26. * 功能说明: SDIO读写SD卡初始化,初始化成高速4线模式、读写以512字节大小进行
  27. * 输 入: uint32_t freq SDIO_CLK时钟频率
  28. * 输 出: uint32_t SD_RES_OK 操作成功 SD_RES_ERR 操作失败 SD_RES_TIMEOUT 操作超时
  29. * 注意事项: 无
  30. ******************************************************************************************************************************************/
  31. uint32_t SDIO_Init(uint32_t freq)
  32. {
  33. uint32_t res;
  34. uint32_t resp, resps[4];
  35. SYS->CLKSEL &= ~SYS_CLKSEL_SDIO_Msk;
  36. if(SystemCoreClock > 80000000) //SDIO时钟需要小于52MHz
  37. SYS->CLKSEL |= (2 << SYS_CLKSEL_SDIO_Pos); //SDCLK = SYSCLK / 4
  38. else
  39. SYS->CLKSEL |= (0 << SYS_CLKSEL_SDIO_Pos); //SDCLK = SYSCLK / 2
  40. SYS->CLKEN0 |= (0x01 << SYS_CLKEN0_SDIO_Pos);
  41. SDIO->CR2 = (1 << SDIO_CR2_RSTALL_Pos);
  42. SDIO->CR1 = (1 << SDIO_CR1_CDSRC_Pos) |
  43. (0 << SDIO_CR1_8BIT_Pos) |
  44. (0 << SDIO_CR1_4BIT_Pos) |
  45. (1 << SDIO_CR1_PWRON_Pos) |
  46. (7 << SDIO_CR1_VOLT_Pos);
  47. SDIO->CR2 = (1 << SDIO_CR2_CLKEN_Pos) |
  48. (1 << SDIO_CR2_SDCLKEN_Pos) |
  49. (calcSDCLKDiv(100000) << SDIO_CR2_SDCLKDIV_Pos) |
  50. (0xC << SDIO_CR2_TIMEOUT_Pos); // 2**25 SDIO_CLK
  51. while((SDIO->CR2 & SDIO_CR2_CLKRDY_Msk) == 0);
  52. SDIO->IM = 0xFFFFFFFF;
  53. SDIO_SendCmd(SD_CMD_GO_IDLE_STATE, 0x00, SD_RESP_NO, 0); //CMD0: GO_IDLE_STATE
  54. res = SDIO_SendCmd(SD_CMD_SEND_IF_COND, 0x1AA, SD_RESP_32b, &resp); //CMD8: SEND_IF_COND, 检测工作电压、检测是否支持SD 2.0
  55. if(res != SD_RES_OK)
  56. return res;
  57. if(resp == 0x1AA) SD_cardInfo.CardType = SDIO_STD_CAPACITY_SD_CARD_V2_0;
  58. else SD_cardInfo.CardType = SDIO_STD_CAPACITY_SD_CARD_V1_1;
  59. do //ACMD41: SD_CMD_SD_APP_OP_COND
  60. {
  61. res = SDIO_SendCmd(SD_CMD_APP_CMD, 0x00, SD_RESP_32b, &resp);
  62. if(res != SD_RES_OK)
  63. return res;
  64. if(resp != 0x120) return SD_RES_ERR; //不是SD卡,可能是MMC卡
  65. if(SD_cardInfo.CardType == SDIO_STD_CAPACITY_SD_CARD_V2_0)
  66. SDIO_SendCmd(SD_CMD_SD_APP_OP_COND, 0x80100000|0x40000000, SD_RESP_32b, &resp);
  67. else
  68. SDIO_SendCmd(SD_CMD_SD_APP_OP_COND, 0x80100000|0x00000000, SD_RESP_32b, &resp);
  69. } while(((resp >> 31) & 0x01) == 0); //上电没完成时resp[31] == 0
  70. if(((resp >> 30) & 0x01) == 1) SD_cardInfo.CardType = SDIO_HIGH_CAPACITY_SD_CARD;
  71. SDIO_SendCmd(SD_CMD_ALL_SEND_CID, 0x00, SD_RESP_128b, resps); //CMD2: SD_CMD_ALL_SEND_CID,获取CID
  72. parseCID(resps);
  73. SDIO_SendCmd(SD_CMD_SET_REL_ADDR, 0x00, SD_RESP_32b, &resp); //CMD3: SD_CMD_SET_REL_ADDR,设置RCA
  74. SD_cardInfo.RCA = resp >> 16;
  75. SDIO_SendCmd(SD_CMD_SEND_CSD, SD_cardInfo.RCA << 16, SD_RESP_128b, resps); //CMD9: SD_CMD_SEND_CSD,获取CSD
  76. parseCSD(resps);
  77. if(SD_cardInfo.CardBlockSize < 0x200) return SD_RES_ERR; //本驱动只支持以512字节为单位的读写,所以最大读写单位必须不小于512
  78. SDIO->CR2 &= ~(SDIO_CR2_SDCLKEN_Msk | SDIO_CR2_SDCLKDIV_Msk);
  79. SDIO->CR2 |= (1 << SDIO_CR2_SDCLKEN_Pos) |
  80. (calcSDCLKDiv(freq) << SDIO_CR2_SDCLKDIV_Pos); //初始化完成,SDCLK切换到高速
  81. SDIO_SendCmd(SD_CMD_SEL_DESEL_CARD, SD_cardInfo.RCA << 16, SD_RESP_32b_busy, &resp); //CMD7: 选中卡,从Standy模式进入Transfer模式
  82. SDIO->IF = SDIO_IF_TRXDONE_Msk;
  83. SDIO_SendCmd(SD_CMD_APP_CMD, SD_cardInfo.RCA << 16, SD_RESP_32b, &resp);
  84. SDIO_SendCmd(SD_CMD_APP_SD_SET_BUSWIDTH, SD_BUSWIDTH_4b, SD_RESP_32b, &resp); //切换成4位总线模式
  85. SDIO->CR1 |= (1 << SDIO_CR1_4BIT_Pos);
  86. SDIO_SendCmd(SD_CMD_SET_BLOCKLEN, 512, SD_RESP_32b, &resp); //固定块大小位512字节
  87. SD_cardInfo.CardBlockSize = 512;
  88. SDIO->BLK = 512;
  89. return SD_RES_OK;
  90. }
  91. /******************************************************************************************************************************************
  92. * 函数名称: SDIO_BlockWrite()
  93. * 功能说明: 向SD卡写入数据
  94. * 输 入: uint32_t block_addr SD卡块地址,每块512字节
  95. * uint32_t buff[] 要写入的数据
  96. * 输 出: uint32_t SD_RES_OK 操作成功 SD_RES_ERR 操作失败 SD_RES_TIMEOUT 操作超时
  97. * 注意事项: 无
  98. ******************************************************************************************************************************************/
  99. uint32_t SDIO_BlockWrite(uint32_t block_addr, uint32_t buff[])
  100. {
  101. uint32_t res, i;
  102. uint32_t addr, resp;
  103. if(SD_cardInfo.CardType == SDIO_HIGH_CAPACITY_SD_CARD) addr = block_addr;
  104. else addr = block_addr * 512;
  105. res = SDIO_SendCmdWithData(SD_CMD_WRITE_SINGLE_BLOCK, addr, SD_RESP_32b, &resp, 0, 1);
  106. if(res != SD_RES_OK)
  107. return res;
  108. while((SDIO->IF & SDIO_IF_BUFWRRDY_Msk) == 0) __NOP();
  109. SDIO->IF = SDIO_IF_BUFWRRDY_Msk;
  110. for(i = 0; i < 512/4; i++) SDIO->DATA = buff[i];
  111. while((SDIO->IF & SDIO_IF_TRXDONE_Msk) == 0) __NOP();
  112. SDIO->IF = SDIO_IF_TRXDONE_Msk;
  113. return SD_RES_OK;
  114. }
  115. /******************************************************************************************************************************************
  116. * 函数名称: SDIO_MultiBlockWrite()
  117. * 功能说明: 向SD卡写入多块数据
  118. * 输 入: uint32_t block_addr SD卡块地址,每块512字节
  119. * uint16_t block_cnt 要写入的块数
  120. * uint32_t buff[] 要写入的数据
  121. * 输 出: uint32_t SD_RES_OK 操作成功 SD_RES_ERR 操作失败 SD_RES_TIMEOUT 操作超时
  122. * 注意事项: 无
  123. ******************************************************************************************************************************************/
  124. uint32_t SDIO_MultiBlockWrite(uint32_t block_addr, uint16_t block_cnt, uint32_t buff[])
  125. {
  126. uint32_t res, i, j;
  127. uint32_t addr, resp;
  128. if(SD_cardInfo.CardType == SDIO_HIGH_CAPACITY_SD_CARD) addr = block_addr;
  129. else addr = block_addr * 512;
  130. res = SDIO_SendCmdWithData(SD_CMD_WRITE_MULT_BLOCK, addr, SD_RESP_32b, &resp, 0, block_cnt);
  131. if(res != SD_RES_OK)
  132. return res;
  133. for(i = 0; i < block_cnt; i++)
  134. {
  135. while((SDIO->IF & SDIO_IF_BUFWRRDY_Msk) == 0) __NOP();
  136. SDIO->IF = SDIO_IF_BUFWRRDY_Msk;
  137. for(j = 0; j < 512/4; j++) SDIO->DATA = buff[i*(512/4) + j];
  138. }
  139. while((SDIO->IF & SDIO_IF_TRXDONE_Msk) == 0) __NOP();
  140. SDIO->IF = SDIO_IF_TRXDONE_Msk;
  141. return SD_RES_OK;
  142. }
  143. /******************************************************************************************************************************************
  144. * 函数名称: SDIO_DMABlockWrite()
  145. * 功能说明: 通过DMA向SD卡写入多块数据
  146. * 输 入: uint32_t block_addr SD卡块地址,每块512字节
  147. * uint16_t block_cnt 要写入的块数
  148. * uint32_t buff[] 要写入的数据
  149. * 输 出: uint32_t SD_RES_OK 操作成功 SD_RES_ERR 操作失败 SD_RES_TIMEOUT 操作超时
  150. * 注意事项: 无
  151. ******************************************************************************************************************************************/
  152. uint32_t SDIO_DMABlockWrite(uint32_t block_addr, uint16_t block_cnt, uint32_t buff[])
  153. {
  154. uint32_t res;
  155. uint32_t addr, resp;
  156. if(SD_cardInfo.CardType == SDIO_HIGH_CAPACITY_SD_CARD) addr = block_addr;
  157. else addr = block_addr * 512;
  158. SDIO->DMA_MEM_ADDR = (uint32_t) buff;
  159. res = SDIO_SendCmdWithDataByDMA(SD_CMD_WRITE_MULT_BLOCK, addr, SD_RESP_32b, &resp, 0, block_cnt);
  160. if(res != SD_RES_OK)
  161. return res;
  162. while((SDIO->IF & SDIO_IF_TRXDONE_Msk) == 0) __NOP();
  163. SDIO->IF = SDIO_IF_TRXDONE_Msk;
  164. return SD_RES_OK;
  165. }
  166. /******************************************************************************************************************************************
  167. * 函数名称: SDIO_BlockRead()
  168. * 功能说明: 从SD卡读出数据
  169. * 输 入: uint32_t block_addr SD卡块地址,每块512字节
  170. * uint32_t buff[] 读出的数据
  171. * 输 出: uint32_t SD_RES_OK 操作成功 SD_RES_ERR 操作失败 SD_RES_TIMEOUT 操作超时
  172. * 注意事项: 无
  173. ******************************************************************************************************************************************/
  174. uint32_t SDIO_BlockRead(uint32_t block_addr, uint32_t buff[])
  175. {
  176. uint32_t res, i;
  177. uint32_t addr, resp;
  178. if(SD_cardInfo.CardType == SDIO_HIGH_CAPACITY_SD_CARD) addr = block_addr;
  179. else addr = block_addr * 512;
  180. res = SDIO_SendCmdWithData(SD_CMD_READ_SINGLE_BLOCK, addr, SD_RESP_32b, &resp, 1, 1);
  181. if(res != SD_RES_OK)
  182. return res;
  183. while((SDIO->IF & SDIO_IF_BUFRDRDY_Msk) == 0) __NOP();
  184. SDIO->IF = SDIO_IF_BUFRDRDY_Msk;
  185. for(i = 0; i < 512/4; i++) buff[i] = SDIO->DATA;
  186. while((SDIO->IF & SDIO_IF_TRXDONE_Msk) == 0) __NOP();
  187. SDIO->IF = SDIO_IF_TRXDONE_Msk;
  188. return SD_RES_OK;
  189. }
  190. /******************************************************************************************************************************************
  191. * 函数名称: SDIO_MultiBlockRead()
  192. * 功能说明: 从SD卡读出多块数据
  193. * 输 入: uint32_t block_addr SD卡块地址,每块512字节
  194. * uint16_t block_cnt 要读出的块数
  195. * uint32_t buff[] 读出的数据
  196. * 输 出: uint32_t SD_RES_OK 操作成功 SD_RES_ERR 操作失败 SD_RES_TIMEOUT 操作超时
  197. * 注意事项: 无
  198. ******************************************************************************************************************************************/
  199. uint32_t SDIO_MultiBlockRead(uint32_t block_addr, uint16_t block_cnt, uint32_t buff[])
  200. {
  201. uint32_t res, i, j;
  202. uint32_t addr, resp;
  203. if(SD_cardInfo.CardType == SDIO_HIGH_CAPACITY_SD_CARD) addr = block_addr;
  204. else addr = block_addr * 512;
  205. res = SDIO_SendCmdWithData(SD_CMD_READ_MULT_BLOCK, addr, SD_RESP_32b, &resp, 1, block_cnt);
  206. if(res != SD_RES_OK)
  207. return res;
  208. for(i = 0; i < block_cnt; i++)
  209. {
  210. while((SDIO->IF & SDIO_IF_BUFRDRDY_Msk) == 0) __NOP();
  211. SDIO->IF = SDIO_IF_BUFRDRDY_Msk;
  212. for(j = 0; j < 512/4; j++) buff[i*(512/4) + j] = SDIO->DATA;
  213. }
  214. while((SDIO->IF & SDIO_IF_TRXDONE_Msk) == 0) __NOP();
  215. SDIO->IF = SDIO_IF_TRXDONE_Msk;
  216. return SD_RES_OK;
  217. }
  218. /******************************************************************************************************************************************
  219. * 函数名称: SDIO_DMABlockRead()
  220. * 功能说明: 通过DMA从SD卡读出多块数据
  221. * 输 入: uint32_t block_addr SD卡块地址,每块512字节
  222. * uint16_t block_cnt 要读出的块数
  223. * uint32_t buff[] 读出的数据
  224. * 输 出: uint32_t SD_RES_OK 操作成功 SD_RES_ERR 操作失败 SD_RES_TIMEOUT 操作超时
  225. * 注意事项: 无
  226. ******************************************************************************************************************************************/
  227. uint32_t SDIO_DMABlockRead(uint32_t block_addr, uint16_t block_cnt, uint32_t buff[])
  228. {
  229. uint32_t res;
  230. uint32_t addr, resp;
  231. if(SD_cardInfo.CardType == SDIO_HIGH_CAPACITY_SD_CARD) addr = block_addr;
  232. else addr = block_addr * 512;
  233. SDIO->DMA_MEM_ADDR = (uint32_t)buff;
  234. res = SDIO_SendCmdWithDataByDMA(SD_CMD_READ_MULT_BLOCK, addr, SD_RESP_32b, &resp, 1, block_cnt);
  235. if(res != SD_RES_OK)
  236. return res;
  237. while((SDIO->IF & SDIO_IF_TRXDONE_Msk) == 0) __NOP();
  238. SDIO->IF = SDIO_IF_TRXDONE_Msk;
  239. return SD_RES_OK;
  240. }
  241. /******************************************************************************************************************************************
  242. * 函数名称: _SDIO_SendCmd()
  243. * 功能说明: SDIO向SD卡发送命令
  244. * 输 入: uint32_t cmd 命令索引
  245. * uint32_t arg 命令参数
  246. * uint32_t resp_type 响应类型,取值SD_RESP_NO、SD_RESP_32b、SD_RESP_128b、SD_RESP_32b_busy
  247. * uint32_t *resp_data 响应内容
  248. * uint32_t have_data 是否有数据传输
  249. * uint32_t data_read 1 读SD卡 0 写SD卡
  250. * uint16_t block_cnt 读写块个数
  251. * uint32_t use_dma 1 使用DMA搬运数据
  252. * 输 出: uint32_t SD_RES_OK 操作成功 SD_RES_ERR 操作失败 SD_RES_TIMEOUT 操作超时
  253. * 注意事项: 无
  254. ******************************************************************************************************************************************/
  255. uint32_t _SDIO_SendCmd(uint32_t cmd, uint32_t arg, uint32_t resp_type, uint32_t *resp_data, uint32_t have_data, uint32_t data_read, uint16_t block_cnt, uint32_t use_dma)
  256. {
  257. SDIO->BLK &= ~SDIO_BLK_COUNT_Msk;
  258. SDIO->BLK |= (block_cnt << SDIO_BLK_COUNT_Pos);
  259. SDIO->ARG = arg;
  260. SDIO->CMD = (cmd << SDIO_CMD_CMDINDX_Pos) |
  261. (0 << SDIO_CMD_CMDTYPE_Pos) |
  262. (0 << SDIO_CMD_IDXCHECK_Pos) |
  263. (0 << SDIO_CMD_CRCCHECK_Pos) |
  264. (resp_type << SDIO_CMD_RESPTYPE_Pos) |
  265. (have_data << SDIO_CMD_HAVEDATA_Pos) |
  266. (data_read << SDIO_CMD_DIRREAD_Pos) |
  267. ((block_cnt > 1) << SDIO_CMD_MULTBLK_Pos) |
  268. ((block_cnt > 1) << SDIO_CMD_BLKCNTEN_Pos) |
  269. ((block_cnt > 1) << SDIO_CMD_AUTOCMD12_Pos) |
  270. (use_dma << SDIO_CMD_DMAEN_Pos);
  271. while((SDIO->IF & SDIO_IF_CMDDONE_Msk) == 0)
  272. {
  273. if(SDIO->IF & SDIO_IF_CMDTIMEOUT_Msk)
  274. {
  275. SDIO->IF = SDIO_IF_CMDTIMEOUT_Msk;
  276. return SD_RES_TIMEOUT;
  277. }
  278. else if(SDIO->IF & SDIO_IF_ERROR_Msk)
  279. {
  280. SDIO->IF = 0xFFFFFFFF;
  281. return SD_RES_ERR;
  282. }
  283. }
  284. SDIO->IF = SDIO_IF_CMDDONE_Msk;
  285. if(resp_type == SD_RESP_32b)
  286. {
  287. resp_data[0] = SDIO->RESP[0];
  288. }
  289. else if(resp_type == SD_RESP_128b)
  290. {
  291. //寄存器中将CID/CSD[127-8]依次存放在了RESP3-0[119-0],最低位的CRC被丢掉
  292. //读出数据时调整了顺序,将CID/CSD[127-8]存放在resp_data0-3[127-8],最低8位填充0x00
  293. resp_data[0] = (SDIO->RESP[3] << 8) + ((SDIO->RESP[2] >> 24) & 0xFF);
  294. resp_data[1] = (SDIO->RESP[2] << 8) + ((SDIO->RESP[1] >> 24) & 0xFF);
  295. resp_data[2] = (SDIO->RESP[1] << 8) + ((SDIO->RESP[0] >> 24) & 0xFF);
  296. resp_data[3] = (SDIO->RESP[0] << 8) + 0x00;
  297. }
  298. return SD_RES_OK;
  299. }
  300. void parseCID(uint32_t CID_Tab[4])
  301. {
  302. uint8_t tmp = 0;
  303. /*!< Byte 0 */
  304. tmp = (uint8_t)((CID_Tab[0] & 0xFF000000) >> 24);
  305. SD_cardInfo.SD_cid.ManufacturerID = tmp;
  306. /*!< Byte 1 */
  307. tmp = (uint8_t)((CID_Tab[0] & 0x00FF0000) >> 16);
  308. SD_cardInfo.SD_cid.OEM_AppliID = tmp << 8;
  309. /*!< Byte 2 */
  310. tmp = (uint8_t)((CID_Tab[0] & 0x000000FF00) >> 8);
  311. SD_cardInfo.SD_cid.OEM_AppliID |= tmp;
  312. /*!< Byte 3 */
  313. tmp = (uint8_t)(CID_Tab[0] & 0x000000FF);
  314. SD_cardInfo.SD_cid.ProdName1 = tmp << 24;
  315. /*!< Byte 4 */
  316. tmp = (uint8_t)((CID_Tab[1] & 0xFF000000) >> 24);
  317. SD_cardInfo.SD_cid.ProdName1 |= tmp << 16;
  318. /*!< Byte 5 */
  319. tmp = (uint8_t)((CID_Tab[1] & 0x00FF0000) >> 16);
  320. SD_cardInfo.SD_cid.ProdName1 |= tmp << 8;
  321. /*!< Byte 6 */
  322. tmp = (uint8_t)((CID_Tab[1] & 0x0000FF00) >> 8);
  323. SD_cardInfo.SD_cid.ProdName1 |= tmp;
  324. /*!< Byte 7 */
  325. tmp = (uint8_t)(CID_Tab[1] & 0x000000FF);
  326. SD_cardInfo.SD_cid.ProdName2 = tmp;
  327. /*!< Byte 8 */
  328. tmp = (uint8_t)((CID_Tab[2] & 0xFF000000) >> 24);
  329. SD_cardInfo.SD_cid.ProdRev = tmp;
  330. /*!< Byte 9 */
  331. tmp = (uint8_t)((CID_Tab[2] & 0x00FF0000) >> 16);
  332. SD_cardInfo.SD_cid.ProdSN = tmp << 24;
  333. /*!< Byte 10 */
  334. tmp = (uint8_t)((CID_Tab[2] & 0x0000FF00) >> 8);
  335. SD_cardInfo.SD_cid.ProdSN |= tmp << 16;
  336. /*!< Byte 11 */
  337. tmp = (uint8_t)(CID_Tab[2] & 0x000000FF);
  338. SD_cardInfo.SD_cid.ProdSN |= tmp << 8;
  339. /*!< Byte 12 */
  340. tmp = (uint8_t)((CID_Tab[3] & 0xFF000000) >> 24);
  341. SD_cardInfo.SD_cid.ProdSN |= tmp;
  342. /*!< Byte 13 */
  343. tmp = (uint8_t)((CID_Tab[3] & 0x00FF0000) >> 16);
  344. SD_cardInfo.SD_cid.Reserved1 |= (tmp & 0xF0) >> 4;
  345. SD_cardInfo.SD_cid.ManufactDate = (tmp & 0x0F) << 8;
  346. /*!< Byte 14 */
  347. tmp = (uint8_t)((CID_Tab[3] & 0x0000FF00) >> 8);
  348. SD_cardInfo.SD_cid.ManufactDate |= tmp;
  349. }
  350. void parseCSD(uint32_t CSD_Tab[4])
  351. {
  352. uint8_t tmp = 0;
  353. /*!< Byte 0 */
  354. tmp = (uint8_t)((CSD_Tab[0] & 0xFF000000) >> 24);
  355. SD_cardInfo.SD_csd.CSDStruct = (tmp & 0xC0) >> 6;
  356. SD_cardInfo.SD_csd.SysSpecVersion = (tmp & 0x3C) >> 2;
  357. SD_cardInfo.SD_csd.Reserved1 = tmp & 0x03;
  358. /*!< Byte 1 */
  359. tmp = (uint8_t)((CSD_Tab[0] & 0x00FF0000) >> 16);
  360. SD_cardInfo.SD_csd.TAAC = tmp;
  361. /*!< Byte 2 */
  362. tmp = (uint8_t)((CSD_Tab[0] & 0x0000FF00) >> 8);
  363. SD_cardInfo.SD_csd.NSAC = tmp;
  364. /*!< Byte 3 */
  365. tmp = (uint8_t)(CSD_Tab[0] & 0x000000FF);
  366. SD_cardInfo.SD_csd.MaxBusClkFrec = tmp;
  367. /*!< Byte 4 */
  368. tmp = (uint8_t)((CSD_Tab[1] & 0xFF000000) >> 24);
  369. SD_cardInfo.SD_csd.CardComdClasses = tmp << 4;
  370. /*!< Byte 5 */
  371. tmp = (uint8_t)((CSD_Tab[1] & 0x00FF0000) >> 16);
  372. SD_cardInfo.SD_csd.CardComdClasses |= (tmp & 0xF0) >> 4;
  373. SD_cardInfo.SD_csd.RdBlockLen = tmp & 0x0F;
  374. /*!< Byte 6 */
  375. tmp = (uint8_t)((CSD_Tab[1] & 0x0000FF00) >> 8);
  376. SD_cardInfo.SD_csd.PartBlockRead = (tmp & 0x80) >> 7;
  377. SD_cardInfo.SD_csd.WrBlockMisalign = (tmp & 0x40) >> 6;
  378. SD_cardInfo.SD_csd.RdBlockMisalign = (tmp & 0x20) >> 5;
  379. SD_cardInfo.SD_csd.DSRImpl = (tmp & 0x10) >> 4;
  380. SD_cardInfo.SD_csd.Reserved2 = 0; /*!< Reserved */
  381. if ((SD_cardInfo.CardType == SDIO_STD_CAPACITY_SD_CARD_V1_1) ||
  382. (SD_cardInfo.CardType == SDIO_STD_CAPACITY_SD_CARD_V2_0))
  383. {
  384. SD_cardInfo.SD_csd.DeviceSize = (tmp & 0x03) << 10;
  385. /*!< Byte 7 */
  386. tmp = (uint8_t)(CSD_Tab[1] & 0x000000FF);
  387. SD_cardInfo.SD_csd.DeviceSize |= (tmp) << 2;
  388. /*!< Byte 8 */
  389. tmp = (uint8_t)((CSD_Tab[2] & 0xFF000000) >> 24);
  390. SD_cardInfo.SD_csd.DeviceSize |= (tmp & 0xC0) >> 6;
  391. SD_cardInfo.SD_csd.MaxRdCurrentVDDMin = (tmp & 0x38) >> 3;
  392. SD_cardInfo.SD_csd.MaxRdCurrentVDDMax = (tmp & 0x07);
  393. /*!< Byte 9 */
  394. tmp = (uint8_t)((CSD_Tab[2] & 0x00FF0000) >> 16);
  395. SD_cardInfo.SD_csd.MaxWrCurrentVDDMin = (tmp & 0xE0) >> 5;
  396. SD_cardInfo.SD_csd.MaxWrCurrentVDDMax = (tmp & 0x1C) >> 2;
  397. SD_cardInfo.SD_csd.DeviceSizeMul = (tmp & 0x03) << 1;
  398. /*!< Byte 10 */
  399. tmp = (uint8_t)((CSD_Tab[2] & 0x0000FF00) >> 8);
  400. SD_cardInfo.SD_csd.DeviceSizeMul |= (tmp & 0x80) >> 7;
  401. SD_cardInfo.CardCapacity = (SD_cardInfo.SD_csd.DeviceSize + 1) ;
  402. SD_cardInfo.CardCapacity *= (1 << (SD_cardInfo.SD_csd.DeviceSizeMul + 2));
  403. SD_cardInfo.CardBlockSize = 1 << (SD_cardInfo.SD_csd.RdBlockLen);
  404. SD_cardInfo.CardCapacity *= SD_cardInfo.CardBlockSize;
  405. }
  406. else if (SD_cardInfo.CardType == SDIO_HIGH_CAPACITY_SD_CARD)
  407. {
  408. /*!< Byte 7 */
  409. tmp = (uint8_t)(CSD_Tab[1] & 0x000000FF);
  410. SD_cardInfo.SD_csd.DeviceSize = (tmp & 0x3F) << 16;
  411. /*!< Byte 8 */
  412. tmp = (uint8_t)((CSD_Tab[2] & 0xFF000000) >> 24);
  413. SD_cardInfo.SD_csd.DeviceSize |= (tmp << 8);
  414. /*!< Byte 9 */
  415. tmp = (uint8_t)((CSD_Tab[2] & 0x00FF0000) >> 16);
  416. SD_cardInfo.SD_csd.DeviceSize |= (tmp);
  417. /*!< Byte 10 */
  418. tmp = (uint8_t)((CSD_Tab[2] & 0x0000FF00) >> 8);
  419. SD_cardInfo.CardCapacity = (uint64_t)(SD_cardInfo.SD_csd.DeviceSize + 1) * 512 * 1024;
  420. SD_cardInfo.CardBlockSize = 512;
  421. }
  422. SD_cardInfo.SD_csd.EraseGrSize = (tmp & 0x40) >> 6;
  423. SD_cardInfo.SD_csd.EraseGrMul = (tmp & 0x3F) << 1;
  424. /*!< Byte 11 */
  425. tmp = (uint8_t)(CSD_Tab[2] & 0x000000FF);
  426. SD_cardInfo.SD_csd.EraseGrMul |= (tmp & 0x80) >> 7;
  427. SD_cardInfo.SD_csd.WrProtectGrSize = (tmp & 0x7F);
  428. /*!< Byte 12 */
  429. tmp = (uint8_t)((CSD_Tab[3] & 0xFF000000) >> 24);
  430. SD_cardInfo.SD_csd.WrProtectGrEnable = (tmp & 0x80) >> 7;
  431. SD_cardInfo.SD_csd.ManDeflECC = (tmp & 0x60) >> 5;
  432. SD_cardInfo.SD_csd.WrSpeedFact = (tmp & 0x1C) >> 2;
  433. SD_cardInfo.SD_csd.MaxWrBlockLen = (tmp & 0x03) << 2;
  434. /*!< Byte 13 */
  435. tmp = (uint8_t)((CSD_Tab[3] & 0x00FF0000) >> 16);
  436. SD_cardInfo.SD_csd.MaxWrBlockLen |= (tmp & 0xC0) >> 6;
  437. SD_cardInfo.SD_csd.WriteBlockPaPartial = (tmp & 0x20) >> 5;
  438. SD_cardInfo.SD_csd.Reserved3 = 0;
  439. SD_cardInfo.SD_csd.ContentProtectAppli = (tmp & 0x01);
  440. /*!< Byte 14 */
  441. tmp = (uint8_t)((CSD_Tab[3] & 0x0000FF00) >> 8);
  442. SD_cardInfo.SD_csd.FileFormatGrouop = (tmp & 0x80) >> 7;
  443. SD_cardInfo.SD_csd.CopyFlag = (tmp & 0x40) >> 6;
  444. SD_cardInfo.SD_csd.PermWrProtect = (tmp & 0x20) >> 5;
  445. SD_cardInfo.SD_csd.TempWrProtect = (tmp & 0x10) >> 4;
  446. SD_cardInfo.SD_csd.FileFormat = (tmp & 0x0C) >> 2;
  447. SD_cardInfo.SD_csd.ECC = (tmp & 0x03);
  448. }
  449. uint32_t calcSDCLKDiv(uint32_t freq)
  450. {
  451. uint32_t prediv;
  452. switch((SYS->CLKSEL & SYS_CLKSEL_SDIO_Msk) >> SYS_CLKSEL_SDIO_Pos)
  453. {
  454. case 0: prediv = 1; break;
  455. case 1: prediv = 3; break;
  456. case 2: prediv = 2; break;
  457. case 3: prediv = 0; break;
  458. }
  459. uint32_t clkdiv = (SystemCoreClock / (1 << prediv)) / freq;
  460. uint32_t regdiv = 0;
  461. if(clkdiv > 128) regdiv = 0x80;
  462. else if(clkdiv > 64) regdiv = 0x40;
  463. else if(clkdiv > 32) regdiv = 0x20;
  464. else if(clkdiv > 16) regdiv = 0x10;
  465. else if(clkdiv > 8) regdiv = 0x08;
  466. else if(clkdiv > 4) regdiv = 0x04;
  467. else if(clkdiv > 2) regdiv = 0x02;
  468. else if(clkdiv > 1) regdiv = 0x01;
  469. else regdiv = 0x00;
  470. return regdiv;
  471. }