SWM320_sdio.c 23 KB

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