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

📄 ch375ev4.c

📁 CH375这个经典的芯片
💻 C
字号:
#include <stdio.h>
#include "CH375INC.H"		/* 定义CH375命令代码及返回状态 */
#include <reg51.h>			/* 以下定义适用于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] _at_ 0x0000;	/* 外部RAM数据缓冲区的起始地址,长度不少于一次读写的数据长度 */

UINT32	DiskStart;		/* 逻辑盘的起始绝对扇区号LBA */
UINT8	SecPerClus;		/* 逻辑盘的每簇扇区数 */
UINT8	RsvdSecCnt;		/* 逻辑盘的保留扇区数 */
UINT16	FATSz16;		/* FAT16逻辑盘的FAT表占用的扇区数 */

/* ********** 硬件USB接口层,无论如何这层省不掉,单片机总要与CH375接口吧 ************************************************************ */

void	mDelaymS( UINT8 delay ) {		/* 以毫秒为单位延时,不精确,适用于24MHz时钟MCS51 */
	UINT8	i, j, c;
	for ( i = delay; i != 0; i -- ) {
		for ( j = 200; j != 0; j -- ) c += 3;  /* 在24MHz时钟下延时500uS */
		for ( j = 200; j != 0; j -- ) c += 3;  /* 在24MHz时钟下延时500uS */
	}
}

void CH375_WR_CMD_PORT( UINT8 cmd ) {	/* 向CH375的命令端口写入命令,周期不小于4uS,如果单片机较快则延时 */
	CH375_CMD_PORT=cmd;
	for ( cmd = 2; cmd != 0; cmd -- );	/* 发出命令码前后应该各延时2uS,对于MCS51可以不需要延时 */
}
void CH375_WR_DAT_PORT( UINT8 dat ) {	/* 向CH375的数据端口写入数据,周期不小于1.5uS,如果单片机较快则延时 */
	CH375_DAT_PORT=dat;					/* 因为MCS51单片机较慢所以实际上无需延时 */
}
UINT8 CH375_RD_DAT_PORT( void ) {		/* 从CH375的数据端口读出数据,周期不小于1.5uS,如果单片机较快则延时 */
	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,i,j=0;
	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 );  /* 出现错误 */
	while(1){j++;
		CH375_WR_CMD_PORT( CMD_DISK_SIZE );  /* 获取USB存储器的容量 */
		Status = mWaitInterrupt( );  /* 等待中断并获取状态 */
		if ( Status == USB_INT_SUCCESS ) break;  /* 出现错误 */
/*这里需要加上这个之后才可以,正确的做法也是这样*/		CH375_WR_CMD_PORT( 	CMD_DISK_R_SENSE );  /* 获取USB存储器的容量 */
			mDelaymS( 250 );
		if(j==5) return(Status);
	}
	Status = mWaitInterrupt( );  /* 等待中断并获取状态 */
	if ( Status == USB_INT_SUCCESS ){  /* 出现错误 */
	for(i=0;i!=5;i++){ printf("Ready\n");
	CH375_WR_CMD_PORT( CMD_DISK_READY );  /* 获取USB存储器的容量 */
	Status = mWaitInterrupt( );  /* 等待中断并获取状态 */
	if ( Status == USB_INT_SUCCESS ) break;  /* 出现错误 */
	CH375_WR_CMD_PORT( 	CMD_DISK_R_SENSE );  /* 获取USB存储器的容量 */
	mDelaymS( 250 );
		}
	}
	return( 0 );  /* U盘已经成功初始化 */
}


UINT8	mReadSector( UINT32 iLbaStart, UINT8 iSectorCount, UINT8X *oDataBuffer ) {		/* 从U盘读取数据块到缓冲区 */
/* iLbaStart 起始扇区号, iSectorCount 扇区数, 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 ) {  /* 等待中断并获取状态,USB存储器读数据块,请求数据读出 */
			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 );  /* 操作失败 */
}
/* 将缓冲区中的多个扇区的数据块写入U盘 */
unsigned char	mWriteSector( unsigned long iLbaStart, unsigned char iSectorCount,unsigned char *mBufferPoint )
/* iLbaStart 是写入的线起始性扇区号, iSectorCount 是写入的扇区数 */
{
	unsigned char mIntStatus;
	unsigned int  mBlockCount;
	unsigned char mLength;
	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 );  /* 扇区数 */
