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

📄 ch375ev3.c

📁 USB驱动芯片简介及驱动源程序
💻 C
字号:
/* 2004.03.05
****************************************
**  Copyright  (C)  W.ch  1999-2004   **
**  Web:  http://www.winchiphead.com  **
****************************************
**  USB 1.1 Host Examples for CH375   **
**  KC7.0@MCS-51                      **
****************************************
*/
/* CH375作为USB主机接口的程序示例,中断方式 */

/* MCS-51单片机C语言的示例程序, U盘数据读写 */

#include <reg51.h>
#include <string.h>
#include <stdio.h>

#ifndef	TRUE
#define	TRUE	1
#define	FALSE	0
#endif

/* 定义CH375命令代码及返回状态 */
#include "CH375INC.H"

/* 以下定义适用于MCS-51单片机,其它单片机参照修改,为了提供C语言的速度需要对本程序进行优化 */
#include <reg51.h>
unsigned char volatile xdata	CH375_CMD_PORT _at_ 0xBDF1;	/* CH375命令端口的I/O地址 */
unsigned char volatile xdata	CH375_DAT_PORT _at_ 0xBCF0;	/* CH375数据端口的I/O地址 */
unsigned char xdata				DATA_BUFFER[8192] _at_ 0x0000;	/* 外部RAM数据缓冲区的起始地址,长度不少于一次读写的数据长度 */

/* 在P1.4连接一个LED用于监控演示程序的进度,低电平LED亮,当U盘插入后亮 */
sbit P1_4  = P1^4;
#define LED_OUT_ACT( )		{ P1_4 = 0; }	/* P1.4 低电平驱动LED显示 */
#define LED_OUT_INACT( )	{ P1_4 = 1; }	/* P1.4 低电平驱动LED显示 */

unsigned char volatile			UdiskStatus;	/* 当前U盘状态,定义如下 */
#define		STATUS_DISCONNECT		0	/* U盘尚未连接或者已经断开 */
#define		STATUS_CONNECT			1	/* U盘刚刚连接 */
#define		STATUS_ERROR			2	/* U盘操作错误或者不支持 */
#define		STATUS_WAIT				3	/* U盘正在操作 */
#define		STATUS_READY			4	/* U盘准备好接受操作 */

/*
  下面是USB移动存储设备的简单应用参考。在单片机应用程序中,
  可以定义USB存储设备的4种状态:已断开、已连接、初始化、可读写。其中"已连接"状态可选。
● 已断开是指USB存储设备已经断开(从USB插座中拔出),完全不可用;
● 已连接是指USB存储设备已经连接(插入USB插座中),但是尚未初始化;
● 初始化是指USB存储设备正在初始化或者初始化失败,所以不可以进行读写;
● 可读写是指USB存储设备初始化成功,可以进行数据读写。

单片机处理CH375中断的一般步骤如下:
㈠ CH375将INT#引脚设置为低电平,向单片机请求中断;
㈡ 单片机进入中断服务程序,首先执行GET_STATUS命令获取中断状态;
㈢ CH375在GET_STATUS命令完成后将INT#引脚恢复为高电平,取消中断请求;
㈣ 单片机可以参考下面的顺序分析GET_STATUS命令获取的中断状态:
  ⑴ 如果中断状态是USB_INT_DISCONNECT,则说明USB设备已经断开,置USB存储设备状态为"已断开",
     如果在此之前USB存储设备有读写操作尚未完成,则放弃操作并按操作失败处理,然后退出中断,必要时可以通知单片机主程序。
  ⑵ 如果中断状态是USB_INT_CONNECT,则说明USB设备已经连接,置USB存储设备状态为"已连接",
     接着发出DISK_INIT命令,并置USB存储设备状态为初始化,然后退出中断,必要时可以通知单片机主程序。
  ⑶ 如果当前的USB存储设备状态是"初始化":
    ① 如果中断状态是USB_INT_SUCCESS,则说明USB存储设备初始化成功,置USB存储设备状态为"可读写",
       然后退出中断,必要时可以通知单片机主程序。
    ② 如果中断状态是其它状态,则说明USB存储设备初始化失败,应该通知单片机主程序,
       提示该USB设备不是存储设备或者该USB设备不支持,然后退出中断。或者,单片机通过其它命令自行处理该USB存储设备的通讯协议。
  ⑷ 如果当前的USB存储设备状态是"可读写",则可以保存中断状态并通知单片机主程序处理,
     然后直接退出中断;或者,在中断服务程序中继续处理:
    ① 如果中断状态是USB_INT_DISK_READ,则说明正在进行USB存储设备的读操作,需要取走64个字节的数据,
       可以发出RD_USB_DATA命令取走数据,接着再发出DISK_RD_GO命令使CH375继续读,然后退出中断。
    ② 如果中断状态是USB_INT_DISK_WRITE,则说明正在进行USB存储设备的写操作,需要提供64个字节的数据,
       可以发出WR_USB_DATA7命令提供数据,接着再发出DISK_WR_GO命令使CH375继续写,然后退出中断。
    ③ 如果中断状态是USB_INT_SUCCESS,则说明读写操作成功,直接退出中断并通知主程序该操作成功;
    ④ 如果中断状态是USB_INT_DISK_ERR,则说明读写操作失败,直接退出中断并通知主程序该操作失败;
    ⑤ 通常不会返回其它中断状态,如果有,则说明出现错误。
  ⑸ 通常不会在其它USB存储设备状态下返回其它中断状态,如果有,则说明出现错误,可以参照USB设备断开的状态处理。

当单片机主程序需要从USB存储设备中读写数据时,可以查询USB存储设备状态,
如果是"可读写"状态,则可以发出DISK_READ命令读数据,或者发出DISK_WRITE命令写数据。
后续的数据读写过程可以在单片机的中断服务程序中完成,也可以由中断程序置标志通知等待中的主程序,
由主程序分析中断状态再完成数据读写过程。
*/

unsigned char *mBufferPoint;

/* 延时2微秒,不精确 */
void	delay2us( )
{
	unsigned char i;
	for ( i = 2; i != 0; i -- );
}

/* 延时1微秒,不精确 */
void	delay1us( )
{
	unsigned char i;
	for ( i = 1; i != 0; i -- );
}

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

/* 基本操作 */

void CH375_WR_CMD_PORT( unsigned char cmd ) {  /* 向CH375的命令端口写入命令,周期不小于4uS,如果单片机较快则延时 */
	delay2us();
	CH375_CMD_PORT=cmd;
	delay2us();
}

void CH375_WR_DAT_PORT( unsigned char dat ) {  /* 向CH375的数据端口写入数据,周期不小于1.5uS,如果单片机较快则延时 */
	CH375_DAT_PORT=dat;
	delay1us();  /* 因为MCS51单片机较慢所以实际上无需延时 */
}

unsigned char CH375_RD_DAT_PORT() {  /* 从CH375的数据端口读出数据,周期不小于1.5uS,如果单片机较快则延时 */
	delay1us();  /* 因为MCS51单片机较慢所以实际上无需延时 */
	return( CH375_DAT_PORT );
}

/* CH375的INT#引脚连接51单片机的INT0引脚, 采用中断方式 */
/* CH375中断服务程序,使用寄存器组1 */
void	CH375Interrupt( ) interrupt 0 using 1
{
	unsigned char i, s, len;
	CH375_WR_CMD_PORT( CMD_GET_STATUS );  /* 获取中断状态并取消中断请求 */
	for ( i = 2; i != 0; i -- );  /* 至少延时2uS */
	s = CH375_RD_DAT_PORT( );  /* 获取中断状态 */
	if ( s == USB_INT_SUCCESS ) UdiskStatus = STATUS_READY;  /* 操作成功 */
	else if ( s == USB_INT_DISCONNECT ) {
		UdiskStatus = STATUS_DISCONNECT;  /* 检测到USB设备断开事件 */
		LED_OUT_INACT( );
	}
	else if ( s == USB_INT_CONNECT ) {
		UdiskStatus = STATUS_CONNECT;  /* 检测到USB设备连接事件 */
		LED_OUT_ACT( );
	}
	else if ( s == USB_INT_DISK_READ ) {  /* USB存储器读数据块,请求数据读出 */
		CH375_WR_CMD_PORT( CMD_RD_USB_DATA );  /* 从CH375缓冲区读取数据块 */
		for ( i = 2; i != 0; i -- );  /* 至少延时2uS */
		len = CH375_RD_DAT_PORT( );  /* 后续数据的长度 */
		while ( len ) {  /* 根据长度读取数据 */
			*mBufferPoint = CH375_RD_DAT_PORT( );  /* 读出数据并保存 */
			mBufferPoint ++;
			len --;
		}
		CH375_WR_CMD_PORT( CMD_DISK_RD_GO );  /* 继续执行USB存储器的读操作 */
	}
	else if ( s == USB_INT_DISK_WRITE ) {  /* USB存储器写数据块,请求数据写入 */
		CH375_WR_CMD_PORT( CMD_WR_USB_DATA7 );  /* 向CH375缓冲区写入数据块 */
		for ( i = 2; i != 0; i -- );  /* 至少延时2uS */
		len = CH375_MAX_DATA_LEN;
		CH375_WR_DAT_PORT( len );  /* 后续数据的长度 */
		do {  /* 对于C51,这个DO+WHILE结构比上面的WHILE效率高,速度快 */
			CH375_WR_DAT_PORT( *mBufferPoint );
			mBufferPoint ++;
		} while ( -- len );
		CH375_WR_CMD_PORT( CMD_DISK_WR_GO );  /* 继续执行USB存储器的写操作 */
	}
	else {  /* 操作失败 */
		UdiskStatus = STATUS_ERROR;
	}
/*	CH375_INT_FLAG = 0;  清中断标志 */
}

