您的位置 首页 FPGA

MCS-51单片机读写U盘的程序设计

MCS-51单片机读写U盘的程序设计-这个程序可以支持WINDOWS按FAT16格式化的U盘,因为程序精简,所以只兼容超过50%以上的U盘品牌,如果换成CH375A芯片则兼容性可提高到85%,当然,如果使用WCH公司的子程序库或者正式版本的C源程序兼容性更好。

这个程序用180行C代码就能够读取FAT16文件体系U盘的根目录,能够看到根目录下的文件名,并可显现首文件内容,不过,该程序很不谨慎,也没有任何过错处理,对U盘兼容性较差,仅仅用于简略试验,作为参阅。

这个程序能够支撑WINDOWS按FAT16格局化的U盘,由于程序精简,所以只兼容超越50%以上的U盘品牌,假如换成CH375A芯片则兼容性可提高到85%,当然,假如运用WCH公司的子程序库或许正式版别的C源程序兼容性更好。测验以下U盘经过:郎科/超稳经典64M/超稳迷你128M/U160-64M/超稳遍及128M,爱国者/迷你王16M/邮箱型,黑匣子/64M,微闪/64M,飙王/32M/64M/128M,晶彩/C200-64M,新科/256M,昂达/128M.。欢迎供给测验成果。

未经过U盘:爱国者/智慧棒128M,清华普天/USB2.0-128M,当然,运用WCH的子程序库或CH375A都能够测验经过 。
 

#i nclude

#i nclude “CH375INC.H”

#i nclude /* 以下界说适用于MCS-51单片机,其它单片机参照修正 */

#define UINT8 unsigned char

#define UINT16 unsigned short

#define UINT32 unsigned long

#define UINT8X unsigned char xdata

#define UINT8VX unsigned char volaTIle xdata

UINT8VX CH375_DAT_PORT _at_ 0xBCF0; /* CH375数据端口的I/O地址 */

#define CH375_INT_WIRE INT0 /* P3.2, 衔接CH375的INT#引脚,用于查询中

断状况 */

UINT8X DISK_BUFFER[512*32] _at_ 0x0000; /* 外部RAM数据缓冲区的开始地址 */

UINT32 DiskStart; /* 逻辑盘的开始肯定扇区号LBA */

UINT8 SecPerClus; /* 逻辑盘的每簇扇区数 */

UINT8 RsvdSecCnt; /* 逻辑盘的保存扇区数 */

UINT16 FATSz16; /* FAT16逻辑盘的FAT表占用的扇区数 */

/* ********** 硬件USB接口层,无论如何这层省不掉,单片机总要与CH375接口吧 */

void mDelaymS( UINT8 delay ) {

UINT8 i, j, c;

for ( i = delay; i != 0; i — ) {

for ( j = 200; j != 0; j — ) c += 3;

for ( j = 200; j != 0; j — ) c += 3;

}

}

void CH375_WR_CMD_PORT( UINT8 cmd ) { /* 向CH375的指令端口写入指令 */

CH375_CMD_PORT=cmd;

for ( cmd = 2; cmd != 0; cmd — ); /* 宣布指令码前后应该各延时2uS */

}

void CH375_WR_DAT_PORT( UINT8 dat ) { /* 向CH375的数据端口写入数据 */

CH375_DAT_PORT=dat; /* 由于MCS51单片机较慢所以实践上无需延时 */

}

UINT8 CH375_RD_DAT_PORT( void ) { /* 从CH375的数据端口读出数据 */

return( CH375_DAT_PORT ); /* 由于MCS51单片机较慢所以实践上无需延时 */

}

UINT8 mWaiTInterrupt( 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内置了,可是要写程序宣布指令及收发数据

*/

UINT8 mInitDisk( void ) { /* 初始化磁盘 */

UINT8 Status;

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盘现已成功初始化 */

}

UINT8 mReadSector( UINT32 iLbaStart, UINT8 iSectorCount, UINT8X *oDataBuffer )

{

UINT16 mBlockCount;

UINT8 c;

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存储器的读操作 */

}

else break; /* 回来过错状况 */

}

if ( mBlockCount == 0 ) {

c = mWaitInterrupt( ); /* 等候中止并获取状况 */

if ( c== USB_INT_SUCCESS ) return( 0 ); /* 操作成功 */

}

return( c ); /* 操作失利 */

}

/* ********** FAT文件体系层,这层程序量实践较大,不过,该程序仅演示极简略的功用,所

以精简 */

UINT16 mGetPointWord( UINT8X *iAddr ) { /* 获取字数据,由于MCS51是大端格局 */

return( iAddr[0] | (UINT16)iAddr[1] 《《 8 );

}

UINT8 mIdenDisk( void ) { /* 辨认剖析当时逻辑盘 */

UINT8 Status;

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 ); /* 成功 */

}

UINT16 mLinkCluster( UINT16 iCluster ) { /* 取得指定簇号的链接簇 */

/* 输入: iCluster 当时簇号, 回来: 原链接簇号, 假如为0则阐明过错 */

UINT8 Status;

Status = mReadSector( DiskStart + RsvdSecCnt + iCluster / 256, 1,

DISK_BUFFER );

if ( Status != 0 ) return( 0 ); /* 过错 */

return( mGetPointWord( &DISK_BUFFER[ ( iCluster + iCluster ) & 0x01FF ] ) );

}

UINT32 mClusterToLba( UINT16 iCluster ) { /* 将簇号转换为肯定LBA扇区地址 */

return( DiskStart + RsvdSecCnt + FATSz16 * 2 + 32 + ( iCluster – 2 ) *

SecPerClus );

}

void mInitSTDIO( void ) { /* 仅用于调试用处及显现内容到PC机,与该程序功用完全无

关 */

SCON = 0x50; PCON = 0x80; TMOD = 0x20; TH1 = 0xf3; TR1=1; TI=1; /* 24MHz,

9600bps */

}

void mStopIfError( UINT8 iErrCode ) { /* 假如过错则中止运转并显现过错状况 */

if ( iErrCode == 0 ) return;

printf( “Error status, %02Xn”, (UINT16)iErrCode );

}

main( ) {

UINT8 Status;

UINT8X *CurrentDir;

UINT16 Cluster;

mDelaymS( 200 ); /* 延时200毫秒 */

mInitSTDIO( );

CH375_WR_CMD_PORT( CMD_SET_USB_MODE ); /* 初始化CH375,设置USB作业形式 */

CH375_WR_DAT_PORT( 6 ); /* 形式代码,自动检测USB设备衔接 */

while ( 1 ) {

printf( “Insert USB diskn” );

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: %sn”, 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: %sn”, DISK_BUFFER ); /* 显现首行

*/

Cluster = mLinkCluster( Cluster ); /* 获取链接簇,回来0阐明过错 */

}

}

while ( mWaitInterrupt( ) != USB_INT_DISCONNECT ); /* 等候U盘拔出 */

mDelaymS( 250 );

}

}
责任编辑;zl

声明:本文内容来自网络转载或用户投稿,文章版权归原作者和原出处所有。文中观点,不代表本站立场。若有侵权请联系本站删除(kf@86ic.com)https://www.86ic.net/fangan/fpga/343453.html

为您推荐

联系我们

联系我们

在线咨询: QQ交谈

邮箱: kf@86ic.com

关注微信
微信扫一扫关注我们

微信扫一扫关注我们

返回顶部