找回密码
 立即注册

QQ登录

只需一步,快速开始

搜索
查看: 16013|回复: 11
打印 上一主题 下一主题
收起左侧

终于实现51单片机读SD卡的逻辑0扇区高兴

  [复制链接]
跳转到指定楼层
楼主
ID:83710 发表于 2015-6-23 18:39 | 只看该作者 回帖奖励 |倒序浏览 |阅读模式
呵呵!今天晚上真是走运啊,我12月8日开始的做SD读实验,到今天晚上11:00才做出来啊,其中写片115次,变过四次程序,连着三晚上没睡好觉,两天偷偷的没去上班在宿舍做实验,终于搞定读逻辑0扇区了!我算是受够了,但是这几天每天晚上都在找资料,请老师,可是真是那句话,靠谁也不如靠自己,别人是不了解你的实际情况的,在这里我真的很感谢网络,有了它,我在千千万万的资料中找到了最有用的几分,然我得意找到钥匙。
接下来说下SD卡(microsd)读DBR:

SD卡作为一种存储设备早在应用了,但是关于SD卡我还是在做完12864显示以后才发现,光显示图片有啥意思?不如来点动画,呵呵殊不知存储空间收到限制了,一幅BMP图片要1KB的存储空间,我的存储容量是8K,呵呵,很显然是不行的,有没有一种大容量小体积便携,便于PC操作的存储设备那?指向了SD卡,一代的可以最大存储4GB,这是相当海量的,也是很理想的,决定后看了关于FAT32的文件系统,感觉真是很难,不过,也大抵能找到根目录,FAT表和数据区,接下来是做实验搭模块,开始看了关于SD卡的资料,网上资料很多,我看了大多泛泛而谈,没做过的,即使看了还是无处下手,我经验如下:1.首先要了解SPI通信,因为他是基础,要靠他来完成数据交换,其实SPI也没什么,个人感觉比异步简单老鼻子了,关键是SD卡中数据发送时上升沿,数据接收是下降沿,记好了,期间一定要保证数据的稳定,且在写时MISO线要保持高电平,读时MOSI要保持高电平,2,写命令是公六字节,最先发送的是指令代码(高位先发)然后是参数,对于参数是地址的则按地址字节最高字节在前,以此类推,例如:ADD=0x01020304,那么先要发送01,在发02,在发03在发04;最后发CRC效验,当然这里默认为FF,不用管它;3,先调复位SD卡,然后再调初始化,在调读,这里注意,当你初始化复位时,时钟一定要慢,当要读扇区是一定要提高时钟频率,要不非得操蛋不可,当调好复位初始化后,先做个试验,读取0扇区的值,当然这里指的是读物理扇区,如果读到扇区的最后两个字节是55AA,那恭喜你,你的程序没问题了,但是FAT文件系统要的是逻辑0扇区的BPB(系统引导记录),所以你要用软件打开磁盘找到PHYSICALL SECTOR NO:和LOGICAL SECTOR NO:即物理扇区和逻辑扇区,例如我的是253  0:表示什么呢?表示逻辑扇区0在物理253扇区,所以找到253扇区就找到逻辑0扇区了,那怎样找?又要注意了,SD卡的读操作的寻址是按字节一扇区来的,所以要吧扇区转换成字节,一扇区是512字节,那么逻辑0扇区的物理地址为253*512=129536;吧这个值写入SD卡然后读出来就是逻辑0扇区的值;一个也不搭茬;找到了逻辑0扇区就找到了进入FAT32文件系统的第一步,也是一把非常重要的钥匙,但是离最终的文件寻址还差的远,要是没这一步,文件寻址是不可能的!还证明了,WINheBUbub不能显示物理0扇区,只能显示逻辑0扇区,嘻嘻!

