ch375读u盘?

#include #include

#define SYSCLK 11059200 //系统时钟取自外部晶体的2分频,即11.0592MHz #define BAUDRATE 9600 #define MAX_BYTE_IO 35

#define LIB_CFG_FILE_IO 1 /* 文件读写的数据的复制方式,0为"外部子程序",1为"内部复制" */ #define LIB_CFG_INT_EN 0 /* CH375的INT#引脚连接方式,0为"查询方式",1为"中断方式" */

/* 只使用单片机内置的1KB外部RAM: 0000H-01FFH 为磁盘读写缓冲区, 以字节为单位读写文件不需要文件数据读写缓冲区FILE_DATA_BUF */ #define DISK_BASE_BUF_ADDR 0x0000 /* 外部RAM的磁盘数据缓冲区的起始地址,从该单元开始的缓冲区长度为SECTOR_SIZE */ #define DISK_BASE_BUF_LEN 2048 /* 默认的磁盘数据缓冲区大小为512字节,建议选择为2048甚至4096以支持某些大扇区的U盘,为0则禁止在.H文件中定义缓冲区并由应用程序在pDISK_BASE_BUF中指定 */ #define FILE_DATA_BUF_ADDR 0x0000 /* 外部RAM的文件数据缓冲区的起始地址,缓冲区长度不小于一次读写的数据长度,字节模式不用该缓冲区 */ /* 由于单片机内置的外部RAM只有1KB, 有些单片机还要去掉256字节内部RAM, 只剩下768字节的外部RAM, 其中前512字节由CH375子程序用于磁盘数据缓冲 */ #define FILE_DATA_BUF_LEN 0x0800 /* 外部RAM的文件数据缓冲区,缓冲区长度不小于一次读写的数据长度,字节模式不用该缓冲区 */

#define CH375_INT_WIRE p0.2 /* P0.2, INT0, CH375的中断线INT#引脚,连接CH375的INT#引脚,用于查询中断状态 */

#define NO_DEFAULT_CH375_F_ENUM 1 /* 未调用CH375FileEnumer程序故禁止以节约代码 */ #define NO_DEFAULT_CH375_F_QUERY 1 /* 未调用CH375FileQuery程序故禁止以节约代码 */

/* 在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显示 */

/* 在P1.5连接一个LED用于监控程序的进度,低电平LED亮,当对U盘操作时亮 */ sbit P1_5 = P1^5; #define LED_RUN_ACT( ) { P1_5 = 0; } /* P1.5 低电平驱动LED显示 */ #define LED_RUN_INACT( ) { P1_5 = 1; } /* P1.5 低电平驱动LED显示 */

