久久久久久久999_99精品久久精品一区二区爱城_成人欧美一区二区三区在线播放_国产精品日本一区二区不卡视频_国产午夜视频_欧美精品在线观看免费
標題:
關于stm32spi讀寫sd卡cmd41一直返回0x01的解決辦法
[打印本頁]
作者:
猶太人天涯
時間:
2020-4-27 15:19
標題:
關于stm32spi讀寫sd卡cmd41一直返回0x01的解決辦法
先說一下我的情況,我的代碼是原子哥的,首先執行到cmd0,返回01,進入到idle狀態,這是沒有問題的,然后cmd8判斷是否為sdv2.0,返回01,這也是沒有問題的。然后循環發送cmd41(r1=SD_SendCmd(CMD41,0x40000000,0X01))和cmd55,返回值一直為0x01,正常來說,返回指應該0x00(我的卡是sdhc)。解決辦法:更換電源,我買的這個模塊的是可以用5v的,之前用的3v。推測可能是電流不夠。店家給的例程實在arduino上跑,我試了沒問題,看到電源我才感覺有點東西,這里踩個坑,希望能幫助到大家(第一次在論壇發帖,緊張。。。。)我用的例程放到下邊
51hei.png
(3.6 KB, 下載次數: 71)
下載附件
2020-4-27 15:48 上傳
單片機源程序如下:
#include "sys.h"
#include "sd-spi.h"
#include "spi2.h"
#include "usart.h"
#include<oled.h>
u8 SD_Type=0;//SD卡的類型
u8 SD_SPI_ReadWriteByte(u8 data)
{
return SPI2_ReadWriteByte(data);
}
//SD卡初始化的時候,需要低速
void SD_SPI_SpeedLow(void)
{
SPI2_SetSpeed(SPI_BaudRatePrescaler_256);//設置到低速模式
}
//SD卡正常工作的時候,可以高速了
void SD_SPI_SpeedHigh(void)
{
SPI2_SetSpeed(SPI_BaudRatePrescaler_2);//設置到高速模式
}
void SD_DisSelect(void)
{
SD_CS=1;
SD_SPI_ReadWriteByte(0xff);//提供額外的8個時鐘
}
//選擇sd卡,并且等待卡準備OK
//返回值:0,成功;1,失敗;
u8 SD_Select(void)
{
SD_CS=0;
if(SD_WaitReady()==0)return 0;//等待成功
SD_DisSelect();
return 1;//等待失敗
}
//等待卡準備好
//返回值:0,準備好了;其他,錯誤代碼
u8 SD_WaitReady(void)
{
u32 t=0;
do
{
if(SD_SPI_ReadWriteByte(0XFF)==0XFF)return 0;//OK
t++;
}while(t<0XFFFFFF);//等待
return 1;
}
//等待SD卡回應
//Response:要得到的回應值
//返回值:0,成功得到了該回應值
// 其他,得到回應值失敗
u8 SD_GetResponse(u8 Response)
{
u16 Count=0xFFFF;//等待次數
while ((SD_SPI_ReadWriteByte(0XFF)!=Response)&&Count)Count--;//等待得到準確的回應
if (Count==0)return MSD_RESPONSE_FAILURE;//得到回應失敗
else return MSD_RESPONSE_NO_ERROR;//正確回應
}
//從sd卡讀取一個數據包的內容
//buf:數據緩存區
//len:要讀取的數據長度.
//返回值:0,成功;其他,失敗;
u8 SD_RecvData(u8*buf,u16 len)
{
if(SD_GetResponse(0xFE))return 1;//等待SD卡發回數據起始令牌0xFE
while(len--)//開始接收數據
{
*buf=SPI2_ReadWriteByte(0xFF);
buf++;
}
//下面是2個偽CRC(dummy CRC)
SD_SPI_ReadWriteByte(0xFF);
SD_SPI_ReadWriteByte(0xFF);
return 0;//讀取成功
}
//向sd卡寫入一個數據包的內容 512字節
//buf:數據緩存區
//cmd:指令
//返回值:0,成功;其他,失敗;
u8 SD_SendBlock(u8*buf,u8 cmd)
{
u16 t;
if(SD_WaitReady())return 1;//等待準備失效
SD_SPI_ReadWriteByte(cmd);
if(cmd!=0XFD)//不是結束指令
{
for(t=0;t<512;t++)SPI2_ReadWriteByte(buf[t]);//提高速度,減少函數傳參時間
SD_SPI_ReadWriteByte(0xFF);//忽略crc
SD_SPI_ReadWriteByte(0xFF);
t=SD_SPI_ReadWriteByte(0xFF);//接收響應
if((t&0x1F)!=0x05)return 2;//響應錯誤
}
return 0;//寫入成功
}
//向SD卡發送一個命令
//輸入: u8 cmd 命令
// u32 arg 命令參數
// u8 crc crc校驗值
//返回值:SD卡返回的響應
u8 SD_SendCmd(u8 cmd, u32 arg, u8 crc)
{
u8 r1;
u8 Retry=0;
SD_DisSelect();//取消上次片選
if(SD_Select())return 0XFF;//片選失效
//發送
SD_SPI_ReadWriteByte(cmd | 0x40);//分別寫入命令
SD_SPI_ReadWriteByte(arg >> 24);
SD_SPI_ReadWriteByte(arg >> 16);
SD_SPI_ReadWriteByte(arg >> 8);
SD_SPI_ReadWriteByte(arg);
SD_SPI_ReadWriteByte(crc);
if(cmd==CMD12)SD_SPI_ReadWriteByte(0xff);//Skip a stuff byte when stop reading
//等待響應,或超時退出
Retry=0X1F;
do
{
r1=SD_SPI_ReadWriteByte(0xFF);
}while((r1&0X80) && Retry--);
//返回狀態值
return r1;
}
//獲取SD卡的CID信息,包括制造商信息
//輸入: u8 *cid_data(存放CID的內存,至少16Byte)
//返回值:0:NO_ERR
// 1:錯誤
u8 SD_GetCID(u8 *cid_data)
{
u8 r1;
//發CMD10命令,讀CID
r1=SD_SendCmd(CMD10,0,0x01);
if(r1==0x00)
{
r1=SD_RecvData(cid_data,16);//接收16個字節的數據
}
SD_DisSelect();//取消片選
if(r1)return 1;
else return 0;
}
//獲取SD卡的CSD信息,包括容量和速度信息
//輸入:u8 *cid_data(存放CID的內存,至少16Byte)
//返回值:0:NO_ERR
// 1:錯誤
u8 SD_GetCSD(u8 *csd_data)
{
u8 r1;
r1=SD_SendCmd(CMD9,0,0x01);//發CMD9命令,讀CSD
if(r1==0)
{
r1=SD_RecvData(csd_data, 16);//接收16個字節的數據
}
SD_DisSelect();//取消片選
if(r1)return 1;
else return 0;
}
//獲取SD卡的總扇區數(扇區數)
//返回值:0: 取容量出錯
// 其他:SD卡的容量(扇區數/512字節)
//每扇區的字節數必為512,因為如果不是512,則初始化不能通過.
u32 SD_GetSectorCount(void)
{
u8 csd[16];
u32 Capacity;
u8 n;
u16 csize;
//取CSD信息,如果期間出錯,返回0
if(SD_GetCSD(csd)!=0) return 0;
//如果為SDHC卡,按照下面方式計算
if((csd[0]&0xC0)==0x40) //V2.00的卡
{
csize = csd[9] + ((u16)csd[8] << 8) + 1;
Capacity = (u32)csize << 10;//得到扇區數
}else//V1.XX的卡
{
n = (csd[5] & 15) + ((csd[10] & 128) >> 7) + ((csd[9] & 3) << 1) + 2;
csize = (csd[8] >> 6) + ((u16)csd[7] << 2) + ((u16)(csd[6] & 3) << 10) + 1;
Capacity= (u32)csize << (n - 9);//得到扇區數
}
return Capacity;
}
//初始化SD卡
u8 SD_Init(void)
{
u8 r1; // 存放SD卡的返回值
u16 retry; // 用來進行超時計數
u8 buf[4];
u16 i;
SPI2_Init(); //初始化IO
SD_Select();
SD_SPI_SpeedLow(); //設置到低速模式
for(i=0;i<10;i++)SD_SPI_ReadWriteByte(0XFF);//發送最少74個脈沖
retry=20;
do
{
r1=SD_SendCmd(CMD0,0,0x95);//進入IDLE狀態
}while((r1!=0X01) && retry--);
SD_Type=0;//默認無卡
if(r1==0X01)
{
if(SD_SendCmd(CMD8,0x1AA,0x87)==1)//SD V2.0
{
for(i=0;i<4;i++)buf[i]=SD_SPI_ReadWriteByte(0XFF); //Get trailing return value of R7 resp
if(buf[2]==0X01&&buf[3]==0XAA)//卡是否支持2.7~3.6V
{
retry=0X0FFE;
do
{
SD_SendCmd(CMD55,0,0X01); //發送CMD55
r1=SD_SendCmd(CMD41,0x40000000,0X01);//發送CMD41
OLED_DisplayHex(0,2,16,16,r1);
}while(r1&&retry--);
if(retry&&SD_SendCmd(CMD58,0,0X01)==0)//鑒別SD2.0卡版本開始
{
OLED_DisplayHex(0,0,16,16,0x22);
for(i=0;i<4;i++)buf[i]=SD_SPI_ReadWriteByte(0XFF);//得到OCR值
if(buf[0]&0x40)SD_Type=SD_TYPE_V2HC; //檢查CCS
else SD_Type=SD_TYPE_V2;
}
}
}else//SD V1.x/ MMC V3
{
SD_SendCmd(CMD55,0,0X01); //發送CMD55
r1=SD_SendCmd(CMD41,0,0X01); //發送CMD41
if(r1<=1)
{
SD_Type=SD_TYPE_V1;
retry=0XFFFE;
do //等待退出IDLE模式
{
SD_SendCmd(CMD55,0,0X01); //發送CMD55
r1=SD_SendCmd(CMD41,0,0X01);//發送CMD41
}while(r1&&retry--);
}else//MMC卡不支持CMD55+CMD41識別
{
SD_Type=SD_TYPE_MMC;//MMC V3
retry=0XFFFE;
do //等待退出IDLE模式
{
r1=SD_SendCmd(CMD1,0,0X01);//發送CMD1
}while(r1&&retry--);
}
if(retry==0||SD_SendCmd(CMD16,512,0X01)!=0)SD_Type=SD_TYPE_ERR;//錯誤的卡
}
}
SD_DisSelect();//取消片選
SD_SPI_SpeedHigh();//高速
if(SD_Type)return 0;
else if(r1)return r1;
return 0xaa;//其他錯誤
}
//讀SD卡
//buf:數據緩存區
//sector:扇區
//cnt:扇區數
//返回值:0,ok;其他,失敗.
u8 SD_ReadDisk(u8*buf,u32 sector,u8 cnt)
{
u8 r1;
if(SD_Type!=SD_TYPE_V2HC)sector <<= 9;//轉換為字節地址
if(cnt==1)
{
r1=SD_SendCmd(CMD17,sector,0X01);//讀命令
if(r1==0)//指令發送成功
{
r1=SD_RecvData(buf,512);//接收512個字節
}
}else
{
r1=SD_SendCmd(CMD18,sector,0X01);//連續讀命令
do
{
r1=SD_RecvData(buf,512);//接收512個字節
buf+=512;
}while(--cnt && r1==0);
SD_SendCmd(CMD12,0,0X01); //發送停止命令
}
SD_DisSelect();//取消片選
return r1;//
}
//寫SD卡
//buf:數據緩存區
//sector:起始扇區
//cnt:扇區數
//返回值:0,ok;其他,失敗.
u8 SD_WriteDisk(u8*buf,u32 sector,u8 cnt)
{
u8 r1;
if(SD_Type!=SD_TYPE_V2HC)sector *= 512;//轉換為字節地址
if(cnt==1)
{
r1=SD_SendCmd(CMD24,sector,0X01);//讀命令
if(r1==0)//指令發送成功
{
r1=SD_SendBlock(buf,0xFE);//寫512個字節
}
}else
{
if(SD_Type!=SD_TYPE_MMC)
{
SD_SendCmd(CMD55,0,0X01);
SD_SendCmd(CMD23,cnt,0X01);//發送指令
}
r1=SD_SendCmd(CMD25,sector,0X01);//連續讀命令
if(r1==0)
{
do
{
r1=SD_SendBlock(buf,0xFC);//接收512個字節
buf+=512;
}while(--cnt && r1==0);
r1=SD_SendBlock(0,0xFD);//接收512個字節
}
}
SD_DisSelect();//取消片選
return r1;//
}
復制代碼
所有資料51hei提供下載:
Desktop.zip
(5.84 KB, 下載次數: 36)
2020-4-27 15:18 上傳
點擊文件名下載附件
sd-spi
下載積分: 黑幣 -5
作者:
will__
時間:
2020-11-8 11:21
換了電源也不行呢 ?
作者:
xtiwxk
時間:
2022-10-5 20:22
感謝,一直是0x01幾天下來一直沒找到原因,之前也試過電源不行,今天把開發板和SD模塊同時都接上5v瞬間返回0x00
作者:
xiaoyao166ny
時間:
2024-5-5 21:44
sd卡 不是3.3 供電嗎
歡迎光臨 (http://m.zg4o1577.cn/bbs/)
Powered by Discuz! X3.1
主站蜘蛛池模板:
偷偷操不一样
|
国产精品99久久久久久www
|
欧美日韩网站
|
亚洲狠狠干
|
亚洲综合在线视频
|
国产一区精品视频
|
一级毛片免费播放
|
日韩在线视频一区
|
中文字幕av一区二区三区谷原希美
|
9999精品视频
|
午夜网址
|
亚洲一区二区在线播放
|
成人在线一区二区
|
国产精品hd
|
日本少妇中文字幕
|
欧美一区二区视频在线观看
|
福利在线播放
|
五月婷在线
|
免费在线观看黄
|
天天久久
|
中文在线观看免费视频
|
www.午夜
|
av一区二区三区在线观看
|
亚洲精品久久久久久久久久久
|
高清视频一区二区
|
国产精品美女久久久久av爽
|
国产一区欧美
|
欧美三级三级三级爽爽爽
|
中文字幕少妇
|
国产视频一二区
|
亚洲一区在线看
|
女人av在线
|
天天操天天舔
|
中文精品一区
|
一区免费视频
|
亚洲在线一区二区
|
国产成人区
|
少妇综合
|
国产一区不卡
|
91看片在线观看
|
国产成人在线视频
|