下面是源码、、、我用的是手机内存卡1GB



  1. #include <reg52.h>
  2. #define uchar unsigned char
  3. #define uint unsigned int
  4. sbit p1_7=P1^7;
  5. sbit CS=P2^0; //片选信号(低电平有效)
  6. sbit DATEIN =P2^1;//主-从数据输入
  7. sbit SCLK=P2^2;//时钟信号
  8. sbit DATEOUT=P2^3;//从-主数据输出

  9. uint btime;
  10. uchar c;

  11. sbit led4=P1^0;
  12. sbit led3=P1^1;    //2010年12月14日与天津开发区
  13. sbit led2=P1^2;
  14. sbit led1=P1^3;
  15. uchar g=0,s=0,bw=0,q=0; //显示单元 个位、十位、百位、千位
  16. uchar code tab1[]={0xc0,0xf9,0xa4,0xb0,0x99,0x92,0x82,0xf8,0x80,0x90,0x88,0x83,0xc6,0xa1,0x86,0x8e};//不带小数点显示0~9

  17. unsigned char bdata dat;
  18. sbit dat_0=dat^0;
  19. sbit dat_1=dat^1;
  20. sbit dat_2=dat^2;
  21. sbit dat_3=dat^3;
  22. sbit dat_4=dat^4;
  23. sbit dat_5=dat^5;
  24. sbit dat_6=dat^6;
  25. sbit dat_7=dat^7;

  26. bit is_init;//决定是否延时;
  27.   uchar lpp;
  28. unsigned char fhz;//返回值
  29. uchar fhz_buff; //读返回值中间量
  30. unsigned char xdata tab[512];


  31. void delay(uint time)
  32. {  while(time)
  33.   time--;
  34. }

  35. //**********************************************
  36. /*读sd卡子程序,无返回值,有参函数,参数为要写入DATEIN数据线的字节*/
  37. void write(unsigned char wr_)// 写入一个字节SD卡
  38. {
  39.            dat=wr_;
  40.         DATEIN=dat_7;
  41.    
  42.       SCLK=0;
  43.    if(is_init)delay(200);
  44.      if(!is_init)delay(2);
  45.       SCLK=1;
  46.       if(is_init) delay(200);
  47.        if(!is_init)delay(2);
  48.   
  49.       DATEIN=dat_6;
  50.    
  51.       SCLK=0;
  52.      if(!is_init)delay(2);
  53.     if(is_init)delay(200);
  54.       SCLK=1;
  55.       if(is_init) delay(200);
  56.      if(!is_init)delay(2);

  57.       DATEIN=dat_5;
  58.    
  59.       SCLK=0;
  60.     if(is_init)delay(200);
  61.      if(!is_init)delay(2);
  62.       SCLK=1;
  63.       if(is_init) delay(200);
  64.     if(!is_init)delay(2);

  65.       DATEIN=dat_4;
  66.    
  67.       SCLK=0;
  68.     if(is_init)delay(200);
  69.       if(!is_init)delay(2);
  70.       SCLK=1;
  71.       if(is_init) delay(200);
  72.      if(!is_init)delay(2);

  73.      DATEIN=dat_3;
  74.    
  75.       SCLK=0;
  76.     if(is_init)delay(200);
  77.       if(!is_init)delay(2);
  78.       SCLK=1;
  79.       if(is_init) delay(200);
  80.     if(!is_init)delay(2);

  81.      DATEIN=dat_2;
  82.    
  83.       SCLK=0;
  84.     if(is_init)delay(200);
  85.       if(!is_init)delay(2);
  86.       SCLK=1;
  87.       if(is_init) delay(200);
  88.       if(!is_init)delay(2);

  89.     DATEIN=dat_1;
  90.    
  91.       SCLK=0;
  92.     if(is_init)delay(200);
  93.      if(!is_init)delay(2);
  94.       SCLK=1;
  95.       if(is_init) delay(200);
  96.      if(!is_init)delay(2);

  97.     DATEIN=dat_0;
  98.    
  99.       SCLK=0;
  100.     if(is_init)delay(200);
  101.       if(!is_init)delay(2);
  102.       SCLK=1;
  103.       if(is_init) delay(200);
  104.      if(!is_init)delay(2);
  105. }
  106. unsigned char read()// 读取一个字节SD卡
  107. {
  108.           DATEOUT=1;
  109.          SCLK=1;
  110.    if(is_init)delay(200);
  111.      if(!is_init)delay(2);
  112.    SCLK=0;
  113.     if(is_init)delay(200);
  114.    if(!is_init)delay(2);
  115.     dat_7=DATEOUT;

  116.      SCLK=1;
  117.    if(is_init)delay(200);
  118.    if(!is_init)delay(2);
  119.    SCLK=0;
  120.     if(is_init)delay(200);
  121.      if(!is_init)delay(2);
  122.     dat_6=DATEOUT;

  123.     SCLK=1;
  124.    if(is_init)delay(200);
  125.     if(!is_init)delay(2);
  126.    SCLK=0;
  127.     if(is_init)delay(200);
  128.     if(!is_init)delay(2);
  129.     dat_5=DATEOUT;

  130.     SCLK=1;
  131.    if(is_init)delay(200);
  132.      if(!is_init)delay(2);
  133.    SCLK=0;
  134.     if(is_init)delay(200);
  135.    if(!is_init)delay(2);
  136.     dat_4=DATEOUT;

  137.      SCLK=1;
  138.    if(is_init)delay(200);
  139.     if(!is_init)delay(2);
  140.    SCLK=0;
  141.     if(is_init)delay(200);
  142.      if(!is_init)delay(2);
  143.     dat_3=DATEOUT;

  144.     SCLK=1;
  145.    if(is_init)delay(200);
  146.        if(!is_init)delay(2);
  147.    SCLK=0;
  148.     if(is_init)delay(200);
  149.        if(!is_init)delay(2);
  150.     dat_2=DATEOUT;

  151.      SCLK=1;
  152.    if(is_init)delay(200);
  153.         if(!is_init)delay(2);
  154.    SCLK=0;
  155.     if(is_init)delay(200);
  156.        if(!is_init)delay(2);
  157.     dat_1=DATEOUT;


  158.      SCLK=1;
  159.    if(is_init)delay(200);
  160.          if(!is_init)delay(2);
  161.    SCLK=0;
  162.     if(is_init)delay(200);
  163.           if(!is_init)delay(2);
  164.     dat_0=DATEOUT;


  165.    return (dat);

  166. }

  167. void restsd()//复位SD卡
  168. {  uchar i;
  169. uchar pcmd[6]={0x40,0x00,0x00,0x00,0x00,0x95};
  170. is_init=1;

  171. CS=1;
  172. for(i=0;i<15;i++)
  173. {
  174.        //120时钟
  175.    write(0xff);
  176. }


  177.   
  178.   
  179.       CS=1;
  180.    write(0xff);//据说是提高兼容性
  181.    CS=0;//片选开

  182.    write( pcmd[0]);
  183.    write( pcmd[1]);
  184.    write( pcmd[2]);
  185.    write( pcmd[3]);
  186.    write( pcmd[4]);
  187.    write( pcmd[5]);

  188.    fhz=read();
  189. for(;;)
  190. {
  191.    fhz=read();
  192.    if(fhz==0x01)break;


  193. }

  194.   
  195.   CS=1;
  196.   write(0xff);

  197. }




  198. void initsd()//初始化
  199. {
  200. //
  201.    uchar pcmd[6]={0x41,0x00,0x00,0x00,0x00,0xff};//
  202. //

  203.       CS=1;
  204.    write(0xff);//据说是提高兼容性
  205.    CS=0;//片选开

  206.    write( pcmd[0]);
  207.    write( pcmd[1]);
  208.    write( pcmd[2]);
  209.    write( pcmd[3]);
  210.    write( pcmd[4]);
  211.    write( pcmd[5]);

  212.   
  213.    fhz=read();
  214. for(;;)
  215. {
  216.    fhz=read();
  217.    if(fhz==0x00)break;


  218. }

  219.   
  220.   CS=1;
  221.   write(0xff);








  222. }

  223. void readsd()//读SD卡物理扇区

  224. {
  225.    uchar pcmd[6]={0x51,0x00,0x01,0xfa,0x00,0xff};//原来这里是高地址字节在前地地址在后啊201012月14日
  226.    uint j,n;


  227.    CS=1;
  228.    write(0xff);//据说是提高兼容性
  229.    CS=0;//片选开

  230.    write( pcmd[0]);
  231.    write( pcmd[1]);
  232.    write( pcmd[2]);
  233.    write( pcmd[3]);
  234.    write( pcmd[4]);
  235.    write( pcmd[5]);
  236.    DATEOUT=1;
  237.   
  238. for(;;)
  239. {
  240.    fhz=read();
  241.    if(fhz==0x00)break;


  242. }
  243. DATEOUT=1;

  244.    for(;;)
  245. {
  246.    fhz=read();
  247.    if(fhz==0xfe)break;


  248. }

  249.   DATEOUT=1;
  250. n=0;
  251. for(j=512;j;j--)
  252. {  
  253.    tab[n]=read();
  254.     n++;

  255. }
  256. fhz=read();
  257. fhz=read();
  258. CS=1;
  259. write(0xff);








  260. }
  261. /*void writesd()

  262. {
  263.    uchar pcmd[6]={0x58,0x00,0x00,0x00,0x00,0xff};
  264.    uint j,k;


  265.    CS=1;
  266.    write(0xff);//据说是提高兼容性
  267.    CS=0;//片选开

  268.    write( pcmd[0]);
  269.    write( pcmd[1]);
  270.    write( pcmd[2]);
  271.    write( pcmd[3]);
  272.    write( pcmd[4]);
  273.    write( pcmd[5]);

  274.    DATEOUT=1;
  275. for(;;)
  276. {
  277.    fhz=read();
  278.    if(fhz==0x00)break;


  279. }
  280. DATEOUT=1;

  281.    for(j=20;j;j--)
  282. {
  283.   write(0xff);



  284. }

  285. write(0xfe);
  286. for(j=512;j;j--)
  287. { k=tab[j];
  288.    write(k);


  289. }
  290. write(0xff);
  291. write(0xff);


  292. DATEOUT=1;
  293. for(;;)
  294. {
  295. fhz=read();
  296. if((fhz&0x0f)==0x05)break;
  297. }





  298.   if(!DATEOUT);
  299. {
  300.    write(0xff);

  301. }
  302. CS=1;
  303. write(0xff);








  304. }
  305. */
  306. void delay1(uint z) //延时程序
  307. {
  308. uint x,a,b;
  309.          for (x=0;x<z;x++)
  310.   
  311.        {
  312.        for(b=120;b>0;b--)
  313.     {
  314.         for(a=3;a>0;a--);
  315.    
  316.              }
  317.     }


  318. }
  319. void display(uchar sd) //显示程序

  320. {   


  321.              c=sd;
  322.            g=(sd&0x0f);
  323.    
  324.    c=(c>>4);
  325.    s=(c&0x0f);
  326.     led1=0;
  327.     P0=tab1[g]; //个
  328.     delay1(2);
  329.     led1=1;
  330.      led2=0;
  331.     P0=tab1[s];//十
  332.     delay1(2);
  333.     led2=1;
  334.     led3=0;
  335.     P0=tab1[0];   //百
  336.     delay1(2);
  337.     led3=1;
  338.                  led4=0;
  339.     P0=tab1[0];    //千
  340.     delay1(2);
  341.     led4=1;


  342. }
  343. void delay2(void)   //误差 -0.00000000025us
  344. {
  345.     unsigned char a,b,c,n;
  346.     for(c=217;c>0;c--)
  347.         for(b=225;b>0;b--)
  348.             for(a=225;a>0;a--);
  349.     for(n=10;n>0;n--);
  350. }
  351. void main()
  352. { uint ad;
  353. uchar v;
  354. is_init=1;
  355. restsd();
  356. initsd();
  357. is_init=0;//提高始终频率
  358. p1_7=0;
  359. //writesd();
  360. readsd();//读物理扇区
  361. p1_7=1;
  362.    btime=65536;
  363. while(1)
  364. {   btime=300;
  365.    delay2();



  366. for(;;)
  367. {
  368. v=tab[ad];
  369. display(v);
  370. btime--;
  371. if(btime==0)break;
  372.   }
  373.   ad++;
  374.   if(ad==512)ad=0;

  375. }





  376. }
