当前位置:首页 > 单片机 > 单片机
[导读]找到了一个现成的例子,对于读写U盘,文件系统是个大麻烦。现在一直还没有弄明白,先收藏一下这个程序先,等到后面慢慢研究。 作者这个编程规范不错,注释很丰富,值得学习。#include#include"CH375INC.H"#include/*

找到了一个现成的例子,对于读写U盘,文件系统是个大麻烦。现在一直还没有弄明白,先收藏一下这个程序先,等到后面慢慢研究。

作者这个编程规范不错,注释很丰富,值得学习。

#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数据端口的I/O地址*/
#defineCH375_INT_WIREINT0/*P3.2,连接CH375的INT#引脚,用于查询中
断状态*/
UINT8XDISK_BUFFER[512*32]_at_0x0000;/*外部RAM数据缓冲区的起始地址*/

UINT32DiskStart;/*逻辑盘的起始绝对扇区号LBA*/
UINT8SecPerClus;/*逻辑盘的每簇扇区数*/
UINT8RsvdSecCnt;/*逻辑盘的保留扇区数*/
UINT16;FATSz16;/*FAT16逻辑盘的FAT表占用的扇区数*/


/***********硬件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--);/*发出命令码前后应该各延时2uS*/
}
voidCH375_WR_DAT_PORT(UINT8dat){/*向CH375的数据端口写入数据*/
CH375_DAT_PORT=dat;/*因为MCS51单片机较慢所以实际上无需延时*/
}
UINT8CH375_RD_DAT_PORT(void){/*从CH375的数据端口读出数据*/
return(CH375_DAT_PORT);/*因为MCS51单片机较慢所以实际上无需延时*/
}
UINT8mWaitInterrupt(void){/*等待CH375中断并获取状态,返回操作状态*/
while(CH375_INT_WIRE);/*查询等待CH375操作完成中断(INT#低电平)*/
CH375_WR_CMD_PORT(CMD_GET_STATUS);/*产生操作完成中断,获取中断状态*/
return(CH375_RD_DAT_PORT());
}

/***********BulkOnly传输协议层,被CH375内置了,无需编写单片机程序*/

/***********RBC/SCSI命令层,虽然被CH375内置了,但是要写程序发出命令及收发数据
*/

UINT8mInitDisk(void){/*初始化磁盘*/
UINT8Status;
CH375_WR_CMD_PORT(CMD_GET_STATUS);/*产生操作完成中断,获取中断状态*/
Status=CH375_RD_DAT_PORT();
if(Status==USB_INT_DISCONNECT)return(Status);/*USB设备断开*/
CH375_WR_CMD_PORT(CMD_DISK_INIT);/*初始化USB存储器*/
Status=mWaitInterrupt();/*等待中断并获取状态*/
if(Status!=USB_INT_SUCCESS)return(Status);/*出现错误*/
CH375_WR_CMD_PORT(CMD_DISK_SIZE);/*获取USB存储器的容量*/
Status=mWaitInterrupt();/*等待中断并获取状态*/
if(Status!=USB_INT_SUCCESS){/*出错重试*/
/*对于CH375A芯片,建议在此执行一次CMD_DISK_R_SENSE命令*/
mDelaymS(250);
CH375_WR_CMD_PORT(CMD_DISK_SIZE);/*获取USB存储器的容量*/
Status=mWaitInterrupt();/*等待中断并获取状态*/
}
if(Status!=USB_INT_SUCCESS)return(Status);/*出现错误*/
return(0);/*U盘已经成功初始化*/
}


UINT8mReadSector(UINT32iLbaStart,UINT8iSectorCount,UINT8X*oDataBuffer)
{
UINT16mBlockCount;
UINT8c;
CH375_WR_CMD_PORT(CMD_DISK_READ);/*从USB存储器读数据块*/
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);/*扇区数*/
for(mBlockCount=iSectorCount*8;mBlockCount!=0;mBlockCount--){
c=mWaitInterrupt();/*等待中断并获取状态*/
if(c==USB_INT_DISK_READ){/*等待中断并获取状态,请求数据读出*/
CH375_WR_CMD_PORT(CMD_RD_USB_DATA);/*从CH375缓冲区读取数据块*/
c=CH375_RD_DAT_PORT();/*后续数据的长度*/
while(c--)*oDataBuffer++=CH375_RD_DAT_PORT();
CH375_WR_CMD_PORT(CMD_DISK_RD_GO);/*继续执行USB存储器的读操作*/
}
elsebreak;/*返回错误状态*/
}
if(mBlockCount==0){
c=mWaitInterrupt();/*等待中断并获取状态*/
if(c==USB_INT_SUCCESS)return(0);/*操作成功*/
}
return(c);/*操作失败*/
}


/***********FAT文件系统层,这层程序量实际较大,不过,该程序仅演示极简单的功能,所
以精简*/

UINT16mGetPointWord(UINT8X*iAddr){/*获取字数据,因为MCS51是大端格式*/
return(iAddr[0]|(UINT16)iAddr[1]<<8);
}

