//***************************************************************************** // //! \brief something should do before the test execute of CH376 test. //! //! \return None. // //***************************************************************************** static void CH376Setup(void) { unsigned long i = 0; // // Set SysClk 72MHz using Extern 8M oscillator // //xSysCtlClockSet(72000000, xSYSCTL_OSC_MAIN|xSYSCTL_XTAL_8MHZ); printf("\n\r Please plug in USB disk first.\n\r"); mDelaymS( 100 ); // // Initialize CH376 as USB host // mInitCH376Host( ); // // Check disk connection // while ( CH376DiskConnect( ) != USB_INT_SUCCESS ) { mDelaymS( 100 ); if(++i > 100) break; } if(i == 101) { printf("\n\rDisk connection failed!\n\r"); } }
/* 主机端的主程序简单示例 */ main() { unsigned char xdata data_to_send[200]; /* 缓冲区 */ unsigned char str_to_print[]="OK, support text print\n"; unsigned char s; mDelaymS( 200 ); set_usb_mode( 6 ); /* 设置USB主机模式 */ while ( wait_interrupt()!=USB_INT_CONNECT ); /* 等待USB打印机连接上来 */ /* 如果设备端是CH341转打印口或者是CH37X,那么以下步骤是可选的,如果是其它USB芯片,那么可能需要执行以下步骤 */ #define USB_RESET_FIRST 1 /* USB规范中未要求在USB设备插入后必须复位该设备,但是计算机的WINDOWS总是这样做,所以有些USB设备也要求在插入后必须先复位才能工作 */ #ifdef USB_RESET_FIRST set_usb_mode( 7 ); /* 复位USB设备,CH375向USB信号线的D+和D-输出低电平 */ /* 如果单片机对CH375的INT#引脚采用中断方式而不是查询方式,那么应该在复制USB设备期间禁止CH375中断,在USB设备复位完成后清除CH375中断标志再允许中断 */ mDelaymS( 10 ); /* 复位时间不少于1mS,建议为10mS */ set_usb_mode( 6 ); /* 结束复位 */ mDelaymS( 100 ); while ( wait_interrupt()!=USB_INT_CONNECT ); /* 等待复位之后的设备端再次连接上来 */ #endif mDelaymS( 200 ); /* 有些USB设备要等待数百毫秒才能正常工作 */ if ( init_print()!=USB_INT_SUCCESS ) while(1); /* 错误 */ while ( 1 ) { s = get_port_status( ); if ( s!=0xFF ) { /* if ( s&0x20 ) printf("No paper!\n"); if ( (s&0x08) == 0 ) printf("Print ERROR!\n");*/ } send_data( strlen(str_to_print), str_to_print ); /* 输出数据给打印机,与通过并口方式输出一样 */ send_data( sizeof(data_to_send), data_to_send ); /* 输出的数据必须按照打印机的格式要求或者遵循打印描述语言 */ /* 可以再次继续发送数据或者接收数据 */ } }
/* 检查操作状态,如果错误则显示错误代码并停机,应该替换为实际的处理措施,例如显示错误信息,等待用户确认后重试等 */ void mStopIfError( UINT8 iError ) { if ( iError == USB_INT_SUCCESS ) return; /* 操作成功 */ printf( "Error: %02X\n", (UINT16)iError ); /* 显示错误 */ while ( 1 ) { /* LED_OUT_ACT( );*/ /* LED闪烁 */ mDelaymS( 200 ); /* LED_OUT_INACT( );*/ mDelaymS( 200 ); } }
/* 检查操作状态,如果错误则显示错误代码并停机 */ void mStopIfError( UINT8 iError ) { if ( iError == ERR_SUCCESS ) return; /* 操作成功 */ printf( "Error: %02X\n", (UINT16)iError ); /* 显示错误 */ while ( 1 ) { LED_UDISK_IN( ); /* LED闪烁 */ mDelaymS( 100 ); LED_UDISK_OUT( ); mDelaymS( 100 ); } }
//***************************************************************************** // //! \brief debugging function to show error information. //! //! \param ucError error code other function returned //! //! \details This function is intended for debugging use, error information will send to USART. //! Or you can modify this function to show more information. //! //! \return None // //***************************************************************************** void mStopIfError( UINT8 ucError ) { if ( ucError == USB_INT_SUCCESS ) return; printf( "Error: %02X\n", ucError ); while ( 1 ) { mDelaymS( 200 ); } }
/* 在操作完U盘准备让用户拔出U盘前调用, 防止用户过早拔出U盘丢失数据 */ UINT8 SafeRemoveDisk( void ) { UINT8 i, s; for ( i = 0; i < 20; i ++ ) { /* 有的U盘总是返回未准备好,不过可以被忽略 */ mDelaymS( 50 ); if ( CH376DiskMount( ) == USB_INT_SUCCESS ) break; /* 查询磁盘是否准备好 */ } mDelaymS( 5 ); s = CH376SendCmdDatWaitInt( CMD1H_SET_CONFIG, 0 ); /* 取消配置,会导致很多U盘的LED灯灭 */ mDelaymS( 5 ); if ( i < 20 && s == USB_INT_SUCCESS ) return( USB_INT_SUCCESS ); /* 操作成功 */ else return( 0xFF ); /* 以下取消SOF包会导致绝大多数U盘的LED灯灭 */ /* 如果此处取消SOF, 那么与此对应, 在检测到U盘插入后应该再恢复为模式6即恢复SOF包 */ // xWriteCH376Cmd( CMD11_SET_USB_MODE ); /* 设备USB工作模式 */ // xWriteCH376Data( 0x05 ); /* 停止发出SOF包,检测到U盘插入后必须用模式6恢复SOF包发出 */ // mDelaymS( 2 ); // xReadCH376Data( ); // xEndCH376Cmd( ); }
unsigned char set_addr( unsigned char addr ) { /* 设置设备端的USB地址 */ unsigned char status; CH375_WR_CMD_PORT( CMD_SET_ADDRESS ); /* 设置USB设备端的USB地址 */ CH375_WR_DAT_PORT( addr ); /* 地址, 从1到127之间的任意值, 常用2到20 */ status=wait_interrupt(); /* 等待CH375操作完成 */ if ( status==USB_INT_SUCCESS ) { /* 操作成功 */ CH375_WR_CMD_PORT( CMD_SET_USB_ADDR ); /* 设置USB主机端的USB地址 */ CH375_WR_DAT_PORT( addr ); /* 当目标USB设备的地址成功修改后,应该同步修改主机端的USB地址 */ } mDelaymS( 5 ); return( status ); }
uint8_t USB_Init(void) { uint8_t s; s = mInitCH376Host(USB_Flash_disk); /* 初始化CH376--USB卡模式*/ if( s != USB_INT_SUCCESS ) return 1; for ( i = 0; i < 20; i ++ ) { mDelaymS(100); if( USB_INT_SUCCESS == CH376DiskConnect( ) ) /* 检查U盘是否连接,不支持SD卡 */ { for ( i = 0; i < 10; i ++ ) { delay_us( 50000 ); s = USB_CHAK( ); //初始化磁盘并测试磁盘是否就绪. if ( s == USB_INT_SUCCESS ) /* 准备好 */ { return 0; } else if ( s == ERR_DISK_DISCON )/* 检测到断开,重新检测并计时 */ { return 3; } if ( CH376GetDiskStatus( ) >= DEF_DISK_MOUNTED && i >= 5 ) /* 有的U盘总是返回未准备好,不过可以忽略,只要其建立连接MOUNTED且尝试5*50mS */ { return 4; } } // i = CH376ReadBlock( buf ); /* 如果需要,可以读取数据块CH376_CMD_DATA.DiskMountInq,返回长度 */ // if ( i == sizeof( INQUIRY_DATA ) ) /* U盘的厂商和产品信息 */ // { // buf[ i ] = 0; // } // s = CH376DiskCapacity( (PUINT32) buf ); // s = CH376DiskQuery ( (PUINT32) buf ); /* 查询磁盘剩余空间信息,扇区数 */ } else return 2; } return 5; }
u8 Save_PerSecContent(u32 In_wr,u8 *content ,u16 saveLen) { /* NOTE : Flash 1 page = 512 Bytes ; 1 Record = 32 Bytes ; 1page= 16 Records 1Sector=8Page=128Records */ u32 pageoffset=0; //Page 偏移 u32 InPageoffset; //页内Record偏移 u16 InPageAddr=0; //页内 地址偏移 u8 reg[1]={0}; //---------------------------------------------------------------------------------------------- // 1. Judge Whether needs to Erase page pageoffset=(u32)(In_wr/7); // 计算出 Page 偏移 除以7 InPageoffset=In_wr-(u32)(pageoffset*7); // 计算出 页内偏移地址 InPageAddr=(u16)(InPageoffset*70); // 计算出页内 字节 乘以 70 (每个记录70个字节) if(((pageoffset%8)==0)&&(InPageoffset==0)) // 判断是否需要擦除Block 被移除到下一个Block 1Block=8Page { SST25V_SectorErase_4KByte((pageoffset+AvrgSpdSec_offset)*PageSIZE); // erase Sector mDelaymS(50); // rt_kprintf("\r\n Erase Cycle Block : %d\r\n",(pageoffset>>6)); } // 2. Filter write area DF_ReadFlash(pageoffset+AvrgSpdSec_offset,InPageAddr,reg,1); DF_delay_us(10); if(reg[0]!=0xff) // 如果要写入的区域 dirty ,则地址增然后从新开始寻找知道找到为止 { In_wr++; AvrgSpdPerSec_write++; if(AvrgSpdPerSec_write>=Max_SPDerSec) AvrgSpdPerSec_write=0; // rt_kprintf("\r\n ******* 每秒钟速度 写区域 Write area : %d In_wr=%d is Dirity! \r\n",AvrgSpdPerSec_write,In_wr); //-------------------------- return false; } // 3. Write Record Content DF_WriteFlashDirect(pageoffset+AvrgSpdSec_offset,InPageAddr,content,saveLen); // 写入信息 DF_delay_ms(10); // rt_kprintf("\r\n Write PerMit Starpageoffset=%d PageOffset= %d , InPageAddr= %d \r\n",AverageSpdStart_offset,pageoffset,InPageoffset); return true; //-------------------------------------------------------------------------------------------- }
//SD卡接口初始化 void SD_Init(void) { BOOL SetFlag = FALSE; s = mInitCH376Host(SD_Card); /* 初始化CH376--SD卡模式*/ if( s != 0xFA ) { for ( i = 0; i < 50; i ++ ) { mDelaymS(2); s = CH376DiskMount( ); //初始化磁盘并测试磁盘是否就绪. if ( s == USB_INT_SUCCESS ) /* 准备好 */ { break; } else if ( s == ERR_DISK_DISCON )/* 检测到断开,重新检测并计时 */ { break; } if ( CH376GetDiskStatus( ) >= DEF_DISK_MOUNTED && i >= 5 ) /* 有的U盘总是返回未准备好,不过可以忽略,只要其建立连接MOUNTED且尝试5*50mS */ { break; } } if( ! ByteLoad( "\\SD_AIR\\AIR.TXT", 0, (uint8_t *) bufread, 15 ) ) { Byte_CREAT_CON_DIR("\\SD_AIR"); //创建工作目录 ( 如果已经存在,直接打开 ) Byte_CREAT_WRITE_PATH( "\\SD_AIR\\AIR.TXT", sdinit,16 );// 不正常 重新创建 ByteFill( "\\SD_AIR\\AIR.TXT", 16, (FilePageSize_AIR - sizeof ( sdinit ) + FilePageSize_AIR )); SampleSet[Q_AIR].FileNum = 0u;SetFlag = TRUE; } else if( strncmp( (char*)sdinit, (char*)bufread, 15 ) ) //检测文件是否正常 返回零为正常 { Byte_CREAT_WRITE_PATH( "\\SD_AIR\\AIR.TXT", sdinit,16 );//非零 不正常 重新创建 ByteFill( "\\SD_AIR\\AIR.TXT", 16, ( FilePageSize_AIR - sizeof ( sdinit ) + FilePageSize_AIR )); SampleSet[Q_AIR].FileNum = 0u;SetFlag = TRUE; } if( ! ByteLoad( "\\SD_TSP\\TSP.TXT", 0, (uint8_t *) bufread, 15 ) ) { Byte_CREAT_CON_DIR("\\SD_TSP"); Byte_CREAT_WRITE_PATH( "\\SD_TSP\\TSP.TXT", sdinit,16 );// 不正常 重新创建 ByteFill( "\\SD_TSP\\TSP.TXT",16, ( FilePageSize_TSP - sizeof ( sdinit ) + FilePageSize_TSP )); SampleSet[Q_TSP].FileNum = 0;SetFlag = TRUE; } else if( strncmp( (char*)sdinit, (char*)bufread, 15 ) ) //检测文件是否正常 返回零为正常 { Byte_CREAT_WRITE_PATH( "\\SD_TSP\\TSP.TXT", sdinit,16 );//非零 不正常 重新创建 ByteFill( "\\SD_TSP\\TSP.TXT", 16, (FilePageSize_TSP - sizeof ( sdinit ) + FilePageSize_TSP )); SampleSet[Q_TSP].FileNum = 0;SetFlag = TRUE; } if( ! ByteLoad( "\\SD_R24\\R24.TXT", 0, (uint8_t *) bufread, 15 ) ) { Byte_CREAT_CON_DIR("\\SD_R24"); Byte_CREAT_WRITE_PATH( "\\SD_R24\\R24.TXT", sdinit,16 );// 不正常 重新创建 ByteFill( "\\SD_R24\\R24.TXT", 16, (FilePageSize_R24 - sizeof ( sdinit ) + FilePageSize_R24 )); SampleSet[Q_R24].FileNum = 0;SetFlag = TRUE; } else if( strncmp( (char*)sdinit, (char*)bufread, 15 ) ) //检测文件是否正常 返回零为正常 { Byte_CREAT_WRITE_PATH( "\\SD_R24\\R24.TXT", sdinit,16 );//非零 不正常 重新创建 ByteFill( "\\SD_R24\\R24.TXT", 16, (FilePageSize_R24 - sizeof ( sdinit ) + FilePageSize_R24 )); SampleSet[Q_R24].FileNum = 0;SetFlag = TRUE; } if( ! ByteLoad( "\\SD_SHI\\SHI.TXT", 0, (uint8_t *) bufread, 15 ) ) { Byte_CREAT_CON_DIR("\\SD_SHI"); Byte_CREAT_WRITE_PATH( "\\SD_SHI\\SHI.TXT", sdinit,16 );// 不正常 重新创建 ByteFill( "\\SD_SHI\\SHI.TXT", 16, (FilePageSize_SHI - sizeof ( sdinit ) + FilePageSize_SHI )); SampleSet[Q_SHI].FileNum = 0;SetFlag = TRUE; } else if( strncmp( (char*)sdinit, (char*)bufread, 15 ) ) //检测文件是否正常 返回零为正常 { Byte_CREAT_WRITE_PATH( "\\SD_SHI\\SHI.TXT", sdinit,16 );//非零 不正常 重新创建 ByteFill( "\\SD_SHI\\SHI.TXT", 16, (FilePageSize_SHI - sizeof ( sdinit ) + FilePageSize_SHI )); SampleSet[Q_SHI].FileNum = 0;SetFlag = TRUE; } if( ! ByteLoad( "\\SD_POW\\POWER.TXT", 0, (uint8_t *) bufread, 15 ) ) { Byte_CREAT_CON_DIR("\\SD_POW"); Byte_CREAT_WRITE_PATH( "\\SD_POW\\POWER.TXT", sdinit,16 );// 不正常 重新创建 ByteFill( "\\SD_POW\\POWER.TXT", 16, ( 100 + 4 - sizeof ( sdinit ) )); } else if( strncmp( (char*)sdinit, (char*)bufread, 15 ) ) //检测文件是否正常 返回零为正常 { Byte_CREAT_WRITE_PATH( "\\SD_POW\\POWER.TXT", sdinit,16 );//非零 不正常 重新创建 ByteFill( "\\SD_POW\\POWER.TXT", 16, (100 + 4 - sizeof ( sdinit ) )); } if(SetFlag == TRUE ) { SampleSetSave(); SetFlag = FALSE; } } }
void host( ) { UINT8 i, c, TotalCount; UINT8 code *pCodeStr; UINT16 EnumCount; i = CH375LibInit( ); /* 初始化CH375程序库和CH375芯片,操作成功返回0 */ mStopIfError( i ); while ( 1 ) { printf( "Insert USB disk\n" ); while ( CH375DiskStatus < DISK_CONNECT ) { /* 等待U盘插入 */ if ( IsKeyPress( ) ) { /* 有键按下 */ printf( "Exit USB host mode\n" ); return; } /* if ( CH375_INT_WIRE == 0 ) xQueryInterrupt( );*/ /* 如果CH375中断,那么查询CH375中断并更新中断状态,可以改成中断方式 */ mDelaymS( 100 ); /* 没必要频繁查询 */ if ( CH375DiskConnect( ) == ERR_SUCCESS ) break; /* 查询方式: 检查磁盘是否连接,返回成功说明连接 */ } LED_UDISK_IN( ); /* LED亮 */ mDelaymS( 250 ); /* 延时,可选操作,有的USB存储器需要几十毫秒的延时 */ /* 检查U盘是否准备好,但是某些U盘必须要执行这一步才能工作 */ for ( i = 0; i < 5; i ++ ) { /* 有的U盘总是返回未准备好,不过可以被忽略 */ mDelaymS( 100 ); printf( "Ready ?\n" ); // if ( CH375DiskReady( ) == ERR_SUCCESS ) break; /* 查询磁盘是否准备好,不支持CH375S,节约代码空间 */ if ( CH375sDiskReady( ) == ERR_SUCCESS ) break; /* 查询磁盘是否准备好,支持CH375S和CH375A,但占用更多的代码空间 */ } /* 读取原文件 */ printf( "Open\n" ); mCopyCodeStringToIRAM( mCmdParam.Open.mPathName, "/C51/CH375HFT.C" ); /* 文件名,该文件在C51子目录下 */ i = CH375FileOpen( ); /* 打开文件 */ if ( i == ERR_MISS_DIR || i == ERR_MISS_FILE ) { /* 没有找到C51子目录,没有找到CH375HFT.C文件 */ /* 列出文件 */ if ( i == ERR_MISS_DIR ) pCodeStr = "/*"; /* C51子目录不存在则列出根目录下的所有文件 */ else pCodeStr = "/C51/CH375*"; /* CH375HFT.C文件不存在则列出\C51子目录下的以CH375开头的文件 */ printf( "List file %s\n", pCodeStr ); for ( EnumCount = 0; EnumCount < 10000; EnumCount ++ ) { /* 最多搜索前10000个文件,实际上没有限制 */ i = mCopyCodeStringToIRAM( mCmdParam.Open.mPathName, pCodeStr ); /* 搜索文件名,*为通配符,适用于所有文件或者子目录 */ mCmdParam.Open.mPathName[ i ] = 0xFF; /* 根据字符串长度将结束符替换为搜索的序号,从0到254,如果是0xFF即255则说明搜索序号在CH375vFileSize变量中 */ CH375vFileSize = EnumCount; /* 指定搜索/枚举的序号 */ i = CH375FileOpen( ); /* 打开文件,如果文件名中含有通配符*,则为搜索文件而不打开 */ /* CH375FileEnum 与 CH375FileOpen 的唯一区别是当后者返回ERR_FOUND_NAME时那么对应于前者返回ERR_SUCCESS */ if ( i == ERR_MISS_FILE ) break; /* 再也搜索不到匹配的文件,已经没有匹配的文件名 */ if ( i == ERR_FOUND_NAME ) { /* 搜索到与通配符相匹配的文件名,文件名及其完整路径在命令缓冲区中 */ printf( " match file %04d#: %s\n", (unsigned int)EnumCount, mCmdParam.Open.mPathName ); /* 显示序号和搜索到的匹配文件名或者子目录名 */ continue; /* 继续搜索下一个匹配的文件名,下次搜索时序号会加1 */ } else { /* 出错 */ mStopIfError( i ); break; } } } else { /* 找到文件或者出错 */ mStopIfError( i ); TotalCount = 200; /* 准备读取总长度 */ printf( "从文件中读出的前%d个字符是:\n",(UINT16)TotalCount ); while ( TotalCount ) { /* 如果文件比较大,一次读不完,可以再调用CH375ByteRead继续读取,文件指针自动向后移动 */ if ( TotalCount > MAX_BYTE_IO ) c = MAX_BYTE_IO; /* 剩余数据较多,限制单次读写的长度不能超过 sizeof( mCmdParam.ByteRead.mByteBuffer ) */ else c = TotalCount; /* 最后剩余的字节数 */ mCmdParam.ByteRead.mByteCount = c; /* 请求读出几十字节数据 */ i = CH375ByteRead( ); /* 以字节为单位读取数据块,单次读写的长度不能超过MAX_BYTE_IO,第二次调用时接着刚才的向后读 */ mStopIfError( i ); TotalCount -= mCmdParam.ByteRead.mByteCount; /* 计数,减去当前实际已经读出的字符数 */ for ( i=0; i!=mCmdParam.ByteRead.mByteCount; i++ ) printf( "%C", mCmdParam.ByteRead.mByteBuffer[i] ); /* 显示读出的字符 */ if ( mCmdParam.ByteRead.mByteCount < c ) { /* 实际读出的字符数少于要求读出的字符数,说明已经到文件的结尾 */ printf( "\n" ); printf( "文件已经结束\n" ); break; } } /* 如果希望从指定位置开始读写,可以移动文件指针 mCmdParam.ByteLocate.mByteOffset = 608; 跳过文件的前608个字节开始读写 CH375ByteLocate( ); mCmdParam.ByteRead.mByteCount = 5; 读取5个字节 CH375ByteRead( ); 直接读取文件的第608个字节到612个字节数据,前608个字节被跳过 如果希望将新数据添加到原文件的尾部,可以移动文件指针 CH375FileOpen( ); mCmdParam.ByteLocate.mByteOffset = 0xffffffff; 移到文件的尾部 CH375ByteLocate( ); mCmdParam.ByteWrite.mByteCount = 13; 写入13个字节的数据 CH375ByteWrite( ); 在原文件的后面添加数据,新加的13个字节接着原文件的尾部放置 mCmdParam.ByteWrite.mByteCount = 2; 写入2个字节的数据 CH375ByteWrite( ); 继续在原文件的后面添加数据 mCmdParam.ByteWrite.mByteCount = 0; 写入0个字节的数据,实际上该操作用于通知程序库更新文件长度 CH375ByteWrite( ); 写入0字节的数据,用于自动更新文件的长度,所以文件长度增加15,如果不这样做,那么执行CH375FileClose时也会自动更新文件长度 */ printf( "Close\n" ); i = CH375FileClose( ); /* 关闭文件 */ mStopIfError( i ); } #ifdef EN_DISK_WRITE /* 子程序库支持写操作 */ /* 产生新文件(覆盖原文件数据),关于向原有文件中添加数据的例子请参考EXAM7和EXAM8 */ LED_WR_NOW( ); /* 写操作 */ printf( "Create\n" ); mCopyCodeStringToIRAM( mCmdParam.Create.mPathName, "/NEWFILE.TXT" ); /* 新文件名,在根目录下,中文文件名 */ i = CH375FileCreate( ); /* 新建文件并打开,如果文件已经存在则先删除后再新建 */ mStopIfError( i ); printf( "Write\n" ); pCodeStr = "Note: \xd\xa这个程序是以字节为单位进行U盘文件读写,单片机只需要有600字节的RAM\xd\xa"; while( 1 ) { /* 分多次写入文件数据 */ for ( i=0; i<MAX_BYTE_IO; i++ ) { c = *pCodeStr; mCmdParam.ByteWrite.mByteBuffer[i] = c; if ( c == 0 ) break; /* 源字符串结束 */ pCodeStr++; } if ( i == 0 ) break; /* 源字符串结束,完成写文件 */ mCmdParam.ByteWrite.mByteCount = i; /* 写入数据的字符数,单次读写的长度不能超过MAX_BYTE_IO,第二次调用时接着刚才的向后写 */ i = CH375ByteWrite( ); /* 向文件写入数据 */ mStopIfError( i ); } printf( "Close\n" ); mCmdParam.Close.mUpdateLen = 1; /* 自动计算文件长度,以字节为单位写文件,建议让程序库关闭文件以便自动更新文件长度 */ i = CH375FileClose( ); mStopIfError( i ); LED_NOT_WR( ); #endif printf( "Take out USB disk\n" ); // while ( CH375DiskStatus != DISK_DISCONNECT ) xQueryInterrupt( ); /* 查询CH375中断并更新中断状态,等待U盘拔出 */ while ( CH375DiskStatus >= DISK_CONNECT ) { /* 查询CH375中断并更新中断状态,等待U盘拔出 */ if ( CH375DiskConnect( ) != ERR_SUCCESS ) break; mDelaymS( 100 ); } LED_UDISK_OUT( ); /* LED灭 */ mDelaymS( 100 ); } }
main( ) { UINT8 s; mDelaymS( 100 ); /* 延时100毫秒 */ mInitSTDIO( ); /* 为了让计算机通过串口监控演示过程 */ printf( "Start\n" ); s = mInitCH376Host( ); /* 初始化CH376 */ mStopIfError( s ); /* 其它电路初始化 */ while ( 1 ) { printf( "Wait Udisk\n" ); // while ( CH376DiskConnect( ) != USB_INT_SUCCESS ) { /* 检查U盘是否连接,等待U盘插入,对于SD卡,可以由单片机直接查询SD卡座的插拔状态引脚 */ // mDelaymS( 100 ); // } while ( 1 ) { /* 与前几行的方法不同,此处是等到中断通知后再查询USB设备连接 */ if ( Query376Interrupt( ) ) { /* 查询CH376中断(INT#引脚为低电平) */ if ( CH376DiskConnect( ) == USB_INT_SUCCESS ) break; /* USB设备连接 */ } mDelaymS( 100 ); } mDelaymS( 200 ); /* 延时,可选操作,有的USB存储器需要几十毫秒的延时 */ /* 对于检测到USB设备的,最多等待10*50mS */ for ( s = 0; s < 10; s ++ ) { /* 最长等待时间,10*50mS */ mDelaymS( 50 ); printf( "Ready ?\n" ); if ( CH376DiskMount( ) == USB_INT_SUCCESS ) break; /* 初始化磁盘并测试磁盘是否就绪 */ } s = CH376ReadBlock( buf ); /* 如果需要,可以读取数据块CH376_CMD_DATA.DiskMountInq,返回长度 */ if ( s == sizeof( INQUIRY_DATA ) ) { /* U盘的厂商和产品信息 */ buf[ s ] = 0; printf( "UdiskInfo: %s\n", ((P_INQUIRY_DATA)buf) -> VendorIdStr ); } printf( "Check Disk Write Protect ? ...\n" ); s = IsDiskWriteProtect( ); /* 检查U盘是否写保护, 返回USB_INT_SUCCESS说明可以写,返回0xFF说明只读/写保护,返回其它值说明是错误代码 */ if ( s != USB_INT_SUCCESS && s != 0xFF ) { /* 操作失败 */ printf( "Again ...\n" ); mDelaymS( 100 ); s = IsDiskWriteProtect( ); /* 再试一次 */ } if ( s == USB_INT_SUCCESS ) { /* 可以写 */ printf( "... No !\n" ); printf( "Create a File\n" ); s = CH376FileCreate( "\\NEWFILE.TXT" ); /* 在根目录下新建文件并打开,如果文件已经存在则先删除后再新建 */ mStopIfError( s ); buf[0] = 'O'; buf[1] = 'K'; s = CH376ByteWrite( buf, 2, NULL ); /* 向文件写入数据 */ mStopIfError( s ); printf( "Close\n" ); s = CH376FileClose( TRUE ); /* 自动计算文件长度 */ mStopIfError( s ); if ( SafeRemoveDisk( ) == USB_INT_SUCCESS ) { /* 安全移除U盘 */ printf( "Safe Remove !\n" ); } else { printf( "Unsafe Remove !\n" ); } } else if ( s == 0xFF ) { /* 写保护 */ printf( "... Yes !\n" ); } else { mStopIfError( s ); /* 显示错误代码 */ } printf( "Take out\n" ); // while ( CH376DiskConnect( ) == USB_INT_SUCCESS ) { /* 检查U盘是否连接,等待U盘拔出 */ // mDelaymS( 100 ); // } while ( 1 ) { /* 与前几行的方法不同,此处是等到中断通知后再查询USB设备连接 */ if ( Query376Interrupt( ) ) { /* 查询CH376中断(INT#引脚为低电平) */ if ( CH376DiskConnect( ) != USB_INT_SUCCESS ) break; /* USB设备断开 */ } mDelaymS( 100 ); } mDelaymS( 200 ); } }
void host( ) { UINT8 i, s; UINT8 TotalCount; UINT16 RealCount; P_FAT_DIR_INFO pDir; s = mInitCH376Host( ); /* 初始化CH376 */ mStopIfError( s ); /* 其它电路初始化 */ while ( 1 ) { printf( "Wait Udisk/SD\n" ); while ( CH376DiskConnect( ) != USB_INT_SUCCESS ) { /* 检查U盘是否连接,等待U盘插入,对于SD卡,可以由单片机直接查询SD卡座的插拔状态引脚 */ if ( IsKeyPress( ) ) { /* 有键按下 */ printf( "Exit USB host mode\n" ); return; } mDelaymS( 100 ); /* 没必要频繁查询 */ } LED_UDISK_IN( ); /* LED亮 */ mDelaymS( 200 ); /* 延时,可选操作,有的USB存储器需要几十毫秒的延时 */ /* 对于检测到USB设备的,最多等待100*50mS,主要针对有些MP3太慢,对于检测到USB设备并且连接DISK_MOUNTED的,最多等待5*50mS,主要针对DiskReady不过的 */ for ( i = 0; i < 100; i ++ ) { /* 最长等待时间,100*50mS */ mDelaymS( 50 ); printf( "Ready ?\n" ); s = CH376DiskMount( ); /* 初始化磁盘并测试磁盘是否就绪 */ if ( s == USB_INT_SUCCESS ) break; /* 准备好 */ else if ( s == ERR_DISK_DISCON ) break; /* 检测到断开,重新检测并计时 */ if ( CH376GetDiskStatus( ) >= DEF_DISK_MOUNTED && i >= 5 ) break; /* 有的U盘总是返回未准备好,不过可以忽略,只要其建立连接MOUNTED且尝试5*50mS */ } if ( s == ERR_DISK_DISCON ) { /* 检测到断开,重新检测并计时 */ printf( "Device gone\n" ); continue; } if ( CH376GetDiskStatus( ) < DEF_DISK_MOUNTED ) { /* 未知USB设备,例如USB键盘、打印机等 */ printf( "Unknown device\n" ); goto UnknownUsbDevice; } i = CH376ReadBlock( buf ); /* 如果需要,可以读取数据块CH376_CMD_DATA.DiskMountInq,返回长度 */ if ( i == sizeof( INQUIRY_DATA ) ) { /* U盘的厂商和产品信息 */ buf[ i ] = 0; printf( "UdiskInfo: %s\n", ((P_INQUIRY_DATA)buf) -> VendorIdStr ); } /* 读取原文件 */ printf( "Open\n" ); strcpy( buf, "\\C51\\CH376HFT.C" ); /* 源文件名,多级目录下的文件名和路径名必须复制到RAM中再处理,而根目录或者当前目录下的文件名可以在RAM或者ROM中 */ s = CH376FileOpenPath( buf ); /* 打开文件,该文件在C51子目录下 */ if ( s == ERR_MISS_DIR || s == ERR_MISS_FILE ) { /* 没有找到目录或者没有找到文件 */ /* 列出文件,完整枚举可以参考EXAM13全盘枚举 */ if ( s == ERR_MISS_DIR ) strcpy( buf, "\\*" ); /* C51子目录不存在则列出根目录下的文件 */ else strcpy( buf, "\\C51\\CH376*" ); /* CH376HFT.C文件不存在则列出\C51子目录下的以CH376开头的文件 */ printf( "List file %s\n", buf ); s = CH376FileOpenPath( buf ); /* 枚举多级目录下的文件或者目录,输入缓冲区必须在RAM中 */ while ( s == USB_INT_DISK_READ ) { /* 枚举到匹配的文件 */ CH376ReadBlock( buf ); /* 读取枚举到的文件的FAT_DIR_INFO结构,返回长度总是sizeof( FAT_DIR_INFO ) */ pDir = (P_FAT_DIR_INFO)buf; /* 当前文件目录信息 */ if ( pDir -> DIR_Name[0] != '.' ) { /* 不是本级或者上级目录名则继续,否则必须丢弃不处理 */ if ( pDir -> DIR_Name[0] == 0x05 ) pDir -> DIR_Name[0] = 0xE5; /* 特殊字符替换 */ pDir -> DIR_Attr = 0; /* 强制文件名字符串结束以便打印输出 */ printf( "*** EnumName: %s\n", pDir -> DIR_Name ); /* 打印名称,原始8+3格式,未整理成含小数点分隔符 */ } xWriteCH376Cmd( CMD0H_FILE_ENUM_GO ); /* 继续枚举文件和目录 */ xEndCH376Cmd( ); s = Wait376Interrupt( ); } if ( s != ERR_MISS_FILE ) mStopIfError( s ); /* 操作出错 */ } else { /* 找到文件或者出错 */ mStopIfError( s ); TotalCount = 200; /* 准备读取总长度 */ printf( "从文件中读出的前%d个字符是:\n",(UINT16)TotalCount ); while ( TotalCount ) { /* 如果文件比较大,一次读不完,可以再调用CH376ByteRead继续读取,文件指针自动向后移动 */ if ( TotalCount > sizeof(buf) ) i = sizeof(buf); /* 剩余数据较多,限制单次读写的长度不能超过缓冲区大小 */ else i = TotalCount; /* 最后剩余的字节数 */ s = CH376ByteRead( buf, i, &RealCount ); /* 以字节为单位读取数据块,单次读写的长度不能超过缓冲区大小,第二次调用时接着刚才的向后读 */ mStopIfError( s ); TotalCount -= (UINT8)RealCount; /* 计数,减去当前实际已经读出的字符数 */ for ( s=0; s!=RealCount; s++ ) printf( "%C", buf[s] ); /* 显示读出的字符 */ if ( RealCount < i ) { /* 实际读出的字符数少于要求读出的字符数,说明已经到文件的结尾 */ printf( "\n" ); printf( "文件已经结束\n" ); break; } } printf( "Close\n" ); s = CH376FileClose( FALSE ); /* 关闭文件 */ mStopIfError( s ); } UnknownUsbDevice: printf( "Take out\n" ); while ( CH376DiskConnect( ) == USB_INT_SUCCESS ) { /* 检查U盘是否连接,等待U盘拔出 */ mDelaymS( 100 ); } LED_UDISK_OUT( ); /* LED灭 */ mDelaymS( 100 ); } }
//***************************************************************************** // //! \brief CH376 test execute main body. //! //! \return None. // //***************************************************************************** static void CH376Execute(void) { unsigned char s; UINT16 usDataCnt; unsigned int ulFileSize, ulSetcorCnt; //waiting for USB device get ready for ( s = 0; s < 10; s ++ ) { mDelaymS( 50 ); printf( "Ready ?\n\r" ); // // Initialize disk and check if disk is ready // if ( CH376DiskMount( ) == USB_INT_SUCCESS ) break; } s = CH376ReadBlock( ucReadData ); TestAssert((s == sizeof( INQUIRY_DATA )), "CH376 API \"CH376ReadBlock()\"error!"); if ( s == sizeof( INQUIRY_DATA ) ) { // // get device information // ucReadData[ s ] = 0; printf( "UdiskInfo: %s\n\r", ((P_INQUIRY_DATA)ucReadData)->VendorIdStr); } s = CH376DiskCapacity(&ulSetcorCnt); TestAssert((s == USB_INT_SUCCESS), "CH376 API \"CH376DiskCapacity()\"error!"); printf("Capacity of U-disk: %dMB\n\r", ulSetcorCnt / (1048576 / DEF_SECTOR_SIZE)); s = CH376DiskQuery(&ulSetcorCnt); TestAssert((s == USB_INT_SUCCESS), "CH376 API \"CH376DiskCapacity()\"error!"); printf("Available capacity: %dMB\n\r", ulSetcorCnt / (1048576 / DEF_SECTOR_SIZE)); s = CH376FileOpenPath("/CH376TST/TST.TXT"); if ( s != USB_INT_SUCCESS ) { // // if file path not found, firstly create the folder then create the file // s = CH376DirCreate("/CH376TST"); TestAssert((s == USB_INT_SUCCESS), "CH376 API \"CH376DirCreate()\"error!"); s = CH376FileCreate("TST.TXT"); TestAssert((s == USB_INT_SUCCESS), "CH376 API \"CH376FileCreate()\"error!"); s = CH376ByteWrite(ucWriteData, 35, &usDataCnt); TestAssert((s == USB_INT_SUCCESS), "CH376 API \"CH376ByteWrite()\"error!"); // // After modifying a file, the parameter of CH376FileClose() must be TRUE to update // file size or you will find the file not available when you open the file on computer. // CH376FileClose(TRUE); } else { // // if file found, move file pointer to the end of file, then append data to the file // CH376ByteLocate(0x0); s = CH376ByteWrite(ucWriteData, 35, &usDataCnt); TestAssert((s == USB_INT_SUCCESS), "CH376 API \"CH376ByteWrite()\"error!"); // // enable auto update file size // CH376FileClose(TRUE); } CH376FileOpenPath("/CH376TST/TST.TXT"); ulFileSize = CH376GetFileSize(); CH376ByteRead(ucReadData, 512, &usDataCnt); for(s = 0; s < ulFileSize; s++) { TestAssert((ucReadData[s] == ucWriteData[s]), "CH376 API \"CH376ByteRead()\"error!"); } CH376FileClose(FALSE); printf("\n\r All test over!\n\r"); }