当前位置:首页 > 单片机 > 单片机
[导读]/* 这个程序用180行C代码就能够读取FAT16文件系统U盘的根目录,可以看到根目录下的文件名,并可显示首文件内容,不过,该程序很不严谨,也没有任何错误处理,对U盘兼容性较差,只是用于简单试验,作为参考.这个程序可以支持W

/* 这个程序用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_CMD_PORT _at_ 0xBDF1; /* CH375命令端口的I/O地址 */
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 );
}
}

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

9月2日消息,不造车的华为或将催生出更大的独角兽公司,随着阿维塔和赛力斯的入局,华为引望愈发显得引人瞩目。

关键字: 阿维塔 塞力斯 华为

加利福尼亚州圣克拉拉县2024年8月30日 /美通社/ -- 数字化转型技术解决方案公司Trianz今天宣布,该公司与Amazon Web Services (AWS)签订了...

关键字: AWS AN BSP 数字化

伦敦2024年8月29日 /美通社/ -- 英国汽车技术公司SODA.Auto推出其旗舰产品SODA V,这是全球首款涵盖汽车工程师从创意到认证的所有需求的工具,可用于创建软件定义汽车。 SODA V工具的开发耗时1.5...

关键字: 汽车 人工智能 智能驱动 BSP

北京2024年8月28日 /美通社/ -- 越来越多用户希望企业业务能7×24不间断运行,同时企业却面临越来越多业务中断的风险,如企业系统复杂性的增加,频繁的功能更新和发布等。如何确保业务连续性,提升韧性,成...

关键字: 亚马逊 解密 控制平面 BSP

8月30日消息,据媒体报道,腾讯和网易近期正在缩减他们对日本游戏市场的投资。

关键字: 腾讯 编码器 CPU

8月28日消息,今天上午,2024中国国际大数据产业博览会开幕式在贵阳举行,华为董事、质量流程IT总裁陶景文发表了演讲。

关键字: 华为 12nm EDA 半导体

8月28日消息,在2024中国国际大数据产业博览会上,华为常务董事、华为云CEO张平安发表演讲称,数字世界的话语权最终是由生态的繁荣决定的。

关键字: 华为 12nm 手机 卫星通信

要点: 有效应对环境变化,经营业绩稳中有升 落实提质增效举措,毛利润率延续升势 战略布局成效显著,战新业务引领增长 以科技创新为引领,提升企业核心竞争力 坚持高质量发展策略,塑强核心竞争优势...

关键字: 通信 BSP 电信运营商 数字经济

北京2024年8月27日 /美通社/ -- 8月21日,由中央广播电视总台与中国电影电视技术学会联合牵头组建的NVI技术创新联盟在BIRTV2024超高清全产业链发展研讨会上宣布正式成立。 活动现场 NVI技术创新联...

关键字: VI 传输协议 音频 BSP

北京2024年8月27日 /美通社/ -- 在8月23日举办的2024年长三角生态绿色一体化发展示范区联合招商会上,软通动力信息技术(集团)股份有限公司(以下简称"软通动力")与长三角投资(上海)有限...

关键字: BSP 信息技术
关闭
关闭