UINT8mIdenDisk(void){/*识别分析当前逻辑盘*/
UINT8Status;
DiskStart=0;/*以下是非常简单的FAT文件系统的分析,正式应用绝对不应该如此简
单*/
Status=mReadSector(0,1,DISK_BUFFER);/*读取逻辑盘引导信息*/
if(Status!=0)return(Status);
if(DISK_BUFFER[0]!=0xEB&&DISK_BUFFER[0]!=0xE9){/*不是逻辑引导扇
区*/
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];/*每簇扇区数*/
RsvdSecCnt=DISK_BUFFER[0x0E];/*逻辑盘的保留扇区数*/
FATSz16=mGetPointWord(&DISK_BUFFER[0x16]);/*FAT表占用扇区数*/
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扇区地址*/
return(DiskStart+RsvdSecCnt+FATSz16*2+32+(iCluster-2)*
SecPerClus);
}


voidmInitSTDIO(void){/*仅用于调试用途及显示内容到PC机,与该程序功能完全无
关*/
SCON=0x50;PCON=0x80;TMOD=0x20;TH1=0xf3;TR1=1;TI=1;/*24MHz,
9600bps*/
}
voidmStopIfError(UINT8iErrCode){/*如果错误则停止运行并显示错误状态*/
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盘进入正常工作状态*/
Status=mInitDisk();/*初始化U盘,实际是识别U盘的类型,必须进行此步骤*/
mStopIfError(Status);
Status=mIdenDisk();/*识别分析U盘文件系统,必要操作*/
mStopIfError(Status);
Status=mReadSector(DiskStart+RsvdSecCnt+FATSz16*2,32,
DISK_BUFFER);
mStopIfError(Status);/*读取FAT16逻辑盘的根目录,通常根目录占用32个扇区
*/
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);/*读取首簇到缓冲区*/
DISK_BUFFER[30]=0;printf("Data:%s
",DISK_BUFFER);/*显示首行
*/
Cluster=mLinkCluster(Cluster);/*获取链接簇,返回0说明错误*/
}
}
while(mWaitInterrupt()!=USB_INT_DISCONNECT);/*等待U盘拔出*/
mDelaymS(250);
}
}

本站声明: 本文章由作者或相关机构授权发布,目的在于传递更多信息,并不代表本站赞同其观点,本站亦不保证或承诺内容真实性等。需要转载请联系该专栏作者,如若文章内容侵犯您的权益,请及时联系本站删除。
换一批
延伸阅读

单片机内部有很多的特殊功能寄存器,每个寄存器在单片机内部都分配有唯一的地址,一般我们会根据寄存器功能的不同给寄存器赋予各自的名称,当我们需要在程序中操作这些特殊功能寄存器时,必须要在程序的最前面将这些名称加以声明,声明的...

关键字: C51 数据类型 扩充定义

数据元(Data Element),也称为数据元素,是用一组属性描述其定义、标识、表示和允许值的数据单元,在一定语境下,通常用于构建一个语义正确、独立且无歧义的特定概念语义的信息单元。数据元可以理解为数据的基本单元,将若...

关键字: C51 数据类型

▼点击下方名片,关注公众号▼欢迎关注【玩转单片机与嵌入式】公众号,回复关键字获取更多免费资料。回复【加群】,限时免费进入知识共享群;回复【3D封装库】,常用元器件的3D封装库;回复【电容】,获取电容、元器件选型相关的内容...

关键字: C51 MDK RealView

在Keil C51软件中51单片机的中断服务和外设驱动程序的开发

关键字: keil5 编译 C51

Intel公司1980年推出了MCS-51系列单片机:集成 8位CPU、4K字节ROM、128字节RAM、4个8位并口、1个全双工串行口、2个16位定时/计数器。寻址范围64K,并有控制功能较强的布尔处理器。 80C5...

关键字: C51 KEIL 编程

c上标3下标5怎么算用计算机,c上标3下标5怎么算

关键字: C51 KEIL

▼点击下方名片,关注公众号▼大家好,很高兴和各位一起分享我的第16篇原创文章,喜欢和支持我的工程师,一定记得给我点赞、收藏、分享。加微信[xyzn3333]与作者沟通交流,免费获取更多单片机与嵌入式的海量电子资料。很多初...

关键字: 51单片机 C51

常看见初学者要求使用_at_,这是一种谬误,把C当作ASM看待了。在C中变量的定位是编译器的事情,初学者只要定义变量和变量的作 用域,编译器就把一个固定地址给这个变量。

关键字: C51 单片机 误区 注意事项

简介:编程首要是要考虑程序的可行性,然后是可读性、可移植性、健壮性以及可测试性。这是总则。但是很多人忽略了可读性、可移植性和健壮性(可调试的方法可能歌不相同),这是不对的。

关键字: C51 编程规范 文件配置

如果你用 Keil C51 进行编译,记住一点:它不区分大小写!!!卧槽,今天编程序那个调错啊,就因为一个数组名和一个变量名完全一样,只是大小写不一样罢了,标准 C 我怎么记得这样可以啊……上网一查,卧槽,Keil C5...

关键字: C51 单片机 编程要点
关闭