/* 设置CH375为USB主机方式 */
unsigned char	mCH375Init( )
{
	unsigned char	i;
	UdiskStatus = STATUS_DISCONNECT;
	CH375_WR_CMD_PORT( CMD_SET_USB_MODE );  /* 设置USB工作模式 */
	CH375_WR_DAT_PORT( 6 );  /* 模式代码,自动检测USB设备连接 */
	for ( i = 0xff; i != 0; i -- ) {  /* 等待操作成功,通常需要等待10uS-20uS */
		if ( CH375_RD_DAT_PORT( ) == CMD_RET_SUCCESS ) break;  /* 操作成功 */
	}
	if ( i != 0 ) return( TRUE );  /* 操作成功 */
	else return( FALSE );  /* CH375出错,例如芯片型号错或者处于串口方式或者不支持 */
}

/* 等待U盘准备好或者等待上次操作结束 */
unsigned char	mWaitReady( )
{
	while( UdiskStatus == STATUS_WAIT );  /* 等待上次操作结束 */
	if ( UdiskStatus == STATUS_READY ) return( TRUE );  /* U盘已经准备好操作,上次操作成功 */
	else if ( UdiskStatus == STATUS_ERROR ) {  /* 上次操作失败 */
		UdiskStatus = STATUS_WAIT;
		CH375_WR_CMD_PORT( CMD_DISK_RESET );  /* 复位U盘 */
		while( UdiskStatus == STATUS_WAIT );  /* 等待中断状态 */
		if ( UdiskStatus == STATUS_READY ) return( TRUE );  /* 等待中断状态,操作成功 */
		return( FALSE );  /* UdiskStatus = STATUS_ERROR */
	}
	else if ( UdiskStatus == STATUS_DISCONNECT ) {  /* U盘已经断开 */
/*		while ( UdiskStatus == STATUS_DISCONNECT );*/
		return( FALSE );
	}
	else if ( UdiskStatus == STATUS_CONNECT ) {  /* U盘已经连接 */
		mDelay100mS( );  /* 如果U盘刚插入那么最好稍等一下再操作 */
		mDelay100mS( );
		CH375_WR_CMD_PORT( CMD_DISK_INIT );  /* 初始化USB存储器 */
		while ( UdiskStatus == STATUS_CONNECT );  /* 等待中断状态 */
		if ( UdiskStatus == STATUS_READY ) {  /* 操作成功,磁盘已经初始化并安装到系统中 */
/* 检查U盘是否准备好,大多数U盘不需要这一步,但是某些U盘必须要执行这一步才能工作 */
//	do {
//		mDelay100mS( );
//		printf( "Disk Ready ?\n" );
//		i = CH375DiskReady( );  /* 查询磁盘是否准备好,如果省掉这个子程序可以节约将近1KB的程序代码 */
//	} while ( i != ERR_SUCCESS );
/* CH375DiskReady 在CH375的U盘文件子程序库中,因为代码较多,所以此处省去 */
			return( TRUE );
		}
		else return( FALSE );
	}
	return( FALSE );
}

/* 从U盘读取多个扇区的数据块到缓冲区 */
unsigned char	mReadSector( unsigned long iLbaStart, unsigned char iSectorCount, unsigned char *iBuffer )
/* iLbaStart 是准备读取的线性起始扇区号, iSectorCount 是准备读取的扇区数, iBuffer 是存放返回数据的缓冲区的起址 */
{
	if ( mWaitReady( ) == FALSE ) return( FALSE );
	mBufferPoint = iBuffer;  /* 指向缓冲区起始地址 */
	UdiskStatus = STATUS_WAIT;
	CH375_WR_CMD_PORT( CMD_DISK_READ );  /* 从USB存储器读数据块 */
	CH375_WR_DAT_PORT( (unsigned char)iLbaStart );  /* LBA的最低8位 */
	CH375_WR_DAT_PORT( (unsigned char)( iLbaStart >> 8 ) );
	CH375_WR_DAT_PORT( (unsigned char)( iLbaStart >> 16 ) );
	CH375_WR_DAT_PORT( (unsigned char)( iLbaStart >> 24 ) );  /* LBA的最高8位 */
	CH375_WR_DAT_PORT( iSectorCount );  /* 扇区数 */
/* 如果是纯中断方式,那么可以做其它事情,中断服务程序会读取数据 */
	return( mWaitReady( ) );
}

/* 将缓冲区中的多个扇区的数据块写入U盘 */
unsigned char	mWriteSector( unsigned long iLbaStart, unsigned char iSectorCount, unsigned char *iBuffer )
/* iLbaStart 是写入的线起始性扇区号, iSectorCount 是写入的扇区数, iBuffer 是存放准备写入数据的缓冲区的起址 */
{
	if ( mWaitReady( ) == FALSE ) return( FALSE );
	mBufferPoint = iBuffer;  /* 指向缓冲区起始地址 */
	UdiskStatus = STATUS_WAIT;
	CH375_WR_CMD_PORT( CMD_DISK_WRITE );  /* 向USB存储器写数据块 */
	CH375_WR_DAT_PORT( (unsigned char)iLbaStart );  /* LBA的最低8位 */
	CH375_WR_DAT_PORT( (unsigned char)( iLbaStart >> 8 ) );
	CH375_WR_DAT_PORT( (unsigned char)( iLbaStart >> 16 ) );
	CH375_WR_DAT_PORT( (unsigned char)( iLbaStart >> 24 ) );  /* LBA的最高8位 */
	CH375_WR_DAT_PORT( iSectorCount );  /* 扇区数 */
/* 如果是纯中断方式,那么可以做其它事情,中断服务程序会读取数据 */
	return( mWaitReady( ) );
}

struct _HD_MBR_DPT {
	unsigned char	PartState;
	unsigned char	StartHead;
	unsigned int	StartSec;
	unsigned char	PartType;
	unsigned char	EndHead;
	unsigned int	EndSec;
	unsigned long	StartSector;
	unsigned long	TotalSector;
};

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

main( ) {
	unsigned char	c;
	LED_OUT_ACT( );  /* 开机后LED亮一下以示工作 */
	mDelay100mS( );  /* 延时100毫秒 */
	LED_OUT_INACT( );
	mInitSTDIO( );
	printf( "Start\n" );
	c = mCH375Init( );  /* 初始化CH375 */
	if ( c == FALSE ) printf( "Error @CH375Init\n" );
	printf( "Insert USB disk\n" );
	while ( 1 ) {  /* 主程序 */
		mDelay100mS( );
		mDelay100mS( );
		mDelay100mS( );
		mDelay100mS( );
		mDelay100mS( );
/* 做其它工作 */
/* 如果打算从U盘读取数据 */
		printf( "Read\n" );
		c = mReadSector( 0, 5, DATA_BUFFER );
		if ( c == FALSE ) printf( "Error @ReadSector\n" );
/* 如果打算向U盘写入数据 */
		c = mWriteSector( 1, 1, DATA_BUFFER );
		if ( c == FALSE ) printf( "Error @WriteSector\n" );
/* 做其它工作 */
		mDelay100mS( );
		mDelay100mS( );
		mDelay100mS( );
		mDelay100mS( );
		mDelay100mS( );
	}
}

⌨️ 快捷键说明

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