一、前言
在現(xiàn)代科技飛速發(fā)展的背景下,存儲(chǔ)解決方案的創(chuàng)新與進(jìn)步成為了推動(dòng)各行各業(yè)發(fā)展的重要力量。這篇講解CS SD NAND貼片式TF卡的特點(diǎn)、技術(shù)參數(shù)及其與傳統(tǒng)TF卡的區(qū)別,并通過(guò)實(shí)際使用案例展示其在不同應(yīng)用場(chǎng)景中的表現(xiàn)。
二、實(shí)物
前段時(shí)間有幸免費(fèi)得到了雷龍出品的貼片式的TF卡的芯片及轉(zhuǎn)接板,兩片貼片式nand芯片+一個(gè)轉(zhuǎn)接板,一種一個(gè)已讓官方焊接完好;如下圖所示:
從獲得的相關(guān)資料看,雷龍出品的貼片式芯片分為兩類,即BOW型和AOW型,其中BOW型為第一代產(chǎn)品,屬商業(yè)級(jí);AOW型則是第二代產(chǎn)品,屬工業(yè)級(jí)或接近工業(yè)級(jí)。
本次收到的具體型號(hào)是CSNP16GCR01-AOW使用pSLC技術(shù),擁有高容量的同時(shí)兼具SLC的特性,不用寫驅(qū)動(dòng)程序自帶壞塊管理的NAND Flash(貼片式TF卡),尺寸小巧,簡(jiǎn)單易用,兼容性強(qiáng),穩(wěn)定可靠,固件可定制,LGA-8封裝,標(biāo)準(zhǔn)SDIO接口,兼容SPI,兼容拔插式TF卡/SD卡,可替代普通TF卡/SD卡,尺寸6.2x8mm,內(nèi)置平均讀寫算法,通過(guò)1萬(wàn)次隨機(jī)掉電測(cè)試,耐高低溫,機(jī)貼手貼都非常方便,速度級(jí)別Class10,標(biāo)準(zhǔn)的SD 2.0協(xié)議使得用戶可以直接移植標(biāo)準(zhǔn)驅(qū)動(dòng)代碼,省去了驅(qū)動(dòng)代碼編程環(huán)節(jié)。支持TF卡啟動(dòng)的SOC都可以用SD NAND,提供STM32參考例程及原廠技術(shù)支持,容量:2GB,比TF卡穩(wěn)定,比eMMC易用。
雷龍CS SD NAND要遠(yuǎn)遠(yuǎn)比TF卡小。芯片能支持兩種讀寫方式,即SD模式和SPI模式在使用時(shí),它有2種工作模式,即默認(rèn)模式和高速模式。
三、SD NAND特征
3.1 SD卡簡(jiǎn)介
雷龍公司推出的SD NAND存儲(chǔ)解決方案,如CSNP4GCR01-AMW與CSNP32GCR01-AOW等型號(hào),是專為滿足現(xiàn)代嵌入式系統(tǒng)需求而設(shè)計(jì)的一種高效存儲(chǔ)介質(zhì)。這些SD NAND存儲(chǔ)芯片結(jié)合了高性能NAND閃存與先進(jìn)的SD控制器技術(shù),能夠在確保數(shù)據(jù)完整性的同時(shí)提供出色的讀寫速度。特別是它們具備強(qiáng)大的壞塊管理能力和先進(jìn)的糾錯(cuò)算法,即使在遭遇意外斷電的情況下也能有效保護(hù)數(shù)據(jù)免受損失。
從技術(shù)規(guī)格來(lái)看,這些SD NAND產(chǎn)品兼容SD2.0標(biāo)準(zhǔn),支持2線或4線接口配置,適用于各種不同的應(yīng)用場(chǎng)景。它們的工作電壓范圍為2.7V至3.6V,這使得它們非常適合作為便攜式設(shè)備或電池供電系統(tǒng)的理想選擇。此外,這些芯片提供了兩種工作模式以適應(yīng)不同的性能需求:默認(rèn)模式下,最大接口傳輸速度可達(dá)12.5MB/s;而在高速模式下,這一數(shù)值翻倍達(dá)到25MB/s,極大地提升了數(shù)據(jù)處理效率。值得注意的是,無(wú)論處于哪種模式,均需配合4條并行數(shù)據(jù)線來(lái)實(shí)現(xiàn)最佳性能表現(xiàn)。
雷龍的SD NAND存儲(chǔ)方案還特別強(qiáng)調(diào)了環(huán)境適應(yīng)性,其工作溫度范圍寬泛,從-40°C到+85°C,確保了即便是在極端條件下也能穩(wěn)定運(yùn)行。同時(shí),這些芯片擁有極低的待機(jī)電流消耗(<250uA),有助于延長(zhǎng)移動(dòng)設(shè)備的電池壽命。安全性方面,除了基本的數(shù)據(jù)保護(hù)措施外,該系列產(chǎn)品還集成了符合SDMI最高安全標(biāo)準(zhǔn)的內(nèi)容保護(hù)機(jī)制,支持通過(guò)CMD42命令實(shí)現(xiàn)SDNAND密碼保護(hù),增強(qiáng)了用戶數(shù)據(jù)的安全性。另外,物理層面的寫保護(hù)功能可以通過(guò)外部機(jī)械開(kāi)關(guān)激活,而內(nèi)部則同時(shí)提供了永久性和臨時(shí)性的寫保護(hù)選項(xiàng),進(jìn)一步增加了靈活性。
在通信協(xié)議的支持上,雷龍SD NAND芯片既支持SDIO讀寫也兼容SPI協(xié)議,這為開(kāi)發(fā)者提供了更多樣化的接入方式選擇。盡管官方標(biāo)稱的最大讀寫速度可達(dá)25MB/s,但實(shí)際能達(dá)到的速度水平將取決于具體的MCU性能及所用接口類型等因素。對(duì)于那些對(duì)讀寫速率沒(méi)有過(guò)高要求的應(yīng)用場(chǎng)景而言,采用SPI協(xié)議往往是一個(gè)更為簡(jiǎn)便且成本效益更高的解決方案。無(wú)論是采取哪種協(xié)議,都必須嚴(yán)格遵循相關(guān)規(guī)范,以確保正確地建立文件系統(tǒng)并維持良好的數(shù)據(jù)交換質(zhì)量。憑借其卓越的性能指標(biāo)、豐富的功能特性以及廣泛的應(yīng)用潛力,雷龍SD NAND存儲(chǔ)解決方案無(wú)疑成為了當(dāng)今嵌入式領(lǐng)域內(nèi)極具競(jìng)爭(zhēng)力的產(chǎn)品之一。
3.2 SD卡Block圖
該SD卡封裝為L(zhǎng)GA-8;引腳分配與定義如下;在這里插入圖片描述:
四、編寫SD NAND驅(qū)動(dòng)代碼
SD NAND 的驅(qū)動(dòng)代碼與正常的SD卡協(xié)議是一樣的,支持標(biāo)準(zhǔn)的SD 2.0協(xié)議,下面我就直接貼出寫好的驅(qū)動(dòng)代碼。
4.1 SPI模擬時(shí)序驅(qū)動(dòng)方式
(1)整體工程代碼
這是當(dāng)前工程的截圖: 代碼采用寄存器風(fēng)格編寫,非常簡(jiǎn)潔。
當(dāng)前工程完成SD NAND卡初始化,扇區(qū)的讀寫,測(cè)試芯片基本的使用情況。
(2) sd.c
#include "sdcard.h"
static u8 SD_Type=0; //存放SD卡的類型
/*
函數(shù)功能:SD卡底層接口,通過(guò)SPI時(shí)序向SD卡讀寫一個(gè)字節(jié)
函數(shù)參數(shù):data是要寫入的數(shù)據(jù)
返 回 值:讀到的數(shù)據(jù)
*/
u8 SDCardReadWriteOneByte(u8 DataTx)
{
u8 i;
u8 data=0;
for(i=0;i<8;i++)
{
SDCARD_SCK=0;
if(DataTx&0x80)SDCARD_MOSI=1;
else SDCARD_MOSI=0;
SDCARD_SCK=1;
DataTx<<=1;
data<<=1;
if(SDCARD_MISO)data|=0x01;
}
return data;
}
//4種: 邊沿兩種、電平是兩種
/*
函數(shù)功能:底層SD卡接口初始化
本程序SPI接口如下:
PC11 片選 SDCardCS
PC12 時(shí)鐘 SDCardSCLK
PD2 輸出 SPI_MOSI--主機(jī)輸出從機(jī)輸入
PC8 輸入 SPI_MISO--主機(jī)輸入從機(jī)輸出
*/
void SDCardSpiInit(void)
{
/*1. 開(kāi)啟時(shí)鐘*/
RCC->APB2ENR|=1<<5; //使能PORTD時(shí)鐘
RCC->APB2ENR|=1<<4; //使能PORTC時(shí)鐘
/*2. 配置GPIO口模式*/
GPIOC->CRH&=0xFFF00FF0;
GPIOC->CRH|=0x00033008;
GPIOD->CRL&=0xFFFFF0FF;
GPIOD->CRL|=0x00000300;
/*3. 上拉*/
GPIOC->ODR|=1<<8;
GPIOC->ODR|=1<<11;
GPIOC->ODR|=1<<12;
GPIOD->ODR|=1<<2;
}
/*
函數(shù)功能:取消選擇,釋放SPI總線
*/
void SDCardCancelCS(void)
{
SDCARD_CS=1;
SDCardReadWriteOneByte(0xff);//提供額外的8個(gè)時(shí)鐘
}
/*
函數(shù) 功 能:選擇sd卡,并且等待卡準(zhǔn)備OK
函數(shù)返回值:0,成功;1,失敗;
*/
void SDCardSelectCS(void)
{
SDCARD_CS=0;
SDCardWaitBusy();//等待成功
}
/*
函數(shù) 功 能:等待卡準(zhǔn)備好
函數(shù)返回值:0,準(zhǔn)備好了;其他,錯(cuò)誤代碼
*/
void SDCardWaitBusy(void)
{
while(SDCardReadWriteOneByte(0XFF)!=0XFF){}
}
/*
函數(shù)功能:等待SD卡回應(yīng)
函數(shù)參數(shù):
Response:要得到的回應(yīng)值
返 回 值:
0,成功得到了該回應(yīng)值
其他,得到回應(yīng)值失敗
*/
u8 SDCardGetAck(u8 Response)
{
u16 Count=0xFFFF;//等待次數(shù)
while((SDCardReadWriteOneByte(0XFF)!=Response)&&Count)Count--;//等待得到準(zhǔn)確的回應(yīng)
if(Count==0)return SDCard_RESPONSE_FAILURE;//得到回應(yīng)失敗
else return SDCard_RESPONSE_NO_ERROR;//正確回應(yīng)
}
/*
函數(shù)功能:從sd卡讀取一個(gè)數(shù)據(jù)包的內(nèi)容
函數(shù)參數(shù):
buf:數(shù)據(jù)緩存區(qū)
len:要讀取的數(shù)據(jù)長(zhǎng)度.
返回值:
0,成功;其他,失敗;
*/
u8 SDCardRecvData(u8*buf,u16 len)
{
if(SDCardGetAck(0xFE))return 1;//等待SD卡發(fā)回?cái)?shù)據(jù)起始令牌0xFE
while(len--)//開(kāi)始接收數(shù)據(jù)
{
*buf=SDCardReadWriteOneByte(0xFF);
buf++;
}
//下面是2個(gè)偽CRC(dummy CRC)
SDCardReadWriteOneByte(0xFF);
SDCardReadWriteOneByte(0xFF);
return 0;//讀取成功
}
/*
函數(shù)功能:向sd卡寫入一個(gè)數(shù)據(jù)包的內(nèi)容 512字節(jié)
函數(shù)參數(shù):
buf 數(shù)據(jù)緩存區(qū)
cmd 指令
返 回 值:0表示成功;其他值表示失敗;
*/
u8 SDCardSendData(u8*buf,u8 cmd)
{
u16 t;
SDCardWaitBusy(); //等待忙狀態(tài)
SDCardReadWriteOneByte(cmd);
if(cmd!=0XFD)//不是結(jié)束指令
{
for(t=0;t<512;t++)SDCardReadWriteOneByte(buf[t]);//提高速度,減少函數(shù)傳參時(shí)間
SDCardReadWriteOneByte(0xFF); //忽略crc
SDCardReadWriteOneByte(0xFF);
t=SDCardReadWriteOneByte(0xFF); //接收響應(yīng)
if((t&0x1F)!=0x05)return 2; //響應(yīng)錯(cuò)誤
}
return 0;//寫入成功
}
/*
函數(shù)功能:向SD卡發(fā)送一個(gè)命令
函數(shù)參數(shù):
u8 cmd 命令
u32 arg 命令參數(shù)
u8 crc crc校驗(yàn)值
返回值:SD卡返回的響應(yīng)
*/
u8 SendSDCardCmd(u8 cmd, u32 arg, u8 crc)
{
u8 r1;
SDCardCancelCS(); //取消上次片選
SDCardSelectCS(); //選中SD卡
//發(fā)送數(shù)據(jù)
SDCardReadWriteOneByte(cmd | 0x40);//分別寫入命令
SDCardReadWriteOneByte(arg >> 24);
SDCardReadWriteOneByte(arg >> 16);
SDCardReadWriteOneByte(arg >> 8);
SDCardReadWriteOneByte(arg);
SDCardReadWriteOneByte(crc);
if(cmd==SDCard_CMD12)SDCardReadWriteOneByte(0xff);//Skip a stuff byte when stop reading
do
{
r1=SDCardReadWriteOneByte(0xFF);
}while(r1&0x80); //等待響應(yīng),或超時(shí)退出
return r1; //返回狀態(tài)值
}
/*
函數(shù)功能:獲取SD卡的CID信息,包括制造商信息
函數(shù)參數(shù):u8 *cid_data(存放CID的內(nèi)存,至少16Byte)
返 回 值:
0:成功,1:錯(cuò)誤
*/
u8 GetSDCardCISDCardOutnfo(u8 *cid_data)
{
u8 r1;
//發(fā)SDCard_CMD10命令,讀CID
r1=SendSDCardCmd(SDCard_CMD10,0,0x01);
if(r1==0x00)
{
r1=SDCardRecvData(cid_data,16);//接收16個(gè)字節(jié)的數(shù)據(jù)
}
SDCardCancelCS();//取消片選
if(r1)return 1;
else return 0;
}
/*
函數(shù)說(shuō)明:
獲取SD卡的CSD信息,包括容量和速度信息
函數(shù)參數(shù):
u8 *cid_data(存放CID的內(nèi)存,至少16Byte)
返 回 值:
0:成功,1:錯(cuò)誤
*/
u8 GetSDCardCSSDCardOutnfo(u8 *csd_data)
{
u8 r1;
r1=SendSDCardCmd(SDCard_CMD9,0,0x01); //發(fā)SDCard_CMD9命令,讀CSD
if(r1==0)
{
r1=SDCardRecvData(csd_data, 16);//接收16個(gè)字節(jié)的數(shù)據(jù)
}
SDCardCancelCS();//取消片選
if(r1)return 1;
else return 0;
}
/*
函數(shù)功能:獲取SD卡的總扇區(qū)數(shù)(扇區(qū)數(shù))
返 回 值:
0表示容量檢測(cè)出錯(cuò),其他值表示SD卡的容量(扇區(qū)數(shù)/512字節(jié))
說(shuō) 明:
每扇區(qū)的字節(jié)數(shù)必為512字節(jié),如果不是512字節(jié),則初始化不能通過(guò).
*/
u32 GetSDCardSectorCount(void)
{
u8 csd[16];
u32 Capacity;
u16 csize;
if(GetSDCardCSSDCardOutnfo(csd)!=0) return 0; //取CSD信息,如果期間出錯(cuò),返回0
if((csd[0]&0xC0)==0x40) //SDHC卡,按照下面方式計(jì)算
{
csize = csd[9] + ((u16)csd[8] << 8) + 1;
Capacity = (u32)csize << 10;//得到扇區(qū)數(shù)
}
return Capacity;
}
/*
函數(shù)功能: 初始化SD卡
返 回 值: 非0表示初始化失敗!
*/
u8 SDCardDeviceInit(void)
{
u8 r1; // 存放SD卡的返回值
u8 buf[4];
u16 i;
SDCardSpiInit();//初始化底層IO口
for(i=0;i<10;i++)SDCardReadWriteOneByte(0xFF); //發(fā)送最少74個(gè)脈沖
do
{
r1=SendSDCardCmd(SDCard_CMD0,0,0x95);//進(jìn)入IDLE狀態(tài) 閑置
}while(r1!=0X01);
SD_Type=0; //默認(rèn)無(wú)卡
if(r1==0X01)
{
if(SendSDCardCmd(SDCard_CMD8,0x1AA,0x87)==1) //SD V2.0
{
for(i=0;i<4;i++)buf[i]=SDCardReadWriteOneByte(0XFF);
if(buf[2]==0X01&&buf[3]==0XAA) //卡是否支持2.7~3.6V
{
do
{
SendSDCardCmd(SDCard_CMD55,0,0X01); //發(fā)送SDCard_CMD55
r1=SendSDCardCmd(SDCard_CMD41,0x40000000,0X01);//發(fā)送SDCard_CMD41
}while(r1);
if(SendSDCardCmd(SDCard_CMD58,0,0X01)==0)//鑒別SD2.0卡版本開(kāi)始
{
for(i=0;i<4;i++)buf[i]=SDCardReadWriteOneByte(0XFF);//得到OCR值
if(buf[0]&0x40)SD_Type=SDCard_TYPE_V2HC; //檢查CCS
else SD_Type=SDCard_TYPE_V2;
}
}
}
}
printf("SD_Type=0x%Xrn",SD_Type);
SDCardCancelCS(); //取消片選
if(SD_Type)return 0; //初始化成功返回0
else if(r1)return r1; //返回值錯(cuò)誤值
return 0xaa; //其他錯(cuò)誤
}
/*
函數(shù)功能:讀SD卡
函數(shù)參數(shù):
buf:數(shù)據(jù)緩存區(qū)
sector:扇區(qū)
cnt:扇區(qū)數(shù)
返回值:
0,ok;其他,失敗.
說(shuō) 明:
SD卡一個(gè)扇區(qū)大小512字節(jié)
*/
u8 SDCardReadData(u8*buf,u32 sector,u32 cnt)
{
u8 r1;
if(SD_Type!=SDCard_TYPE_V2HC)sector<<=9;//轉(zhuǎn)換為字節(jié)地址
if(cnt==1)
{
r1=SendSDCardCmd(SDCard_CMD17,sector,0X01);//讀命令
if(r1==0) //指令發(fā)送成功
{
r1=SDCardRecvData(buf,512); //接收512個(gè)字節(jié)
}
}else
{
r1=SendSDCardCmd(SDCard_CMD18,sector,0X01);//連續(xù)讀命令
do
{
r1=SDCardRecvData(buf,512);//接收512個(gè)字節(jié)
buf+=512;
}while(--cnt && r1==0);
SendSDCardCmd(SDCard_CMD12,0,0X01); //發(fā)送停止命令
}
SDCardCancelCS();//取消片選
return r1;//
}
/*
函數(shù)功能:向SD卡寫數(shù)據(jù)
函數(shù)參數(shù):
buf:數(shù)據(jù)緩存區(qū)
sector:起始扇區(qū)
cnt:扇區(qū)數(shù)
返回值:
0,ok;其他,失敗.
說(shuō) 明:
SD卡一個(gè)扇區(qū)大小512字節(jié)
*/
u8 SDCardWriteData(u8*buf,u32 sector,u32 cnt)
{
u8 r1;
if(SD_Type!=SDCard_TYPE_V2HC)sector *= 512;//轉(zhuǎn)換為字節(jié)地址
if(cnt==1)
{
r1=SendSDCardCmd(SDCard_CMD24,sector,0X01);//讀命令
if(r1==0)//指令發(fā)送成功
{
r1=SDCardSendData(buf,0xFE);//寫512個(gè)字節(jié)
}
}
else
{
if(SD_Type!=SDCard_TYPE_MMC)
{
SendSDCardCmd(SDCard_CMD55,0,0X01);
SendSDCardCmd(SDCard_CMD23,cnt,0X01);//發(fā)送指令
}
r1=SendSDCardCmd(SDCard_CMD25,sector,0X01);//連續(xù)讀命令
if(r1==0)
{
do
{
r1=SDCardSendData(buf,0xFC);//接收512個(gè)字節(jié)
buf+=512;
}while(--cnt && r1==0);
r1=SDCardSendData(0,0xFD);//接收512個(gè)字節(jié)
}
}
SDCardCancelCS();//取消片選
return r1;//
}
(3) sd.h
#ifndef SD_H
#define SD_H_
#include "stm32f10x.h"
#include "led.h"
#include "usart.h"
/*----------------------------------------------
本程序SPI接口如下:
PC11 片選 SDCardCS
PC12 時(shí)鐘 SDCardSCLK
PD2 輸出 SPI_MOSI--主機(jī)輸出從機(jī)輸入
PC8 輸入 SPI_MISO--主機(jī)輸入從機(jī)輸出
------------------------------------------------*/
#define SDCARD_CS PCout(11)
#define SDCARD_SCK PCout(12)
#define SDCARD_MOSI PDout(2)
#define SDCARD_MISO PCin(8)
// SD卡類型定義
#define SDCard_TYPE_ERR 0X00 //卡類型錯(cuò)誤
#define SDCard_TYPE_MMC 0X01 //MMC卡
#define SDCard_TYPE_V1 0X02
#define SDCard_TYPE_V2 0X04
#define SDCard_TYPE_V2HC 0X06
// SD卡指令表
#define SDCard_CMD0 0 //卡復(fù)位
#define SDCard_CMD1 1
#define SDCard_CMD8 8 //命令8 ,SEND_IF_COND
#define SDCard_CMD9 9 //命令9 ,讀CSD數(shù)據(jù)
#define SDCard_CMD10 10 //命令10,讀CID數(shù)據(jù)
#define SDCard_CMD12 12 //命令12,停止數(shù)據(jù)傳輸
#define SDCard_CMD13 16 //命令16,設(shè)置扇區(qū)大小 應(yīng)返回0x00
#define SDCard_CMD17 17 //命令17,讀扇區(qū)
#define SDCard_CMD18 18 //命令18,讀Multi 扇區(qū)
#define SDCard_CMD23 23 //命令23,設(shè)置多扇區(qū)寫入前預(yù)先擦除N個(gè)block
#define SDCard_CMD24 24 //命令24,寫扇區(qū)
#define SDCard_CMD25 25 //命令25,寫多個(gè)扇區(qū)
#define SDCard_CMD41 41 //命令41,應(yīng)返回0x00
#define SDCard_CMD55 55 //命令55,應(yīng)返回0x01
#define SDCard_CMD58 58 //命令58,讀OCR信息
#define SDCard_CMD59 59 //命令59,使能/禁止CRC,應(yīng)返回0x00、
/*SD卡回應(yīng)標(biāo)記字*/
#define SDCard_RESPONSE_NO_ERROR 0x00 //正確回應(yīng)
#define SDCard_SD_IN_IDLE_STATE 0x01 //閑置狀態(tài)
#define SDCard_SD_ERASE_RESET 0x02 //擦除復(fù)位
#define SDCard_RESPONSE_FAILURE 0xFF //響應(yīng)失敗
//函數(shù)聲明
u8 SDCardReadWriteOneByte(u8 data); //底層接口,SPI讀寫字節(jié)函數(shù)
void SDCardWaitBusy(void); //等待SD卡準(zhǔn)備
u8 SDCardGetAck(u8 Response); //獲得應(yīng)答
u8 SDCardDeviceInit(void); //初始化
u8 SDCardReadData(u8*buf,u32 sector,u32 cnt); //讀塊(扇區(qū))
u8 SDCardWriteData(u8*buf,u32 sector,u32 cnt); //寫塊(扇區(qū))
u32 GetSDCardSectorCount(void); //讀扇區(qū)數(shù)
u8 GetSDCardCISDCardOutnfo(u8 *cid_data); //讀SD卡CID
u8 GetSDCardCSSDCardOutnfo(u8 *csd_data); //讀SD卡CSD
#endif
(4)運(yùn)行效果
4.2 SDIO方式
如果想提高SD NAND的讀寫速度,可以采用SDIO協(xié)議,STM32本身有SDIO的硬件支持,配置好SDIO的寄存器即可完成SD NAND的操作。 SDIO的數(shù)據(jù)線都比SPI多,讀寫速度自然沒(méi)法比的。
下面貼出STM32F103ZE上面編寫的SDIO協(xié)議讀寫SD NAND的驅(qū)動(dòng)代碼。
(1)整體工程代碼
(2)sdio.c
#include "sdio_sdcard.h"
#include "string.h"
#include "sys.h"
#include "usart.h"
static u8 CardType=SDIO_STD_CAPACITY_SD_CARD_V1_1; //SD卡類型(默認(rèn)為1.x卡)
static u32 CSD_Tab[4],CID_Tab[4],RCA=0; //SD卡CSD,CID以及相對(duì)地址(RCA)數(shù)據(jù)
static u8 DeviceMode=SD_DMA_MODE; //工作模式,注意,工作模式必須通過(guò)SDIO_SdCardSetDeviceMode,后才算數(shù).這里只是定義一個(gè)默認(rèn)的模式(SD_DMA_MODE)
static u8 StopCondition=0; //是否發(fā)送停止傳輸標(biāo)志位,DMA多塊讀寫的時(shí)候用到
volatile SDIO_SD_ERROR_INFO TransferError=SD_OK; //數(shù)據(jù)傳輸錯(cuò)誤標(biāo)志,DMA讀寫時(shí)使用
volatile u8 TransferEnd=0; //傳輸結(jié)束標(biāo)志,DMA讀寫時(shí)使用
SD_CardInfo SDCardInfo; //SD卡信息
//SDIO_SdCardReadDiskSector/SDIO_SdCardWriteDiskSector函數(shù)專用buf,當(dāng)這兩個(gè)函數(shù)的數(shù)據(jù)緩存區(qū)地址不是4字節(jié)對(duì)齊的時(shí)候,
//需要用到該數(shù)組,確保數(shù)據(jù)緩存區(qū)地址是4字節(jié)對(duì)齊的.
__align(4) u8 SDIO_DATA_BUFFER[512];
/*
SD卡與開(kāi)發(fā)板的SDIO方式接線關(guān)系如下:
DATA0---PC8
DATA1---PC9
DATA2---PC10
DATA3---PC11
CLK-----PC1
CMD-----PD2
*/
/*
函數(shù)功能:SDIO方式初始化SD卡
返回值 :錯(cuò)誤代碼;(0,無(wú)錯(cuò)誤)
*/
SDIO_SD_ERROR_INFO SDIO_SdCardInit(void)
{
u8 clkdiv=0;
SDIO_SD_ERROR_INFO errorstatus=SD_OK;
//SDIO IO口初始化
RCC->APB2ENR|=1<<4; //使能PORTC時(shí)鐘
RCC->APB2ENR|=1<<5; //使能PORTD時(shí)鐘
RCC->AHBENR|=1<<10; //使能SDIO時(shí)鐘
RCC->AHBENR|=1<<1; //使能DMA2時(shí)鐘
GPIOC->CRH&=0XFFF00000;
GPIOC->CRH|=0X000BBBBB; //PC.8~12 復(fù)用輸出
GPIOD->CRL&=0XFFFFF0FF;
GPIOD->CRL|=0X00000B00; //PD2復(fù)用輸出,PD7 上拉輸入
//SDIO外設(shè)寄存器設(shè)置為默認(rèn)值
SDIO->POWER=0x00000000;
SDIO->CLKCR=0x00000000;
SDIO->ARG=0x00000000;
SDIO->CMD=0x00000000;
SDIO->DTIMER=0x00000000;
SDIO->DLEN=0x00000000;
SDIO->DCTRL=0x00000000;
SDIO->ICR=0x00C007FF;
SDIO->MASK=0x00000000;
STM32_NVIC_SetPriority(SDIO_IRQn,0,0); //SDIO中斷配置
errorstatus=SDIO_SdPowerON(); //SD卡上電
SDIO_SdCardInitializeCards(); //初始化SD卡
SDIO_SdCardGetInfo(&SDCardInfo); //獲取卡信息
SDIO_SdCardSelectAddr((u32)(SDCardInfo.RCA<<16));//選中SD卡
SDIO_SdCardEnableWideBusOperation(1); //4位寬度,如果是MMC卡,則不能用4位模式
if((errorstatus==SD_OK)||(SDIO_MULTIMEDIA_CARD==CardType))
{
if(SDCardInfo.CardType==SDIO_STD_CAPACITY_SD_CARD_V1_1||SDCardInfo.CardType==SDIO_STD_CAPACITY_SD_CARD_V2_0)
{
clkdiv=SDIO_TRANSFER_CLK_DIV+6; //V1.1/V2.0卡,設(shè)置最高72/12=6Mhz
}else clkdiv=SDIO_TRANSFER_CLK_DIV; //SDHC等其他卡,設(shè)置最高72/6=12Mhz
SDIO_ClockSet(clkdiv); //設(shè)置時(shí)鐘頻率,SDIO時(shí)鐘計(jì)算公式:SDIO_CK時(shí)鐘=SDIOCLK/[clkdiv+2];其中,SDIOCLK固定為48Mhz
errorstatus=SDIO_SdCardSetDeviceMode(SD_POLLING_MODE); //設(shè)置為查詢模式
}
return errorstatus;
}
/*
函數(shù)功能: SDIO時(shí)鐘初始化設(shè)置
函數(shù)參數(shù):
clkdiv:時(shí)鐘分頻系數(shù)
CK時(shí)鐘=SDIOCLK/[clkdiv+2];(SDIOCLK時(shí)鐘固定為48Mhz)
*/
void SDIO_ClockSet(u8 clkdiv)
{
u32 tmpreg=SDIO->CLKCR;
tmpreg&=0XFFFFFF00;
tmpreg|=clkdiv;
SDIO->CLKCR=tmpreg;
}
/*
函數(shù)功能: SDIO發(fā)送命令函數(shù)
函數(shù)參數(shù):
cmdindex:命令索引,低六位有效
waitrsp:期待的相應(yīng).00/10,無(wú)響應(yīng);01,短響應(yīng);11,長(zhǎng)響應(yīng)
arg:參數(shù)
*/
void SDIO_SendCmd(u8 cmdindex,u8 waitrsp,u32 arg)
{
u32 tmpreg;
SDIO->ARG=arg;
tmpreg=SDIO->CMD;
tmpreg&=0XFFFFF800; //清除index和waitrsp
tmpreg|=cmdindex&0X3F; //設(shè)置新的index
tmpreg|=waitrsp<<6; //設(shè)置新的wait rsp
tmpreg|=0<<8; //無(wú)等待
tmpreg|=1<<10; //命令通道狀態(tài)機(jī)使能
SDIO->CMD=tmpreg;
}
/*
函數(shù)功能: SDIO發(fā)送數(shù)據(jù)配置函數(shù)
函數(shù)參數(shù):
datatimeout:超時(shí)時(shí)間設(shè)置
datalen:傳輸數(shù)據(jù)長(zhǎng)度,低25位有效,必須為塊大小的整數(shù)倍
blksize:塊大小.實(shí)際大小為:2^blksize字節(jié)
dir:數(shù)據(jù)傳輸方向:0,控制器到卡;1,卡到控制器;
*/
void SDIO_SendDataConfig(u32 datatimeout,u32 datalen,u8 blksize,u8 dir)
{
u32 tmpreg;
SDIO->DTIMER=datatimeout;
SDIO->DLEN=datalen&0X1FFFFFF; //低25位有效
tmpreg=SDIO->DCTRL;
tmpreg&=0xFFFFFF08; //清除之前的設(shè)置.
tmpreg|=blksize<<4; //設(shè)置塊大小
tmpreg|=0<<2; //塊數(shù)據(jù)傳輸
tmpreg|=(dir&0X01)<<1; //方向控制
tmpreg|=1<<0; //數(shù)據(jù)傳輸使能,DPSM狀態(tài)機(jī)
SDIO->DCTRL=tmpreg;
}
/*
函數(shù)功能:卡上電
查詢所有SDIO接口上的卡設(shè)備,并查詢其電壓和配置時(shí)鐘
返回值:錯(cuò)誤代碼;(0,無(wú)錯(cuò)誤)
*/
SDIO_SD_ERROR_INFO SDIO_SdPowerON(void)
{
u8 i=0;
SDIO_SD_ERROR_INFO errorstatus=SD_OK;
u32 response=0,count=0,validvoltage=0;
u32 SDType=SD_STD_CAPACITY;
//配置CLKCR寄存器
SDIO->CLKCR=0; //清空CLKCR之前的設(shè)置
SDIO->CLKCR|=0<<9; //非省電模式
SDIO->CLKCR|=0<<10; //關(guān)閉旁路,CK根據(jù)分頻設(shè)置輸出
SDIO->CLKCR|=0<<11; //1位數(shù)據(jù)寬度
SDIO->CLKCR|=0<<13; //SDIOCLK上升沿產(chǎn)生SDIOCK
SDIO->CLKCR|=0<<14; //關(guān)閉硬件流控制
SDIO_ClockSet(SDIO_INIT_CLK_DIV);//設(shè)置時(shí)鐘頻率(初始化的時(shí)候,不能超過(guò)400Khz)
SDIO->POWER=0X03; //上電狀態(tài),開(kāi)啟卡時(shí)鐘
SDIO->CLKCR|=1<<8; //SDIOCK使能
for(i=0;i<74;i++)
{
SDIO_SendCmd(SD_CMD_GO_IDLE_STATE,0,0);//發(fā)送CMD0進(jìn)入IDLE STAGE模式命令.
errorstatus=SDIO_CmdErrorCheck();
if(errorstatus==SD_OK)break;
}
if(errorstatus)return errorstatus;//返回錯(cuò)誤狀態(tài)
SDIO_SendCmd(SDIO_SEND_IF_COND,1,SD_CHECK_PATTERN);//發(fā)送CMD8,短響應(yīng),檢查SD卡接口特性.
//arg[11:8]:01,支持電壓范圍,2.7~3.6V
//arg[7:0]:默認(rèn)0XAA
//返回響應(yīng)7
errorstatus=SDIO_CmdResp7Error(); //等待R7響應(yīng)
if(errorstatus==SD_OK) //R7響應(yīng)正常
{
CardType=SDIO_STD_CAPACITY_SD_CARD_V2_0; //SD 2.0卡
SDType=SD_HIGH_CAPACITY; //高容量卡
}
SDIO_SendCmd(SD_CMD_APP_CMD,1,0); //發(fā)送CMD55,短響應(yīng)
errorstatus=SDIO_CmdResp1Error(SD_CMD_APP_CMD); //等待R1響應(yīng)
if(errorstatus==SD_OK)//SD2.0/SD 1.1
{
//SD卡,發(fā)送ACMD41 SD_APP_OP_COND,參數(shù)為:0x80100000
while((!validvoltage)&&(count<SD_MAX_VOLT_TRIAL))
{
SDIO_SendCmd(SD_CMD_APP_CMD,1,0); //發(fā)送CMD55,短響應(yīng)
errorstatus=SDIO_CmdResp1Error(SD_CMD_APP_CMD); //等待R1響應(yīng)
if(errorstatus!=SD_OK)return errorstatus; //響應(yīng)錯(cuò)誤
SDIO_SendCmd(SD_CMD_SD_APP_OP_COND,1,SD_VOLTAGE_WINDOW_SD|SDType);//發(fā)送ACMD41,短響應(yīng)
errorstatus=SDIO_CmdResp3Error(); //等待R3響應(yīng)
if(errorstatus!=SD_OK)return errorstatus; //響應(yīng)錯(cuò)誤
response=SDIO->RESP1;; //得到響應(yīng)
validvoltage=(((response>>31)==1)?1:0); //判斷SD卡上電是否完成
count++;
}
if(count>=SD_MAX_VOLT_TRIAL)
{
errorstatus=SD_INVALID_VOLTRANGE;
return errorstatus;
}
if(response&=SD_HIGH_CAPACITY)
{
CardType=SDIO_HIGH_CAPACITY_SD_CARD;
}
}
return(errorstatus);
}
/*
函數(shù)功能: SD卡斷電
返回值:錯(cuò)誤代碼;(0,無(wú)錯(cuò)誤)
*/
SDIO_SD_ERROR_INFO SD_PowerOFF(void)
{
SDIO->POWER&=~(3<<0);//SDIO電源關(guān)閉,時(shí)鐘停止
return SD_OK;
}
/*
函數(shù)功能:初始化所有的卡,并讓卡進(jìn)入就緒狀態(tài)
返回值:錯(cuò)誤代碼
*/
SDIO_SD_ERROR_INFO SDIO_SdCardInitializeCards(void)
{
SDIO_SD_ERROR_INFO errorstatus=SD_OK;
u16 rca = 0x01;
if((SDIO->POWER&0X03)==0)return SD_REQUEST_NOT_APPLICABLE;//檢查電源狀態(tài),確保為上電狀態(tài)
if(SDIO_SECURE_DIGITAL_IO_CARD!=CardType) //非SECURE_DIGITAL_IO_CARD
{
SDIO_SendCmd(SD_CMD_ALL_SEND_CID,3,0); //發(fā)送CMD2,取得CID,長(zhǎng)響應(yīng)
errorstatus=SDIO_CmdResp2Error(); //等待R2響應(yīng)
if(errorstatus!=SD_OK)return errorstatus; //響應(yīng)錯(cuò)誤
CID_Tab[0]=SDIO->RESP1;
CID_Tab[1]=SDIO->RESP2;
CID_Tab[2]=SDIO->RESP3;
CID_Tab[3]=SDIO->RESP4;
}
if((SDIO_STD_CAPACITY_SD_CARD_V1_1==CardType)||(SDIO_STD_CAPACITY_SD_CARD_V2_0==CardType)||(SDIO_SECURE_DIGITAL_IO_COMBO_CARD==CardType)||(SDIO_HIGH_CAPACITY_SD_CARD==CardType))//判斷卡類型
{
SDIO_SendCmd(SD_CMD_SET_REL_ADDR,1,0); //發(fā)送CMD3,短響應(yīng)
errorstatus=SDIO_CmdResp6Error(SD_CMD_SET_REL_ADDR,&rca);//等待R6響應(yīng)
if(errorstatus!=SD_OK)return errorstatus; //響應(yīng)錯(cuò)誤
}
if(SDIO_MULTIMEDIA_CARD==CardType)
{
SDIO_SendCmd(SD_CMD_SET_REL_ADDR,1,(u32)(rca<<16));//發(fā)送CMD3,短響應(yīng)
errorstatus=SDIO_CmdResp2Error(); //等待R2響應(yīng)
if(errorstatus!=SD_OK)return errorstatus; //響應(yīng)錯(cuò)誤
}
if(SDIO_SECURE_DIGITAL_IO_CARD!=CardType) //非SECURE_DIGITAL_IO_CARD
{
RCA = rca;
SDIO_SendCmd(SD_CMD_SEND_CSD,3,(u32)(rca<<16));//發(fā)送CMD9+卡RCA,取得CSD,長(zhǎng)響應(yīng)
errorstatus=SDIO_CmdResp2Error(); //等待R2響應(yīng)
if(errorstatus!=SD_OK)return errorstatus; //響應(yīng)錯(cuò)誤
CSD_Tab[0]=SDIO->RESP1;
CSD_Tab[1]=SDIO->RESP2;
CSD_Tab[2]=SDIO->RESP3;
CSD_Tab[3]=SDIO->RESP4;
}
return SD_OK;//卡初始化成功
}
/*
函數(shù)功能:得到卡信息
函數(shù)參數(shù):
cardinfo:卡信息存儲(chǔ)區(qū)
返回值:錯(cuò)誤狀態(tài)
*/
SDIO_SD_ERROR_INFO SDIO_SdCardGetInfo(SD_CardInfo *cardinfo)
{
SDIO_SD_ERROR_INFO errorstatus=SD_OK;
u8 tmp=0;
cardinfo->CardType=(u8)CardType; //卡類型
cardinfo->RCA=(u16)RCA; //卡RCA值
tmp=(u8)((CSD_Tab[0]&0xFF000000)>>24);
cardinfo->SD_csd.CSDStruct=(tmp&0xC0)>>6; //CSD結(jié)構(gòu)
cardinfo->SD_csd.SysSpecVersion=(tmp&0x3C)>>2; //2.0協(xié)議還沒(méi)定義這部分(為保留),應(yīng)該是后續(xù)協(xié)議定義的
cardinfo->SD_csd.Reserved1=tmp&0x03; //2個(gè)保留位
tmp=(u8)((CSD_Tab[0]&0x00FF0000)>>16); //第1個(gè)字節(jié)
cardinfo->SD_csd.TAAC=tmp; //數(shù)據(jù)讀時(shí)間1
tmp=(u8)((CSD_Tab[0]&0x0000FF00)>>8); //第2個(gè)字節(jié)
cardinfo->SD_csd.NSAC=tmp; //數(shù)據(jù)讀時(shí)間2
tmp=(u8)(CSD_Tab[0]&0x000000FF); //第3個(gè)字節(jié)
cardinfo->SD_csd.MaxBusClkFrec=tmp; //傳輸速度
tmp=(u8)((CSD_Tab[1]&0xFF000000)>>24); //第4個(gè)字節(jié)
cardinfo->SD_csd.CardComdClasses=tmp<<4; //卡指令類高四位
tmp=(u8)((CSD_Tab[1]&0x00FF0000)>>16); //第5個(gè)字節(jié)
cardinfo->SD_csd.CardComdClasses|=(tmp&0xF0)>>4;//卡指令類低四位
cardinfo->SD_csd.RdBlockLen=tmp&0x0F; //最大讀取數(shù)據(jù)長(zhǎng)度
tmp=(u8)((CSD_Tab[1]&0x0000FF00)>>8); //第6個(gè)字節(jié)
cardinfo->SD_csd.PartBlockRead=(tmp&0x80)>>7; //允許分塊讀
cardinfo->SD_csd.WrBlockMisalign=(tmp&0x40)>>6; //寫塊錯(cuò)位
cardinfo->SD_csd.RdBlockMisalign=(tmp&0x20)>>5; //讀塊錯(cuò)位
cardinfo->SD_csd.DSRImpl=(tmp&0x10)>>4;
cardinfo->SD_csd.Reserved2=0; //保留
if((CardType==SDIO_STD_CAPACITY_SD_CARD_V1_1)||(CardType==SDIO_STD_CAPACITY_SD_CARD_V2_0)||(SDIO_MULTIMEDIA_CARD==CardType))//標(biāo)準(zhǔn)1.1/2.0卡/MMC卡
{
cardinfo->SD_csd.DeviceSize=(tmp&0x03)<<10; //C_SIZE(12位)
tmp=(u8)(CSD_Tab[1]&0x000000FF); //第7個(gè)字節(jié)
cardinfo->SD_csd.DeviceSize|=(tmp)<<2;
tmp=(u8)((CSD_Tab[2]&0xFF000000)>>24); //第8個(gè)字節(jié)
cardinfo->SD_csd.DeviceSize|=(tmp&0xC0)>>6;
cardinfo->SD_csd.MaxRdCurrentVDDMin=(tmp&0x38)>>3;
cardinfo->SD_csd.MaxRdCurrentVDDMax=(tmp&0x07);
tmp=(u8)((CSD_Tab[2]&0x00FF0000)>>16); //第9個(gè)字節(jié)
cardinfo->SD_csd.MaxWrCurrentVDDMin=(tmp&0xE0)>>5;
cardinfo->SD_csd.MaxWrCurrentVDDMax=(tmp&0x1C)>>2;
cardinfo->SD_csd.DeviceSizeMul=(tmp&0x03)<<1;//C_SIZE_MULT
tmp=(u8)((CSD_Tab[2]&0x0000FF00)>>8); //第10個(gè)字節(jié)
cardinfo->SD_csd.DeviceSizeMul|=(tmp&0x80)>>7;
cardinfo->CardCapacity=(cardinfo->SD_csd.DeviceSize+1);//計(jì)算卡容量
cardinfo->CardCapacity*=(1<<(cardinfo->SD_csd.DeviceSizeMul+2));
cardinfo->CardBlockSize=1<<(cardinfo->SD_csd.RdBlockLen);//塊大小
cardinfo->CardCapacity*=cardinfo->CardBlockSize;
}else if(CardType==SDIO_HIGH_CAPACITY_SD_CARD) //高容量卡
{
tmp=(u8)(CSD_Tab[1]&0x000000FF); //第7個(gè)字節(jié)
cardinfo->SD_csd.DeviceSize=(tmp&0x3F)<<16;//C_SIZE
tmp=(u8)((CSD_Tab[2]&0xFF000000)>>24); //第8個(gè)字節(jié)
cardinfo->SD_csd.DeviceSize|=(tmp<<8);
tmp=(u8)((CSD_Tab[2]&0x00FF0000)>>16); //第9個(gè)字節(jié)
cardinfo->SD_csd.DeviceSize|=(tmp);
tmp=(u8)((CSD_Tab[2]&0x0000FF00)>>8); //第10個(gè)字節(jié)
cardinfo->CardCapacity=(long long)(cardinfo->SD_csd.DeviceSize+1)*512*1024;//計(jì)算卡容量
cardinfo->CardBlockSize=512; //塊大小固定為512字節(jié)
}
cardinfo->SD_csd.EraseGrSize=(tmp&0x40)>>6;
cardinfo->SD_csd.EraseGrMul=(tmp&0x3F)<<1;
tmp=(u8)(CSD_Tab[2]&0x000000FF); //第11個(gè)字節(jié)
cardinfo->SD_csd.EraseGrMul|=(tmp&0x80)>>7;
cardinfo->SD_csd.WrProtectGrSize=(tmp&0x7F);
tmp=(u8)((CSD_Tab[3]&0xFF000000)>>24); //第12個(gè)字節(jié)
cardinfo->SD_csd.WrProtectGrEnable=(tmp&0x80)>>7;
cardinfo->SD_csd.ManDeflECC=(tmp&0x60)>>5;
cardinfo->SD_csd.WrSpeedFact=(tmp&0x1C)>>2;
cardinfo->SD_csd.MaxWrBlockLen=(tmp&0x03)<<2;
tmp=(u8)((CSD_Tab[3]&0x00FF0000)>>16); //第13個(gè)字節(jié)
cardinfo->SD_csd.MaxWrBlockLen|=(tmp&0xC0)>>6;
cardinfo->SD_csd.WriteBlockPaPartial=(tmp&0x20)>>5;
cardinfo->SD_csd.Reserved3=0;
cardinfo->SD_csd.ContentProtectAppli=(tmp&0x01);
tmp=(u8)((CSD_Tab[3]&0x0000FF00)>>8); //第14個(gè)字節(jié)
cardinfo->SD_csd.FileFormatGrouop=(tmp&0x80)>>7;
cardinfo->SD_csd.CopyFlag=(tmp&0x40)>>6;
cardinfo->SD_csd.PermWrProtect=(tmp&0x20)>>5;
cardinfo->SD_csd.TempWrProtect=(tmp&0x10)>>4;
cardinfo->SD_csd.FileFormat=(tmp&0x0C)>>2;
cardinfo->SD_csd.ECC=(tmp&0x03);
tmp=(u8)(CSD_Tab[3]&0x000000FF); //第15個(gè)字節(jié)
cardinfo->SD_csd.CSD_CRC=(tmp&0xFE)>>1;
cardinfo->SD_csd.Reserved4=1;
tmp=(u8)((CID_Tab[0]&0xFF000000)>>24); //第0個(gè)字節(jié)
cardinfo->SD_cid.ManufacturerID=tmp;
tmp=(u8)((CID_Tab[0]&0x00FF0000)>>16); //第1個(gè)字節(jié)
cardinfo->SD_cid.OEM_AppliID=tmp<<8;
tmp=(u8)((CID_Tab[0]&0x000000FF00)>>8); //第2個(gè)字節(jié)
cardinfo->SD_cid.OEM_AppliID|=tmp;
tmp=(u8)(CID_Tab[0]&0x000000FF); //第3個(gè)字節(jié)
cardinfo->SD_cid.ProdName1=tmp<<24;
tmp=(u8)((CID_Tab[1]&0xFF000000)>>24); //第4個(gè)字節(jié)
cardinfo->SD_cid.ProdName1|=tmp<<16;
tmp=(u8)((CID_Tab[1]&0x00FF0000)>>16); //第5個(gè)字節(jié)
cardinfo->SD_cid.ProdName1|=tmp<<8;
tmp=(u8)((CID_Tab[1]&0x0000FF00)>>8); //第6個(gè)字節(jié)
cardinfo->SD_cid.ProdName1|=tmp;
tmp=(u8)(CID_Tab[1]&0x000000FF); //第7個(gè)字節(jié)
cardinfo->SD_cid.ProdName2=tmp;
tmp=(u8)((CID_Tab[2]&0xFF000000)>>24); //第8個(gè)字節(jié)
cardinfo->SD_cid.ProdRev=tmp;
tmp=(u8)((CID_Tab[2]&0x00FF0000)>>16); //第9個(gè)字節(jié)
cardinfo->SD_cid.ProdSN=tmp<<24;
tmp=(u8)((CID_Tab[2]&0x0000FF00)>>8); //第10個(gè)字節(jié)
cardinfo->SD_cid.ProdSN|=tmp<<16;
tmp=(u8)(CID_Tab[2]&0x000000FF); //第11個(gè)字節(jié)
cardinfo->SD_cid.ProdSN|=tmp<<8;
tmp=(u8)((CID_Tab[3]&0xFF000000)>>24); //第12個(gè)字節(jié)
cardinfo->SD_cid.ProdSN|=tmp;
tmp=(u8)((CID_Tab[3]&0x00FF0000)>>16); //第13個(gè)字節(jié)
cardinfo->SD_cid.Reserved1|=(tmp&0xF0)>>4;
cardinfo->SD_cid.ManufactDate=(tmp&0x0F)<<8;
tmp=(u8)((CID_Tab[3]&0x0000FF00)>>8); //第14個(gè)字節(jié)
cardinfo->SD_cid.ManufactDate|=tmp;
tmp=(u8)(CID_Tab[3]&0x000000FF); //第15個(gè)字節(jié)
cardinfo->SD_cid.CID_CRC=(tmp&0xFE)>>1;
cardinfo->SD_cid.Reserved2=1;
return errorstatus;
}
/*
函數(shù)功能: 設(shè)置SDIO總線寬度
函數(shù)參數(shù):
wmode:位寬模式.0,1位數(shù)據(jù)寬度;1,4位數(shù)據(jù)寬度;2,8位數(shù)據(jù)寬度
返回值:SD卡錯(cuò)誤狀態(tài)
*/
SDIO_SD_ERROR_INFO SDIO_SdCardEnableWideBusOperation(u32 wmode)
{
SDIO_SD_ERROR_INFO errorstatus=SD_OK;
if((SDIO_STD_CAPACITY_SD_CARD_V1_1==CardType)||(SDIO_STD_CAPACITY_SD_CARD_V2_0==CardType)||(SDIO_HIGH_CAPACITY_SD_CARD==CardType))
{
if(wmode>=2)return SD_UNSUPPORTED_FEATURE;//不支持8位模式
else
{
errorstatus=SDIO_SdCardEnWideBus(wmode);
if(SD_OK==errorstatus)
{
SDIO->CLKCR&=~(3<<11); //清除之前的位寬設(shè)置
SDIO->CLKCR|=(u16)wmode<<11;//1位/4位總線寬度
SDIO->CLKCR|=0<<14; //不開(kāi)啟硬件流控制
}
}
}
return errorstatus;
}
/*
函數(shù)功能:設(shè)置SD卡工作模式
返回值:錯(cuò)誤狀態(tài)
*/
SDIO_SD_ERROR_INFO SDIO_SdCardSetDeviceMode(u32 Mode)
{
SDIO_SD_ERROR_INFO errorstatus = SD_OK;
if((Mode==SD_DMA_MODE)||(Mode==SD_POLLING_MODE))DeviceMode=Mode;
else errorstatus=SD_INVALID_PARAMETER;
return errorstatus;
}
/*
函數(shù)功能:選卡,發(fā)送CMD7,選擇相對(duì)地址(rca)為addr的卡,取消其他卡.如果為0,則都不選擇.
函數(shù)參數(shù):
addr:卡的RCA地址
*/
SDIO_SD_ERROR_INFO SDIO_SdCardSelectAddr(u32 addr)
{
SDIO_SendCmd(SD_CMD_SEL_DESEL_CARD,1,addr); //發(fā)送CMD7,選擇卡,短響應(yīng)
return SDIO_CmdResp1Error(SD_CMD_SEL_DESEL_CARD);
}
/*
函數(shù)功能: SD卡讀取一個(gè)塊
函數(shù)參數(shù):
buf:讀數(shù)據(jù)緩存區(qū)(必須4字節(jié)對(duì)齊!!)
addr:讀取地址
blksize:塊大小
*/
SDIO_SD_ERROR_INFO SDIO_SdCardReadBlock(u8 *buf,long long addr,u16 blksize)
{
SDIO_SD_ERROR_INFO errorstatus=SD_OK;
u8 power;
u32 count=0,*tempbuff=(u32*)buf;//轉(zhuǎn)換為u32指針
u32 timeout=SDIO_DATATIMEOUT;
if(NULL==buf)return SD_INVALID_PARAMETER;
SDIO->DCTRL=0x0; //數(shù)據(jù)控制寄存器清零(關(guān)DMA)
if(CardType==SDIO_HIGH_CAPACITY_SD_CARD)//大容量卡
{
blksize=512;
addr>>=9;
}
SDIO_SendDataConfig(SD_DATATIMEOUT,0,0,0); //清除DPSM狀態(tài)機(jī)配置
if(SDIO->RESP1&SD_CARD_LOCKED)return SD_LOCK_UNLOCK_FAILED;//卡鎖了
if((blksize>0)&&(blksize<=2048)&&((blksize&(blksize-1))==0))
{
power=convert_from_bytes_to_power_of_two(blksize);
SDIO_SendCmd(SD_CMD_SET_BLOCKLEN,1,blksize); //發(fā)送CMD16+設(shè)置數(shù)據(jù)長(zhǎng)度為blksize,短響應(yīng)
errorstatus=SDIO_CmdResp1Error(SD_CMD_SET_BLOCKLEN); //等待R1響應(yīng)
if(errorstatus!=SD_OK)return errorstatus; //響應(yīng)錯(cuò)誤
}else return SD_INVALID_PARAMETER;
SDIO_SendDataConfig(SD_DATATIMEOUT,blksize,power,1); //blksize,卡到控制器
SDIO_SendCmd(SD_CMD_READ_SINGLE_BLOCK,1,addr); //發(fā)送CMD17+從addr地址出讀取數(shù)據(jù),短響應(yīng)
errorstatus=SDIO_CmdResp1Error(SD_CMD_READ_SINGLE_BLOCK);//等待R1響應(yīng)
if(errorstatus!=SD_OK)return errorstatus; //響應(yīng)錯(cuò)誤
if(DeviceMode==SD_POLLING_MODE) //查詢模式,輪詢數(shù)據(jù)
{
// INTX_DISABLE();//關(guān)閉總中斷(POLLING模式,嚴(yán)禁中斷打斷SDIO讀寫操作!!!)
while(!(SDIO->STA&((1<<5)|(1<<1)|(1<<3)|(1<<10)|(1<<9))))//無(wú)上溢/CRC/超時(shí)/完成(標(biāo)志)/起始位錯(cuò)誤
{
if(SDIO->STA&(1<<15)) //接收區(qū)半滿,表示至少存了8個(gè)字
{
for(count=0;count<8;count++) //循環(huán)讀取數(shù)據(jù)
{
*(tempbuff+count)=SDIO->FIFO;
}
tempbuff+=8;
timeout=0X7FFFFF; //讀數(shù)據(jù)溢出時(shí)間
}else //處理超時(shí)
{
if(timeout==0)return SD_DATA_TIMEOUT;
timeout--;
}
}
if(SDIO->STA&(1<<3)) //數(shù)據(jù)超時(shí)錯(cuò)誤
{
SDIO->ICR|=1<<3; //清錯(cuò)誤標(biāo)志
return SD_DATA_TIMEOUT;
}else if(SDIO->STA&(1<<1)) //數(shù)據(jù)塊CRC錯(cuò)誤
{
SDIO->ICR|=1<<1; //清錯(cuò)誤標(biāo)志
return SD_DATA_CRC_FAIL;
}else if(SDIO->STA&(1<<5)) //接收f(shuō)ifo上溢錯(cuò)誤
{
SDIO->ICR|=1<<5; //清錯(cuò)誤標(biāo)志
return SD_RX_OVERRUN;
}else if(SDIO->STA&(1<<9)) //接收起始位錯(cuò)誤
{
SDIO->ICR|=1<<9; //清錯(cuò)誤標(biāo)志
return SD_START_BIT_ERR;
}
while(SDIO->STA&(1<<21)) //FIFO里面,還存在可用數(shù)據(jù)
{
*tempbuff=SDIO->FIFO; //循環(huán)讀取數(shù)據(jù)
tempbuff++;
}
// INTX_ENABLE();//開(kāi)啟總中斷
SDIO->ICR=0X5FF; //清除所有標(biāo)記
}else if(DeviceMode==SD_DMA_MODE)
{
SDIO_SdCard_DMAConfig((u32*)buf,blksize,0);
TransferError=SD_OK;
StopCondition=0; //單塊讀,不需要發(fā)送停止傳輸指令
TransferEnd=0; //傳輸結(jié)束標(biāo)置位,在中斷服務(wù)置1
SDIO->MASK|=(1<<1)|(1<<3)|(1<<8)|(1<<5)|(1<<9); //配置需要的中斷
SDIO->DCTRL|=1<<3; //SDIO DMA使能
while(((DMA2->ISR&0X2000)==RESET)&&(TransferEnd==0)&&(TransferError==SD_OK)&&timeout)timeout--;//等待傳輸完成
if(timeout==0)return SD_DATA_TIMEOUT;//超時(shí)
if(TransferError!=SD_OK)errorstatus=TransferError;
}
return errorstatus;
}
/*
函數(shù)功能: SD卡讀取多個(gè)塊
函數(shù)參數(shù):
buf:讀數(shù)據(jù)緩存區(qū)
addr:讀取地址
blksize:塊大小
nblks:要讀取的塊數(shù)
返回值:錯(cuò)誤狀態(tài)
*/
__align(4) u32 *tempbuff;
SDIO_SD_ERROR_INFO SDIO_SdCardReadMultiBlocks(u8 *buf,long long addr,u16 blksize,u32 nblks)
{
SDIO_SD_ERROR_INFO errorstatus=SD_OK;
u8 power;
u32 count=0;
u32 timeout=SDIO_DATATIMEOUT;
tempbuff=(u32*)buf; //轉(zhuǎn)換為u32指針
SDIO->DCTRL=0x0; //數(shù)據(jù)控制寄存器清零(關(guān)DMA)
if(CardType==SDIO_HIGH_CAPACITY_SD_CARD)//大容量卡
{
blksize=512;
addr>>=9;
}
SDIO_SendDataConfig(SD_DATATIMEOUT,0,0,0); //清除DPSM狀態(tài)機(jī)配置
if(SDIO->RESP1&SD_CARD_LOCKED)return SD_LOCK_UNLOCK_FAILED;//卡鎖了
if((blksize>0)&&(blksize<=2048)&&((blksize&(blksize-1))==0))
{
power=convert_from_bytes_to_power_of_two(blksize);
SDIO_SendCmd(SD_CMD_SET_BLOCKLEN,1,blksize); //發(fā)送CMD16+設(shè)置數(shù)據(jù)長(zhǎng)度為blksize,短響應(yīng)
errorstatus=SDIO_CmdResp1Error(SD_CMD_SET_BLOCKLEN); //等待R1響應(yīng)
if(errorstatus!=SD_OK)return errorstatus; //響應(yīng)錯(cuò)誤
}else return SD_INVALID_PARAMETER;
if(nblks>1) //多塊讀
{
if(nblks*blksize>SD_MAX_DATA_LENGTH)return SD_INVALID_PARAMETER;//判斷是否超過(guò)最大接收長(zhǎng)度
SDIO_SendDataConfig(SD_DATATIMEOUT,nblks*blksize,power,1);//nblks*blksize,512塊大小,卡到控制器
SDIO_SendCmd(SD_CMD_READ_MULT_BLOCK,1,addr); //發(fā)送CMD18+從addr地址出讀取數(shù)據(jù),短響應(yīng)
errorstatus=SDIO_CmdResp1Error(SD_CMD_READ_MULT_BLOCK);//等待R1響應(yīng)
if(errorstatus!=SD_OK)return errorstatus; //響應(yīng)錯(cuò)誤
if(DeviceMode==SD_POLLING_MODE)
{
// INTX_DISABLE();//關(guān)閉總中斷(POLLING模式,嚴(yán)禁中斷打斷SDIO讀寫操作!!!)
while(!(SDIO->STA&((1<<5)|(1<<1)|(1<<3)|(1<<8)|(1<<9))))//無(wú)上溢/CRC/超時(shí)/完成(標(biāo)志)/起始位錯(cuò)誤
{
if(SDIO->STA&(1<<15)) //接收區(qū)半滿,表示至少存了8個(gè)字
{
for(count=0;count<8;count++) //循環(huán)讀取數(shù)據(jù)
{
*(tempbuff+count)=SDIO->FIFO;
}
tempbuff+=8;
timeout=0X7FFFFF; //讀數(shù)據(jù)溢出時(shí)間
}else //處理超時(shí)
{
if(timeout==0)return SD_DATA_TIMEOUT;
timeout--;
}
}
if(SDIO->STA&(1<<3)) //數(shù)據(jù)超時(shí)錯(cuò)誤
{
SDIO->ICR|=1<<3; //清錯(cuò)誤標(biāo)志
return SD_DATA_TIMEOUT;
}else if(SDIO->STA&(1<<1)) //數(shù)據(jù)塊CRC錯(cuò)誤
{
SDIO->ICR|=1<<1; //清錯(cuò)誤標(biāo)志
return SD_DATA_CRC_FAIL;
}else if(SDIO->STA&(1<<5)) //接收f(shuō)ifo上溢錯(cuò)誤
{
SDIO->ICR|=1<<5; //清錯(cuò)誤標(biāo)志
return SD_RX_OVERRUN;
}else if(SDIO->STA&(1<<9)) //接收起始位錯(cuò)誤
{
SDIO->ICR|=1<<9; //清錯(cuò)誤標(biāo)志
return SD_START_BIT_ERR;
}
while(SDIO->STA&(1<<21)) //FIFO里面,還存在可用數(shù)據(jù)
{
*tempbuff=SDIO->FIFO; //循環(huán)讀取數(shù)據(jù)
tempbuff++;
}
if(SDIO->STA&(1<<8)) //接收結(jié)束
{
if((SDIO_STD_CAPACITY_SD_CARD_V1_1==CardType)||(SDIO_STD_CAPACITY_SD_CARD_V2_0==CardType)||(SDIO_HIGH_CAPACITY_SD_CARD==CardType))
{
SDIO_SendCmd(SD_CMD_STOP_TRANSMISSION,1,0); //發(fā)送CMD12+結(jié)束傳輸
errorstatus=SDIO_CmdResp1Error(SD_CMD_STOP_TRANSMISSION);//等待R1響應(yīng)
if(errorstatus!=SD_OK)return errorstatus;
}
}
// INTX_ENABLE();//開(kāi)啟總中斷
SDIO->ICR=0X5FF; //清除所有標(biāo)記
}
else if(DeviceMode==SD_DMA_MODE)
{
SDIO_SdCard_DMAConfig((u32*)buf,nblks*blksize,0);
TransferError=SD_OK;
StopCondition=1; //多塊讀,需要發(fā)送停止傳輸指令
TransferEnd=0; //傳輸結(jié)束標(biāo)置位,在中斷服務(wù)置1
SDIO->MASK|=(1<<1)|(1<<3)|(1<<8)|(1<<5)|(1<<9); //配置需要的中斷
SDIO->DCTRL|=1<<3; //SDIO DMA使能
while(((DMA2->ISR&0X2000)==RESET)&&timeout)timeout--;//等待傳輸完成
if(timeout==0)return SD_DATA_TIMEOUT;//超時(shí)
while((TransferEnd==0)&&(TransferError==SD_OK));
if(TransferError!=SD_OK)errorstatus=TransferError;
}
}
return errorstatus;
}
/*
函數(shù)功能:SD卡寫1個(gè)塊
函數(shù)參數(shù):
buf:數(shù)據(jù)緩存區(qū)
addr:寫地址
blksize:塊大小
返回值:錯(cuò)誤狀態(tài)
*/
SDIO_SD_ERROR_INFO SDIO_SdCardWriteBlock(u8 *buf,long long addr, u16 blksize)
{
SDIO_SD_ERROR_INFO errorstatus = SD_OK;
u8 power=0,cardstate=0;
u32 timeout=0,bytestransferred=0;
u32 cardstatus=0,count=0,restwords=0;
u32 tlen=blksize; //總長(zhǎng)度(字節(jié))
u32*tempbuff=(u32*)buf;
if(buf==NULL)return SD_INVALID_PARAMETER;//參數(shù)錯(cuò)誤
SDIO->DCTRL=0x0; //數(shù)據(jù)控制寄存器清零(關(guān)DMA)
SDIO_SendDataConfig(SD_DATATIMEOUT,0,0,0); //清除DPSM狀態(tài)機(jī)配置
if(SDIO->RESP1&SD_CARD_LOCKED)return SD_LOCK_UNLOCK_FAILED;//卡鎖了
if(CardType==SDIO_HIGH_CAPACITY_SD_CARD) //大容量卡
{
blksize=512;
addr>>=9;
}
if((blksize>0)&&(blksize<=2048)&&((blksize&(blksize-1))==0))
{
power=convert_from_bytes_to_power_of_two(blksize);
SDIO_SendCmd(SD_CMD_SET_BLOCKLEN,1,blksize); //發(fā)送CMD16+設(shè)置數(shù)據(jù)長(zhǎng)度為blksize,短響應(yīng)
errorstatus=SDIO_CmdResp1Error(SD_CMD_SET_BLOCKLEN); //等待R1響應(yīng)
if(errorstatus!=SD_OK)return errorstatus; //響應(yīng)錯(cuò)誤
}else return SD_INVALID_PARAMETER;
SDIO_SendCmd(SD_CMD_SEND_STATUS,1,(u32)RCA<<16); //發(fā)送CMD13,查詢卡的狀態(tài),短響應(yīng)
errorstatus=SDIO_CmdResp1Error(SD_CMD_SEND_STATUS); //等待R1響應(yīng)
if(errorstatus!=SD_OK)return errorstatus;
cardstatus=SDIO->RESP1;
timeout=SD_DATATIMEOUT;
while(((cardstatus&0x00000100)==0)&&(timeout>0)) //檢查READY_FOR_DATA位是否置位
{
timeout--;
SDIO_SendCmd(SD_CMD_SEND_STATUS,1,(u32)RCA<<16);//發(fā)送CMD13,查詢卡的狀態(tài),短響應(yīng)
errorstatus=SDIO_CmdResp1Error(SD_CMD_SEND_STATUS); //等待R1響應(yīng)
if(errorstatus!=SD_OK)return errorstatus;
cardstatus=SDIO->RESP1;
}
if(timeout==0)return SD_ERROR;
SDIO_SendCmd(SD_CMD_WRITE_SINGLE_BLOCK,1,addr); //發(fā)送CMD24,寫單塊指令,短響應(yīng)
errorstatus=SDIO_CmdResp1Error(SD_CMD_WRITE_SINGLE_BLOCK);//等待R1響應(yīng)
if(errorstatus!=SD_OK)return errorstatus;
StopCondition=0; //單塊寫,不需要發(fā)送停止傳輸指令
SDIO_SendDataConfig(SD_DATATIMEOUT,blksize,power,0); //blksize, 控制器到卡
timeout=SDIO_DATATIMEOUT;
if(DeviceMode == SD_POLLING_MODE)
{
// INTX_DISABLE();//關(guān)閉總中斷(POLLING模式,嚴(yán)禁中斷打斷SDIO讀寫操作!!!)
while(!(SDIO->STA&((1<<10)|(1<<4)|(1<<1)|(1<<3)|(1<<9))))//數(shù)據(jù)塊發(fā)送成功/下溢/CRC/超時(shí)/起始位錯(cuò)誤
{
if(SDIO->STA&(1<<14)) //發(fā)送區(qū)半空,表示至少存了8個(gè)字
{
if((tlen-bytestransferred)<SD_HALFFIFOBYTES)//不夠32字節(jié)了
{
restwords=((tlen-bytestransferred)%4==0)?((tlen-bytestransferred)/4):((tlen-bytestransferred)/4+1);
for(count=0;count<restwords;count++,tempbuff++,bytestransferred+=4)
{
SDIO->FIFO=*tempbuff;
}
}else
{
for(count=0;count<8;count++)
{
SDIO->FIFO=*(tempbuff+count);
}
tempbuff+=8;
bytestransferred+=32;
}
timeout=0X3FFFFFFF; //寫數(shù)據(jù)溢出時(shí)間
}else
{
if(timeout==0)return SD_DATA_TIMEOUT;
timeout--;
}
}
if(SDIO->STA&(1<<3)) //數(shù)據(jù)超時(shí)錯(cuò)誤
{
SDIO->ICR|=1<<3; //清錯(cuò)誤標(biāo)志
return SD_DATA_TIMEOUT;
}else if(SDIO->STA&(1<<1)) //數(shù)據(jù)塊CRC錯(cuò)誤
{
SDIO->ICR|=1<<1; //清錯(cuò)誤標(biāo)志
return SD_DATA_CRC_FAIL;
}else if(SDIO->STA&(1<<4)) //接收f(shuō)ifo下溢錯(cuò)誤
{
SDIO->ICR|=1<<4; //清錯(cuò)誤標(biāo)志
return SD_TX_UNDERRUN;
}else if(SDIO->STA&(1<<9)) //接收起始位錯(cuò)誤
{
SDIO->ICR|=1<<9; //清錯(cuò)誤標(biāo)志
return SD_START_BIT_ERR;
}
// INTX_ENABLE();//開(kāi)啟總中斷
SDIO->ICR=0X5FF; //清除所有標(biāo)記
}else if(DeviceMode==SD_DMA_MODE)
{
SDIO_SdCard_DMAConfig((u32*)buf,blksize,1);//SDIO DMA配置
TransferError=SD_OK;
StopCondition=0; //單塊寫,不需要發(fā)送停止傳輸指令
TransferEnd=0; //傳輸結(jié)束標(biāo)置位,在中斷服務(wù)置1
SDIO->MASK|=(1<<1)|(1<<3)|(1<<8)|(1<<4)|(1<<9); //配置產(chǎn)生數(shù)據(jù)接收完成中斷
SDIO->DCTRL|=1<<3; //SDIO DMA使能.
while(((DMA2->ISR&0X2000)==RESET)&&timeout)timeout--;//等待傳輸完成
if(timeout==0)
{
SDIO_SdCardInit(); //重新初始化SD卡,可以解決寫入死機(jī)的問(wèn)題
return SD_DATA_TIMEOUT; //超時(shí)
}
timeout=SDIO_DATATIMEOUT;
while((TransferEnd==0)&&(TransferError==SD_OK)&&timeout)timeout--;
if(timeout==0)return SD_DATA_TIMEOUT; //超時(shí)
if(TransferError!=SD_OK)return TransferError;
}
SDIO->ICR=0X5FF; //清除所有標(biāo)記
errorstatus=SDIO_SdCardProgrammingState(&cardstate);
while((errorstatus==SD_OK)&&((cardstate==SD_CARD_PROGRAMMING)||(cardstate==SD_CARD_RECEIVING)))
{
errorstatus=SDIO_SdCardProgrammingState(&cardstate);
}
return errorstatus;
}
/*
函數(shù)功能:SD卡寫多個(gè)塊
函數(shù)參數(shù):
buf:數(shù)據(jù)緩存區(qū)
addr:寫地址
blksize:塊大小
nblks:要寫入的塊數(shù)
返回值:錯(cuò)誤狀態(tài)
*/
SDIO_SD_ERROR_INFO SDIO_SdCardWriteMultiBlocks(u8 *buf,long long addr,u16 blksize,u32 nblks)
{
SDIO_SD_ERROR_INFO errorstatus = SD_OK;
u8 power = 0, cardstate = 0;
u32 timeout=0,bytestransferred=0;
u32 count = 0, restwords = 0;
u32 tlen=nblks*blksize; //總長(zhǎng)度(字節(jié))
u32 *tempbuff = (u32*)buf;
if(buf==NULL)return SD_INVALID_PARAMETER; //參數(shù)錯(cuò)誤
SDIO->DCTRL=0x0; //數(shù)據(jù)控制寄存器清零(關(guān)DMA)
SDIO_SendDataConfig(SD_DATATIMEOUT,0,0,0); //清除DPSM狀態(tài)機(jī)配置
if(SDIO->RESP1&SD_CARD_LOCKED)return SD_LOCK_UNLOCK_FAILED;//卡鎖了
if(CardType==SDIO_HIGH_CAPACITY_SD_CARD)//大容量卡
{
blksize=512;
addr>>=9;
}
if((blksize>0)&&(blksize<=2048)&&((blksize&(blksize-1))==0))
{
power=convert_from_bytes_to_power_of_two(blksize);
SDIO_SendCmd(SD_CMD_SET_BLOCKLEN,1,blksize); //發(fā)送CMD16+設(shè)置數(shù)據(jù)長(zhǎng)度為blksize,短響應(yīng)
errorstatus=SDIO_CmdResp1Error(SD_CMD_SET_BLOCKLEN); //等待R1響應(yīng)
if(errorstatus!=SD_OK)return errorstatus; //響應(yīng)錯(cuò)誤
}else return SD_INVALID_PARAMETER;
if(nblks>1)
{
if(nblks*blksize>SD_MAX_DATA_LENGTH)return SD_INVALID_PARAMETER;
if((SDIO_STD_CAPACITY_SD_CARD_V1_1==CardType)||(SDIO_STD_CAPACITY_SD_CARD_V2_0==CardType)||(SDIO_HIGH_CAPACITY_SD_CARD==CardType))
{
//提高性能
SDIO_SendCmd(SD_CMD_APP_CMD,1,(u32)RCA<<16); //發(fā)送ACMD55,短響應(yīng)
errorstatus=SDIO_CmdResp1Error(SD_CMD_APP_CMD); //等待R1響應(yīng)
if(errorstatus!=SD_OK)return errorstatus;
SDIO_SendCmd(SD_CMD_SET_BLOCK_COUNT,1,nblks); //發(fā)送CMD23,設(shè)置塊數(shù)量,短響應(yīng)
errorstatus=SDIO_CmdResp1Error(SD_CMD_SET_BLOCK_COUNT);//等待R1響應(yīng)
if(errorstatus!=SD_OK)return errorstatus;
}
SDIO_SendCmd(SD_CMD_WRITE_MULT_BLOCK,1,addr); //發(fā)送CMD25,多塊寫指令,短響應(yīng)
errorstatus=SDIO_CmdResp1Error(SD_CMD_WRITE_MULT_BLOCK); //等待R1響應(yīng)
if(errorstatus!=SD_OK)return errorstatus;
SDIO_SendDataConfig(SD_DATATIMEOUT,nblks*blksize,power,0);//blksize, 控制器到卡
if(DeviceMode==SD_POLLING_MODE)
{
timeout=SDIO_DATATIMEOUT;
while(!(SDIO->STA&((1<<4)|(1<<1)|(1<<8)|(1<<3)|(1<<9))))//下溢/CRC/數(shù)據(jù)結(jié)束/超時(shí)/起始位錯(cuò)誤
{
if(SDIO->STA&(1<<14)) //發(fā)送區(qū)半空,表示至少存了8字(32字節(jié))
{
if((tlen-bytestransferred)<SD_HALFFIFOBYTES)//不夠32字節(jié)了
{
restwords=((tlen-bytestransferred)%4==0)?((tlen-bytestransferred)/4):((tlen-bytestransferred)/4+1);
for(count=0;count<restwords;count++,tempbuff++,bytestransferred+=4)
{
SDIO->FIFO=*tempbuff;
}
}else //發(fā)送區(qū)半空,可以發(fā)送至少8字(32字節(jié))數(shù)據(jù)
{
for(count=0;count<SD_HALFFIFO;count++)
{
SDIO->FIFO=*(tempbuff+count);
}
tempbuff+=SD_HALFFIFO;
bytestransferred+=SD_HALFFIFOBYTES;
}
timeout=0X3FFFFFFF; //寫數(shù)據(jù)溢出時(shí)間
}else
{
if(timeout==0)return SD_DATA_TIMEOUT;
timeout--;
}
}
if(SDIO->STA&(1<<3)) //數(shù)據(jù)超時(shí)錯(cuò)誤
{
SDIO->ICR|=1<<3; //清錯(cuò)誤標(biāo)志
return SD_DATA_TIMEOUT;
}else if(SDIO->STA&(1<<1)) //數(shù)據(jù)塊CRC錯(cuò)誤
{
SDIO->ICR|=1<<1; //清錯(cuò)誤標(biāo)志
return SD_DATA_CRC_FAIL;
}else if(SDIO->STA&(1<<4)) //接收f(shuō)ifo下溢錯(cuò)誤
{
SDIO->ICR|=1<<4; //清錯(cuò)誤標(biāo)志
return SD_TX_UNDERRUN;
}else if(SDIO->STA&(1<<9)) //接收起始位錯(cuò)誤
{
SDIO->ICR|=1<<9; //清錯(cuò)誤標(biāo)志
return SD_START_BIT_ERR;
}
if(SDIO->STA&(1<<8)) //發(fā)送結(jié)束
{
if((SDIO_STD_CAPACITY_SD_CARD_V1_1==CardType)||(SDIO_STD_CAPACITY_SD_CARD_V2_0==CardType)||(SDIO_HIGH_CAPACITY_SD_CARD==CardType))
{
SDIO_SendCmd(SD_CMD_STOP_TRANSMISSION,1,0); //發(fā)送CMD12+結(jié)束傳輸
errorstatus=SDIO_CmdResp1Error(SD_CMD_STOP_TRANSMISSION);//等待R1響應(yīng)
if(errorstatus!=SD_OK)return errorstatus;
}
}
// INTX_ENABLE();//開(kāi)啟總中斷
SDIO->ICR=0X5FF; //清除所有標(biāo)記
}
else if(DeviceMode==SD_DMA_MODE)
{
SDIO_SdCard_DMAConfig((u32*)buf,nblks*blksize,1);//SDIO DMA配置
TransferError=SD_OK;
StopCondition=1; //多塊寫,需要發(fā)送停止傳輸指令
TransferEnd=0; //傳輸結(jié)束標(biāo)置位,在中斷服務(wù)置1
SDIO->MASK|=(1<<1)|(1<<3)|(1<<8)|(1<<4)|(1<<9); //配置產(chǎn)生數(shù)據(jù)接收完成中斷
SDIO->DCTRL|=1<<3; //SDIO DMA使能.
timeout=SDIO_DATATIMEOUT;
while(((DMA2->ISR&0X2000)==RESET)&&timeout)timeout--;//等待傳輸完成
if(timeout==0) //超時(shí)
{
SDIO_SdCardInit(); //重新初始化SD卡,可以解決寫入死機(jī)的問(wèn)題
return SD_DATA_TIMEOUT; //超時(shí)
}
timeout=SDIO_DATATIMEOUT;
while((TransferEnd==0)&&(TransferError==SD_OK)&&timeout)timeout--;
if(timeout==0)return SD_DATA_TIMEOUT; //超時(shí)
if(TransferError!=SD_OK)return TransferError;
}
}
SDIO->ICR=0X5FF; //清除所有標(biāo)記
errorstatus=SDIO_SdCardProgrammingState(&cardstate);
while((errorstatus==SD_OK)&&((cardstate==SD_CARD_PROGRAMMING)||(cardstate==SD_CARD_RECEIVING)))
{
errorstatus=SDIO_SdCardProgrammingState(&cardstate);
}
return errorstatus;
}
/*
函數(shù)功能: SDIO中斷服務(wù)函數(shù)
*/
void SDIO_IRQHandler(void)
{
SDIO_SdCardProcessIRQSrc();//處理所有SDIO相關(guān)中斷
}
/*
函數(shù)功能: SDIO中斷處理函數(shù)
函數(shù)參數(shù): 處理SDIO傳輸過(guò)程中的各種中斷事務(wù)
返回值:錯(cuò)誤代碼
*/
SDIO_SD_ERROR_INFO SDIO_SdCardProcessIRQSrc(void)
{
if(SDIO->STA&(1<<8))//接收完成中斷
{
if(StopCondition==1)
{
SDIO_SendCmd(SD_CMD_STOP_TRANSMISSION,1,0); //發(fā)送CMD12,結(jié)束傳輸
TransferError=SDIO_CmdResp1Error(SD_CMD_STOP_TRANSMISSION);
}else TransferError = SD_OK;
SDIO->ICR|=1<<8;//清除完成中斷標(biāo)記
SDIO->MASK&=~((1<<1)|(1<<3)|(1<<8)|(1<<14)|(1<<15)|(1<<4)|(1<<5)|(1<<9));//關(guān)閉相關(guān)中斷
TransferEnd = 1;
return(TransferError);
}
if(SDIO->STA&(1<<1))//數(shù)據(jù)CRC錯(cuò)誤
{
SDIO->ICR|=1<<1;//清除中斷標(biāo)記
SDIO->MASK&=~((1<<1)|(1<<3)|(1<<8)|(1<<14)|(1<<15)|(1<<4)|(1<<5)|(1<<9));//關(guān)閉相關(guān)中斷
TransferError = SD_DATA_CRC_FAIL;
return(SD_DATA_CRC_FAIL);
}
if(SDIO->STA&(1<<3))//數(shù)據(jù)超時(shí)錯(cuò)誤
{
SDIO->ICR|=1<<3;//清除中斷標(biāo)記
SDIO->MASK&=~((1<<1)|(1<<3)|(1<<8)|(1<<14)|(1<<15)|(1<<4)|(1<<5)|(1<<9));//關(guān)閉相關(guān)中斷
TransferError = SD_DATA_TIMEOUT;
return(SD_DATA_TIMEOUT);
}
if(SDIO->STA&(1<<5))//FIFO上溢錯(cuò)誤
{
SDIO->ICR|=1<<5;//清除中斷標(biāo)記
SDIO->MASK&=~((1<<1)|(1<<3)|(1<<8)|(1<<14)|(1<<15)|(1<<4)|(1<<5)|(1<<9));//關(guān)閉相關(guān)中斷
TransferError = SD_RX_OVERRUN;
return(SD_RX_OVERRUN);
}
if(SDIO->STA&(1<<4))//FIFO下溢錯(cuò)誤
{
SDIO->ICR|=1<<4;//清除中斷標(biāo)記
SDIO->MASK&=~((1<<1)|(1<<3)|(1<<8)|(1<<14)|(1<<15)|(1<<4)|(1<<5)|(1<<9));//關(guān)閉相關(guān)中斷
TransferError = SD_TX_UNDERRUN;
return(SD_TX_UNDERRUN);
}
if(SDIO->STA&(1<<9))//起始位錯(cuò)誤
{
SDIO->ICR|=1<<9;//清除中斷標(biāo)記
SDIO->MASK&=~((1<<1)|(1<<3)|(1<<8)|(1<<14)|(1<<15)|(1<<4)|(1<<5)|(1<<9));//關(guān)閉相關(guān)中斷
TransferError = SD_START_BIT_ERR;
return(SD_START_BIT_ERR);
}
return(SD_OK);
}
/*
函數(shù)功能: 檢查CMD0的執(zhí)行狀態(tài)
返回值: sd卡錯(cuò)誤碼
*/
SDIO_SD_ERROR_INFO SDIO_CmdErrorCheck(void)
{
SDIO_SD_ERROR_INFO errorstatus = SD_OK;
u32 timeout=SDIO_CMD0TIMEOUT;
while(timeout--)
{
if(SDIO->STA&(1<<7))break; //命令已發(fā)送(無(wú)需響應(yīng))
}
if(timeout==0)return SD_CMD_RSP_TIMEOUT;
SDIO->ICR=0X5FF; //清除標(biāo)記
return errorstatus;
}
/*
函數(shù)功能: 檢查R7響應(yīng)的錯(cuò)誤狀態(tài)
函數(shù)參數(shù): 返回值:sd卡錯(cuò)誤碼
*/
SDIO_SD_ERROR_INFO SDIO_CmdResp7Error(void)
{
SDIO_SD_ERROR_INFO errorstatus=SD_OK;
u32 status;
u32 timeout=SDIO_CMD0TIMEOUT;
while(timeout--)
{
status=SDIO->STA;
if(status&((1<<0)|(1<<2)|(1<<6)))break;//CRC錯(cuò)誤/命令響應(yīng)超時(shí)/已經(jīng)收到響應(yīng)(CRC校驗(yàn)成功)
}
if((timeout==0)||(status&(1<<2))) //響應(yīng)超時(shí)
{
errorstatus=SD_CMD_RSP_TIMEOUT; //當(dāng)前卡不是2.0兼容卡,或者不支持設(shè)定的電壓范圍
SDIO->ICR|=1<<2; //清除命令響應(yīng)超時(shí)標(biāo)志
return errorstatus;
}
if(status&1<<6) //成功接收到響應(yīng)
{
errorstatus=SD_OK;
SDIO->ICR|=1<<6; //清除響應(yīng)標(biāo)志
}
return errorstatus;
}
/*
函數(shù)功能:檢查R1響應(yīng)的錯(cuò)誤狀態(tài)
函數(shù)參數(shù):
cmd:當(dāng)前命令
返回值:sd卡錯(cuò)誤碼
*/
SDIO_SD_ERROR_INFO SDIO_CmdResp1Error(u8 cmd)
{
u32 status;
while(1)
{
status=SDIO->STA;
if(status&((1<<0)|(1<<2)|(1<<6)))break;//CRC錯(cuò)誤/命令響應(yīng)超時(shí)/已經(jīng)收到響應(yīng)(CRC校驗(yàn)成功)
}
if(status&(1<<2)) //響應(yīng)超時(shí)
{
SDIO->ICR=1<<2; //清除命令響應(yīng)超時(shí)標(biāo)志
return SD_CMD_RSP_TIMEOUT;
}
if(status&(1<<0)) //CRC錯(cuò)誤
{
SDIO->ICR=1<<0; //清除標(biāo)志
return SD_CMD_CRC_FAIL;
}
if(SDIO->RESPCMD!=cmd)return SD_ILLEGAL_CMD;//命令不匹配
SDIO->ICR=0X5FF; //清除標(biāo)記
return (SDIO_SD_ERROR_INFO)(SDIO->RESP1&SD_OCR_ERRORBITS);//返回卡響應(yīng)
}
/*
函數(shù)功能: 檢查R3響應(yīng)的錯(cuò)誤狀態(tài)
返回值: 錯(cuò)誤狀態(tài)
*/
SDIO_SD_ERROR_INFO SDIO_CmdResp3Error(void)
{
u32 status;
while(1)
{
status=SDIO->STA;
if(status&((1<<0)|(1<<2)|(1<<6)))break;//CRC錯(cuò)誤/命令響應(yīng)超時(shí)/已經(jīng)收到響應(yīng)(CRC校驗(yàn)成功)
}
if(status&(1<<2)) //響應(yīng)超時(shí)
{
SDIO->ICR|=1<<2; //清除命令響應(yīng)超時(shí)標(biāo)志
return SD_CMD_RSP_TIMEOUT;
}
SDIO->ICR=0X5FF; //清除標(biāo)記
return SD_OK;
}
/*
函數(shù)功能: 檢查R2響應(yīng)的錯(cuò)誤狀態(tài)
返回值:錯(cuò)誤狀態(tài)
*/
SDIO_SD_ERROR_INFO SDIO_CmdResp2Error(void)
{
SDIO_SD_ERROR_INFO errorstatus=SD_OK;
u32 status;
u32 timeout=SDIO_CMD0TIMEOUT;
while(timeout--)
{
status=SDIO->STA;
if(status&((1<<0)|(1<<2)|(1<<6)))break;//CRC錯(cuò)誤/命令響應(yīng)超時(shí)/已經(jīng)收到響應(yīng)(CRC校驗(yàn)成功)
}
if((timeout==0)||(status&(1<<2))) //響應(yīng)超時(shí)
{
errorstatus=SD_CMD_RSP_TIMEOUT;
SDIO->ICR|=1<<2; //清除命令響應(yīng)超時(shí)標(biāo)志
return errorstatus;
}
if(status&1<<0) //CRC錯(cuò)誤
{
errorstatus=SD_CMD_CRC_FAIL;
SDIO->ICR|=1<<0; //清除響應(yīng)標(biāo)志
}
SDIO->ICR=0X5FF; //清除標(biāo)記
return errorstatus;
}
/*
函數(shù)功能: 檢查R6響應(yīng)的錯(cuò)誤狀態(tài)
函數(shù)參數(shù):
cmd:之前發(fā)送的命令
prca:卡返回的RCA地址
返回值:錯(cuò)誤狀態(tài)
*/
SDIO_SD_ERROR_INFO SDIO_CmdResp6Error(u8 cmd,u16*prca)
{
SDIO_SD_ERROR_INFO errorstatus=SD_OK;
u32 status;
u32 rspr1;
while(1)
{
status=SDIO->STA;
if(status&((1<<0)|(1<<2)|(1<<6)))break;//CRC錯(cuò)誤/命令響應(yīng)超時(shí)/已經(jīng)收到響應(yīng)(CRC校驗(yàn)成功)
}
if(status&(1<<2)) //響應(yīng)超時(shí)
{
SDIO->ICR|=1<<2; //清除命令響應(yīng)超時(shí)標(biāo)志
return SD_CMD_RSP_TIMEOUT;
}
if(status&1<<0) //CRC錯(cuò)誤
{
SDIO->ICR|=1<<0; //清除響應(yīng)標(biāo)志
return SD_CMD_CRC_FAIL;
}
if(SDIO->RESPCMD!=cmd) //判斷是否響應(yīng)cmd命令
{
return SD_ILLEGAL_CMD;
}
SDIO->ICR=0X5FF; //清除所有標(biāo)記
rspr1=SDIO->RESP1; //得到響應(yīng)
if(SD_ALLZERO==(rspr1&(SD_R6_GENERAL_UNKNOWN_ERROR|SD_R6_ILLEGAL_CMD|SD_R6_COM_CRC_FAILED)))
{
*prca=(u16)(rspr1>>16); //右移16位得到,rca
return errorstatus;
}
if(rspr1&SD_R6_GENERAL_UNKNOWN_ERROR)return SD_GENERAL_UNKNOWN_ERROR;
if(rspr1&SD_R6_ILLEGAL_CMD)return SD_ILLEGAL_CMD;
if(rspr1&SD_R6_COM_CRC_FAILED)return SD_COM_CRC_FAILED;
return errorstatus;
}
/*
函數(shù)功能:SDIO使能寬總線模式
函數(shù)參數(shù):
enx:0,不使能;1,使能;
返回值:錯(cuò)誤狀態(tài)
*/
SDIO_SD_ERROR_INFO SDIO_SdCardEnWideBus(u8 enx)
{
SDIO_SD_ERROR_INFO errorstatus = SD_OK;
u32 scr[2]={0,0};
u8 arg=0X00;
if(enx)arg=0X02;
else arg=0X00;
if(SDIO->RESP1&SD_CARD_LOCKED)return SD_LOCK_UNLOCK_FAILED;//SD卡處于LOCKED狀態(tài)
errorstatus=SDIO_SdCardFindSCR(RCA,scr); //得到SCR寄存器數(shù)據(jù)
if(errorstatus!=SD_OK)return errorstatus;
if((scr[1]&SD_WIDE_BUS_SUPPORT)!=SD_ALLZERO) //支持寬總線
{
SDIO_SendCmd(SD_CMD_APP_CMD,1,(u32)RCA<<16); //發(fā)送CMD55+RCA,短響應(yīng)
errorstatus=SDIO_CmdResp1Error(SD_CMD_APP_CMD);
if(errorstatus!=SD_OK)return errorstatus;
SDIO_SendCmd(SD_CMD_APP_SD_SET_BUSWIDTH,1,arg);//發(fā)送ACMD6,短響應(yīng),參數(shù):10,4位;00,1位.
errorstatus=SDIO_CmdResp1Error(SD_CMD_APP_SD_SET_BUSWIDTH);
return errorstatus;
}else return SD_REQUEST_NOT_APPLICABLE; //不支持寬總線設(shè)置
}
/*
函數(shù)功能: 檢查卡是否正在執(zhí)行寫操作
函數(shù)參數(shù): pstatus:當(dāng)前狀態(tài)
返回值:錯(cuò)誤代碼
*/
SDIO_SD_ERROR_INFO SDIO_SdCardProgrammingState(u8 *pstatus)
{
vu32 respR1 = 0, status = 0;
SDIO_SendCmd(SD_CMD_SEND_STATUS,1,(u32)RCA<<16); //發(fā)送CMD13
status=SDIO->STA;
while(!(status&((1<<0)|(1<<6)|(1<<2))))status=SDIO->STA;//等待操作完成
if(status&(1<<0)) //CRC檢測(cè)失敗
{
SDIO->ICR|=1<<0; //清除錯(cuò)誤標(biāo)記
return SD_CMD_CRC_FAIL;
}
if(status&(1<<2)) //命令超時(shí)
{
SDIO->ICR|=1<<2; //清除錯(cuò)誤標(biāo)記
return SD_CMD_RSP_TIMEOUT;
}
if(SDIO->RESPCMD!=SD_CMD_SEND_STATUS)return SD_ILLEGAL_CMD;
SDIO->ICR=0X5FF; //清除所有標(biāo)記
respR1=SDIO->RESP1;
*pstatus=(u8)((respR1>>9)&0x0000000F);
return SD_OK;
}
/*
函數(shù)功能: 讀取當(dāng)前卡狀態(tài)
函數(shù)參數(shù):
pcardstatus:卡狀態(tài)
返回值 :錯(cuò)誤代碼
*/
SDIO_SD_ERROR_INFO SDIO_SdCardSendStatus(uint32_t *pcardstatus)
{
SDIO_SD_ERROR_INFO errorstatus = SD_OK;
if(pcardstatus==NULL)
{
errorstatus=SD_INVALID_PARAMETER;
return errorstatus;
}
SDIO_SendCmd(SD_CMD_SEND_STATUS,1,RCA<<16); //發(fā)送CMD13,短響應(yīng)
errorstatus=SDIO_CmdResp1Error(SD_CMD_SEND_STATUS); //查詢響應(yīng)狀態(tài)
if(errorstatus!=SD_OK)return errorstatus;
*pcardstatus=SDIO->RESP1;//讀取響應(yīng)值
return errorstatus;
}
/*
函數(shù)功能: 返回SD卡的狀態(tài)
返回值 : SD卡狀態(tài)
*/
SDCardState SDIO_SdCardGetState(void)
{
u32 resp1=0;
if(SDIO_SdCardSendStatus(&resp1)!=SD_OK)return SD_CARD_ERROR;
else return (SDCardState)((resp1>>9) & 0x0F);
}
/*
函數(shù)功能:查找SD卡的SCR寄存器值
函數(shù)參數(shù):
rca:卡相對(duì)地址
pscr:數(shù)據(jù)緩存區(qū)(存儲(chǔ)SCR內(nèi)容)
返回值:錯(cuò)誤狀態(tài)
*/
SDIO_SD_ERROR_INFO SDIO_SdCardFindSCR(u16 rca,u32 *pscr)
{
u32 index = 0;
SDIO_SD_ERROR_INFO errorstatus = SD_OK;
u32 tempscr[2]={0,0};
SDIO_SendCmd(SD_CMD_SET_BLOCKLEN,1,8); //發(fā)送CMD16,短響應(yīng),設(shè)置Block Size為8字節(jié)
errorstatus=SDIO_CmdResp1Error(SD_CMD_SET_BLOCKLEN);
if(errorstatus!=SD_OK)return errorstatus;
SDIO_SendCmd(SD_CMD_APP_CMD,1,(u32)rca<<16); //發(fā)送CMD55,短響應(yīng)
errorstatus=SDIO_CmdResp1Error(SD_CMD_APP_CMD);
if(errorstatus!=SD_OK)return errorstatus;
SDIO_SendDataConfig(SD_DATATIMEOUT,8,3,1); //8個(gè)字節(jié)長(zhǎng)度,block為8字節(jié),SD卡到SDIO.
SDIO_SendCmd(SD_CMD_SD_APP_SEND_SCR,1,0); //發(fā)送ACMD51,短響應(yīng),參數(shù)為0
errorstatus=SDIO_CmdResp1Error(SD_CMD_SD_APP_SEND_SCR);
if(errorstatus!=SD_OK)return errorstatus;
while(!(SDIO->STA&(SDIO_FLAG_RXOVERR|SDIO_FLAG_DCRCFAIL|SDIO_FLAG_DTIMEOUT|SDIO_FLAG_DBCKEND|SDIO_FLAG_STBITERR)))
{
if(SDIO->STA&(1<<21))//接收FIFO數(shù)據(jù)可用
{
*(tempscr+index)=SDIO->FIFO; //讀取FIFO內(nèi)容
index++;
if(index>=2)break;
}
}
if(SDIO->STA&(1<<3)) //接收數(shù)據(jù)超時(shí)
{
SDIO->ICR|=1<<3; //清除標(biāo)記
return SD_DATA_TIMEOUT;
}
else if(SDIO->STA&(1<<1)) //已發(fā)送/接收的數(shù)據(jù)塊CRC校驗(yàn)錯(cuò)誤
{
SDIO->ICR|=1<<1; //清除標(biāo)記
return SD_DATA_CRC_FAIL;
}
else if(SDIO->STA&(1<<5)) //接收FIFO溢出
{
SDIO->ICR|=1<<5; //清除標(biāo)記
return SD_RX_OVERRUN;
}
else if(SDIO->STA&(1<<9)) //起始位檢測(cè)錯(cuò)誤
{
SDIO->ICR|=1<<9; //清除標(biāo)記
return SD_START_BIT_ERR;
}
SDIO->ICR=0X5FF; //清除標(biāo)記
//把數(shù)據(jù)順序按8位為單位倒過(guò)來(lái).
*(pscr+1)=((tempscr[0]&SD_0TO7BITS)<<24)|((tempscr[0]&SD_8TO15BITS)<<8)|((tempscr[0]&SD_16TO23BITS)>>8)|((tempscr[0]&SD_24TO31BITS)>>24);
*(pscr)=((tempscr[1]&SD_0TO7BITS)<<24)|((tempscr[1]&SD_8TO15BITS)<<8)|((tempscr[1]&SD_16TO23BITS)>>8)|((tempscr[1]&SD_24TO31BITS)>>24);
return errorstatus;
}
/*
函數(shù)功能: 得到NumberOfBytes以2為底的指數(shù)
函數(shù)參數(shù): NumberOfBytes:字節(jié)數(shù)
返回值:以2為底的指數(shù)值
*/
u8 convert_from_bytes_to_power_of_two(u16 NumberOfBytes)
{
u8 count=0;
while(NumberOfBytes!=1)
{
NumberOfBytes>>=1;
count++;
}
return count;
}
/*
函數(shù)功能: 配置SDIO DMA
函數(shù)參數(shù):
mbuf:存儲(chǔ)器地址
bufsize:傳輸數(shù)據(jù)量
dir:方向;1,存儲(chǔ)器-->SDIO(寫數(shù)據(jù));0,SDIO-->存儲(chǔ)器(讀數(shù)據(jù));
*/
void SDIO_SdCard_DMAConfig(u32*mbuf,u32 bufsize,u8 dir)
{
DMA2->IFCR|=(0XF<<12); //清除DMA2通道4的各種標(biāo)記
DMA2_Channel4->CCR&=~(1<<0); //關(guān)閉DMA 通道4
DMA2_Channel4->CCR&=~(0X7FF<<4); //清除之前的設(shè)置,DIR,CIRC,PINC,MINC,PSIZE,MSIZE,PL,MEM2MEM
DMA2_Channel4->CCR|=dir<<4; //從存儲(chǔ)器讀
DMA2_Channel4->CCR|=0<<5; //普通模式
DMA2_Channel4->CCR|=0<<6; //外設(shè)地址非增量模式
DMA2_Channel4->CCR|=1<<7; //存儲(chǔ)器增量模式
DMA2_Channel4->CCR|=2<<8; //外設(shè)數(shù)據(jù)寬度為32位
DMA2_Channel4->CCR|=2<<10; //存儲(chǔ)器數(shù)據(jù)寬度32位
DMA2_Channel4->CCR|=2<<12; //高優(yōu)先級(jí)
DMA2_Channel4->CNDTR=bufsize/4; //DMA2,傳輸數(shù)據(jù)量
DMA2_Channel4->CPAR=(u32)&SDIO->FIFO;//DMA2 外設(shè)地址
DMA2_Channel4->CMAR=(u32)mbuf; //DMA2,存儲(chǔ)器地址
DMA2_Channel4->CCR|=1<<0; //開(kāi)啟DMA通道
}
/*
函數(shù)功能: 讀SD卡
函數(shù)參數(shù):
buf:讀數(shù)據(jù)緩存區(qū)
sector:扇區(qū)地址
cnt:扇區(qū)個(gè)數(shù)
返回值:錯(cuò)誤狀態(tài);0,正常;其他,錯(cuò)誤代碼;
*/
u8 SDIO_SdCardReadDiskSector(u8*buf,u32 sector,u8 cnt)
{
u8 sta=SD_OK;
long long lsector=sector;
u8 n;
lsector<<=9;
if((u32)buf%4!=0)
{
for(n=0;n<cnt;n++)
{
sta=SDIO_SdCardReadBlock(SDIO_DATA_BUFFER,lsector+512*n,512);//單個(gè)sector的讀操作
memcpy(buf,SDIO_DATA_BUFFER,512);
buf+=512;
}
}else
{
if(cnt==1)sta=SDIO_SdCardReadBlock(buf,lsector,512); //單個(gè)sector的讀操作
else sta=SDIO_SdCardReadMultiBlocks(buf,lsector,512,cnt);//多個(gè)sector
}
return sta;
}
/*
函數(shù)功能:寫SD卡
函數(shù)參數(shù):
buf:寫數(shù)據(jù)緩存區(qū)
sector:扇區(qū)地址
cnt:扇區(qū)個(gè)數(shù)
返回值:錯(cuò)誤狀態(tài);0,正常;其他,錯(cuò)誤代碼;
*/
u8 SDIO_SdCardWriteDiskSector(u8*buf,u32 sector,u8 cnt)
{
u8 sta=SD_OK;
u8 n;
long long lsector=sector;
lsector<<=9;
if((u32)buf%4!=0)
{
for(n=0;n<cnt;n++)
{
memcpy(SDIO_DATA_BUFFER,buf,512);
sta=SDIO_SdCardWriteBlock(SDIO_DATA_BUFFER,lsector+512*n,512);//單個(gè)sector的寫操作
buf+=512;
}
}else
{
if(cnt==1)sta=SDIO_SdCardWriteBlock(buf,lsector,512); //單個(gè)sector的寫操作
else sta=SDIO_SdCardWriteMultiBlocks(buf,lsector,512,cnt); //多個(gè)sector
}
return sta;
}
(3)sdio.h
#ifndef __SDIO_SDCARD_H
#define __SDIO_SDCARD_H
#include "stm32f10x.h"
//SDIO相關(guān)標(biāo)志位
#define SDIO_FLAG_CCRCFAIL ((uint32_t)0x00000001)
#define SDIO_FLAG_DCRCFAIL ((uint32_t)0x00000002)
#define SDIO_FLAG_CTIMEOUT ((uint32_t)0x00000004)
#define SDIO_FLAG_DTIMEOUT ((uint32_t)0x00000008)
#define SDIO_FLAG_TXUNDERR ((uint32_t)0x00000010)
#define SDIO_FLAG_RXOVERR ((uint32_t)0x00000020)
#define SDIO_FLAG_CMDREND ((uint32_t)0x00000040)
#define SDIO_FLAG_CMDSENT ((uint32_t)0x00000080)
#define SDIO_FLAG_DATAEND ((uint32_t)0x00000100)
#define SDIO_FLAG_STBITERR ((uint32_t)0x00000200)
#define SDIO_FLAG_DBCKEND ((uint32_t)0x00000400)
#define SDIO_FLAG_CMDACT ((uint32_t)0x00000800)
#define SDIO_FLAG_TXACT ((uint32_t)0x00001000)
#define SDIO_FLAG_RXACT ((uint32_t)0x00002000)
#define SDIO_FLAG_TXFIFOHE ((uint32_t)0x00004000)
#define SDIO_FLAG_RXFIFOHF ((uint32_t)0x00008000)
#define SDIO_FLAG_TXFIFOF ((uint32_t)0x00010000)
#define SDIO_FLAG_RXFIFOF ((uint32_t)0x00020000)
#define SDIO_FLAG_TXFIFOE ((uint32_t)0x00040000)
#define SDIO_FLAG_RXFIFOE ((uint32_t)0x00080000)
#define SDIO_FLAG_TXDAVL ((uint32_t)0x00100000)
#define SDIO_FLAG_RXDAVL ((uint32_t)0x00200000)
#define SDIO_FLAG_SDIOIT ((uint32_t)0x00400000)
#define SDIO_FLAG_CEATAEND ((uint32_t)0x00800000)
//用戶配置區(qū)
//SDIO時(shí)鐘計(jì)算公式:SDIO_CK時(shí)鐘=SDIOCLK/[clkdiv+2];其中,SDIOCLK一般為72Mhz
//使用DMA模式的時(shí)候,傳輸速率可以到24Mhz,不過(guò)如果你的卡不是高速卡,可能也會(huì)出錯(cuò)
//出錯(cuò)就請(qǐng)降低時(shí)鐘,使用查詢模式的話,推薦SDIO_TRANSFER_CLK_DIV設(shè)置為3或者更大
#define SDIO_INIT_CLK_DIV 0xB2 //SDIO初始化頻率,最大400Kh
#define SDIO_TRANSFER_CLK_DIV 0x04 //SDIO傳輸頻率,該值太小可能會(huì)導(dǎo)致讀寫文件出錯(cuò)
//SDIO工作模式定義,通過(guò)SDIO_SdCardSetDeviceMode函數(shù)設(shè)置.
#define SD_POLLING_MODE 0 //查詢模式,該模式下,如果讀寫有問(wèn)題,建議增大SDIO_TRANSFER_CLK_DIV的設(shè)置.
#define SD_DMA_MODE 1 //DMA模式,該模式下,如果讀寫有問(wèn)題,建議增大SDIO_TRANSFER_CLK_DIV的設(shè)置.
//SDIO 各種錯(cuò)誤枚舉定義
typedef enum
{
//特殊錯(cuò)誤定義
SD_CMD_CRC_FAIL = (1), /*!< Command response received (but CRC check failed) */
SD_DATA_CRC_FAIL = (2), /*!< Data bock sent/received (CRC check Failed) */
SD_CMD_RSP_TIMEOUT = (3), /*!< Command response timeout */
SD_DATA_TIMEOUT = (4), /*!< Data time out */
SD_TX_UNDERRUN = (5), /*!< Transmit FIFO under-run */
SD_RX_OVERRUN = (6), /*!< Receive FIFO over-run */
SD_START_BIT_ERR = (7), /*!< Start bit not detected on all data signals in widE bus mode */
SD_CMD_OUT_OF_RANGE = (8), /*!< CMD's argument was out of range.*/
SD_ADDR_MISALIGNED = (9), /*!< Misaligned address */
SD_BLOCK_LEN_ERR = (10), /*!< Transferred block length is not allowed for the card or the number of transferred bytes does not match the block length */
SD_ERASE_SEQ_ERR = (11), /*!< An error in the sequence of erase command occurs.*/
SD_BAD_ERASE_PARAM = (12), /*!< An Invalid selection for erase groups */
SD_WRITE_PROT_VIOLATION = (13), /*!< Attempt to program a write protect block */
SD_LOCK_UNLOCK_FAILED = (14), /*!< Sequence or password error has been detected in unlock command or if there was an attempt to access a locked card */
SD_COM_CRC_FAILED = (15), /*!< CRC check of the previous command failed */
SD_ILLEGAL_CMD = (16), /*!< Command is not legal for the card state */
SD_CARD_ECC_FAILED = (17), /*!< Card internal ECC was applied but failed to correct the data */
SD_CC_ERROR = (18), /*!< Internal card controller error */
SD_GENERAL_UNKNOWN_ERROR = (19), /*!< General or Unknown error */
SD_STREAM_READ_UNDERRUN = (20), /*!< The card could not sustain data transfer in stream read operation. */
SD_STREAM_WRITE_OVERRUN = (21), /*!< The card could not sustain data programming in stream mode */
SD_CID_CSD_OVERWRITE = (22), /*!< CID/CSD overwrite error */
SD_WP_ERASE_SKIP = (23), /*!< only partial address space was erased */
SD_CARD_ECC_DISABLED = (24), /*!< Command has been executed without using internal ECC */
SD_ERASE_RESET = (25), /*!< Erase sequence was cleared before executing because an out of erase sequence command was received */
SD_AKE_SEQ_ERROR = (26), /*!< Error in sequence of authentication. */
SD_INVALID_VOLTRANGE = (27),
SD_ADDR_OUT_OF_RANGE = (28),
SD_SWITCH_ERROR = (29),
SD_SDIO_DISABLED = (30),
SD_SDIO_FUNCTION_BUSY = (31),
SD_SDIO_FUNCTION_FAILED = (32),
SD_SDIO_UNKNOWN_FUNCTION = (33),
//標(biāo)準(zhǔn)錯(cuò)誤定義
SD_INTERNAL_ERROR,
SD_NOT_CONFIGURED,
SD_REQUEST_PENDING,
SD_REQUEST_NOT_APPLICABLE,
SD_INVALID_PARAMETER,
SD_UNSUPPORTED_FEATURE,
SD_UNSUPPORTED_HW,
SD_ERROR,
SD_OK = 0
} SDIO_SD_ERROR_INFO;
//SD卡CSD寄存器數(shù)據(jù)
typedef struct
{
u8 CSDStruct; /*!< CSD structure */
u8 SysSpecVersion; /*!< System specification version */
u8 Reserved1; /*!< Reserved */
u8 TAAC; /*!< Data read access-time 1 */
u8 NSAC; /*!< Data read access-time 2 in CLK cycles */
u8 MaxBusClkFrec; /*!< Max. bus clock frequency */
u16 CardComdClasses; /*!< Card command classes */
u8 RdBlockLen; /*!< Max. read data block length */
u8 PartBlockRead; /*!< Partial blocks for read allowed */
u8 WrBlockMisalign; /*!< Write block misalignment */
u8 RdBlockMisalign; /*!< Read block misalignment */
u8 DSRImpl; /*!< DSR implemented */
u8 Reserved2; /*!< Reserved */
u32 DeviceSize; /*!< Device Size */
u8 MaxRdCurrentVDDMin; /*!< Max. read current @ VDD min */
u8 MaxRdCurrentVDDMax; /*!< Max. read current @ VDD max */
u8 MaxWrCurrentVDDMin; /*!< Max. write current @ VDD min */
u8 MaxWrCurrentVDDMax; /*!< Max. write current @ VDD max */
u8 DeviceSizeMul; /*!< Device size multiplier */
u8 EraseGrSize; /*!< Erase group size */
u8 EraseGrMul; /*!< Erase group size multiplier */
u8 WrProtectGrSize; /*!< Write protect group size */
u8 WrProtectGrEnable; /*!< Write protect group enable */
u8 ManDeflECC; /*!< Manufacturer default ECC */
u8 WrSpeedFact; /*!< Write speed factor */
u8 MaxWrBlockLen; /*!< Max. write data block length */
u8 WriteBlockPaPartial; /*!< Partial blocks for write allowed */
u8 Reserved3; /*!< Reserded */
u8 ContentProtectAppli; /*!< Content protection application */
u8 FileFormatGrouop; /*!< File format group */
u8 CopyFlag; /*!< Copy flag (OTP) */
u8 PermWrProtect; /*!< Permanent write protection */
u8 TempWrProtect; /*!< Temporary write protection */
u8 FileFormat; /*!< File Format */
u8 ECC; /*!< ECC code */
u8 CSD_CRC; /*!< CSD CRC */
u8 Reserved4; /*!< always 1*/
} SD_CSD;
//SD卡CID寄存器數(shù)據(jù)
typedef struct
{
u8 ManufacturerID; /*!< ManufacturerID */
u16 OEM_AppliID; /*!< OEM/Application ID */
u32 ProdName1; /*!< Product Name part1 */
u8 ProdName2; /*!< Product Name part2*/
u8 ProdRev; /*!< Product Revision */
u32 ProdSN; /*!< Product Serial Number */
u8 Reserved1; /*!< Reserved1 */
u16 ManufactDate; /*!< Manufacturing Date */
u8 CID_CRC; /*!< CID CRC */
u8 Reserved2; /*!< always 1 */
} SD_CID;
//SD卡狀態(tài)
typedef enum
{
SD_CARD_READY = ((uint32_t)0x00000001),
SD_CARD_IDENTIFICATION = ((uint32_t)0x00000002),
SD_CARD_STANDBY = ((uint32_t)0x00000003),
SD_CARD_TRANSFER = ((uint32_t)0x00000004),
SD_CARD_SENDING = ((uint32_t)0x00000005),
SD_CARD_RECEIVING = ((uint32_t)0x00000006),
SD_CARD_PROGRAMMING = ((uint32_t)0x00000007),
SD_CARD_DISCONNECTED = ((uint32_t)0x00000008),
SD_CARD_ERROR = ((uint32_t)0x000000FF)
}SDCardState;
//SD卡信息,包括CSD,CID等數(shù)據(jù)
typedef struct
{
SD_CSD SD_csd;
SD_CID SD_cid;
long long CardCapacity; //SD卡容量,單位:字節(jié),最大支持2^64字節(jié)大小的卡.
u32 CardBlockSize; //SD卡塊大小
u16 RCA; //卡相對(duì)地址
u8 CardType; //卡類型
} SD_CardInfo;
extern SD_CardInfo SDCardInfo;//SD卡信息
//SDIO 指令集
#define SD_CMD_GO_IDLE_STATE ((u8)0)
#define SD_CMD_SEND_OP_COND ((u8)1)
#define SD_CMD_ALL_SEND_CID ((u8)2)
#define SD_CMD_SET_REL_ADDR ((u8)3) /*!< SDIO_SEND_REL_ADDR for SD Card */
#define SD_CMD_SET_DSR ((u8)4)
#define SD_CMD_SDIO_SEN_OP_COND ((u8)5)
#define SD_CMD_HS_SWITCH ((u8)6)
#define SD_CMD_SEL_DESEL_CARD ((u8)7)
#define SD_CMD_HS_SEND_EXT_CSD ((u8)8)
#define SD_CMD_SEND_CSD ((u8)9)
#define SD_CMD_SEND_CID ((u8)10)
#define SD_CMD_READ_DAT_UNTIL_STOP ((u8)11) /*!< SD Card doesn't support it */
#define SD_CMD_STOP_TRANSMISSION ((u8)12)
#define SD_CMD_SEND_STATUS ((u8)13)
#define SD_CMD_HS_BUSTEST_READ ((u8)14)
#define SD_CMD_GO_INACTIVE_STATE ((u8)15)
#define SD_CMD_SET_BLOCKLEN ((u8)16)
#define SD_CMD_READ_SINGLE_BLOCK ((u8)17)
#define SD_CMD_READ_MULT_BLOCK ((u8)18)
#define SD_CMD_HS_BUSTEST_WRITE ((u8)19)
#define SD_CMD_WRITE_DAT_UNTIL_STOP ((u8)20)
#define SD_CMD_SET_BLOCK_COUNT ((u8)23)
#define SD_CMD_WRITE_SINGLE_BLOCK ((u8)24)
#define SD_CMD_WRITE_MULT_BLOCK ((u8)25)
#define SD_CMD_PROG_CID ((u8)26)
#define SD_CMD_PROG_CSD ((u8)27)
#define SD_CMD_SET_WRITE_PROT ((u8)28)
#define SD_CMD_CLR_WRITE_PROT ((u8)29)
#define SD_CMD_SEND_WRITE_PROT ((u8)30)
#define SD_CMD_SD_ERASE_GRP_START ((u8)32) /*!< To set the address of the first write
block to be erased. (For SD card only) */
#define SD_CMD_SD_ERASE_GRP_END ((u8)33) /*!< To set the address of the last write block of the
continuous range to be erased. (For SD card only) */
#define SD_CMD_ERASE_GRP_START ((u8)35) /*!< To set the address of the first write block to be erased.
(For MMC card only spec 3.31) */
#define SD_CMD_ERASE_GRP_END ((u8)36) /*!< To set the address of the last write block of the
continuous range to be erased. (For MMC card only spec 3.31) */
#define SD_CMD_ERASE ((u8)38)
#define SD_CMD_FAST_IO ((u8)39) /*!< SD Card doesn't support it */
#define SD_CMD_GO_IRQ_STATE ((u8)40) /*!< SD Card doesn't support it */
#define SD_CMD_LOCK_UNLOCK ((u8)42)
#define SD_CMD_APP_CMD ((u8)55)
#define SD_CMD_GEN_CMD ((u8)56)
#define SD_CMD_NO_CMD ((u8)64)
/**
* @brief Following commands are SD Card Specific commands.
* SDIO_APP_CMD :CMD55 should be sent before sending these commands.
*/
#define SD_CMD_APP_SD_SET_BUSWIDTH ((u8)6) /*!< For SD Card only */
#define SD_CMD_SD_APP_STAUS ((u8)13) /*!< For SD Card only */
#define SD_CMD_SD_APP_SEND_NUM_WRITE_BLOCKS ((u8)22) /*!< For SD Card only */
#define SD_CMD_SD_APP_OP_COND ((u8)41) /*!< For SD Card only */
#define SD_CMD_SD_APP_SET_CLR_CARD_DETECT ((u8)42) /*!< For SD Card only */
#define SD_CMD_SD_APP_SEND_SCR ((u8)51) /*!< For SD Card only */
#define SD_CMD_SDIO_RW_DIRECT ((u8)52) /*!< For SD I/O Card only */
#define SD_CMD_SDIO_RW_EXTENDED ((u8)53) /*!< For SD I/O Card only */
/**
* @brief Following commands are SD Card Specific security commands.
* SDIO_APP_CMD should be sent before sending these commands.
*/
#define SD_CMD_SD_APP_GET_MKB ((u8)43) /*!< For SD Card only */
#define SD_CMD_SD_APP_GET_MID ((u8)44) /*!< For SD Card only */
#define SD_CMD_SD_APP_SET_CER_RN1 ((u8)45) /*!< For SD Card only */
#define SD_CMD_SD_APP_GET_CER_RN2 ((u8)46) /*!< For SD Card only */
#define SD_CMD_SD_APP_SET_CER_RES2 ((u8)47) /*!< For SD Card only */
#define SD_CMD_SD_APP_GET_CER_RES1 ((u8)48) /*!< For SD Card only */
#define SD_CMD_SD_APP_SECURE_READ_MULTIPLE_BLOCK ((u8)18) /*!< For SD Card only */
#define SD_CMD_SD_APP_SECURE_WRITE_MULTIPLE_BLOCK ((u8)25) /*!< For SD Card only */
#define SD_CMD_SD_APP_SECURE_ERASE ((u8)38) /*!< For SD Card only */
#define SD_CMD_SD_APP_CHANGE_SECURE_AREA ((u8)49) /*!< For SD Card only */
#define SD_CMD_SD_APP_SECURE_WRITE_MKB ((u8)48) /*!< For SD Card only */
//支持的SD卡定義
#define SDIO_STD_CAPACITY_SD_CARD_V1_1 ((u32)0x00000000)
#define SDIO_STD_CAPACITY_SD_CARD_V2_0 ((u32)0x00000001)
#define SDIO_HIGH_CAPACITY_SD_CARD ((u32)0x00000002)
#define SDIO_MULTIMEDIA_CARD ((u32)0x00000003)
#define SDIO_SECURE_DIGITAL_IO_CARD ((u32)0x00000004)
#define SDIO_HIGH_SPEED_MULTIMEDIA_CARD ((u32)0x00000005)
#define SDIO_SECURE_DIGITAL_IO_COMBO_CARD ((u32)0x00000006)
#define SDIO_HIGH_CAPACITY_MMC_CARD ((u32)0x00000007)
//SDIO相關(guān)參數(shù)定義
#define NULL 0
#define SDIO_STATIC_FLAGS ((u32)0x000005FF)
#define SDIO_CMD0TIMEOUT ((u32)0x00010000)
#define SDIO_DATATIMEOUT ((u32)0xFFFFFFFF)
#define SDIO_FIFO_Address ((u32)0x40018080)
//Mask for errors Card Status R1 (OCR Register)
#define SD_OCR_ADDR_OUT_OF_RANGE ((u32)0x80000000)
#define SD_OCR_ADDR_MISALIGNED ((u32)0x40000000)
#define SD_OCR_BLOCK_LEN_ERR ((u32)0x20000000)
#define SD_OCR_ERASE_SEQ_ERR ((u32)0x10000000)
#define SD_OCR_BAD_ERASE_PARAM ((u32)0x08000000)
#define SD_OCR_WRITE_PROT_VIOLATION ((u32)0x04000000)
#define SD_OCR_LOCK_UNLOCK_FAILED ((u32)0x01000000)
#define SD_OCR_COM_CRC_FAILED ((u32)0x00800000)
#define SD_OCR_ILLEGAL_CMD ((u32)0x00400000)
#define SD_OCR_CARD_ECC_FAILED ((u32)0x00200000)
#define SD_OCR_CC_ERROR ((u32)0x00100000)
#define SD_OCR_GENERAL_UNKNOWN_ERROR ((u32)0x00080000)
#define SD_OCR_STREAM_READ_UNDERRUN ((u32)0x00040000)
#define SD_OCR_STREAM_WRITE_OVERRUN ((u32)0x00020000)
#define SD_OCR_CID_CSD_OVERWRIETE ((u32)0x00010000)
#define SD_OCR_WP_ERASE_SKIP ((u32)0x00008000)
#define SD_OCR_CARD_ECC_DISABLED ((u32)0x00004000)
#define SD_OCR_ERASE_RESET ((u32)0x00002000)
#define SD_OCR_AKE_SEQ_ERROR ((u32)0x00000008)
#define SD_OCR_ERRORBITS ((u32)0xFDFFE008)
//Masks for R6 Response
#define SD_R6_GENERAL_UNKNOWN_ERROR ((u32)0x00002000)
#define SD_R6_ILLEGAL_CMD ((u32)0x00004000)
#define SD_R6_COM_CRC_FAILED ((u32)0x00008000)
#define SD_VOLTAGE_WINDOW_SD ((u32)0x80100000)
#define SD_HIGH_CAPACITY ((u32)0x40000000)
#define SD_STD_CAPACITY ((u32)0x00000000)
#define SD_CHECK_PATTERN ((u32)0x000001AA)
#define SD_VOLTAGE_WINDOW_MMC ((u32)0x80FF8000)
#define SD_MAX_VOLT_TRIAL ((u32)0x0000FFFF)
#define SD_ALLZERO ((u32)0x00000000)
#define SD_WIDE_BUS_SUPPORT ((u32)0x00040000)
#define SD_SINGLE_BUS_SUPPORT ((u32)0x00010000)
#define SD_CARD_LOCKED ((u32)0x02000000)
#define SD_CARD_PROGRAMMING ((u32)0x00000007)
#define SD_CARD_RECEIVING ((u32)0x00000006)
#define SD_DATATIMEOUT ((u32)0xFFFFFFFF)
#define SD_0TO7BITS ((u32)0x000000FF)
#define SD_8TO15BITS ((u32)0x0000FF00)
#define SD_16TO23BITS ((u32)0x00FF0000)
#define SD_24TO31BITS ((u32)0xFF000000)
#define SD_MAX_DATA_LENGTH ((u32)0x01FFFFFF)
#define SD_HALFFIFO ((u32)0x00000008)
#define SD_HALFFIFOBYTES ((u32)0x00000020)
//Command Class Supported
#define SD_CCCC_LOCK_UNLOCK ((u32)0x00000080)
#define SD_CCCC_WRITE_PROT ((u32)0x00000040)
#define SD_CCCC_ERASE ((u32)0x00000020)
//CMD8指令
#define SDIO_SEND_IF_COND ((u32)0x00000008)
//相關(guān)函數(shù)定義
SDIO_SD_ERROR_INFO SDIO_SdCardInit(void);
void SDIO_ClockSet(u8 clkdiv);
void SDIO_SendCmd(u8 cmdindex,u8 waitrsp,u32 arg);
void SDIO_SendDataConfig(u32 datatimeout,u32 datalen,u8 blksize,u8 dir);
SDIO_SD_ERROR_INFO SDIO_SdPowerON(void);
SDIO_SD_ERROR_INFO SD_PowerOFF(void);
SDIO_SD_ERROR_INFO SDIO_SdCardInitializeCards(void);
SDIO_SD_ERROR_INFO SDIO_SdCardGetInfo(SD_CardInfo *cardinfo);
SDIO_SD_ERROR_INFO SDIO_SdCardEnableWideBusOperation(u32 wmode);
SDIO_SD_ERROR_INFO SDIO_SdCardSetDeviceMode(u32 mode);
SDIO_SD_ERROR_INFO SDIO_SdCardSelectAddr(u32 addr);
SDIO_SD_ERROR_INFO SDIO_SdCardSendStatus(uint32_t *pcardstatus);
SDCardState SDIO_SdCardGetState(void);
SDIO_SD_ERROR_INFO SDIO_SdCardReadBlock(u8 *buf,long long addr,u16 blksize);
SDIO_SD_ERROR_INFO SDIO_SdCardReadMultiBlocks(u8 *buf,long long addr,u16 blksize,u32 nblks);
SDIO_SD_ERROR_INFO SDIO_SdCardWriteBlock(u8 *buf,long long addr, u16 blksize);
SDIO_SD_ERROR_INFO SDIO_SdCardWriteMultiBlocks(u8 *buf,long long addr,u16 blksize,u32 nblks);
SDIO_SD_ERROR_INFO SDIO_SdCardProcessIRQSrc(void);
SDIO_SD_ERROR_INFO SDIO_CmdErrorCheck(void);
SDIO_SD_ERROR_INFO SDIO_CmdResp7Error(void);
SDIO_SD_ERROR_INFO SDIO_CmdResp1Error(u8 cmd);
SDIO_SD_ERROR_INFO SDIO_CmdResp3Error(void);
SDIO_SD_ERROR_INFO SDIO_CmdResp2Error(void);
SDIO_SD_ERROR_INFO SDIO_CmdResp6Error(u8 cmd,u16*prca);
SDIO_SD_ERROR_INFO SDIO_SdCardEnWideBus(u8 enx);
SDIO_SD_ERROR_INFO SDIO_SdCardProgrammingState(u8 *pstatus);
SDIO_SD_ERROR_INFO SDIO_SdCardFindSCR(u16 rca,u32 *pscr);
u8 convert_from_bytes_to_power_of_two(u16 NumberOfBytes);
void SDIO_SdCard_DMAConfig(u32*mbuf,u32 bufsize,u8 dir);
u8 SDIO_SdCardReadDiskSector(u8*buf,u32 sector,u8 cnt); //讀SD卡,fatfs/usb調(diào)用
u8 SDIO_SdCardWriteDiskSector(u8*buf,u32 sector,u8 cnt); //寫SD卡,fatfs/usb調(diào)用
#endif
五、總結(jié)
綜上所述,雷龍公司推出的CS SD NAND貼片式TF卡以其緊湊的設(shè)計(jì)、卓越的性能和廣泛的應(yīng)用潛力,為現(xiàn)代數(shù)字存儲(chǔ)解決方案提供了一種全新的選擇。通過(guò)深入的技術(shù)分析可以看出,該產(chǎn)品不僅在尺寸上遠(yuǎn)小于傳統(tǒng)的TF卡,而且在穩(wěn)定性、讀寫速度以及環(huán)境耐受性方面表現(xiàn)優(yōu)異。其采用的pSLC技術(shù)確保了數(shù)據(jù)的高度可靠性和較長(zhǎng)的使用壽命,而標(biāo)準(zhǔn)SDIO接口和SPI通信模式的支持,則進(jìn)一步增強(qiáng)了其兼容性和靈活性。實(shí)際測(cè)試表明,這款CS SD NAND貼片式TF卡無(wú)論是在作為大容量存儲(chǔ)介質(zhì)使用,還是應(yīng)用于如MP3播放器等多媒體設(shè)備中,都能穩(wěn)定高效地工作。它所提供的不僅僅是硬件上的升級(jí),更是對(duì)傳統(tǒng)存儲(chǔ)方式的一種革新。對(duì)于尋求高性能、高可靠性存儲(chǔ)方案的工程師和技術(shù)愛(ài)好者而言,雷龍CS SD NAND無(wú)疑是一個(gè)值得考慮的選擇。希望本文的技術(shù)探討能夠?yàn)榇蠹姨峁┯袃r(jià)值的參考,并激發(fā)更多關(guān)于未來(lái)存儲(chǔ)技術(shù)發(fā)展的思考。