复制代码
以上程序仅在本人试验模块中通过,不知兼容咋样!用LED显示512字节的码形!
2015年12月14日-12月8日历时一星期终于读出BPB,加油啊,离文件系统还差的远啊!


评分

参与人数 1威望 +1 黑币 +5 收起 理由
红尘有你 + 1 + 5

查看全部评分

分享到:  QQ好友和群QQ好友和群 QQ空间QQ空间 腾讯微博腾讯微博 腾讯朋友腾讯朋友
收藏收藏5 分享淘帖 顶 踩
回复

使用道具 举报

沙发
ID:73848 发表于 2015-7-1 09:55 来自手机 | 只看该作者
感受到楼主的喜悦了,接连发了那么多帖子,祝早日成功
回复

使用道具 举报

板凳
ID:73848 发表于 2015-7-1 09:55 来自手机 | 只看该作者
感受到楼主的喜悦了,接连发了那么多帖子,祝早日成功
回复

使用道具 举报

地板
ID:81808 发表于 2015-7-1 12:48 | 只看该作者
时空穿越了啊!现在就15年12月了,幸会,幸会!
回复

使用道具 举报

5#
ID:84485 发表于 2015-7-1 16:50 | 只看该作者
非常好
回复

使用道具 举报

6#
ID:97940 发表于 2015-12-1 22:36 | 只看该作者
有电路吗 求
回复

使用道具 举报

7#
ID:97249 发表于 2015-12-2 12:25 | 只看该作者
楼主精神实在可嘉,加油!!
回复

使用道具 举报

8#
ID:101982 发表于 2016-1-5 22:01 | 只看该作者
多谢!!!!
回复

使用道具 举报

9#
ID:140999 发表于 2016-9-30 17:05 | 只看该作者
"所以你要用软件打开磁盘找到PHYSICALL SECTOR NO:和LOGICAL SECTOR NO:即物理扇区和逻辑扇区",请教一下楼主:你是怎么查看物理扇区和逻辑扇区的?用的是什么软件。
回复

使用道具 举报

10#
ID:140218 发表于 2016-10-17 10:58 来自手机 | 只看该作者
一遍一遍的看
回复

使用道具 举报

11#
ID:140644 发表于 2020-1-25 15:41 | 只看该作者
帮顶,支持楼主
回复

使用道具 举报

12#
ID:65707 发表于 2020-3-25 22:29 | 只看该作者
帮顶,支持楼主.正在学习SD卡图片显示
回复

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

手机版|小黑屋|51黑电子论坛 |51黑电子论坛6群 QQ 管理员QQ:125739409;技术交流QQ群281945664

Powered by 单片机教程网

快速回复 返回顶部 返回列表