📄 hostdisk.c
字号:
/* 获取磁盘特性 */
UINT8 mDiskInquiry( PUINT8 DataBuf )
{
#ifdef BIG_ENDIAN
mBOC.mCBW.mCBW_DataLen = 0x24000000;
#else
mBOC.mCBW.mCBW_DataLen = 0x00000024;
#endif
mBOC.mCBW.mCBW_Flag = 0x80;
mBOC.mCBW.mCBW_CB_Len = 6;
mBOC.mCBW.mCBW_CB_Buf[0] = 0x12; /* 命令码 */
mBOC.mCBW.mCBW_CB_Buf[1] = 0x00;
mBOC.mCBW.mCBW_CB_Buf[2] = 0x00;
mBOC.mCBW.mCBW_CB_Buf[3] = 0x00;
mBOC.mCBW.mCBW_CB_Buf[4] = 0x24;
mBOC.mCBW.mCBW_CB_Buf[5] = 0x00;
return( mBulkOnlyCmd( DataBuf ) ); /* 执行基于BulkOnly协议的命令 */
}
/* 获取磁盘容量 */
UINT8 mDiskCapacity( PUINT8 DataBuf )
{
#ifdef BIG_ENDIAN
mBOC.mCBW.mCBW_DataLen = 0x08000000;
#else
mBOC.mCBW.mCBW_DataLen = 0x00000008;
#endif
mBOC.mCBW.mCBW_Flag = 0x80;
mBOC.mCBW.mCBW_CB_Len = 10;
mBOC.mCBW.mCBW_CB_Buf[0] = 0x25; /* 命令码 */
mBOC.mCBW.mCBW_CB_Buf[1] = 0x00;
mBOC.mCBW.mCBW_CB_Buf[2] = 0x00;
mBOC.mCBW.mCBW_CB_Buf[3] = 0x00;
mBOC.mCBW.mCBW_CB_Buf[4] = 0x00;
mBOC.mCBW.mCBW_CB_Buf[5] = 0x00;
mBOC.mCBW.mCBW_CB_Buf[6] = 0x00;
mBOC.mCBW.mCBW_CB_Buf[7] = 0x00;
mBOC.mCBW.mCBW_CB_Buf[8] = 0x00;
mBOC.mCBW.mCBW_CB_Buf[9] = 0x00;
return( mBulkOnlyCmd( DataBuf ) ); /* 执行基于BulkOnly协议的命令 */
}
/* 测试磁盘是否就绪 */
UINT8 mDiskTestReady( void )
{
mBOC.mCBW.mCBW_DataLen = 0;
mBOC.mCBW.mCBW_Flag = 0x00;
mBOC.mCBW.mCBW_CB_Len = 6;
mBOC.mCBW.mCBW_CB_Buf[0] = 0x00; /* 命令码 */
mBOC.mCBW.mCBW_CB_Buf[1] = 0x00;
mBOC.mCBW.mCBW_CB_Buf[2] = 0x00;
mBOC.mCBW.mCBW_CB_Buf[3] = 0x00;
mBOC.mCBW.mCBW_CB_Buf[4] = 0x00;
mBOC.mCBW.mCBW_CB_Buf[5] = 0x00;
return( mBulkOnlyCmd( NULL ) ); /* 执行基于BulkOnly协议的命令 */
}
/* 以扇区为单位从磁盘读取数据 */
UINT8 mReadSector( UINT32 StartLba, UINT8 SectCount, PUINT8 DataBuf )
{
UINT32 len;
len = (UINT32)SectCount << 9;
#ifdef BIG_ENDIAN
mBOC.mCBW.mCBW_DataLen = mSwapEndian( len );
#else
mBOC.mCBW.mCBW_DataLen = len;
#endif
mBOC.mCBW.mCBW_Flag = 0x80;
mBOC.mCBW.mCBW_CB_Len = 10;
mBOC.mCBW.mCBW_CB_Buf[0] = 0x28; /* 命令码 */
mBOC.mCBW.mCBW_CB_Buf[1] = 0x00;
mBOC.mCBW.mCBW_CB_Buf[2] = (UINT8)( StartLba >> 24 );
mBOC.mCBW.mCBW_CB_Buf[3] = (UINT8)( StartLba >> 16 );
mBOC.mCBW.mCBW_CB_Buf[4] = (UINT8)( StartLba >> 8 );
mBOC.mCBW.mCBW_CB_Buf[5] = (UINT8)( StartLba );
mBOC.mCBW.mCBW_CB_Buf[6] = 0x00;
mBOC.mCBW.mCBW_CB_Buf[7] = 0x00;
mBOC.mCBW.mCBW_CB_Buf[8] = SectCount;
mBOC.mCBW.mCBW_CB_Buf[9] = 0x00;
return( mBulkOnlyCmd( DataBuf ) ); /* 执行基于BulkOnly协议的命令 */
}
/* 以扇区为单位将数据写入磁盘 */
UINT8 mWriteSector( UINT32 StartLba, UINT8 SectCount, PUINT8 DataBuf )
{
UINT32 len;
len = (UINT32)SectCount << 9;
#ifdef BIG_ENDIAN
mBOC.mCBW.mCBW_DataLen = mSwapEndian( len );
#else
mBOC.mCBW.mCBW_DataLen = len;
#endif
mBOC.mCBW.mCBW_Flag = 0x00;
mBOC.mCBW.mCBW_CB_Len = 10;
mBOC.mCBW.mCBW_CB_Buf[0] = 0x2A; /* 命令码 */
mBOC.mCBW.mCBW_CB_Buf[1] = 0x00;
mBOC.mCBW.mCBW_CB_Buf[2] = (UINT8)( StartLba >> 24 );
mBOC.mCBW.mCBW_CB_Buf[3] = (UINT8)( StartLba >> 16 );
mBOC.mCBW.mCBW_CB_Buf[4] = (UINT8)( StartLba >> 8 );
mBOC.mCBW.mCBW_CB_Buf[5] = (UINT8)( StartLba );
mBOC.mCBW.mCBW_CB_Buf[6] = 0x00;
mBOC.mCBW.mCBW_CB_Buf[7] = 0x00;
mBOC.mCBW.mCBW_CB_Buf[8] = SectCount;
mBOC.mCBW.mCBW_CB_Buf[9] = 0x00;
return( mBulkOnlyCmd( DataBuf ) ); /* 执行基于BulkOnly协议的命令 */
}
/* ********** 主程序 ********** */
/* 为printf和getkey输入输出初始化串口 */
void mInitSTDIO( )
{
SCON = 0x50;
PCON = 0x80;
TL2 = RCAP2L = 0 - 13; /* 24MHz晶振, 57600bps */
TH2 = RCAP2H = 0xFF;
T2CON = 0x34; /* 定时器2用于串口的波特率发生器 */
TI = 1;
}
int main( void ) // USB host
{
UINT8 i, s;
UINT8 idata buf[64];
UINT8 xdata DISK_BUF[512];
// P1&=0xF8; // 如果在U盘文件读写模块上试用本程序必须加上本行
mDelaymS( 50 ); // 等待CH374复位完成
CH374_PORT_INIT( ); /* CH374接口初始化 */
mInitSTDIO( ); /* 为了让计算机通过串口监控演示过程 */
printf( "Start CH374 Host\n" );
Init374Host( ); // 初始化USB主机
while ( 1 ) {
HostSetBusFree( ); // 设定USB主机空闲
printf( "Wait Device In\n" );
while ( 1 ) {
if ( Query374Interrupt( ) ) HostDetectInterrupt( ); // 如果有USB主机中断则处理
if ( Query374DeviceIn( ) ) break; // 有USB设备
}
mDelaymS( 200 ); // 由于USB设备刚插入尚未稳定,故等待USB设备数百毫秒,消除插拔抖动
if ( Query374Interrupt( ) ) HostDetectInterrupt( ); // 如果有USB主机中断则处理
printf( "Reset Device\n" );
HostSetBusReset( ); // USB总线复位
for ( i = 0; i < 100; i ++ ) { // 等待USB设备复位后重新连接
if ( Query374DeviceIn( ) ) break; // 有USB设备
mDelaymS( 1 );
}
if ( Query374Interrupt( ) ) HostDetectInterrupt( ); // 如果有USB主机中断则处理
if ( Query374DeviceIn( ) ) { // 有USB设备
if ( Query374DevFullSpeed( ) ) {
printf( "Start Full-Speed Device\n" );
HostSetFullSpeed( ); // 检测到全速USB设备
}
else {
printf( "Start Low-Speed Device\n" );
HostSetLowSpeed( ); // 检测到低速USB设备
}
}
else {
printf( "Device gone !\n" );
continue; // 设备已经断开,继续等待
}
mDelaymS( 20 );
printf( "GetDeviceDescr: " );
s = GetDeviceDescr( buf ); // 获取设备描述符
if ( s != USB_INT_SUCCESS ) {
printf( "ERROR = %02X\n", (UINT16)s );
goto WaitDeviceOut; // 终止操作,等待USB设备拔出
}
for ( i = 0; i < ( (PUSB_SETUP_REQ)SetupGetDevDescr ) -> wLengthL; i ++ ) printf( "%02X ", (UINT16)( buf[i] ) );
printf( "\n" );
printf( "SetUsbAddress: " );
s = SetUsbAddress( 0x02 ); // 设置USB设备地址
if ( s != USB_INT_SUCCESS ) {
printf( "ERROR = %02X\n", (UINT16)s );
goto WaitDeviceOut; // 终止操作,等待USB设备拔出
}
printf( "\n" );
printf( "GetConfigDescr: " );
s = GetConfigDescr( buf ); // 获取配置描述符
if ( s != USB_INT_SUCCESS ) {
printf( "ERROR = %02X\n", (UINT16)s );
goto WaitDeviceOut; // 终止操作,等待USB设备拔出
}
for ( i = 0; i < ( (PUSB_CFG_DESCR)buf ) -> wTotalLengthL; i ++ ) printf( "%02X ", (UINT16)( buf[i] ) );
printf( "\n" );
/* 分析配置描述符,获取端点数据/各端点地址/各端点大小等,更新变量endp_addr和endp_size等 */
mDiskBulkInEndp = 0;
mDiskBulkOutEndp = 0;
for ( i = 0; i < 2; i ++ ) { /* 分析前两个端点 */
if ( ( (PUSB_CFG_DESCR_LONG)buf ) -> endp_descr[ i ].wMaxPacketSize == 0x40 && ( (PUSB_CFG_DESCR_LONG)buf ) -> endp_descr[ i ].bmAttributes == 2 ) { /* 64字节长度的批量端点 */
if ( ( (PUSB_CFG_DESCR_LONG)buf ) -> endp_descr[ i ].bEndpointAddress & 0x80 ) mDiskBulkInEndp = ( (PUSB_CFG_DESCR_LONG)buf ) -> endp_descr[ i ].bEndpointAddress & 0x0F; /* IN端点 */
else mDiskBulkOutEndp = ( (PUSB_CFG_DESCR_LONG)buf ) -> endp_descr[ i ].bEndpointAddress & 0x0F; /* OUT端点 */
}
}
if ( ( (PUSB_CFG_DESCR_LONG)buf ) -> itf_descr.bInterfaceClass != 0x08 || mDiskBulkInEndp == 0 || mDiskBulkOutEndp == 0 ) { /* 不是USB存储类设备,不支持 */
printf( "Not USB Mass Storage Device\n" );
goto WaitDeviceOut; // 终止操作,等待USB设备拔出
}
printf( "SetUsbConfig: " );
s = SetUsbConfig( ( (PUSB_CFG_DESCR)buf ) -> bConfigurationValue ); // 设置USB设备配置
if ( s != USB_INT_SUCCESS ) {
printf( "ERROR = %02X\n", (UINT16)s );
goto WaitDeviceOut; // 终止操作,等待USB设备拔出
}
printf( "\n" );
mSaveDevEndpTog = 0x00; // 清同步标志
printf( "Disk Inquiry: " );
s = mDiskInquiry( buf ); /* 获取磁盘特性 */
if ( s != USB_INT_SUCCESS ) {
printf( "ERROR = %02X\n", (UINT16)s );
goto WaitDeviceOut; // 终止操作,等待USB设备拔出
}
for ( i = 0; i < 8; i ++ ) printf( "%02X ", (UINT16)( buf[i] ) );
printf( ", " );
for ( i = 8; i < 36; i ++ ) printf( "%c", buf[i] );
printf( "\n" );
mDelaymS( 100 );
printf( "Disk Capacity: " );
s = mDiskCapacity( buf ); /* 获取磁盘容量 */
if ( s != USB_INT_SUCCESS ) {
printf( "ERROR = %02X\n", (UINT16)s );
}
else {
for ( i = 0; i < 8; i ++ ) printf( "%02X ", (UINT16)( buf[i] ) );
printf( ", %3d MB\n", (UINT16)( ( (UINT32)( buf[1] ) << 16 | (UINT16)( buf[2] ) << 8 | buf[3] ) >> 11 ) );
}
printf( "Disk Ready: " );
s = mDiskTestReady( ); /* 测试磁盘是否就绪 */
if ( s != USB_INT_SUCCESS ) {
printf( "ERROR = %02X\n", (UINT16)s );
}
else printf( "\n" );
printf( "Disk Read First Sector: " );
s = mReadSector( 0x00000000, 1, DISK_BUF ); /* 以扇区为单位从磁盘读取数据 */
if ( s != USB_INT_SUCCESS ) {
printf( "ERROR = %02X\n", (UINT16)s );
goto WaitDeviceOut; // 终止操作,等待USB设备拔出
}
for ( i = 0; i < 16; i ++ ) printf( "%02X ", (UINT16)( DISK_BUF[i] ) );
printf( "\n" );
DISK_BUF[0] ^= 0xFF;
DISK_BUF[1] ^= 0xFF;
DISK_BUF[510] ^= 0xFF;
DISK_BUF[511] ^= 0xFF;
printf( "Disk Write Second Sector: " );
s = mWriteSector( 0x00000001, 1, DISK_BUF ); /* 以扇区为单位将数据写入磁盘 */
if ( s != USB_INT_SUCCESS ) {
printf( "ERROR = %02X\n", (UINT16)s );
goto WaitDeviceOut; // 终止操作,等待USB设备拔出
}
for ( i = 0; i < 16; i ++ ) printf( "%02X ", (UINT16)( DISK_BUF[i] ) );
printf( "\n" );
WaitDeviceOut: // 等待USB设备拔出
printf( "Wait Device Out\n" );
while ( 1 ) {
if ( Query374Interrupt( ) ) HostDetectInterrupt( ); // 如果有USB主机中断则处理
if ( Query374DeviceIn( ) == FALSE ) break; // 没有USB设备
}
mDelaymS( 100 ); // 等待设备完全断开,消除插拔抖动
if ( Query374DeviceIn( ) ) goto WaitDeviceOut; // 没有完全断开
// HostSetBusFree( ); // 设定USB主机空闲,主要目的是关闭SOF
}
}
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -