有关CH372或CH375的USB通讯的问题解答

我想问问用ch375实现u盘的读写程序,在调试的时候,到等待u盘连接这一部就进行不下去了,这是为什么? 请各位高手指点一下!谢谢! 非常着急!!!


首先查看U盘的供电是不是为5v,USB信号线有没有接反?VCC,D-,D+,GND. U盘接上去之后有没有中断产生?


另: (1)CH375LibInit通过没 插入U盘后,检查 (2)UD+,UD-的电压是多少 (3)INT#的电平是多少,插入U盘后,该引脚应输出低电平


非常感谢! 我还想问一下编好的程序应该烧进芯片里调试?还是有其他的方法可以调试? 刚刚开始学单片机,什么也不知道请见谅!!


可以用仿真器调试 直接烧芯片的话,那么我们的例程中提供了串口调试输出


我用的是keil仿真的,但是生不成hex文件,有用wave6000调试倒能生成hex文件,就是调试的时候等待u盘插入执行的时间很长,没法继续下去。很苦恼。


(1)Keil中生产Hex,需要设置:Project->Option for Target->Output->Create HEX (2)仿真的话,等待U盘连接时需要全速运行 (3)不知道前几贴中的方面是否检查了,另可以电话联系,那样说的清楚点:025-52638368


是不是要把电路通电看看啊!


是的.


1.用keil软件不能生成hex文件,会不会因为keil不是正版软件啊? 2.等待u盘连接即使全速也不行,我想问问这个时间要多长啊?我运行了就在那个循环里出不来啊?

下面是程序,能帮我看看吗? #include #include #include /* 以下定义适用于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 ); /* 操作失败 */ }

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


(1)CH375的2个地址及中断线定义与你的硬件是否一致 (2)设置模式后,读取操作状态,看看模式是否设置成功,若不成功,当然检测不到U盘连接 (3)对于直接读写物理扇区,请参考我们的例程,下载CH375EVT.ZIP,参考\CH375EVT\PUB\MCS51C\MISCEL


通电插入U盘后VD+ VD-的电压为3.17V 我想问问如何检查信号线有没有反应?


(1)通电插入U盘,不对U盘进行其他操作(主机不发送SOF),UD+应为3V以上,而UD-应为0V,可以用电压表测量 (2)在通讯期间可以用示波器测量,UD+、UD-上会有方波信号


非常感谢! 我想问问我已经把程序烧进单片机里面了,插上优盘后,led并不亮啊!是什么原因啊?


LED有谁驱动的? 若是375的ACT#驱动的,那么不亮表示没检测到U盘,应检查: (1)做测试命令(流程查看手册)检查写命令、写数据、读数据函数是否正确;总线则检查地址,模拟I/O则检查时许是否与手册一致 (2)主机模式设置成功没?


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