⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 ch375hft.c

📁 USB驱动芯片简介及驱动源程序
💻 C
📖 第 1 页 / 共 2 页
字号:
/* 2004.06.05
****************************************
**  Copyright  (C)  W.ch  1999-2004   **
**  Web:  http://www.winchiphead.com  **
****************************************
**  USB Host File Interface for CH375 **
**  TC2.0@PC, KC7.0@MCS51             **
****************************************
*/
/* CH375 主机文件系统接口 */
/* 支持: FAT12/FAT16 */

/* MCS-51单片机C语言的U盘文件读写示例程序, 适用于ATMEL/PHILIPS/SST等具有1KB内部RAM以及双DPTR的单片机 */
/* 该程序将U盘中的/C51/CH375HFT.C文件中的每64字节的前4个字符显示出来(显示4字符,跳过60字符,显示4字符...),
   如果找不到原文件CH375HFT.C, 那么该程序将显示C51子目录下所有以CH375开头的文件名,
   如果找不到C51子目录, 那么该程序将显示根目录下的所有文件名,
   最后新建文件"里面全0.TXT",文件内容全部是字符0,如果有ADC,可以用ADC的数据替找字符0
*/
/* CH375的INT#引脚采用查询方式处理, 磁盘数据复制方式为"双DPTR复制", 文件数据复制方式为"外部子程序",
   本例使用512字节的外部RAM作为磁盘数据缓冲区, 而没有文件数据缓冲区, 演示没有外部RAM但是单片机的内部RAM大于1K字节的应用 */


/* C51   CH375HFT.C */
/* LX51  CH375HFT.OBJ , CH375HF4.LIB */
/* OHX51 CH375HFT */

#include <reg52.h>
#include <stdio.h>

/* 以下定义的详细说明请看CH375HF4.H文件 */
#define LIB_CFG_DISK_IO			2		/* 磁盘读写的数据的复制方式,1为"单DPTR复制",2为"双DPTR复制",3为"单DPTR和P2+R0复制" */
#define LIB_CFG_FILE_IO			0		/* 文件读写的数据的复制方式,0为"外部子程序",1为"单DPTR复制",2为"双DPTR复制",3为"单DPTR和P2+R0复制" */
#define LIB_CFG_INT_EN			0		/* CH375的INT#引脚连接方式,0为"查询方式",1为"中断方式" */
/*#define LIB_CFG_FILE_IO_DEFAULT	1*/		/* 使用CH375HF4.H提供的默认"外部子程序" */

#define CH375_CMD_PORT_ADDR		0xBDF1	/* CH375命令端口的I/O地址 */
#define CH375_DAT_PORT_ADDR		0xBCF0	/* CH375数据端口的I/O地址 */
/* 只使用单片机内置的1KB外部RAM: 0000H-01FFH 为磁盘读写缓冲区, 没有文件数据缓冲区 */
#define	DISK_BASE_BUF_ADDR		0x0000	/* 外部RAM的磁盘数据缓冲区的起始地址,从该单元开始的缓冲区长度为SECTOR_SIZE */
/* 由于单片机内置的外部RAM只有1KB, 有些单片机还要去掉256字节内部RAM, 只剩下768字节的外部RAM,
   其中前512字节由CH375子程序用于磁盘数据缓冲, 由于"外部子程序"处理文件数据复制, 所以没有文件数据缓冲区 */

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

#include "..\CH375HF4.H"

UINT8X	mFileReadBuffer[ 64 ]	_at_ 0x0200;	/* 缓冲CH375读文件的一次数据传输,长度为CH375_MAX_DATA_LEN */

void xWriteToExtBuf( UINT8 mLength )	/* 该子程序由CH375的子程序库调用,用于从CH375读取文件数据到外部缓冲区,被CH375FileRead调用 */
{
	UINT8	i;
	PUINT8X	p;
	if ( mLength > 64 ) return;  /* 防止缓冲区溢出 */
	p = mFileReadBuffer;
	while ( mLength ) {  /* 根据长度读取数据,实际上长度总是CH375_MAX_DATA_LEN,也就是64 */
		*p = CH375_DAT_PORT;  /* 将U盘读出数据暂存到内部RAM缓冲区,可以用这种方式将文件数据保存到单片机的各种串行存储器中 */
		p ++;
		mLength --;
	}  /* 复制上述数据的总时间不得超过2mS */
/*	PutDataToDAC( mFileReadBuffer );  将mFileReadBuffer缓冲区中的数据输出给DAC */
	for ( i = 0; i != 4; i ++ ) { while( TI == 0 ); TI = 0; SBUF = mFileReadBuffer[ i ]; }  /* 显示前4个字符 */
}

