[菜农助学交流] M0助学板学习第五帖:SD卡读取实验

[复制链接]
 楼主| mcsgy 发表于 2011-10-16 15:59 | 显示全部楼层 |阅读模式
由于本人实力有限,SD卡方面的知识不太懂。为此本实验核心部分,即SD卡部分的核心函数都是从“正点原子”的STM32,SD卡实验移植过来的(未经他本人同意)。  
      四个文件分别是MMC_SD.h,MMC_SD.C ,spi.h,spi.c。当然其中改了一些定义和函数实现的方法。希望“正点原子”大哥支持!     

(一)实验硬件连接图(需要准备一张SD卡)
        GPA13 -->  SD_CS
        GPA14 -->  SPI_FLASH_CS
        GPB9   -->  SS11(未连接)
        GPC8   -->  SS10
        GPC9   -->  SPICLK10
        GPC10 -->  MISO10
        GPC11 -->  MOSI10
        
(二)SPI相关函数

  1.         #include "spi.h"
  2. #include "NUC1xx.h"
  3. #include "DrvSPI.h"
  4. //////////////////////////////////////////////////////////////////////////////////     
  5. //本程序只供学习使用,未经作者许可,不得用于其它任何用途
  6. //Mini STM32开发板
  7. //SPI 驱动函数      
  8. //正点原子@ALIENTEK
  9. //技术论坛:www.openedv.com
  10. //修改日期:2010/6/13
  11. //版本:V1.0
  12. //版权所有,盗版必究。
  13. //Copyright(C) 正点原子 2009-2019
  14. //All rights reserved
  15. //////////////////////////////////////////////////////////////////////////////////      


  16. //以下是SPI模块的初始化代码,配置成主机模式,访问SD Card/W25X16/24L01/JF24C                              
  17. //SPI口初始化
  18. //这里针是对SPI1的初始化
  19. void SPIx_Init(void)
  20. {     
  21.     DrvSPI_Open(eDRVSPI_PORT1,eDRVSPI_MASTER,eDRVSPI_TYPE1,8,FALSE);//SPI0主机模式打开
  22.     DrvSPI_EnableAutoSS(eDRVSPI_PORT1, eDRVSPI_SS0);
  23.     DrvSPI_SetSlaveSelectActiveLevel(eDRVSPI_PORT1, eDRVSPI_ACTIVE_LOW_FALLING);
  24.     DrvSPI_SetClockFreq(eDRVSPI_PORT1,2000000,0);                   //24Mhz频率
  25.     SPIx_ReadWriteByte(0xff);//启动传输         
  26. }   
  27. //SPI 速度设置函数
  28. //SpeedSet:
  29. //SPI_SPEED_2   2分频   (SPI 36M@sys 72M)
  30. //SPI_SPEED_8   8分频   (SPI 9M@sys 72M)
  31. //SPI_SPEED_16  16分频  (SPI 4.5M@sys 72M)
  32. //SPI_SPEED_256 256分频 (SPI 281.25K@sys 72M)
  33. void SPIx_SetSpeed(u8 SpeedSet)
  34. {
  35. //    SPI1->CR1&=0XFFC7;//Fsck=Fcpu/256
  36.     switch(SpeedSet)
  37.     {
  38.         case SPI_SPEED_2://二分频
  39.             DrvSPI_SetClockFreq(eDRVSPI_PORT0,24000000,0);       //模拟分频设置频率24Mhz
  40. //            SPI1->CR1|=0<<3;//Fsck=Fpclk/2=36Mhz
  41.             break;
  42.         case SPI_SPEED_4://四分频
  43.             DrvSPI_SetClockFreq(eDRVSPI_PORT0,12000000,0);       //12Mhz
  44. //            SPI1->CR1|=1<<3;//Fsck=Fpclk/4=18Mhz
  45.             break;
  46.         case SPI_SPEED_8://八分频
  47.             DrvSPI_SetClockFreq(eDRVSPI_PORT0,6000000,0);       //6Mhz
  48. //            SPI1->CR1|=2<<3;//Fsck=Fpclk/8=9Mhz
  49.             break;
  50.         case SPI_SPEED_16://十六分频
  51.             DrvSPI_SetClockFreq(eDRVSPI_PORT0,2000000,0);       //200Khz
  52. //            SPI1->CR1|=3<<3;//Fsck=Fpclk/16=4.5Mhz
  53.             break;
  54.         case SPI_SPEED_256://256分频
  55.             DrvSPI_SetClockFreq(eDRVSPI_PORT0,300000,0);
  56. //            SPI1->CR1|=7<<3;//Fsck=Fpclk/16=281.25Khz
  57.             break;
  58.     }         
  59. //    SPI1->CR1|=1<<6; //SPI设备使能      
  60. }
  61. //SPIx 读写一个字节
  62. //TxData:要写入的字节
  63. //返回值:读取到的字节
  64. u8 SPIx_ReadWriteByte(u8 TxData)
  65. {        
  66.     u8 retry=0;
  67.     while(SPI1->CNTRL.GO_BUSY==1)
  68.     {
  69.         retry++;
  70.         if(retry>200)
  71.         {
  72.             return 0;
  73.         }
  74.     }
  75.     SPI1->TX[0] = TxData;
  76.     SPI1->CNTRL.GO_BUSY = 1;
  77.     retry=0;
  78.     while(SPI1->CNTRL.GO_BUSY==1)
  79.     {
  80.           retry++;
  81.         if(retry>200)
  82.         {
  83.             return 0;
  84.         }
  85.     }

  86.     return SPI1->RX[0];
  87.    
  88.                         
  89. }
  90.         