/*************端口初始化*******************/ void PORT_Init (void) { XBR0=0x04; //UART0允许,TX0接到P0.0,RX0接到P0.1 XBR1=0x04; //INT0接到P0.2 XBR2=0x40; //使能交叉开关和弱上拉 P0MDOUT |=0x01; //使能TX0作为推挽输出,RX0和INT0设为开漏*/ P1MDOUT=0xff; //P1为推拉方式,以驱动LED灯P1.6 P2MDOUT=0x00; //P2为开漏方式,已上拉 P3MDOUT=0x00; //P3为开漏方式,作为打印机和液晶的控制口,已外接上拉电阻 P74OUT=0x01; //P5、P6为开漏方式,已外接5K的上拉电阻;P4.0~P4.3为推拉,P4.4~P4.7为开漏 //若都设成开漏则不行,P4.4~P4.7初始化时作为输入(第1列~第4列),接上拉到中断,P4.0~P4.3(第1行~第4行)初始化时作为输出,不接上拉 }

/* 串口初始化(设置工作方式和波特率) */

void UART0_Init (void) { SCON0=0xD0; //设置串口为9位数据 TMOD=0x20; //TMOD: T1工作在模式2, 8位重载 //TL1=0; TH1=-(SYSCLK/BAUDRATE)/16; //设置定时1用作波特率9600的重载值,TH1=70H TR1=1; //开T1 CKCON |=0x10; //T1用系统时钟作为它的时基,T1M=1 PCON |=0x80; //SMOD00=1(波特率加倍) }

/*************中断初始化*******************/ void INT_Init(void) { IT0=1; //中断0为下降沿触发 EX0=1; //开外部中断0请求 ES0=1; //允许UART0的接和收中断 IP=0x01; }

/*写指令*/ void xWriteCH375Cmd( unsigned char mCmd ) { TI0 = 0; TB80 = 1; SBUF0 = mCmd; while ( TI0 == 0 ); }

/*写数据*/ void xWriteCH375Data( unsigned char mData ) { TI0 = 0; TB80 = 0; SBUF0 = mData; while ( TI0 == 0 ); }

/*读数据*/ unsigned char xReadCH375Data( void ) { while ( RI0 == 0 ); RI0 = 0; return( SBUF0 ); }

/* 延时100毫秒,不精确 */ void mDelay100mS( ) { unsigned char i, j, c; for ( i = 200; i != 0; i -- ) for ( j = 200; j != 0; j -- ) c+=3; }

/* 将程序空间的字符串复制到内部RAM中,返回字符串长度 */ unsigned char mCopyCodeStringToIRAM( unsigned char idata *iDestination, unsigned char code *iSource ) { unsigned char i = 0; while ( *iDestination = *iSource ) { iDestination ++; iSource ++; i ++; } return( i ); }

/* 检查操作状态,如果错误则显示错误代码并停机 */ void mStopIfError( unsigned char iError ) { if ( iError == ERR_SUCCESS ) return; /* 操作成功 */ while ( 1 ) { LED_OUT_ACT( ); /* LED闪烁 */ mDelay100mS( ); LED_OUT_INACT( ); mDelay100mS( ); } }

main( ) { unsigned char i; PORT_Init ( ); UART0_Init ( ); INT_Init( ); LED_OUT_INIT( ); LED_OUT_ACT( ); /* 开机后LED亮一下以示工作 */ mDelay100mS( ); /* 延时100毫秒 */ LED_OUT_INACT( ); #if DISK_BASE_BUF_LEN == 0 pDISK_BASE_BUF = &my_buffer[0]; /* 不在.H文件中定义CH375的专用缓冲区,而是用缓冲区指针指向其它应用程序的缓冲区便于合用以节约RAM */ #endif i = CH375LibInit( ); /* 初始化CH375程序库和CH375芯片,操作成功返回0 */ mStopIfError( i ); /* 其它电路初始化 */ while ( 1 ) { //printf( "Wait Udisk\n" ); while ( CH375DiskStatus != DISK_CONNECT ) ;xQueryInterrupt( ); /* 查询CH375中断并更新中断状态,等待U盘插入 */ LED_OUT_ACT( ); /* LED亮 */ mDelay100mS( ); /* 延时,可选操作,有的USB存储器需要几十毫秒的延时 */ mDelay100mS( );

/* 检查U盘是否准备好,有些U盘不需要这一步,但是某些U盘必须要执行这一步才能工作 */ for ( i = 0; i < 5; i ++ ) { /* 有的U盘总是返回未准备好,不过可以被忽略 */ mDelay100mS( ); if ( CH375DiskReady( ) == ERR_SUCCESS ) break; /* 查询磁盘是否准备好 */ }

#if DISK_BASE_BUF_LEN if ( DISK_BASE_BUF_LEN < CH375vSectorSize ) { /* 检查磁盘数据缓冲区是否足够大,CH375vSectorSize是U盘的实际扇区大小 */ while ( CH375DiskConnect( ) == ERR_SUCCESS ) mDelay100mS( ); continue; } #endif /* 查询磁盘物理容量 */ /* printf( "DiskSize\n" ); i = CH375DiskSize( ); mStopIfError( i ); printf( "TotalSize = %u MB \n", (unsigned int)( mCmdParam.DiskSize.mDiskSizeSec * (CH375vSectorSize/512) / 2048 ) ); // 显示为以MB为单位的容量 // 原计算方法 (unsigned int)( mCmdParam.DiskSize.mDiskSizeSec * CH375vSectorSize / 1000000 ) 有可能前两个数据相乘后导致溢出, 所以修改成上式 */

LED_RUN_ACT( ); /* 运行灯亮,开始操作U盘 */ mCopyCodeStringToIRAM( mCmdParam.Open.mPathName, "/MINE.TXT" ); /*mine.TXT文件名,该文件在根目录下 */ i = CH375FileOpen( ); /* 打开文件 */ if ( i == ERR_SUCCESS ) { /* 文件存在并且已经被打开,移动文件指针到尾部以便添加数据 */ CH375vFileSize ); /* V1.5以上子程序库在成功打开文件后,全局变量CH375vFileSize中是文件当前长度 */ mCmdParam.ByteLocate.mByteOffset = 0x00000000; /* 移到文件的开始 */ i = CH375FileReadX( ); mStopIfError( i ); } i = CH375FileClose( ); /* 关闭文件 */ mStopIfError( i ); LED_RUN_INACT( ); while ( CH375DiskStatus != DISK_DISCONNECT ) xQueryInterrupt( ); /* 查询CH375中断并更新中断状态,等待U盘拔出 */ LED_OUT_INACT( ); /* LED灭 */ mDelay100mS( ); mDelay100mS( ); } }

这是我组织的用单片机读u盘的程序,我想把u盘中MINE.txt中的内容读到File_DATA_BUF中,请帮我看以看这个程序对不对?

(1)你做实验没有,结果对不对? (2)mCmdParam.ByteLocate.mByteOffset = 0x00000000; /* 移到文件的开始 */ i = CH375FileReadX( );你是想移动文件指针还是想读数据呢? CH375vFileSize ); 这是什么意思,编译能过么? (3)参考\CH375LIB\MCS51\FILELIB4\下的例程,如:EXAM1,EXAM2


没做过实验,因为编译没通过,我是想读MINE.TXT中的数据。


main( ) { unsigned char i, c,SecCount; UINT16 NewSize, count; /* 因为演示板的RAM容量只有32KB,所以NewSize限制为16位,实际上如果文件大于32256字节,应该分几次读写并且将NewSize改为UINT32以便累计 */ UINT8 code *pCodeStr; PORT_Init ( ); UART0_Init ( ); INT_Init( ); LED_OUT_INIT( ); LED_OUT_ACT( ); /* 开机后LED亮一下以示工作 */ mDelay100mS( ); /* 延时100毫秒 */ LED_OUT_INACT( ); #if DISK_BASE_BUF_LEN == 0 pDISK_BASE_BUF = &my_buffer[0]; /* 不在.H文件中定义CH375的专用缓冲区,而是用缓冲区指针指向其它应用程序的缓冲区便于合用以节约RAM */ #endif i = CH375LibInit( ); /* 初始化CH375程序库和CH375芯片,操作成功返回0 */ mStopIfError( i ); /* 其它电路初始化 */ while ( 1 ) { while ( CH375DiskStatus != DISK_CONNECT ) ;xQueryInterrupt( ); /* 查询CH375中断并更新中断状态,等待U盘插入 */ LED_OUT_ACT( ); /* LED亮 */ mDelay100mS( ); /* 延时,可选操作,有的USB存储器需要几十毫秒的延时 */ mDelay100mS( );

/* 检查U盘是否准备好,有些U盘不需要这一步,但是某些U盘必须要执行这一步才能工作 */ for ( i = 0; i < 5; i ++ ) { /* 有的U盘总是返回未准备好,不过可以被忽略 */ mDelay100mS( ); if ( CH375DiskReady( ) == ERR_SUCCESS ) break; /* 查询磁盘是否准备好 */ }

#if DISK_BASE_BUF_LEN if ( DISK_BASE_BUF_LEN < CH375vSectorSize ) { /* 检查磁盘数据缓冲区是否足够大,CH375vSectorSize是U盘的实际扇区大小 */ while ( CH375DiskConnect( ) == ERR_SUCCESS ) mDelay100mS( ); continue; } #endif /* 查询磁盘物理容量 */ /* printf( "DiskSize\n" ); i = CH375DiskSize( ); mStopIfError( i ); printf( "TotalSize = %u MB \n", (unsigned int)( mCmdParam.DiskSize.mDiskSizeSec * (CH375vSectorSize/512) / 2048 ) ); // 显示为以MB为单位的容量 // 原计算方法 (unsigned int)( mCmdParam.DiskSize.mDiskSizeSec * CH375vSectorSize / 1000000 ) 有可能前两个数据相乘后导致溢出, 所以修改成上式 */

LED_RUN_ACT( ); /* 运行灯亮,开始操作U盘 */ mCopyCodeStringToIRAM( mCmdParam.Open.mPathName, "/MINE.TXT" ); /*mine.TXT文件名,该文件在根目录下 */ i = CH375FileOpen( ); /* 打开文件 */ if ( i == ERR_SUCCESS ) { /* 找到文件或者出错 */ mStopIfError( i ); /* printf( "Query\n" ); i = CH375FileQuery( ); 查询当前文件的信息 mStopIfError( i );*/ if ( CH375vFileSize > FILE_DATA_BUF_LEN ) { /* 由于演示板用的62256只有32K字节,其中CH375子程序用512字节,所以只读取不超过63个扇区,也就是不超过32256字节 */ SecCount = FILE_DATA_BUF_LEN / CH375vSectorSize; /* 由于演示板用的62256只有32K字节,其中CH375子程序用512字节,所以只读取不超过63个扇区,也就是不超过32256字节 */ NewSize = FILE_DATA_BUF_LEN; /* 由于RAM有限所以限制长度 */ } else { /* 如果原文件较小,那么使用原长度 */ SecCount = ( CH375vFileSize + CH375vSectorSize - 1 ) / CH375vSectorSize; /* 计算文件的扇区数,因为读写是以扇区为单位的,先加CH375vSectorSize-1是为了读出文件尾部不足1个扇区的部分 */ NewSize = (UINT16)CH375vFileSize; /* 原文件的长度 */ } mCmdParam.Read.mSectorCount = SecCount; /* 读取全部数据,如果超过60个扇区则只读取60个扇区 */ // current_buffer = & FILE_DATA_BUF[0]; 如果文件读写的数据的复制方式为"外部子程序",那么需要设置存放数据的缓冲区的起始地址 */ CH375vFileSize += CH375vSectorSize - 1; /* 默认情况下,以扇区方式读取数据时,无法读出文件尾部不足1个扇区的部分,所以必须临时加大文件长度以读取尾部零头 */ i = CH375FileRead( ); /* 从文件读取数据 */ CH375vFileSize -= CH375vSectorSize - 1; /* 恢复原文件长度 */ mStopIfError( i ); } i = CH375FileClose( ); /* 关闭文件 */ mStopIfError( i ); LED_RUN_INACT( );

while ( CH375DiskStatus != DISK_DISCONNECT ) xQueryInterrupt( ); /* 查询CH375中断并更新中断状态,等待U盘拔出 */ LED_OUT_INACT( ); /* LED灭 */ mDelay100mS( ); mDelay100mS( ); } }