UINT8 GetDataFromADC( )  /* 没有ADC,所以总是模拟返回数据30H */
{
/*	while ( RI == 0 ); RI = 0; return( SBUF );  通过串口获得数据或者从各种串行存储器中取出数据 */
	return( '0' );
}

void xReadFromExtBuf( UINT8 mLength )	/* 该子程序由CH375的子程序库调用,用于从外部缓冲区读取文件数据到CH375,被CH375FileWrite调用 */
{
	while ( mLength ) {  /* 根据长度写入数据,实际上长度总是CH375_MAX_DATA_LEN,也就是64 */
		CH375_DAT_PORT = GetDataFromADC( );  /* 从ADC获取一字节数据写入U盘,可以用这种方式从单片机的各种串行存储器中取出文件数据 */
		mLength --;
	}  /* 复制上述数据的总时间不得超过2mS,如果超过2mS请参考下述方法 */
}

#if	0	/* 如果xReadFromExtBuf子程序执行总时间超过1mS,那么应该用下述方法避免CH375长时间等待数据 */
UINT8X	mFileWriteBuffer[ 64 ]	_at_ 0x0240;	/* 缓冲CH375写文件的一次数据传输,长度为CH375_MAX_DATA_LEN */
void xReadFromExtBuf( UINT8 mLength )	/* 该子程序由CH375的子程序库调用,用于从外部缓冲区读取文件数据到CH375,被CH375FileWrite调用 */
{
	UINT8	i;
	while ( mLength ) CH375_DAT_PORT = mLength --;  /* 由于CH375的命令必须在2mS之内完成,所以先写入无意义数据结束前一个命令 */
	for ( i = 0; i != 64; i ++ ) mFileWriteBuffer[ i ] = GetDataFromADC( );  /* 从ADC获得数据暂存到临时缓冲区, 没有时间限制 */
	CH375_CMD_PORT = CMD_WR_USB_DATA7;  /* 重新发出写数据命令,向CH375的主机端点发送缓冲区写入数据块 */
	if ( CH375IntStatus ) CH375IntStatus = 0;  /* 仅作延时,不短于2uS */
	CH375_DAT_PORT = 64;  /* 后续数据的长度,对于写U盘必须是64 */
	for ( i = 0; i != 64; i ++ ) CH375_DAT_PORT = mFileWriteBuffer[ i ];  /* 将临时缓冲区中的数据写入CH375,覆盖前面写入的无意义数据 */
}
#endif

/* 在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显示 */
sbit P1_5  = P1^5;
/* 在P1.5连接一个LED用于监控演示程序的进度,低电平LED亮,当对U盘操作时亮 */
#define LED_RUN_ACT( )		{ P1_5 = 0; }	/* P1.5 低电平驱动LED显示 */
#define LED_RUN_INACT( )	{ P1_5 = 1; }	/* P1.5 低电平驱动LED显示 */
sbit P1_6  = P1^6;
/* 在P1.6连接一个LED用于监控演示程序的进度,低电平LED亮,当对U盘写操作时亮 */
#define LED_WR_ACT( )		{ P1_6 = 0; }	/* P1.6 低电平驱动LED显示 */
#define LED_WR_INACT( )		{ P1_6 = 1; }	/* P1.6 低电平驱动LED显示 */

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

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

/* 检查操作状态,如果错误则显示错误代码并停机 */
void	mStopIfError( UINT8 iError )
{
	if ( iError == ERR_SUCCESS ) return;  /* 操作成功 */
	printf( "Error: %02X\n", (UINT16)iError );  /* 显示错误 */
	while ( 1 ) {
		LED_OUT_ACT( );  /* LED闪烁 */
		mDelay100mS( );
		LED_OUT_INACT( );
		mDelay100mS( );
	}
}

/* 为printf和getkey输入输出初始化串口 */
void	mInitSTDIO( )
{
	SCON = 0x50;
	PCON = 0x80;
	TMOD = 0x20;
	TH1 = 0xf3;  /* 24MHz晶振, 9600bps */
	TR1 = 1;
	TI = 1;
}

main( ) {
	UINT8	i, c, SecCount;
	UINT16	NewSize;  /* 因为演示板的RAM容量只有32KB,所以NewSize限制为16位,实际上如果文件大于32256字节,应该分几次读写并且将NewSize改为UINT32以便累计 */
	UINT8	code *pCodeStr;
	LED_OUT_INIT( );
	LED_OUT_ACT( );  /* 开机后LED亮一下以示工作 */

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -