sdcmd.c
上传用户:pudewen666
上传日期:2007-06-08
资源大小:75k
文件大小:34k
源码类别:

微处理器开发

开发平台:

Windows_Unix

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