?? exam11.c
字號:
/* 2004.06.05
****************************************
** Copyright (C) W.ch 1999-2005 **
** Web: http://www.winchiphead.com **
****************************************
** USB Host File Interface for CH375 **
** TC2.0@PC, KC7.0@MCS51 **
****************************************
*/
/* CH375 主機文件系統接口 */
/* 支持: FAT12/FAT16 */
/* MCS-51單片機C語言的U盤文件讀寫示例程序, 適用于89C52或者更大程序空間的單片機 */
/* 本程序用于演示創建/打開/刪除長文件名文件 */
/* CH375的INT#引腳采用查詢方式處理, 數據復制方式為"單DPTR復制", 所以速度較慢, 適用于所有MCS51單片機
本例適用于V2.8及以上版本的CH375子程序庫 */
/* 僅供有FAT文件系統和長文件名相關技術知識基礎的用戶參考,支持小寫字母或者漢字等不超過256個字符的文件名,
對該類知識不了解的用戶建議不要涉及長文件名,建議只使用現成的短文件名,短文件名支持8+3格式,大寫字母或者漢字 */
/* C51 CH375HFT.C */
/* LX51 CH375HFT.OBJ , CH375HF6.LIB */
/* OHX51 CH375HFT */
#include <reg52.h>
#include <stdio.h>
#include <string.h>
/* 以下定義的詳細說明請看CH375HF6.H文件 */
#define LIB_CFG_DISK_IO 1 /* 磁盤讀寫的數據的復制方式,1為"單DPTR復制",2為"雙DPTR復制",3為"單DPTR和P2+R0復制" */
#define LIB_CFG_FILE_IO 1 /* 文件讀寫的數據的復制方式,0為"外部子程序",1為"單DPTR復制",2為"雙DPTR復制",3為"單DPTR和P2+R0復制" */
#define LIB_CFG_INT_EN 0 /* CH375的INT#引腳連接方式,0為"查詢方式",1為"中斷方式" */
/*#define LIB_CFG_FILE_IO_DEFAULT 1*/ /* 使用CH375HF6.H提供的默認"外部子程序" */
/*#define LIB_CFG_UPD_SIZE 1*/ /* 在添加數據后是否自動更新文件長度: 0為"不更新",1為"自動更新" */
/* 默認情況下,如果扇區數/字節數不為0那么CH375FileWrite/CH375ByteWrite只負責寫入數據而不修改文件長度,
如果需要每次寫完數據后會自動修改/更新文件長度,那么可以使全局變量CH375LibConfig的位4為1,
如果長時間不寫入數據則應該更新文件長度,防止突然斷電后前面寫入的數據與文件長度不相符,
如果確定不會突然斷電或者后面很快有數據不斷寫入則不必更新文件長度,可以提高速度并減少U盤損耗(U盤內部的內存壽命有限,不宜頻繁改寫) */
#define CH375_CMD_PORT_ADDR 0xBDF1 /* CH375命令端口的I/O地址 */
#define CH375_DAT_PORT_ADDR 0xBCF0 /* CH375數據端口的I/O地址 */
/* 62256提供的32KB的RAM分為兩部分: 0000H-01FFH為磁盤讀寫緩沖區, 0200H-7FFFH為文件數據緩沖區 */
#define DISK_BASE_BUF_ADDR 0x0000 /* 外部RAM的磁盤數據緩沖區的起始地址,從該單元開始的緩沖區長度為SECTOR_SIZE */
#define FILE_DATA_BUF_ADDR 0x0200 /* 外部RAM的文件數據緩沖區的起始地址,緩沖區長度不小于一次讀寫的數據長度 */
/* 由于演示板用的62256只有32K字節,其中CH375子程序用512字節,所以外部RAM剩余長度為32256字節 */
#define FILE_DATA_BUF_LEN 0x200 /* 外部RAM的文件數據緩沖區,緩沖區長度不小于一次讀寫的數據長度,本例要求不小于0x400即可 */
#define FILE_DATA_BUF_ADDR1 0x400
unsigned char xdata FileDataBuf1[FILE_DATA_BUF_LEN] _at_ FILE_DATA_BUF_ADDR1 ;
#define CH375_INT_WIRE INT0 /* P3.2, INT0, CH375的中斷線INT#引腳,連接CH375的INT#引腳,用于查詢中斷狀態 */
#define EN_CH375LIB_MORE 1
//#define EN_DISK_FAT32 1 /* 使用CH375HF6.LIB及CH375HF4.H時請啟用該常量定義 */
#include "..\CH375HF6.H"
#include "FILELONG.H"
/* 在P1.4連接一個LED用于監控演示程序的進度,低電平LED亮,當U盤插入后亮 */
sbit P1_4 = P1^4;
#define LED_OUT_INIT( ) { P1_4 = 1; } /* P1.4 高電平 */
#define LED_OUT_ACT( ) { P1_4 = 0; } /* P1.4 低電平驅動LED顯示 */
#define LED_OUT_INACT( ) { P1_4 = 1; } /* P1.4 低電平驅動LED顯示 */
sbit P1_5 = P1^5;
/* 在P1.5連接一個LED用于監控演示程序的進度,低電平LED亮,當對U盤操作時亮 */
#define LED_RUN_ACT( ) { P1_5 = 0; } /* P1.5 低電平驅動LED顯示 */
#define LED_RUN_INACT( ) { P1_5 = 1; } /* P1.5 低電平驅動LED顯示 */
sbit P1_6 = P1^6;
/* 在P1.6連接一個LED用于監控演示程序的進度,低電平LED亮,當對U盤寫操作時亮 */
#define LED_WR_ACT( ) { P1_6 = 0; } /* P1.6 低電平驅動LED顯示 */
#define LED_WR_INACT( ) { P1_6 = 1; } /* P1.6 低電平驅動LED顯示 */
/* 以毫秒為單位延時,不精確,適用于24MHz時鐘 */
void mDelaymS( unsigned char delay )
{
unsigned char i, j, c;
for ( i = delay; i != 0; i -- ) {
for ( j = 200; j != 0; j -- ) c += 3; /* 在24MHz時鐘下延時500uS */
for ( j = 200; j != 0; j -- ) c += 3; /* 在24MHz時鐘下延時500uS */
}
}
/* 將程序空間的字符串復制到內部RAM中,返回字符串長度 */
UINT8 mCopyCodeStringToIRAM( UINT8 idata *iDestination, UINT8 code *iSource )
{
UINT8 i = 0;
while ( *iDestination = *iSource ) {
iDestination ++;
iSource ++;
i ++;
}
return( i );
}
/* 將程序空間的字符串復制到內部RAM中,返回字符串長度 */
UINT8 mCopyCodeStringToIRAM1( UINT8 idata *iDestination, UINT8 xdata *iSource )
{
UINT8 i = 0;
while ( *iDestination = *iSource ) {
iDestination ++;
iSource ++;
i ++;
}
return( i );
}
/* 將程序空間的字符串復制到內部RAM中,返回字符串長度 */
UINT8 mCopyCodeStringToXRAM( UINT8 xdata *iDestination, UINT8 code *iSource )
{
UINT8 i = 0;
while ( *iDestination = *iSource ) {
iDestination ++;
iSource ++;
i ++;
}
return( i );
}
/* 檢查操作狀態,如果錯誤則顯示錯誤代碼并停機 */
void mStopIfError( UINT8 iError )
{
if ( iError == ERR_SUCCESS ) return; /* 操作成功 */
printf( "Error: %02X\n", (UINT16)iError ); /* 顯示錯誤 */
while ( 1 ) {
LED_OUT_ACT( ); /* LED閃爍 */
mDelaymS( 200 );
LED_OUT_INACT( );
mDelaymS( 200 );
}
}
/* 為printf和getkey輸入輸出初始化串口 */
void mInitSTDIO( )
{
SCON = 0x50;
PCON = 0x80;
TMOD = 0x21;
TH1 = 0xf3; /* 24MHz晶振, 9600bps */
TR1 = 1;
TI = 1;
}
/*計算短文件名的校驗和,*/
unsigned char ChkSum (P_FAT_DIR_INFO pDir1)
{
unsigned char FcbNameLen;
unsigned char Sum;
Sum = 0;
for (FcbNameLen=0; FcbNameLen!=11; FcbNameLen++) {
//if(pDir1->DIR_Name[FcbNameLen]==0x20)continue;
Sum = ((Sum & 1) ? 0x80 : 0) + (Sum >> 1) + pDir1->DIR_Name[FcbNameLen];
}
return (Sum);
}
/*分析緩沖區的目錄項和長文件名是否相同,返回00-15為找到長文件名相同的文件00-15表示對應短文件名在目錄
項的位置,返回0X80-8F表示分析到目錄項的結尾,以后都是未用的目錄項,返回0FF表示此扇區無匹配的目錄項*/
UINT8 mLDirCheck(P_FAT_DIR_INFO pDir2,F_LONG_NAME xdata *pLdir1){
UINT8 i,j,k,sum,nodir,nodir1;
F_LONG_NAME xdata *pLdir2;
unsigned int xdata *pLName;
unsigned char data1;
for(i=0;i!=16;i++){
if(pDir2->DIR_Name[0]==0xe5){pDir2+=1;continue;} /*如果此項被刪除繼續分析下一目錄*/ /*是被刪除的文件名則跳過*/
if(pDir2->DIR_Name[0]==0x00){return i|0x80;} /*分析到以下空間沒有文件名存在退出*/
if((pDir2->DIR_Attr==0x0f)|(pDir2->DIR_Attr==0x8)){pDir2+=1;continue;} /*如果找到的是卷標或者長文件名繼續*/
/*找到一個短文件名*/
k=i-1; /*長文件名項應在短文件名上面*/
if(i==0){ /*如果此短文件名在本扇區第一項*/
pLdir2=pLdir1; /*長文件名應在上一扇區的最后一項*/
k=15; /*記錄長文件名位置*/
pLdir2+=15; /*偏移到結尾*/
}
else pLdir2=(F_LONG_NAME xdata *)(pDir2-1); /*取長文件名目錄項*/
sum=ChkSum(pDir2); /*計算累加和*/
pLName=LongFileName; /*指項指定的長文件名*/
nodir=0; /*初始化標志*/
nodir1=1;
while(1){
if((pLdir2->LDIR_Ord!=0xe5)&(pLdir2->LDIR_Attr==ATTR_LONG_NAME)& (pLdir2->LDIR_Chksum==sum)){ /*找到一個長文件名*/
for(j=0;j!=5;j++){
if((pLdir2->LDIR_Name1[j]==0x00)|(*pLName==0))continue; /*分析到長文件名結尾*/
if((pLdir2->LDIR_Name1[j]==0xff)|(*pLName==0))continue; /*分析到長文件名結尾*/
if(pLdir2->LDIR_Name1[j]!=*pLName){ /*不等則設置標志*/
nodir=1;
break;
}
pLName++;
}
if(nodir==1)break; /*文件名不同退出*/
for(j=0;j!=6;j++){
if((pLdir2->LDIR_Name2[j]==0x00)|(*pLName==0))continue;
if((pLdir2->LDIR_Name2[j]==0xff)|(*pLName==0))continue;
if(*pLName!=pLdir2->LDIR_Name2[j]){nodir=1;break;}
pLName++;
}
if(nodir==1)break; /*文件名不同退出*/
for(j=0;j!=2;j++){
if((pLdir2->LDIR_Name3[j]==0x00)|(*pLName==0))continue;
if((pLdir2->LDIR_Name3[j]==0xff)|(*pLName==0))continue;
if(*pLName!=pLdir2->LDIR_Name3[j]){nodir=1;break;}
pLName++;
}
if(nodir==1)break; /*文件名不同退出*/
if((data1=pLdir2->LDIR_Ord&0x40)==0x40){nodir1=0;break;} /*找到長文件名,并且比較結束*/
}
else break; /*不是連續對應的長文件名退出*/
if(k==0){
pLdir2=pLdir1;
pLdir2+=15;
k=15;
}
else {
k=k-1;
pLdir2-=1;
}
}
if(nodir1==0) return i; /*表示找到長文件名,返回短文件名在的目錄項*/
pDir2+=1;
}
return 0xff; /*指搜索完這一個扇區沒找到響應的長文件名*/
}
/*檢查上級子目錄并打開*/
UINT8 mChkName(unsigned char data *pJ){
UINT8 i,j;
j = 0xFF;
for ( i = 0; i != sizeof( mCmdParam.Create.mPathName ); i ++ ) { /* 檢查目錄路徑 */
if ( mCmdParam.Create.mPathName[ i ] == 0 ) break;
if ( mCmdParam.Create.mPathName[ i ] == PATH_SEPAR_CHAR1 || mCmdParam.Create.mPathName[ i ] == PATH_SEPAR_CHAR2 ) j = i; /* 記錄上級目錄 */
}
i = ERR_SUCCESS;
if ( j == 0 || j == 2 && mCmdParam.Create.mPathName[1] == ':' ) { /* 在根目錄下創建 */
mCmdParam.Open.mPathName[ 0]='/';
mCmdParam.Open.mPathName[ 1]=0;
i=CH375FileOpen(); /*打開根目錄*/
if ( i == ERR_OPEN_DIR ) i = ERR_SUCCESS; /* 成功打開上級目錄 */
}
else {
if ( j != 0xFF ) { /* 對于絕對路徑應該獲取上級目錄的起始簇號 */
mCmdParam.Create.mPathName[ j ] = 0;
i = CH375FileOpen( ); /* 打開上級目錄 */
if ( i == ERR_SUCCESS ) i = ERR_MISS_DIR; /* 是文件而非目錄 */
else if ( i == ERR_OPEN_DIR ) i = ERR_SUCCESS; /* 成功打開上級目錄 */
mCmdParam.Create.mPathName[ j ] = PATH_SEPAR_CHAR1; /* 恢復目錄分隔符 */
}
}
*pJ=j; /*指針中返回一組數據*/
return i;
}
/*根據指定的長文件名搜索對應的短文件名長文件名空間放入長文件名,短文件名空間放入路徑00結尾*/
UINT8 mLoopkUpSName(){
UINT8 BlockSer1; /*定義兩個扇區塊內記數*/
unsigned char xdata ParData[MAX_PATH_LEN]; /**/
UINT16 tempSec; /*扇區偏移*/
UINT8 i,j,k;
F_LONG_NAME xdata *pLDirName;
P_FAT_DIR_INFO pDirName;
bit FBuf;
unsigned char data *pBuf1;
CH375DirtyBuffer();
for(k=0;k!=MAX_PATH_LEN;k++)ParData[k]=mCmdParam.Other.mBuffer[k]; /*保存當前路徑*/
i=mChkName(&j);
if ( i == ERR_SUCCESS ) { /* 成功獲取上級目錄的起始簇號 */
BlockSer1=0;
FBuf=0; /*初始化*/
tempSec=0;
FileDataBuf1[0]=0xe5; /*第一次用,無效緩沖區*/
k=0xff;
while(1){ /*下面是讀取并分析目錄項*/
pDirName=FBuf?FILE_DATA_BUF_ADDR1:FILE_DATA_BUF_ADDR; /*短文件名指針指向緩沖區*/
pLDirName=FBuf?FILE_DATA_BUF_ADDR:FILE_DATA_BUF_ADDR1;
/*當前處理的文件緩沖區*/ /*這里使用雙向緩沖區,去處理文件名*/
mCmdParam.ReadX.mSectorCount=1; /*讀取一扇區數據*/
mCmdParam.ReadX.mDataBuffer=FBuf?FILE_DATA_BUF_ADDR1:FILE_DATA_BUF_ADDR;
FBuf=!FBuf; /*緩沖區標志翻轉*/
i=CH375FileReadX( );
if(mCmdParam.ReadX.mSectorCount==0){k=0xff;break;}
k=mLDirCheck(pDirName,pLDirName);
if(k!=0x0ff){break;} /*找到文件或者找到文件結尾退出*/
}
if(k<16){
pDirName+=k; /*所找的文件短文件名在此目錄項*/
if(j!=0xff){
for(k=0;k!=j+1;k++)mCmdParam.Other.mBuffer[k]=ParData[k];
}
pBuf1=&mCmdParam.Other.mBuffer[j+1]; /*取文件名的地址*/
for(i=0;i!=8;i++){
if(pDirName->DIR_Name[i]==0x20)continue;
else{
*pBuf1=pDirName->DIR_Name[i];
pBuf1++;
}
}
if(pDirName->DIR_Name[i]!=0x20){
*pBuf1='.';
pBuf1++; /*不是目錄則有擴展名*/
}
for(;i!=11;i++){
if(pDirName->DIR_Name[i]==0x20)continue;
else {
*pBuf1=pDirName->DIR_Name[i];
pBuf1++;
}
}
*pBuf1=00;
for(k=0;k!=j+1;k++)ParData[k]=mCmdParam.Other.mBuffer[k];
/*復制短文件名*/
i=CH375FileClose( );
for(k=0;k!=j+1;k++)mCmdParam.Other.mBuffer[k]=ParData[k];
}
else k=0xff; /*返回沒找到指定的長文件名文件???????*/
i=CH375FileClose( );
}
else {k=i;};
return k;
}
/*這里可以創建文件的長文件名,在短文件名空間輸入路徑以及參考短文件名,在長文件名空間輸入該文件長文件名的UNICODE代碼,
返回狀態,00表示成功,并且在短文件名空間返回真實的短文件名,其他為不成功狀態*/
/*創建并打開*/
UINT8 mCreatLName(){
UINT8 BlockSer1; /*定義兩個扇區塊內記數*/
unsigned char xdata ParData[MAX_PATH_LEN]; /**/
UINT16 tempSec; /*扇區偏移*/
UINT8 i,j,k,x,sum,y,z;
F_LONG_NAME xdata *pLDirName;
P_FAT_DIR_INFO pDirName,pDirName1;
bit FBuf;
unsigned char data *pBuf1;
unsigned int xdata *pBuf;
CH375DirtyBuffer( );
for(k=0;k!=MAX_PATH_LEN;k++)ParData[k]=mCmdParam.Other.mBuffer[k]; /**/
i=mChkName(&j);
if ( i == ERR_SUCCESS ) { /* 成功獲取上級目錄的起始簇號 */
BlockSer1=0;
FBuf=0; /*初始化*/
tempSec=0;
?? 快捷鍵說明
復制代碼
Ctrl + C
搜索代碼
Ctrl + F
全屏模式
F11
切換主題
Ctrl + Shift + D
顯示快捷鍵
?
增大字號
Ctrl + =
減小字號
Ctrl + -