//	mBufferPoint = DISK_BUFFER;  /* 指向缓冲区起始地址 */
	for ( mBlockCount = iSectorCount *8; mBlockCount != 0; mBlockCount -- ) {  /* 数据块计数 */
		mIntStatus = mWaitInterrupt( );  /* 等待中断并获取状态 */
		if ( mIntStatus == USB_INT_DISK_WRITE ) {  /* USB存储器写数据块,请求数据写入 */
			CH375_WR_CMD_PORT( CMD_WR_USB_DATA7 );  /* 向CH375缓冲区写入数据块 */
			mLength = 64;
			CH375_WR_DAT_PORT( mLength );  /* 后续数据的长度 */
			do { 
				CH375_WR_DAT_PORT( *mBufferPoint );
				mBufferPoint ++;
			} while ( -- mLength );
			CH375_WR_CMD_PORT( CMD_DISK_WR_GO );  /* 继续执行USB存储器的写操作 */
		}
		else break;  /* 返回错误状态 */
	}
	if ( mBlockCount == 0 ) {
		mIntStatus = mWaitInterrupt( );  /* 等待中断并获取状态 */
		if ( mIntStatus == USB_INT_SUCCESS ) return( 0 );  /* 操作成功 */
	}
	return( mIntStatus );  /* 操作失败 */
}


/* ********** FAT文件系统层,这层程序量实际较大,不过,该程序仅演示极简单的功能,所以精简 ************************************************************ */

UINT16	mGetPointWord( UINT8X *iAddr ) {	/* 获取字数据,因为MCS51是大端格式,U盘FAT通常是小端格式,所以转换 */
	return( iAddr[0] | (UINT16)iAddr[1] << 8 );
}

UINT8	mIdenDisk( void ) {		/* 识别分析当前逻辑盘 */
	UINT8	Status;
	DiskStart = 0;  /* 以下是非常简单的FAT文件系统的分析,正式应用绝对不应该如此简单,否则兼容性和容错性差 */
	Status = mReadSector( 0, 1, DISK_BUFFER );  /* 读取逻辑盘引导信息 */printf("1\n");
	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 );  /* 读取簇号所在的FAT扇区 */
	if ( Status != 0 ) return( 0 );  /* 错误 */
	return( mGetPointWord( &DISK_BUFFER[ ( iCluster + iCluster ) & 0x01FF ] ) );  /* 取原簇链接 */
}

UINT32	mClusterToLba( UINT16 iCluster ) {	/* 将簇号转换为绝对LBA扇区地址 */
	return( DiskStart + RsvdSecCnt + FATSz16 + FATSz16 + 32 + ( iCluster - 2 ) * SecPerClus );  /* 将簇号转换为LBA,得当前操作的起始LBA */
}

void	mInitSTDIO( void ) {	/* 仅用于调试用途及显示内容到PC机,与该程序功能完全无关,为printf和getkey输入输出初始化串口 */
	SCON = 0x50; PCON = 0x80; TMOD = 0x20; TH1 = 0xf4; TR1 = 1; TI = 1;  /* 24MHz晶振, 9600bps */
}
void	mStopIfError( UINT8 iErrCode ) {	/* 如果错误则停止运行并显示错误状态,正式应用还需要分析处理 */
	if ( iErrCode == 0 ) return;
	printf( "Error status, %02X\n", (UINT16)iErrCode );
	while(1);
}

main( ) {
	UINT8	Status;
	UINT16	len,i;
	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" );
//again:	
	while ( mWaitInterrupt( ) != USB_INT_CONNECT );  /* 等待U盘连接 */
		mDelaymS( 250 );  /* 延时等待U盘进入正常工作状态 */
		mDelaymS( 250 );  /* 延时等待U盘进入正常工作状态 */
		mDelaymS( 250 );  /* 延时等待U盘进入正常工作状态 */
		mDelaymS( 250 );  /* 延时等待U盘进入正常工作状态 */
		mDelaymS( 250 );  /* 延时等待U盘进入正常工作状态 */
		Status = mInitDisk( );  /* 初始化U盘,实际是识别U盘的类型,不影响U盘中的数据,在所有读写操作之前必须进行此步骤 */
		printf("status=%02x\n",(unsigned short)Status);
//		mStopIfError( Status );
//		if(Status==0x16)
//		goto again;
//		Status = mIdenDisk( );  /* 识别分析U盘文件系统,必要操作 */
//		mStopIfError( Status );
		Status = mReadSector( 63, 1, DISK_BUFFER );  /* 读取FAT16逻辑盘的根目录,通常根目录占用32个扇区 */
		for(len=0;len!=512;len++)
			printf("%02x ",(unsigned short)DISK_BUFFER[len]);
			printf("\n");
		for(i=0;i!=512;i++)
			DISK_BUFFER[i]=0;
		for(i=0;i!=10000;i++){
		len = mWriteSector(i,1,DISK_BUFFER);
		printf("i=%02x\n",(unsigned short)i);
		}
		printf("take out\n");
		while(mWaitInterrupt( )!=USB_INT_DISCONNECT);
		}
}
 

⌨️ 快捷键说明

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