(三)MMC_SD相关函数
        由于该文件代码比较多,我只贴我修改的部分
        一处是SD卡初始化部分

  1.        u8 SD_Init(void)
  2. {                                 
  3.     u8 r1;      // 存放SD卡的返回值
  4.     u16 retry;  // 用来进行超时计数
  5.     u8 buff[6];
  6.     //设置硬件上与SD卡相关联的控制引脚输出
  7.     //避免NRF24L01/W25X16等的影响
  8. //    RCC->APB2ENR|=1<<2;       //PORTA时钟使能
  9. //    GPIOA->CRL&=0XFFF000FF;
  10. //    GPIOA->CRL|=0X00033300;//PA2.3.4 推挽         
  11. //    GPIOA->ODR|=0X7<<2;    //PA2.3.4上拉  
  12. //********************************************************/
  13. //其实就只是修改了IO口这部分,其余的都未动
  14.     DrvGPIO_Open(E_GPA,13,E_IO_OUTPUT);         //SD卡得CS片选
  15.     DrvGPIO_Open(E_GPA,14,E_IO_OUTPUT);         //SPI_FLASH_CS片选
  16.     DrvGPIO_Open(E_GPC,8,E_IO_OUTPUT);         //SS10从机选择位0
  17.     DrvGPIO_Open(E_GPB,9,E_IO_OUTPUT);         //SS11从机选择位1
  18. //*******************************************************/
  19.     SPIx_Init();
  20.      SPIx_SetSpeed(SPI_SPEED_256);//设置到低速模式         
  21.     SD_CSDIS;
  22. //    SD_CS=1;   
  23.     if(SD_Idle_Sta()) return 1;//超时返回1 设置到idle 模式失败      
  24.     //-----------------SD卡复位到idle结束-----------------     
  25.     //获取卡片的SD版本信息
  26.     SD_CSEN;
  27. //     SD_CS=0;   
  28.     r1 = SD_SendCommand_NoDeassert(8, 0x1aa,0x87);         
  29.     //如果卡片版本信息是v1.0版本的,即r1=0x05,则进行以下初始化
  30.     if(r1 == 0x05)
  31.     {
  32.         //设置卡类型为SDV1.0,如果后面检测到为MMC卡,再修改为MMC
  33.         SD_Type = SD_TYPE_V1;      
  34.         //如果是V1.0卡,CMD8指令后没有后续数据
  35.         //片选置高,结束本次命令
  36.         SD_CSDIS;
  37. //        SD_CS=1;
  38.         //多发8个CLK,让SD结束后续操作
  39.         SPIx_ReadWriteByte(0xFF);      
  40.         //-----------------SD卡、MMC卡初始化开始-----------------     
  41.         //发卡初始化指令CMD55+ACMD41
  42.         // 如果有应答,说明是SD卡,且初始化完成
  43.         // 没有回应,说明是MMC卡,额外进行相应初始化
  44.         retry = 0;
  45.         do
  46.         {
  47.             //先发CMD55,应返回0x01;否则出错
  48.             r1 = SD_SendCommand(CMD55, 0, 0);
  49.             if(r1 == 0XFF)return r1;//只要不是0xff,就接着发送      
  50.             //得到正确响应后,发ACMD41,应得到返回值0x00,否则重试200次
  51.             r1 = SD_SendCommand(ACMD41, 0, 0);
  52.             retry++;
  53.         }while((r1!=0x00) && (retry<400));
  54.         // 判断是超时还是得到正确回应
  55.         // 若有回应:是SD卡;没有回应:是MMC卡      
  56.         //----------MMC卡额外初始化操作开始------------
  57.         if(retry==400)
  58.         {
  59.             retry = 0;
  60.             //发送MMC卡初始化命令(没有测试)
  61.             do
  62.             {
  63.                 r1 = SD_SendCommand(1,0,0);
  64.                 retry++;
  65.             }while((r1!=0x00)&& (retry<400));
  66.             if(retry==400)return 1;   //MMC卡初始化超时            
  67.             //写入卡类型
  68.             SD_Type = SD_TYPE_MMC;
  69.         }
  70.         //----------MMC卡额外初始化操作结束------------        
  71.         //设置SPI为高速模式
  72.         SPIx_SetSpeed(SPI_SPEED_4);   
  73.         SPIx_ReadWriteByte(0xFF);     
  74.         //禁止CRC校验      
  75.         r1 = SD_SendCommand(CMD59, 0, 0x95);
  76.         if(r1 != 0x00)return r1;  //命令错误,返回r1         
  77.         //设置Sector Size
  78.         r1 = SD_SendCommand(CMD16, 512, 0x95);
  79.         if(r1 != 0x00)return r1;//命令错误,返回r1         
  80.         //-----------------SD卡、MMC卡初始化结束-----------------

  81.     }//SD卡为V1.0版本的初始化结束     
  82.     //下面是V2.0卡的初始化
  83.     //其中需要读取OCR数据,判断是SD2.0还是SD2.0HC卡
  84.     else if(r1 == 0x01)
  85.     {
  86.         //V2.0的卡,CMD8命令后会传回4字节的数据,要跳过再结束本命令
  87.         buff[0] = SPIx_ReadWriteByte(0xFF);  //should be 0x00
  88.         buff[1] = SPIx_ReadWriteByte(0xFF);  //should be 0x00
  89.         buff[2] = SPIx_ReadWriteByte(0xFF);  //should be 0x01
  90.         buff[3] = SPIx_ReadWriteByte(0xFF);  //should be 0xAA        
  91.         SD_CSDIS;
  92. //        SD_CS=1;      
  93.         SPIx_ReadWriteByte(0xFF);//the next 8 clocks            
  94.         //判断该卡是否支持2.7V-3.6V的电压范围
  95.         //if(buff[2]==0x01 && buff[3]==0xAA) //不判断,让其支持的卡更多
  96.         {      
  97.             retry = 0;
  98.             //发卡初始化指令CMD55+ACMD41
  99.             do
  100.             {
  101.                 r1 = SD_SendCommand(CMD55, 0, 0);
  102.                 if(r1!=0x01)return r1;      
  103.                 r1 = SD_SendCommand(ACMD41, 0x40000000, 0);
  104.                 if(retry>200)return r1;  //超时则返回r1状态  
  105.             }while(r1!=0);         
  106.             //初始化指令发送完成,接下来获取OCR信息           
  107.             //-----------鉴别SD2.0卡版本开始-----------
  108.             r1 = SD_SendCommand_NoDeassert(CMD58, 0, 0);
  109.             if(r1!=0x00)
  110.             {
  111.                 SD_CSDIS;
  112. //                SD_CS=1;//释放SD片选信号
  113.                 return r1;  //如果命令没有返回正确应答,直接退出,返回应答     
  114.             }//读OCR指令发出后,紧接着是4字节的OCR信息
  115.             buff[0] = SPIx_ReadWriteByte(0xFF);
  116.             buff[1] = SPIx_ReadWriteByte(0xFF);
  117.             buff[2] = SPIx_ReadWriteByte(0xFF);
  118.             buff[3] = SPIx_ReadWriteByte(0xFF);         
  119.             //OCR接收完成,片选置高
  120.             SD_CSDIS;
  121. //            SD_CS=1;
  122.             SPIx_ReadWriteByte(0xFF);      
  123.             //检查接收到的OCR中的bit30位(CCS),确定其为SD2.0还是SDHC
  124.             //如果CCS=1:SDHC   CCS=0:SD2.0
  125.             if(buff[0]&0x40)SD_Type = SD_TYPE_V2HC;    //检查CCS     
  126.             else SD_Type = SD_TYPE_V2;        
  127.             //-----------鉴别SD2.0卡版本结束-----------
  128.             //设置SPI为高速模式
  129.             SPIx_SetSpeed(SPI_SPEED_4);  
  130.         }        
  131.     }
  132.     return r1;
  133. }                                                                                    
  134. //从SD卡中读回指定长度的数据,放置在给定位置
  135. //输入: u8 *data(存放读回数据的内存>len)
  136. //      u16 len(数据长度)
  137. //      u8 release(传输完成后是否释放总线CS置高 0:不释放 1:释放)     
  138. //返回值:0:NO_ERR
  139. //       other:错误信息                                                         
  140. u8 SD_ReceiveData(u8 *data, u16 len, u8 release)
  141. {
  142.     // 启动一次传输
  143.     SD_CSEN;
  144. //    SD_CS=0;                        
  145.     if(SD_GetResponse(0xFE))//等待SD卡发回数据起始令牌0xFE
  146.     {      
  147.         SD_CSDIS;
  148. //        SD_CS=1;
  149.         return 1;
  150.     }
  151.     while(len--)//开始接收数据
  152.     {
  153.         *data=SPIx_ReadWriteByte(0xFF);
  154.         data++;
  155.     }
  156.     //下面是2个伪CRC(dummy CRC)
  157.     SPIx_ReadWriteByte(0xFF);
  158.     SPIx_ReadWriteByte(0xFF);
  159.     if(release==RELEASE)//按需释放总线,将CS置高
  160.     {
  161.         SD_CSDIS;   
  162. //        SD_CS=1;//传输结束
  163.         SPIx_ReadWriteByte(0xFF);
  164.     }                                                                     
  165.     return 0;
  166. }                        
  167.       

还有一处是宏定义
       原来的SD_CS片选IO口的定义为位寻址模式,但是NUC120不支持位寻址,所以我用了个很笨的方法,定义了两个宏
      #define SD_CSEN    (DrvGPIO_ClrBit(E_GPA,13))
      #define SD_CSDIS    (DrvGPIO_SetBit(E_GPA,13))
      并将MMC_SD.C文件中所有的SD_CD=1用SD_CSDIS替换,所有的SD_CD=0用SD_CSEN替换
      我想不出其他更好的办法了。
(四)main()函数
      main函数的功能是初始化外设,然后在死循环里读取SD卡的第一个扇区,并通过串口把数据发送到上位机,通过串口调试助手接收。和”正点原子“的主函数功能是一样的。
      运行的结果如下图:
      
      有兴趣的网友可以在这里下载整个工程代码:

本帖子中包含更多资源

您需要 登录 才可以下载或查看,没有账号?注册

×
weshiluwei6 发表于 2011-10-16 17:04 | 显示全部楼层
niubi 牛人 支持支持啊
Swallow_0322 发表于 2011-10-16 18:58 | 显示全部楼层
顶!
lixiaoxu2meng 发表于 2011-10-17 07:35 | 显示全部楼层
顶啊
lwslws201 发表于 2011-10-19 09:30 | 显示全部楼层
:victory:
lixingzhe8 发表于 2012-6-4 09:50 | 显示全部楼层
强,分享
冰茶爽爽 发表于 2012-8-10 10:32 | 显示全部楼层
支持楼主的经验分享和资料!
Lro 发表于 2014-4-12 22:36 | 显示全部楼层
多谢楼主分享~~~学习了~~~~
您需要登录后才可以回帖 登录 | 注册

本版积分规则

个人签名:从零开始

6

主题

232

帖子

1

粉丝
快速回复 在线客服 返回列表 返回顶部