📄 exam11.c
字号:
/* 2004.06.05
****************************************
** Copyright (C) W.ch 1999-2005 **
** Web: http://www.winchiphead.com **
****************************************
** USB Host File Interface for CH374 **
** TC2.0@PC, KC7.0@MCS51 **
****************************************
*/
/* CH374 主机文件系统接口 */
/* 支持: FAT12/FAT16/FAT32 */
/* MCS-51单片机C语言的U盘文件读写示例程序, 适用于89C52或者更大程序空间的单片机 */
/* 本程序用于演示创建/打开/删除长文件名文件 */
/* CH374的INT#引脚采用查询方式处理, 数据复制方式为"单DPTR复制", 所以速度较慢, 适用于所有MCS51单片机 */
/* 仅供有FAT文件系统和长文件名相关技术知识基础的用户参考,支持小写字母或者汉字等不超过256个字符的文件名,
对该类知识不了解的用户建议不要涉及长文件名,建议只使用现成的短文件名,短文件名支持8+3格式,大写字母或者汉字 */
/* C51 CH374HFT.C */
/* LX51 CH374HFT.OBJ , CH374HF6.LIB, C51DPTR1.LIB */
/* OHX51 CH374HFT */
#include <reg52.h>
#include <stdio.h>
#include <string.h>
/* 以下定义的详细说明请看CH374HF6.H文件 */
#define LIB_CFG_INT_EN 0 /* CH374的INT#引脚连接方式,0为"查询方式",1为"中断方式" */
#define CH374_IDX_PORT_ADDR 0xBDF1 /* CH374索引端口的I/O地址 */
#define CH374_DAT_PORT_ADDR 0xBCF0 /* CH374数据端口的I/O地址 */
/* 62256提供的32KB的RAM分为两部分: 0000H-01FFH为磁盘读写缓冲区, 0200H-7FFFH为文件数据缓冲区 */
#define DISK_BASE_BUF_ADDR 0x0000 /* 外部RAM的磁盘数据缓冲区的起始地址,从该单元开始的缓冲区长度为SECTOR_SIZE */
#define CH374_INT_WIRE INT0 /* P3.2, INT0, CH374的中断线INT#引脚,连接CH374的INT#引脚,用于查询中断状态 */
/* 如果未连接CH374的中断引脚,那么应该去掉上述定义,自动使用寄存器查询方式 */
#define NO_DEFAULT_CH374_F_ENUM 1 /* 未调用CH374FileEnumer程序故禁止以节约代码 */
#define NO_DEFAULT_CH374_F_QUERY 1 /* 未调用CH374FileQuery程序故禁止以节约代码 */
#define NO_DEFAULT_CH374_RESET 1 /* 未调用CH374Reset程序故禁止以节约代码 */
#include "..\CH374HF6.H"
#include "FILELONG.H"
#define FILE_DATA_BUF_ADDR0 0x0200 /* 外部RAM的文件数据缓冲区的起始地址,缓冲区长度不小于一次读写的数据长度 */
#define FILE_DATA_BUF_ADDR1 0x0400
UINT8X FileDataBuf0[0x0200] _at_ FILE_DATA_BUF_ADDR0 ;
UINT8X FileDataBuf1[0x0200] _at_ FILE_DATA_BUF_ADDR1 ;
/* 在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显示 */
/* 检查操作状态,如果错误则显示错误代码并停机 */
void mStopIfError( UINT8 iError )
{
if ( iError == ERR_SUCCESS ) return; /* 操作成功 */
printf( "Error: %02X\n", (UINT16)iError ); /* 显示错误 */
/* 遇到错误后,应该分析错误码以及CH374DiskStatus状态,例如调用CH374DiskConnect查询当前U盘是否连接,如果U盘已断开那么就重新等待U盘插上再操作,
建议出错后的处理步骤:
1、调用一次CH374DiskReady,成功则继续操作,例如Open,Read/Write等,在CH374DiskReady中会自动调用CH374DiskConnect,不必另外调用
2、如果CH374DiskReady不成功,那么强行将CH374DiskStatus置为DISK_CONNECT状态,然后从头开始操作(等待U盘连接CH374DiskConnect,CH374DiskReady等) */
while ( 1 ) {
LED_OUT_ACT( ); /* LED闪烁 */
CH374DelaymS( 200 );
LED_OUT_INACT( );
CH374DelaymS( 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表示此扇区无匹配的目录项*/
unsigned char mLDirCheck(P_FAT_DIR_INFO pDir2,F_LONG_NAME xdata *pLdir1){
unsigned char 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; /*指搜索完这一个扇区没找到响应的长文件名*/
}
/*检查上级子目录并打开*/
unsigned char mChkName(unsigned char data *pJ){
unsigned char 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=CH374FileOpen(); /*打开根目录*/
if ( i == ERR_OPEN_DIR ) i = ERR_SUCCESS; /* 成功打开上级目录 */
}
else {
if ( j != 0xFF ) { /* 对于绝对路径应该获取上级目录的起始簇号 */
mCmdParam.Create.mPathName[ j ] = 0;
i = CH374FileOpen( ); /* 打开上级目录 */
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结尾*/
unsigned char mLoopkUpSName(){
unsigned char BlockSer1; /*定义两个扇区块内记数*/
unsigned char xdata ParData[MAX_PATH_LEN]; /**/
unsigned char i,j,k;
F_LONG_NAME xdata *pLDirName;
P_FAT_DIR_INFO pDirName;
bit FBuf;
unsigned char data *pBuf1;
CH374DirtyBuffer();
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; /*初始化*/
FileDataBuf1[0]=0xe5; /*第一次用,无效缓冲区*/
k=0xff;
while(1){ /*下面是读取并分析目录项*/
pDirName=FBuf?FILE_DATA_BUF_ADDR1:FILE_DATA_BUF_ADDR0; /*短文件名指针指向缓冲区*/
pLDirName=FBuf?FILE_DATA_BUF_ADDR0:FILE_DATA_BUF_ADDR1;
/*当前处理的文件缓冲区*/ /*这里使用双向缓冲区,去处理文件名*/
mCmdParam.ReadX.mSectorCount=1; /*读取一扇区数据*/
mCmdParam.ReadX.mDataBuffer=FBuf?FILE_DATA_BUF_ADDR1:FILE_DATA_BUF_ADDR0;
FBuf=!FBuf; /*缓冲区标志翻转*/
i=CH374FileReadX( );
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=CH374FileClose( );
for(k=0;k!=j+1;k++)mCmdParam.Other.mBuffer[k]=ParData[k];
}
else k=0xff; /*返回没找到指定的长文件名文件???????*/
i=CH374FileClose( );
}
else {k=i;};
return k;
}
/*这里可以创建文件的长文件名,在短文件名空间输入路径以及参考短文件名,在长文件名空间输入该文件长文件名的UNICODE代码,
返回状态,00表示成功,并且在短文件名空间返回真实的短文件名,其他为不成功状态*/
/*创建并打开*/
unsigned char mCreatLName(){
unsigned char BlockSer1; /*定义两个扇区块内记数*/
unsigned char xdata ParData[MAX_PATH_LEN]; /**/
unsigned int tempSec; /*扇区偏移*/
unsigned char 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;
CH374DirtyBuffer( );
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_ADDR0; /*短文件名指针指向缓冲区*/
pLDirName=FBuf?FILE_DATA_BUF_ADDR0:FILE_DATA_BUF_ADDR1;
mCmdParam.ReadX.mSectorCount=1; /*读取一扇区数据*/
mCmdParam.ReadX.mDataBuffer=FBuf?FILE_DATA_BUF_ADDR1:FILE_DATA_BUF_ADDR0; /*当前处理的文件缓冲区,这里使用双向缓冲区,去处理文件名*/
FBuf=!FBuf; /*缓冲区标志翻转*/
i=CH374FileReadX( );
if(mCmdParam.ReadX.mSectorCount==0){k=0xff;break;}
tempSec+=1;
k=mLDirCheck(pDirName,pLDirName);
z=k;
z&=0x0f;
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]; /*取文件名的地址*/
//else pBuf1=&mCmdParam.Other.mBuffer;
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++;
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -