sdcmd.c
上传用户:tsengnan
上传日期:2010-03-31
资源大小:22k
文件大小:34k
源码类别:

uCOS

开发平台:

C/C++

  1. /**--------------File Info-------------------------------------------------------------------------------
  2. ** File name: sddriver.c
  3. ** Last modified Date: 2005-1-6
  4. ** Last Version: V1.0
  5. ** Descriptions: sd 卡驱动软件包: SD卡物理层 ---- SD卡SPI模式支持的命令 
  6. ** Soft Packet of SD Card Driver: commands that sd card supported in spi mode
  7. **
  8. **------------------------------------------------------------------------------------------------------
  9. ** Created by: gan peng
  10. ** Created date: 2005-1-6
  11. ** Version: V1.0
  12. ** Descriptions: The original version
  13. **
  14. **------------------------------------------------------------------------------------------------------
  15. ** Modified by:
  16. ** Modified date:
  17. ** Version:
  18. ** Descriptions:
  19. **
  20. **------------------------------------------------------------------------------------------------------
  21. ** Modified by: 
  22. ** Modified date:
  23. ** Version:
  24. ** Descriptions: 
  25. **
  26. ********************************************************************************************************/
  27. #include "config.h"
  28. #include "sdconfig.h"
  29. #include "sdhal.h"
  30. #include "sdcrc.h"
  31. #include "sdcmd.h"
  32. #include "sddriver.h"
  33. /********************************************************************************************************************
  34. ** 函数名称: INT8U SD_SendCmd() Name:   INT8U SD_SendCmd()
  35. ** 功能描述: 向卡发送命令,并取得响应 Function: send command to the card,and get a response
  36. ** 输   入: INT8U cmd     : 命令字 Input:   INT8U cmd     : command byte
  37.  INT8U *param : 命令参数,长度为4字节   INT8U *param : command parameter,length is 4 bytes  
  38.  INT8U resptype : 响应类型   INT8U resptype: response type
  39.  INT8U *resp : 响应,长度为1-5字节   INT8U *resp : response,length is 1-5 bytes
  40. ** 输   出: 0:   正确    >0:   错误码    Output:   0:  right >0:  error code
  41. ********************************************************************************************************************/
  42. INT8U SD_SendCmd(INT8U cmd, INT8U *param, INT8U resptype, INT8U *resp)
  43. {
  44. INT32 i,rlen;
  45. INT8U tmp;
  46.     
  47. SPI_CS_Assert();
  48.    
  49.     SPI_SendByte((cmd & 0x3F) | 0x40);  /* 发送命令头和命令字 send command header and word */
  50.     
  51.     for (i = 3; i >= 0; i--)
  52.         SPI_SendByte(param[i]);  /* 发送参数 send parameters */
  53. #if SD_CRC_EN
  54. tmp = SD_GetCmdByte6((cmd & 0x3F) | 0x40, param);
  55. SPI_SendByte(tmp);
  56. #else
  57.     SPI_SendByte(0x95);    /* CRC校验码,只用于第1个命令 CRC,only used for the first command */
  58. #endif 
  59.     
  60.     rlen = 0;
  61.     switch (resptype)  /* 根据不同的命令,得到不同的响应长度 */
  62.     {  /* according various command,get the various response length */
  63.    case R1:
  64.       case R1B: rlen = 1;  break;
  65.          
  66.      case R2:  rlen = 2;  break;
  67.          
  68.     case R3:  rlen = 5;  break;
  69.          
  70.      default:  SPI_SendByte(0xFF);
  71.              SPI_CS_Deassert();  
  72.                return SD_ERR_CMD_RESPTYPE;  /* 返回命令响应类型错误 return error of command response type */
  73.            break;
  74.     }
  75.     
  76.     i = 0;
  77.     do   /* 等待响应,响应的开始位为0 */
  78.     {  /* Wait for a response,a response is a start bit(zero) */ 
  79.         tmp = SPI_RecByte();
  80.         i++;
  81.     }while (((tmp & 0x80) != 0) && (i < SD_CMD_TIMEOUT));
  82.     
  83.     if (i >= SD_CMD_TIMEOUT)
  84.     {  
  85.         SPI_CS_Deassert();
  86.         return SD_ERR_CMD_TIMEOUT;  /* 返回命令超时 return response timeout of command */
  87.     }
  88.     
  89.     for (i = rlen - 1; i >= 0; i--)
  90.     {
  91.         resp[i] = tmp;
  92.         tmp = SPI_RecByte();    /* 循环的最后发送8clock  at the last recycle,clock out 8 clock */
  93.     }
  94.       
  95.     SPI_CS_Deassert();
  96.     return SD_NO_ERR;  /* 返回执行成功 return perform sucessfully */
  97. }
  98. /********************************************************************************************************************
  99. ** 函数名称: void SD_PackParam() Name:   void SD_PackParam()
  100. ** 功能描述: 将32位的参数转为字节形式 Function: change 32bit parameter to bytes form 
  101. ** 输   入: INT8U *parameter: 字节参数缓冲区 Input:   INT8U *parameter: the buffer of bytes parameter
  102.  INT32U value    : 32位参数   INT32U value    : 32bit parameter
  103. ** 输   出: 无 Output:   NULL
  104. *********************************************************************************************************************/
  105. void SD_PackParam(INT8U *parameter, INT32U value)
  106. {
  107.     parameter[3] = (INT8U)(value >> 24);
  108.     parameter[2] = (INT8U)(value >> 16);
  109.     parameter[1] = (INT8U)(value >> 8);
  110.     parameter[0] = (INT8U)(value);
  111. }
  112. /********************************************************************************************************************
  113. ** 函数名称: INT8U SD_BlockCommand() Name:   INT8U SD_BlockCommand()
  114. ** 功能描述: 块命令 Function: command about block operation
  115. ** 输   入: INT8U cmd      : 命令字 Input:   INT8U cmd       : command byte 
  116.  INT8U resptype  : 响应类型   INT8U resptype  : response type
  117.  INT32U parameter: 块操作参数     INT32U parameter: parameter of block operation
  118. ** 输   出: 0:   正确    >0:   错误码    Output:   0:  right >0:  error code
  119. *********************************************************************************************************************/
  120. INT8U SD_BlockCommand(INT8U cmd, INT8U resptype, INT32U parameter)
  121. {
  122. INT8U param[4],resp,ret;
  123. parameter <<= SD_BLOCKSIZE_NBITS;  /* 调整地址:左移9位 adjust address: move 9 bits left */
  124. SD_PackParam(param, parameter);  /* 将参数转化为字节形式 change the parameter to bytes form */
  125.    /* 结束数据传输失败 stop transmission operation fail */
  126. if (resp != 0)
  127.  return SD_ERR_CMD_RESP;    /* 响应错误 response is error */
  128.  
  129. return SD_NO_ERR;
  130. }
  131. /*
  132. ************************************************
  133.   下面为SD卡SPI命令
  134. ************************************************
  135. */
  136. /********************************************************************************************************************
  137. ** 函数名称: INT8U SD_ResetSD() Name:   INT8U SD_ResetSD()
  138. ** 功能描述: 复位SD卡 Function: reset sd card
  139. ** 输   入: 无 Input:   NULL
  140. ** 输   出: 0:   正确    >0:   错误码    Output:   0:  right >0:  error code
  141. *********************************************************************************************************************/
  142. INT8U SD_ResetSD(void)
  143. {
  144. INT8U param[4] = {0,0,0,0},resp;
  145.     return (SD_SendCmd(CMD0, param, CMD0_R, &resp)); /* 复位命令 command that reset sd card */
  146. }
  147. /********************************************************************************************************************
  148. ** 函数名称: INT8U SD_ReadCSD() Name:   INT8U SD_ReadCSD()
  149. ** 功能描述: 读SD卡的CSD寄存器 Function: read CSD register of sd card 
  150. ** 输   入: INT8U csdlen  : 寄存器长度(固定为16)   INT8U csdlen  : len of register (fixed,is 16)
  151.  INT8U *recbuf : 接收缓冲区       INT8U *recbuf : recbuffer
  152. ** 输   出: 0:   正确    >0:   错误码    Output:   0:  right >0:  error code
  153. *********************************************************************************************************************/
  154. INT8U SD_ReadCSD(INT8U csdlen, INT8U *recbuf)
  155. {
  156.     INT8U param[4] = {0,0,0,0},resp,ret;
  157.       
  158.   
  159.     if (resp != 0)
  160.         return SD_ERR_CMD_RESP; /* 响应错误 response is error */
  161.     
  162. return (SD_ReadRegister(csdlen, recbuf));
  163. }
  164. /*******************************************************************************************************************
  165. ** 函数名称: INT8U SD_ReadCID() Name:   INT8U SD_ReadCID()
  166. ** 功能描述: 读SD卡的CID寄存器 Function: read CID register of sd card
  167. ** 输   入: INT8U cidlen  : 寄存器长度(固定为16)   INT8U cidlen  : len of register (fixed,is 16)
  168.  INT8U *recbuf : 接收缓冲区       INT8U *recbuf : recbuffer
  169. ** 输   出: 0:   正确    >0:   错误码    Output:   0:  right >0:  error code
  170. ********************************************************************************************************************/
  171. #if SD_ReadCID_EN
  172. INT8U SD_ReadCID(INT8U cidlen, INT8U *recbuf)
  173. {
  174.     INT8U param[4] = {0,0,0,0},resp,ret;
  175.  
  176.     ret = SD_SendCmd(CMD10, param, CMD10_R, &resp); /* 读CID寄存器命令 command that read CID register */
  177.     if ( ret != SD_NO_ERR)
  178.     return ret;   
  179.    
  180.     if (resp != 0)
  181.         return SD_ERR_CMD_RESP; /* 响应错误 response is error */
  182.       
  183.    return (SD_ReadRegister(cidlen, recbuf));
  184. }
  185. #endif
  186. /********************************************************************************************************************
  187. ** 函数名称: INT8U SD_StopTransmission() Name:   INT8U SD_StopTransmission()
  188. ** 功能描述: 停止数据传输 Function: stop data transmission 
  189. ** 输   入: 无   Input:    NULL
  190. ** 输   出: 0:   正确    >0:   错误码    Output:   0:  right >0:  error code
  191. *********************************************************************************************************************/
  192. INT8U SD_StopTransmission(void)
  193. {
  194. INT8U param[4] = {0,0,0,0},resp;
  195.   return (SD_SendCmd(CMD12, param, CMD12_R, &resp)); /* 结束数据传输命令失败 stop transmission command fail */
  196. }
  197. /*********************************************************************************************************************
  198. ** 函数名称: INT8U SD_ReadCard_Status() Name:   INT8U SD_ReadCard_Status()
  199. ** 功能描述: 读SD卡的 Card Status 寄存器 Function: read Card Status register of sd card 
  200. ** 输   入: INT8U len:   寄存器长度(固定为2)   INT8U len:      len of register (fixed,is 2)
  201.  INT8U *recbuf : 接收缓冲区       INT8U *recbuf : recbuffer
  202. ** 输   出: 0:   正确    >0:   错误码    Output:   0:  right >0:  error code
  203. **********************************************************************************************************************/
  204. INT8U SD_ReadCard_Status(INT8U len, INT8U *buffer)
  205. {
  206.     INT8U param[4] = {0,0,0,0};
  207.     return (SD_SendCmd(CMD13, param, CMD13_R, buffer)); /* 读 Card Status 寄存器 */
  208.          /* read register of Card Status */
  209. }
  210. /********************************************************************************************************************
  211. ** 函数名称: INT8U Sd_SetBlockLen() Name:   INT8U Sd_SetBlockLen()
  212. ** 功能描述: 设置一个块的长度 Function: set a block len of sd card 
  213. ** 输   入: INT32U length : 块的长度值 Input:   INT32U length : the length of a block
  214. ** 输   出: 0:   正确    >0:   错误码    Output:   0:  right >0:  error code
  215. *********************************************************************************************************************/
  216. INT8U SD_SetBlockLen(INT32U length)
  217. {
  218. INT8U param[4],resp,ret;
  219.   
  220.     SD_PackParam(param, length); /* 将参数转化为字节形式 change the parameter to bytes form */
  221.           
  222.     ret = SD_SendCmd(CMD16, param, CMD16_R, &resp);
  223.     if (ret != SD_NO_ERR)
  224.   return ret; /* 设置块的长度为length失败 set the length of block to length fail */
  225. if (resp != 0)
  226.      return SD_ERR_CMD_RESP;     /* 响应错误 response is error */
  227.     
  228.     return SD_NO_ERR;  /* 返回执行成功 return perform sucessfully */
  229. }
  230. /********************************************************************************************************************
  231. ** 函数名称: INT8U SD_ReadSingleBlock() Name:   INT8U SD_ReadSingleBlock()
  232. ** 功能描述: 读单块命令 Function: read single block command
  233. ** 输   入: INT32U blockaddr: 块地址 Input:   INT32U blockaddr: block address
  234. ** 输   出: 0:   正确    >0:   错误码    Output:   0:  right >0:  error code
  235. *********************************************************************************************************************/
  236. INT8U SD_ReadSingleBlock(INT32U blockaddr)
  237. {
  238. return (SD_BlockCommand(CMD17, CMD17_R, blockaddr)); /* 读单块命令 command that read single block */
  239. }
  240. /********************************************************************************************************************
  241. ** 函数名称: INT8U SD_ReadMultipleBlock() Name:   INT8U SD_ReadMultipleBlock()
  242. ** 功能描述: 读多块命令 Function: read multiple block command 
  243. ** 输   入: INT32U blockaddr: 块地址 Input:   INT32U blockaddr: block address
  244. ** 输   出: 0:   正确    >0:   错误码    Output:   0:  right >0:  error code
  245. *********************************************************************************************************************/
  246. INT8U SD_ReadMultipleBlock(INT32U blockaddr)
  247. {
  248. return (SD_BlockCommand(CMD18, CMD18_R, blockaddr)); /* 读多块命令 command that read multiple block */
  249. }
  250. /********************************************************************************************************************
  251. ** 函数名称: INT8U SD_WriteSingleBlock() Name:   INT8U SD_WriteSingleBlock()
  252. ** 功能描述: 写单块命令 Function: write single block command
  253. ** 输   入: INT32U blockaddr: block address Input:   INT32U blockaddr: block address
  254. ** 输   出: 0:   正确    >0:   错误码    Output:   0:  right >0:  error code
  255. *********************************************************************************************************************/
  256. INT8U SD_WriteSingleBlock(INT32U blockaddr)
  257. {
  258. return (SD_BlockCommand(CMD24, CMD24_R, blockaddr)); /* 写单块命令 command that write single block */
  259. }
  260. /********************************************************************************************************************
  261. ** 函数名称: INT8U SD_WriteMultipleBlock() Name:   INT8U SD_WriteMultipleBlock()
  262. ** 功能描述: 写多块命令 Function: write multiple block command
  263. ** 输   入: INT32U blockaddr: 块地址 Input:   INT32U blockaddr: block address
  264. ** 输   出: 0:   正确    >0:   错误码    Output:   0:  right >0:  error code
  265. *********************************************************************************************************************/
  266. INT8U SD_WriteMultipleBlock(INT32U blockaddr)
  267. {
  268. return (SD_BlockCommand(CMD25, CMD25_R, blockaddr)); /* 写多块命令 command that write multiple block */
  269. }
  270. /********************************************************************************************************************
  271. ** 函数名称: INT8U SD_ProgramCSD() Name:   INT8U SD_ProgramCSD()
  272. ** 功能描述: 写CSD寄存器 Function: write CSD register
  273. ** 输   入: INT8U *buff     : CSD寄存器内容 Input:   INT8U *buff      : the content of CSD register
  274.  INT8U len     : CSD寄存器长度      INT8U len   : the length of CSD register
  275. ** 输   出: 0:   正确    >0:   错误码    Output:   0:  right >0:  error code
  276. ********************************************************************************************************************/
  277. #if SD_ProgramCSD_EN
  278. INT8U SD_ProgramCSD(INT8U len, INT8U *buff)
  279. {
  280. INT8U param[4] = {0,0,0,0},resp,ret;
  281. if (len != 16) return SD_ERR_USER_PARAM;
  282. ret = SD_SendCmd(CMD27, param, CMD27_R, &resp);  /* 发送写CSD寄存器命令 send command that write CSD */
  283. if (ret != SD_NO_ERR)
  284. return ret;
  285.         
  286.     if (resp != 0)    
  287.         return SD_ERR_CMD_RESP;
  288. buff[15] = (SD_GetCRC7(buff, 15) << 1) + 0x01;   /* 计算CSD中的crc 位域 calculate crc field in CSD */
  289. return(SD_WriteBlockData(0, 16, buff));
  290. }
  291. /********************************************************************************************************************
  292. ** 函数名称: INT8U SD_GetCRC7() Name:   INT8U SD_GetCRC7()
  293. ** 功能描述: 计算CRC7 Function: calculate crc7
  294. ** 输   入: INT8U *pSource: 数据 Input:    INT8U *pSource: data
  295.  INT16U len    : 数据长度   INT16U len   : data length
  296. ** 输   出: CRC7码 Output:   CRC7 code
  297. *********************************************************************************************************************/
  298. INT8U SD_GetCRC7(INT8U *pSource, INT16U len)
  299. {
  300. INT8U i = 0, j;
  301. INT8U reg = 0;
  302. do
  303. {
  304.     for (j = 0; j < 8; j++)
  305.     {
  306. reg <<= 1;
  307. reg ^= ((((pSource[i] << j) ^ reg) & 0x80) ? 0x9 : 0);
  308.     }
  309.     
  310.     i++;
  311.     
  312. }while(i < len);
  313. return reg;
  314. }
  315. #endif
  316. #if SD_EraseBlock_EN
  317. /********************************************************************************************************************
  318. ** 函数名称: INT8U SD_EraseStartBlock() Name:   INT8U SD_EraseStartBlock()
  319. ** 功能描述: 设置块擦除起始地址 Function: select the start block address of erasing operation 
  320. ** 输   入: INT32U startblock: 块地址 Input:    INT32U startblock : block address
  321. ** 输   出: 0:   正确    >0:   错误码    Output:   0:  right    >0:  error code
  322. *********************************************************************************************************************/
  323. INT8U SD_EraseStartBlock(INT32U startblock)
  324. {
  325. return (SD_BlockCommand(CMD32, CMD32_R, startblock)); /* 发送擦除起始块地址 send the start block address of erasing operation */
  326. }
  327. /********************************************************************************************************************
  328. ** 函数名称: INT8U SD_EraseEndBlock() Name:   INT8U SD_EraseEndBlock()
  329. ** 功能描述: 设置块擦除终止地址 Function: select the end block address of erasing operation  
  330. ** 输   入: INT32U endblock: 块地址 Input:   INT32U Length : block address
  331. ** 输   出: 0:   正确    >0:   错误码    Output:   0:  right    >0:  error code
  332. *********************************************************************************************************************/
  333. INT8U SD_EraseEndBlock(INT32U endblock)
  334. {
  335. return (SD_BlockCommand(CMD33, CMD33_R, endblock));     /* 发送擦除终止块地址 send the end block address of erasing operation */
  336. }
  337. /********************************************************************************************************************
  338. ** 函数名称: INT8U SD_EraseSelectedBlock() Name:   INT8U SD_EraseSelectedBlock()
  339. ** 功能描述: 擦除已选中的块 Function: erase block selected
  340. ** 输   入: 无 Input:   NULL
  341. ** 输   出: 0:   正确    >0:   错误码    Output:   0:  right >0:  error code
  342. *********************************************************************************************************************/
  343. INT8U SD_EraseSelectedBlock(void)
  344. {
  345. INT8U param[4],resp,tmp;
  346. SD_PackParam(param, 0);
  347. tmp = SD_SendCmd(CMD38, param, CMD38_R, &resp);       /* 擦除所选择的块  erase blocks selected */
  348. if (tmp != SD_NO_ERR)
  349. return tmp;  
  350. if (SD_WaitBusy(SD_WAIT_ERASE) != SD_NO_ERR) /* 等待擦除完成 wait for finishing erasing */
  351. return SD_ERR_TIMEOUT_ERASE;
  352. else
  353. return SD_NO_ERR;
  354. }
  355. #endif
  356. /*********************************************************************************************************************
  357. ** 函数名称: INT8U SD_ReadOCR() Name:   INT8U SD_ReadOCR()
  358. ** 功能描述: 读操作条件寄存器OCR Function: read OCR register of sd card
  359. ** 输   入: INT8U ocrlen  : 寄存器长度(固定为4) Input:   INT8U ocrlen  : len of register (fixed,is 4)
  360.  INT8U *recbuf : 接收缓冲区       INT8U *recbuf : recbuffer
  361. ** 输   出: 0:   正确    >0:   错误码    Output:   0:  right >0:  error code
  362. **********************************************************************************************************************/
  363. INT8U SD_ReadOCR(INT8U ocrlen, INT8U *recbuf)
  364. {
  365.     INT8U param[4] = {0,0,0,0},resp[5],tmp;
  366.     tmp = SD_SendCmd(CMD58, param, CMD58_R, resp); /* 读 OCR 寄存器命令 */
  367.     if (tmp != SD_NO_ERR) /* read OCR register command */
  368.      return tmp;  
  369.     
  370.     if (resp[0] != 0)
  371.         return SD_ERR_CMD_RESP;   /* 响应错误 response is error */
  372.     
  373.     for (tmp = 0; tmp < 4; tmp++)
  374.      recbuf[tmp] = resp[tmp + 1]; /* 复制OCR寄存器内容到接收缓冲区 */
  375.     
  376.     return SD_NO_ERR;
  377. }
  378. /********************************************************************************************************************
  379. ** 函数名称: INT8U SD_EnableCRC() Name:   INT8U SD_EnableCRC()
  380. ** 功能描述: 使能SD卡的CRC校验功能 Function: enable crc check of SD Card 
  381. ** 输   入: INT8U bEnable : 1:使能 0:禁止 Input:   INT8U bEnable : 1:enable  0: disable
  382. ** 输   出: 0:   正确    >0:   错误码    Output:   0:  right >0:  error code
  383. *********************************************************************************************************************/
  384. #if SD_CRC_EN
  385. INT8U SD_EnableCRC(INT8U bEnable)
  386. {
  387. INT8U param[4],resp,ret;
  388. if (bEnable == 1)
  389. param[0] = 1; /* 使能crc enable crc */
  390. else
  391. param[1] = 0;  /* 禁止crc disalbe crc */
  392. ret = SD_SendCmd(CMD59, param, CMD59_R, &resp); /* "使能/禁止CRC"命令 enable/disable crc command */
  393. if (ret != SD_NO_ERR)
  394.         return ret;  
  395.     
  396.     if (resp != 0)    
  397.         return SD_ERR_CMD_RESP; /* 响应错误 response is error */
  398.  
  399. return SD_NO_ERR;
  400. }
  401. #endif
  402. /*********************************************************************************************************************
  403. ** 函数名称: INT8U SD_Read_SD_Status()      Name:    INT8U SD_Read_SD_Status()
  404. ** 功能描述: 读SD卡的 SD_Status 寄存器      Function: read SD_Status register of sd card 
  405. ** 输   入: INT8U sdslen   : 寄存器长度(固定为64)  Input:    INT8U sdslen: len of register (fixed,is 64)
  406.  INT8U *recbuf  : 接收缓冲区        INT8U *recbuf: recbuffer
  407. ** 输   出: 0:   正确    >0:   错误码     Output:   0:  right >0:  error code
  408. **********************************************************************************************************************/
  409. #if SD_ReadSD_Status_EN
  410. INT8U SD_ReadSD_Status(INT8U sdslen, INT8U *recbuf)
  411. {
  412.     INT8U param[4] = {0,0,0,0},resp[2],ret;
  413.     
  414.     ret = SD_SendCmd(CMD55, param, CMD55_R, resp); /* 后续命令为一个应用命令 */
  415.     if (ret != SD_NO_ERR)
  416.      return ret; /* command that the followed commnad is a specific application */
  417.       
  418.     if (resp[0] != 0)
  419.         return SD_ERR_CMD_RESP; /* 响应错误 response is error */
  420.     
  421.     ret = SD_SendCmd(ACMD13, param, ACMD13_R, resp); /* 读 SD_Status 命令 */
  422.     if (ret != SD_NO_ERR)
  423.      return ret; /* command that read SD_Status register */
  424.    
  425.     if ((resp[0] != 0) || (resp[1] != 0))
  426.         return SD_ERR_CMD_RESP; /* 响应错误 response is error */
  427.         
  428. return (SD_ReadBlockData(sdslen, recbuf)); /* 读出寄存器内容 read the content of the register */
  429. }
  430. #endif
  431. /*******************************************************************************************************************
  432. ** 函数名称: INT8U SD_ReadSCR() Name:   INT8U SD_ReadSCR()
  433. ** 功能描述: 读SD卡的 SCR 寄存器 Function: read SCR register of sd card 
  434. ** 输   入: INT8U scrlen   : 寄存器长度(固定为8)  Input:    INT8U scrlen  : len of register (fixed,is 8)
  435.  INT8U *recbuf  : 接收缓冲区   INT8U *recbuf  : recbuffer
  436. ** 输   出: 0:   正确    >0:   错误码    Output:   0:  right >0:  error code
  437. ********************************************************************************************************************/
  438. #if SD_ReadSCR_EN
  439. INT8U SD_ReadSCR(INT8U scrlen, INT8U *recbuf)
  440. {
  441.     INT8U param[4] = {0,0,0,0},resp,ret;
  442.     
  443.     ret = SD_SendCmd(CMD55, param, CMD55_R, &resp); /* 后续命令为一个应用命令 */
  444.     if (ret != SD_NO_ERR) /* command that the followed commnad is a specific application */
  445.      return ret;
  446.       
  447.     if (resp != 0)
  448.         return SD_ERR_CMD_RESP; /* 响应错误 response is error */
  449.     
  450.     ret = SD_SendCmd(ACMD51, param, ACMD51_R, &resp);    /* 发送读 SD_Status 命令*/
  451.     if (ret != SD_NO_ERR) /* command that read SD_Status register */
  452.     return ret;
  453.     
  454.     if (resp != 0)
  455.         return SD_ERR_CMD_RESP;   /* 响应错误 response is error */
  456.         
  457. return (SD_ReadBlockData(scrlen, recbuf));   /* 读出寄存器内容 read the content of the register */
  458. }
  459. #endif
  460. /********************************************************************************************************************
  461. ** 函数名称: INT8U SD_GetNumWRBlcoks() Name:   INT8U SD_GetNumWRBlcoks()
  462. ** 功能描述: 得到正确写入的块数 Function: get the block numbers that written correctly
  463. ** 输   入: INT32U *blocknum: 返回的块数 Input:   INT32U blocknum : the block numbers returned
  464. ** 输   出: 0:   正确    >0:   错误码    Output:   0:  right >0:  error code
  465. *********************************************************************************************************************/
  466. #if SD_WriteMultiBlock_EN
  467. INT8U SD_GetNumWRBlcoks(INT32U *blocknum)
  468. {
  469.     INT8U tmp[4] = {0,0,0,0},resp,ret;
  470.   
  471.     ret = SD_SendCmd(CMD55, tmp, CMD55_R, &resp);    /* 后续命令为一个应用命令 */
  472.     if (ret != SD_NO_ERR)  /* command that the followed commnad is a specific application */
  473.      return ret;
  474.       
  475.     if (resp != 0)
  476.      return SD_ERR_CMD_RESP;    
  477.  
  478.     ret = SD_SendCmd(ACMD22, tmp, ACMD22_R, &resp);   /* 读取正确写入的块数命令 */
  479.     if (ret != SD_NO_ERR) /* command that read the numbers of block written correctly */
  480.     return ret;     
  481.      
  482. if (resp != 0)
  483.      return SD_ERR_CMD_RESP; /* 响应错误 response is error */
  484.     
  485.     ret = SD_ReadBlockData(4, tmp); /* 读块数 read the numbvers of block */
  486.     if (ret != SD_NO_ERR)
  487.      return ret;
  488.     
  489.     *blocknum = (tmp[0] << 24) + (tmp[1] << 16) + (tmp[2] << 8) + tmp[3];
  490.      /* 转换为32位 change to 32 bits */
  491. return SD_NO_ERR;     /* 返回执行成功 return perform sucessfully */
  492. }
  493. #endif
  494. /*********************************************************
  495.      下面为一些数据传输函数
  496. **********************************************************/
  497. /********************************************************************************************************************
  498. ** 函数名称: INT8U SD_ReadRegister() Name:   INT8U SD_ReadRegister()
  499. ** 功能描述: 从SD卡读取数据 Function: read data from SD card
  500. ** 输   入: INT32U len   : 长度 Input:   INT32U len   : length
  501.  INT8U *recbuf: 接收缓冲区     INT8U *recbuf: receive buffer
  502. ** 输   出: 0:   正确    >0:   错误码    Output:   0:  right >0:  error code
  503. *********************************************************************************************************************/
  504. INT8U SD_ReadRegister(INT32U len, INT8U *recbuf)
  505. {
  506. INT32U i = 0;
  507. INT8U resp;
  508.     SPI_CS_Assert();
  509.     do{     /* 等待数据起始令牌 wait for data start token */
  510.         resp = SPI_RecByte();
  511.      i++;
  512.     }while((resp == 0xFF) && (i < SD_READREG_TIMEOUT));    
  513.     
  514.     if (i >= SD_READREG_TIMEOUT)
  515.     {
  516.      SPI_CS_Deassert();
  517.      return SD_ERR_TIMEOUT_READ; /* 超时, 返回错误 timeout, return error */
  518.    }
  519.   
  520.     if (resp != SD_TOK_READ_STARTBLOCK)
  521.     { /* 不是收到数据起始令牌 not receive data start token */
  522. recbuf[0] = resp;
  523. i = 1; /* 还有len - 1个字节要接收 still len - 1 bytes will be received */
  524.     }
  525.     else
  526.     i = 0; /* 收到数据起始令牌,还有len个字节要接收 received data start token,still len bytes will be received */
  527.       
  528.     for (; i < len; i++)
  529.     recbuf[i] = SPI_RecByte(); /* 接收数据 receive data */
  530.    
  531.     i = SPI_RecByte();
  532.     i = (i << 8) + SPI_RecByte();     /* 读取16位CRC get 16-bit CRC */
  533. #if SD_CRC_EN 
  534.     if (i != SD_GetCRC16(recbuf, len))
  535.     { /* CRC校验错误 CRC check is error */
  536.     SPI_CS_Deassert();
  537.     SPI_SendByte(0xFF);    
  538.    return SD_ERR_DATA_CRC16; /* 返回RCR16错误  return error of CRC16 */
  539.    }    
  540. #endif   
  541.   
  542.     SPI_SendByte(0xFF); /* 返回之前发送8个clock  clock out 8 clk before return */
  543.     SPI_CS_Deassert();
  544.     
  545. return SD_NO_ERR;
  546. }
  547. /*******************************************************************************************************************
  548. ** 函数名称: INT8U SD_ReadBlockData() Name:   INT8U SD_ReadBlockData()
  549. ** 功能描述: 从SD卡中读取数据块 Function: read block data from sd card
  550. ** 输   入: INT32U len    : 长度 Input:   INT32U len    : length
  551.  INT8U *recbuf : 接收缓冲区   INT8U *recbuf : the buffer of receive
  552. ** 输   出: 0:   正确    >0:   错误码    Output:   0:  right >0:  error code
  553. *******************************************************************************************************************/
  554. INT8U SD_ReadBlockData(INT32U len, INT8U *recbuf)
  555. {
  556.     INT8U tmp;
  557.     INT32U i = 0;
  558.     
  559.     SPI_CS_Assert();    
  560.     do
  561.     {      /* 等待接收数据开始令牌0xFE  wait for receiving data start token 0xFE */
  562.         tmp = SPI_RecByte();
  563.         i++;
  564.      }while((tmp == 0xFF) && (i < sds.timeout_read));
  565. if (i >= sds.timeout_read)
  566. {
  567. SPI_CS_Deassert();
  568.         return SD_ERR_TIMEOUT_READ; /* 返回读超时错误码  return error timeout error code of reading */
  569. }
  570. if (tmp != SD_TOK_READ_STARTBLOCK) /* 块读开始令牌错误 read start block token is error */
  571. {
  572. SPI_SendByte(0xFF);
  573. SPI_CS_Deassert();
  574. return SD_ERR_DATA_START_TOK;
  575. }
  576. for (i = 0; i < len; i++)
  577.     recbuf[i] = SPI_RecByte(); /* 接收数据 receive data */
  578.    
  579.     i = SPI_RecByte();
  580.     i = (i << 8) + SPI_RecByte();     /* 读取16位CRC get 16-bit CRC */
  581. #if SD_CRC_EN 
  582.     if (i != SD_GetCRC16(recbuf, len))
  583.     { /* CRC校验错误 CRC check is error */
  584.     SPI_SendByte(0xFF); 
  585.     SPI_CS_Deassert();  
  586.    return SD_ERR_DATA_CRC16; /* 返回CRC16错误  return error of CRC16 */
  587.    }    
  588. #endif   
  589. SPI_SendByte(0xFF); 
  590. SPI_CS_Deassert();
  591.    return SD_NO_ERR; /* 返回函数执行成功 return function perform sucessfully */
  592. }
  593. /*******************************************************************************************************************
  594. ** 函数名称: INT8U SD_WriteBlockData() Name:   INT8U SD_WriteBlockData()
  595. ** 功能描述: 向sd卡写数据块 Function: write block data to sd card
  596. ** 输   入: INT8U bmulti  : 是否为多块操作1:是0:否 Input:   INT8U bmulti   : multi blocks operate 1:Y 0:N 
  597.  INT32U len    : 长度      INT32U len     : length
  598.  INT8U *sendbuf: 发送缓冲区     INT8U *sendbuf : the buffer of send
  599. ** 输   出: 0:   正确    >0:   错误码    Output:   0:  right >0:  error code
  600. ********************************************************************************************************************/
  601. INT8U SD_WriteBlockData(INT8U bmulti, INT32U len, INT8U *sendbuf)
  602. {
  603. INT16U i;
  604. INT8U tmp;
  605. SPI_CS_Assert();
  606.     SPI_SendByte(0xFF); /* 开始发送数据之前发送8个clock clock out 8 clk before start */
  607.     
  608.     if (bmulti == 1)
  609.         SPI_SendByte(SD_TOK_WRITE_STARTBLOCK_M); /* 写多块开始令牌 start token of write multi blocks */
  610. else
  611. SPI_SendByte(SD_TOK_WRITE_STARTBLOCK); /* 写单块开始令牌 start token of write single block */
  612. for (i = 0; i < len; i++)
  613.         SPI_SendByte(sendbuf[i]); /* 发送数据 send data */
  614. #if SD_CRC_EN
  615. i = SD_GetCRC16(sendbuf,len); /* 计算CRC16 calculate CRC16 */
  616. #endif
  617. SPI_SendByte((i >> 8) & 0xFF);
  618. SPI_SendByte(i & 0xFF);  /* 发送CRC16校验码 send CRC16 check code */
  619.     
  620. tmp = SPI_RecByte();
  621.    if ((tmp & SD_RESP_DATA_MSK) != SD_RESP_DATA_ACCETPTED)
  622.    {
  623.     SPI_SendByte(0xFF); /* 返回之前发送8个clock  clock out 8 clk before return */
  624.     SPI_CS_Deassert();
  625.     return SD_ERR_DATA_RESP; /* 数据响应错误 data response error */
  626.     }
  627.         
  628.     SPI_CS_Deassert();
  629.      
  630.     if (SD_WaitBusy(SD_WAIT_WRITE) != SD_NO_ERR)
  631.      return SD_ERR_TIMEOUT_WRITE; /* 写入超时 write time out */
  632.     else
  633.      return SD_NO_ERR;  /* 写入正确 write right */
  634.  }
  635. /*******************************************************************************************************************
  636. ** 函数名称: void SD_StopMultiToken() Name:   void SD_StopMultiToken(void)
  637. ** 功能描述: 发送多块写停止令牌 Function: send the token that stop multiple block write
  638. ** 输   入: 无     Input:   NULL
  639. ** 输   出: 无 Output:   NULL
  640. ********************************************************************************************************************/
  641. void SD_StopMultiToken(void)
  642. {
  643. SPI_CS_Assert();
  644. SPI_SendByte(0xFF); /* 先发送8个clock send 8 clock first */
  645. SPI_SendByte(SD_TOK_STOP_MULTI); /* 发送停止数据传输令牌 send stop transmission token */
  646. SPI_RecByte();
  647.     SPI_CS_Deassert();
  648. }
  649. /********************************************************************************************************************
  650. ** 函数名称: void SD_WaitBusy() Name:   void SD_WaitBusy()
  651. ** 功能描述: 查询SD卡是否处于忙状态 Function: poll SD Card wheather it is busy
  652. ** 输   入: INT32U waittype: 超时类型 Input:   INT32U timeout: time out type
  653. ** 输   出: 0: 未超时  >0: 错误码 Output:   0: not time out   > 0: error code
  654. *********************************************************************************************************************/
  655. INT8U SD_WaitBusy(INT8U waittype)
  656. {
  657.     INT32U timeout, i = 0;
  658.     INT8U tmp;
  659.     
  660.     if (waittype == SD_WAIT_WRITE)
  661.      timeout = sds.timeout_write;
  662.     else
  663.      timeout = sds.timeout_erase;
  664. SPI_CS_Assert();
  665.     do
  666.     {  /* 等待忙结束 wait for being busy end */
  667.         tmp = SPI_RecByte();
  668.         i++;
  669.     }while ((tmp != 0xFF) && (i < timeout)); /* 忙时收到的值为0xFF always receive 0xFF when card is busy */    
  670. SPI_CS_Deassert(); /* 有错 */
  671. if(i < timeout) 
  672. return SD_NO_ERR; /* 返回0,表示没超时 return 0 indicate that operation is not time out */
  673. else 
  674. return SD_ERR_TIMEOUT_WAIT; /* 返回错误码,表示超时 return error code indicate that operation is time out */
  675. }
  676. /********************************************************************************************************************
  677. ** 函数名称: void SD_SPIDelay() Name:   void SD_SPIDelay()
  678. ** 功能描述: SPI总线延时 Function: SPI Bus delay 
  679. ** 输   入: INT8U value: 延时值,不超过255     Input:   INT8U value : delay value,do not beyond 255
  680. ** 输   出: 无 Output:   NULL
  681. *********************************************************************************************************************/
  682. void SD_SPIDelay(INT8U value)
  683. {
  684.     INT8U i;
  685.     for (i = 0; i < value; i++)
  686.         SPI_SendByte(0xFF);   /* 发送0xFF clock out 0xFF */
  687. }