#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, %02X\n", (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 disk\n" );

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\n", 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\n", DISK_BUFFER );  /* 显示首行

*/

Cluster = mLinkCluster( Cluster );  /* 获取链接簇,返回0说明错误 */

}

}

while ( mWaitInterrupt( ) != USB_INT_DISCONNECT

);  /* 等待U盘拔出 */

mDelaymS( 250 );

}

}

单片机读tf卡c语言程序,单片机读写U盘闪盘超精简C源程序相关推荐

  1. 单片机读tf卡c语言程序,单片机读写SD卡,MMC卡,MiniSD,TF卡,MicroSD卡

    图一 一:系统构成: 1:89C51单片机(ISP 单片机,可不用编程器) 2:128x64LCD 3: 带SD ,MiniSD ,MicroSD(TFSD)卡座 送32MByte SD卡 4:支持红 ...

  2. 52单片机led灯闪烁c语言程序,单片机LED灯闪烁程序

    原标题:单片机LED灯闪烁程序 点亮 小灯的程序我们第二课完成了,就是让 LED = 0.熄灭小灯的程序也很简单,就是 LED = 1.点亮和熄灭都会了,那么如果我们在亮和灭中间加个延时,反复不停的点 ...

  3. 52单片机led灯闪烁c语言程序,单片机 LED 灯闪烁程序

    点亮 LED 小灯的程序我们第二课完成了,就是让 LED = 0.熄灭小灯的程序也很简单,就是 LED = 1.点亮和熄灭都会了,那么如果我们在亮和灭中间加个延时,反复不停的点亮和熄灭小灯,就成了闪烁 ...

  4. 单片机六位抢答器c语言程序,单片机八人抢答器程序设计

    有一个<单片机八人抢答器程序设计>的题目. 设计要求: 利用 KEIL 软件编制控制程序. 利用 PROTEUS 仿真软件,绘制单片机控制系统: P1 口接 8 个按钮,编号为1.2.3. ...

  5. 单片机六位抢答器c语言程序,单片机六人抢答器程序.doc

    题目如下:六人抢答器,以拨动开关K0~K5中的某个开关为ON作为抢答按键,无人抢答时,6只数码管循环轮流显示1~6跑马,谁先抢答,数码管停止跑马,6个数码管同时亮谁的编号,气候再有按键按下,系统不予回 ...

  6. 液晶显示屏的C语言编码,单片机液晶汉字显示c语言程序

    单片机液晶汉字显示c语言程序 单片机液晶汉字显示c语言程序 /*液晶屏分为4行*12列汉字,全部使用模拟接口方式. /* TGLCMLIMIT64A接口程序(模拟方式) ;************** ...

  7. DSB算法C语言程序,单片机中使用DSB温度传感器C语言程序.doc

    单片机中使用DSB温度传感器C语言程序 单片机中使用DS18B20温度传感器C语言程序(参考1) /************************************************** ...

  8. C语言 int 转单精度浮点,单精度浮点数与十六进制转换 C语言程序 单片机也可用...

    单精度浮点数与十六进制转换 C语言程序 单片机也可用 #include float Hex_To_Decimal(unsigned char *Byte,int num)//十六进制到浮点数 { // ...

  9. 超声波c语言程序,51单片机超声波模块的C语言程序.doc

    51单片机超声波模块的C语言程序.doc 超声波模块程序 超声波模块程序 Trig P20 Echo P32 include define uchar unsigned char define uin ...

最新文章

  1. 容器云之K8s自动化安装方式的选择
  2. 承受海量访问建站方案
  3. amd插帧技术如何开启_联想ThinkPad笔记本电脑如何开启CPU的虚拟化技术图文教程...
  4. 整合Spring框架和Hibernate框架
  5. 华为上架自主地图应用Here WeGo 或登录HMS系统成为预装应用
  6. react-native调用安卓原生,如果调用没有参数的方法会报错,会提示没有加入参数.
  7. awk内置字符串函数详解
  8. 全向轮机器人运动模型及应用分析(图片版)
  9. AR增强现实开发介绍
  10. c++ STL之queue
  11. [免费专栏] Android安全之Android工程模式
  12. 广告系统数据采集方法介绍
  13. 在C#中使用OpenCV(使用OpenCVSharp)
  14. 解决Mysql:Incorrect string value: '\xF0\x9F\x8D\x83\xF0\x9F...' for column
  15. 《算法0基础100讲》(第7讲)素数判定——866.回文素数
  16. 自媒体短视频快速吸引粉丝的诀窍
  17. 斜杠'/' 和反斜杠'\'
  18. 人工智能之机器学习简介
  19. 0. 一字一句的搞懂vue-cli之vue webpack template配置
  20. ubuntu-关机时黑屏左上角只有一个横着的光标一直闪烁长时间无法关机

热门文章

  1. 按钮button加超链接
  2. 局域网IP地址查看软件V1.2使用教程 Python办公自动化
  3. Kafka集群部署搭建完美标准版
  4. android 设置setmultichoiceitems设置初始化勾选_Linux内核启动:虚拟盘空间设置和内存管理结构初始化...
  5. 台式计算机更新不了,台式机更新造成电脑关不了机怎么办
  6. linux执行.sql脚本 db2,DB2 如何执行sql脚本
  7. linux禁止客户端上传文件_Linux 几种上传文件到linux服务的方法
  8. c++ udp多线程 例子_[内附完整源码和文档] 基于udp实现tcp功能进行大文件传输
  9. python中str是什么_python的str()字符串类型的方法详解
  10. switch语句php,PHPswitch 语句 - PHP教程