找到了一個現(xiàn)成的例子,對于讀寫U盤,文件系統(tǒng)是個大麻煩?,F(xiàn)在一直還沒有弄明白,先收藏一下這個程序先,等到后面慢慢研究。
作者這個編程規(guī)范不錯,注釋很豐富,值得學習。
#include
#include"CH375INC.H"
#include/*以下定義適用于MCS-51單片機*/
#defineUINT8unsignedchar
#defineUINT16unsignedshort
#defineUINT32unsignedlong
#defineUINT8Xunsignedcharxdata
#defineUINT8VXunsignedcharvolatilexdata
UINT8VXCH375_CMD_PORT_at_0xBDF1;/*CH375命令端口的I/O地址*/
UINT8VXCH375_DAT_PORT_at_0xBCF0;/*CH375數(shù)據(jù)端口的I/O地址*/
#defineCH375_INT_WIREINT0/*P3.2,連接CH375的INT#引腳,用于查詢中
斷狀態(tài)*/
UINT8XDISK_BUFFER[512*32]_at_0x0000;/*外部RAM數(shù)據(jù)緩沖區(qū)的起始地址*/
UINT32DiskStart;/*邏輯盤的起始絕對扇區(qū)號LBA*/
UINT8SecPerClus;/*邏輯盤的每簇扇區(qū)數(shù)*/
UINT8RsvdSecCnt;/*邏輯盤的保留扇區(qū)數(shù)*/
UINT16;FATSz16;/*FAT16邏輯盤的FAT表占用的扇區(qū)數(shù)*/
/***********硬件USB接口層,無論如何這層省不掉,單片機總要與CH375接口吧*/
voidmDelaymS(UINT8delay){
UINT8i,j,c;
for(i=delay;i!=0;i--){
for(j=200;j!=0;j--)c+=3;
for(j=200;j!=0;j--)c+=3;
}
}
voidCH375_WR_CMD_PORT(UINT8cmd){/*向CH375的命令端口寫入命令*/
CH375_CMD_PORT=cmd;
for(cmd=2;cmd!=0;cmd--);/*發(fā)出命令碼前后應該各延時2uS*/
}
voidCH375_WR_DAT_PORT(UINT8dat){/*向CH375的數(shù)據(jù)端口寫入數(shù)據(jù)*/
CH375_DAT_PORT=dat;/*因為MCS51單片機較慢所以實際上無需延時*/
}
UINT8CH375_RD_DAT_PORT(void){/*從CH375的數(shù)據(jù)端口讀出數(shù)據(jù)*/
return(CH375_DAT_PORT);/*因為MCS51單片機較慢所以實際上無需延時*/
}
UINT8mWaitInterrupt(void){/*等待CH375中斷并獲取狀態(tài),返回操作狀態(tài)*/
while(CH375_INT_WIRE);/*查詢等待CH375操作完成中斷(INT#低電平)*/
CH375_WR_CMD_PORT(CMD_GET_STATUS);/*產(chǎn)生操作完成中斷,獲取中斷狀態(tài)*/
return(CH375_RD_DAT_PORT());
}
/***********BulkOnly傳輸協(xié)議層,被CH375內(nèi)置了,無需編寫單片機程序*/
/***********RBC/SCSI命令層,雖然被CH375內(nèi)置了,但是要寫程序發(fā)出命令及收發(fā)數(shù)據(jù)
*/
UINT8mInitDisk(void){/*初始化磁盤*/
UINT8Status;
CH375_WR_CMD_PORT(CMD_GET_STATUS);/*產(chǎn)生操作完成中斷,獲取中斷狀態(tài)*/
Status=CH375_RD_DAT_PORT();
if(Status==USB_INT_DISCONNECT)return(Status);/*USB設備斷開*/
CH375_WR_CMD_PORT(CMD_DISK_INIT);/*初始化USB存儲器*/
Status=mWaitInterrupt();/*等待中斷并獲取狀態(tài)*/
if(Status!=USB_INT_SUCCESS)return(Status);/*出現(xiàn)錯誤*/
CH375_WR_CMD_PORT(CMD_DISK_SIZE);/*獲取USB存儲器的容量*/
Status=mWaitInterrupt();/*等待中斷并獲取狀態(tài)*/
if(Status!=USB_INT_SUCCESS){/*出錯重試*/
/*對于CH375A芯片,建議在此執(zhí)行一次CMD_DISK_R_SENSE命令*/
mDelaymS(250);
CH375_WR_CMD_PORT(CMD_DISK_SIZE);/*獲取USB存儲器的容量*/
Status=mWaitInterrupt();/*等待中斷并獲取狀態(tài)*/
}
if(Status!=USB_INT_SUCCESS)return(Status);/*出現(xiàn)錯誤*/
return(0);/*U盤已經(jīng)成功初始化*/
}
UINT8mReadSector(UINT32iLbaStart,UINT8iSectorCount,UINT8X*oDataBuffer)
{
UINT16mBlockCount;
UINT8c;
CH375_WR_CMD_PORT(CMD_DISK_READ);/*從USB存儲器讀數(shù)據(jù)塊*/
CH375_WR_DAT_PORT((UINT8)iLbaStart);/*LBA的最低8位*/
CH375_WR_DAT_PORT((UINT8)(iLbaStart>>8));
CH375_WR_DAT_PORT((UINT8)(iLbaStart>>16));
CH375_WR_DAT_PORT((UINT8)(iLbaStart>>24));/*LBA的最高8位*/
CH375_WR_DAT_PORT(iSectorCount);/*扇區(qū)數(shù)*/
for(mBlockCount=iSectorCount*8;mBlockCount!=0;mBlockCount--){
c=mWaitInterrupt();/*等待中斷并獲取狀態(tài)*/
if(c==USB_INT_DISK_READ){/*等待中斷并獲取狀態(tài),請求數(shù)據(jù)讀出*/
CH375_WR_CMD_PORT(CMD_RD_USB_DATA);/*從CH375緩沖區(qū)讀取數(shù)據(jù)塊*/
c=CH375_RD_DAT_PORT();/*后續(xù)數(shù)據(jù)的長度*/
while(c--)*oDataBuffer++=CH375_RD_DAT_PORT();
CH375_WR_CMD_PORT(CMD_DISK_RD_GO);/*繼續(xù)執(zhí)行USB存儲器的讀操作*/
}
elsebreak;/*返回錯誤狀態(tài)*/
}
if(mBlockCount==0){
c=mWaitInterrupt();/*等待中斷并獲取狀態(tài)*/
if(c==USB_INT_SUCCESS)return(0);/*操作成功*/
}
return(c);/*操作失敗*/
}
/***********FAT文件系統(tǒng)層,這層程序量實際較大,不過,該程序僅演示極簡單的功能,所
以精簡*/
UINT16mGetPointWord(UINT8X*iAddr){/*獲取字數(shù)據(jù),因為MCS51是大端格式*/
return(iAddr[0]|(UINT16)iAddr[1]<<8);
}
UINT8mIdenDisk(void){/*識別分析當前邏輯盤*/
UINT8Status;
DiskStart=0;/*以下是非常簡單的FAT文件系統(tǒng)的分析,正式應用絕對不應該如此簡
單*/
Status=mReadSector(0,1,DISK_BUFFER);/*讀取邏輯盤引導信息*/
if(Status!=0)return(Status);
if(DISK_BUFFER[0]!=0xEB&&DISK_BUFFER[0]!=0xE9){/*不是邏輯引導扇
區(qū)*/
DiskStart=DISK_BUFFER[0x1C6]|(UINT16)DISK_BUFFER[0x1C7]<<8
|(UINT32)DISK_BUFFER[0x1C8]<<16|(UINT32)DISK_BUFFER[0x1C9]<<24;
Status=mReadSector(DiskStart,1,DISK_BUFFER);
if(Status!=0)return(Status);
}
SecPerClus=DISK_BUFFER[0x0D];/*每簇扇區(qū)數(shù)*/
RsvdSecCnt=DISK_BUFFER[0x0E];/*邏輯盤的保留扇區(qū)數(shù)*/
FATSz16=mGetPointWord(&DISK_BUFFER[0x16]);/*FAT表占用扇區(qū)數(shù)*/
return(0);/*成功*/
}
UINT16mLinkCluster(UINT16iCluster){/*獲得指定簇號的鏈接簇*/
/*輸入:iCluster當前簇號,返回:原鏈接簇號,如果為0則說明錯誤*/
UINT8Status;
Status=mReadSector(DiskStart+RsvdSecCnt+iCluster/256,1,
DISK_BUFFER);
if(Status!=0)return(0);/*錯誤*/
return(mGetPointWord(&DISK_BUFFER[(iCluster+iCluster)&0x01FF]));
}
UINT32mClusterToLba(UINT16iCluster){/*將簇號轉換為絕對LBA扇區(qū)地址*/
return(DiskStart+RsvdSecCnt+FATSz16*2+32+(iCluster-2)*
SecPerClus);
}
voidmInitSTDIO(void){/*僅用于調(diào)試用途及顯示內(nèi)容到PC機,與該程序功能完全無
關*/
SCON=0x50;PCON=0x80;TMOD=0x20;TH1=0xf3;TR1=1;TI=1;/*24MHz,
9600bps*/
}
voidmStopIfError(UINT8iErrCode){/*如果錯誤則停止運行并顯示錯誤狀態(tài)*/
if(iErrCode==0)return;
printf("Errorstatus,%02X
",(UINT16)iErrCode);
}
main(){
UINT8Status;
UINT8X*CurrentDir;
UINT16Cluster;
mDelaymS(200);/*延時200毫秒*/
mInitSTDIO();
CH375_WR_CMD_PORT(CMD_SET_USB_MODE);/*初始化CH375,設置USB工作模式*/
CH375_WR_DAT_PORT(6);/*模式代碼,自動檢測USB設備連接*/
while(1){
printf("InsertUSBdisk
");
while(mWaitInterrupt()!=USB_INT_CONNECT);/*等待U盤連接*/
mDelaymS(250);/*延時等待U盤進入正常工作狀態(tài)*/
Status=mInitDisk();/*初始化U盤,實際是識別U盤的類型,必須進行此步驟*/
mStopIfError(Status);
Status=mIdenDisk();/*識別分析U盤文件系統(tǒng),必要操作*/
mStopIfError(Status);
Status=mReadSector(DiskStart+RsvdSecCnt+FATSz16*2,32,
DISK_BUFFER);
mStopIfError(Status);/*讀取FAT16邏輯盤的根目錄,通常根目錄占用32個扇區(qū)
*/
for(CurrentDir=DISK_BUFFER;CurrentDir[0]!=0;CurrentDir+=32){
if((CurrentDir[0x0B]&0x08)==0&&CurrentDir[0]!=0xE5){
CurrentDir[0x0B]=0;/*為了便于顯示,設置文件名或者目錄名的結束標志*/
printf("Name:%s
",CurrentDir);/*通過串口輸出顯示*/
}
}/*以上顯示根目錄下的所有文件名,以下打開第一個文件,如果是C文件的話*/
if((DISK_BUFFER[0x0B]&0x08)==0&&DISK_BUFFER[0]!=0xE5&&DISK_BUFFER[8]
=='C'){
Cluster=mGetPointWord(&DISK_BUFFER[0x1A]);/*文件的首簇*/
while(Cluster<0xFFF8){/*文件簇未結束*/
if(Cluster==0)mStopIfError(0x8F);/*對于首簇,可能是0長度文件
*/
Status=mReadSector(mClusterToLba(Cluster),SecPerClus,
DISK_BUFFER);
mStopIfError(Status);/*讀取首簇到緩沖區(qū)*/
DISK_BUFFER[30]=0;printf("Data:%s
",DISK_BUFFER);/*顯示首行
*/
Cluster=mLinkCluster(Cluster);/*獲取鏈接簇,返回0說明錯誤*/
}
}
while(mWaitInterrupt()!=USB_INT_DISCONNECT);/*等待U盤拔出*/
mDelaymS(250);
}
}