这是我结合exam1中的程序新写的main函数 (1)我编译的时候在CH375HF6.H中 UINT8XV CH375_CMD_PORT _at_ CH375_CMD_PORT_ADDR; /* CH375命令端口的I/O地址 */ UINT8XV CH375_DAT_PORT _at_ CH375_DAT_PORT_ADDR; /* CH375数据端口的I/O地址 */ 会说CH375_CMD_PORT_ADDR和CH375_DAT_PORT_ADDR没有定义? (2)程序中用到CH375vSectorSize即u盘实际扇区大小,用CH375vSectorSize这句话就可以得到u盘的实际大小吗?


(1)对比exam1,看看CH375_CMD_PORT_ADDR和CH375_DAT_PORT_ADDR是怎么定义的 (2)是的


LED_RUN_ACT( ); /* 运行灯亮,开始操作U盘 */ mCopyCodeStringToIRAM( mCmdParam.Open.mPathName, "/MINE.TXT" ); /*mine.TXT文件名,该文件在根目录下 */ i = CH375FileOpen( ); /* 打开文件 */ if ( i == ERR_SUCCESS ) { /* 找到文件或者出错 */ mStopIfError( i ); if ( CH375vFileSize > FILE_DATA_BUF_LEN ) { /* 由于演示板用的62256只有32K字节,其中CH375子程序用512字节,所以只读取不超过63个扇区,也就是不超过32256字节 */ SecCount = FILE_DATA_BUF_LEN / CH375vSectorSize; /* 由于演示板用的62256只有32K字节,其中CH375子程序用512字节,所以只读取不超过63个扇区,也就是不超过32256字节 */ NewSize = FILE_DATA_BUF_LEN; /* 由于RAM有限所以限制长度 */ } else { /* 如果原文件较小,那么使用原长度 */ SecCount = ( CH375vFileSize + CH375vSectorSize - 1 ) / CH375vSectorSize; /* 计算文件的扇区数,因为读写是以扇区为单位的,先加CH375vSectorSize-1是为了读出文件尾部不足1个扇区的部分 */ NewSize = (UINT16)CH375vFileSize; /* 原文件的长度 */ } mCmdParam.Read.mSectorCount = SecCount; /* 读取全部数据,如果超过60个扇区则只读取60个扇区 */

CH375vFileSize += CH375vSectorSize - 1; /* 默认情况下,以扇区方式读取数据时,无法读出文件尾部不足1个扇区的部分,所以必须临时加大文件长度以读取尾部零头 */ i = CH375FileRead( ); /* 从文件读取数据 */ CH375vFileSize -= CH375vSectorSize - 1; /* 恢复原文件长度 */ mStopIfError( i ); } i = CH375FileClose( ); /* 关闭文件 */ mStopIfError( i ); LED_RUN_INACT( );


这是我组合的以扇区方式读u盘的程序,请帮我看看这个程序实现得对不对?


只有登录才能回复,可以选择微信账号登录