//***************************************************************************** // //! \brief Peripheral Initialization. //! //! \param None. //! //! This function is to Initialize joystick, LED, USART1, CH376, USB flash disk and VS1003. //! //! \return None. // //***************************************************************************** void PeriphInit() { unsigned char s; JoystickInit(); xSysCtlPeripheralEnable(xGPIOSPinToPeripheralId(LED_PIN)); xGPIOSPinDirModeSet(LED_PIN, xGPIO_DIR_MODE_OUT); xGPIOSPinWrite( LED_PIN, 0 ); mInitSTDIO( ); s = mInitCH376Host( ); mStopIfError( s ); while ( CH376DiskConnect( ) != USB_INT_SUCCESS ) { xSysCtlDelay( 1000000 ); printf("Waiting for USB flash disk to plug in...\n\r"); } xSysCtlDelay( 2000000 ); for ( s = 0; s < 10; s ++ ) { xSysCtlDelay( 500000 ); if ( CH376DiskMount( ) == USB_INT_SUCCESS ) break; } VS10xxInit(); }
int main( void ) { int i; int j; UINT8 res; UINT8 s; UINT8 NameBuf[20] = {0}; UINT8 CountT = 0; UINT32 U_D; PUINT32 pU_D; pU_D = &U_D; // Stop watchdog timer to prevent time out reset WDTCTL = WDTPW + WDTHOLD; P1DIR_bit.P1DIR0 = 1; P1OUT_bit.P1OUT0 = 1; UartInit(); code_start: CountT ++; for(i = 0;i<30000;i++); printk("Code Start !\r\n"); printk("程序开始 !\r\n"); do{ for(i = 0;i<30000;i++); res = mInitCH376Host(); }while(res != USB_INT_SUCCESS); printk("USB_INT_SUCCESS !\r\n"); do{ for(i = 0;i<30000;i++); res = CH376DiskConnect( ); /* 查询U盘是否连接,返回USB_INT_SUCCESS则说明当前已连接 */ }while(res!=USB_INT_SUCCESS); printk(" Connect SUCCESS\r\n"); do{ for(i = 0;i<30000;i++); res = CH376DiskMount( ); /* 查询U盘或SD卡是否准备好,有些U盘可能需多次调用才能成功 */ }while(res!=USB_INT_SUCCESS); printk(" U is OK !\r\n"); s = CH376DiskCapacity(pU_D); mStopIfError( s ); printk(" U盘大小:"); printHex((unsigned int)U_D); printHex((unsigned int)(U_D>>16)); printk("\r\n"); s = CH376DiskQuery(pU_D); mStopIfError( s ); printk(" 剩余大小:"); printHex((unsigned int)U_D); printHex((unsigned int)(U_D>>16)); printk("\r\n"); strcpy( NameBuf, "\\2012.TXT" ); /* 目标文件名 */ #if 0 printf( "Open\r\n" ); s = CH376FileOpenPath( NameBuf ); mStopIfError( s ); if ( s == ERR_MISS_FILE ){ printf( "Create\r\n" ); s = CH376FileCreatePath( NameBuf ); /* 新建多级目录下的文件,支持多级目录路径,输入缓冲区必须在RAM中 */ mStopIfError( s ); } printf( "CH376ByteLocate\r\n" ); s = CH376ByteLocate(0xFFFFFFFF); mStopIfError( s ); printf( "Write\r\n" ); strcpy( buf, "时间 数据 ---\xd\n" ); s = CH376ByteWrite( buf, strlen(buf), NULL ); /* 以字节为单位向当前位置写入数据块 */ mStopIfError( s ); strcpy( buf, "2012-12-01 01 ---\xd\n" ); buf[9] = '0' + CountT; for(i = 0; i < 100;i ++){ buf[12] = '0' + i/10; buf[13] = '0' + i%10; printf( "Write %s" ,buf); s = CH376ByteWrite( buf, strlen(buf), NULL ); /* 以字节为单位向当前位置写入数据块 */ mStopIfError( s ); } printf( "Close\r\n" ); s = CH376FileClose( TRUE ); /* 关闭文件,对于字节读写建议自动更新文件长度 */ mStopIfError( s ); #endif /* 如果MY_ADC.TXT文件已经存在则添加数据到尾部,如果不存在则新建文件 */ printf( "Open\n" ); s = CH376FileOpen( NameBuf ); /* 打开文件,该文件在根目录下 */ if ( s == USB_INT_SUCCESS ) { /* 文件存在并且已经被打开,移动文件指针到尾部以便添加数据 */ printf( "File size = %ld\n", CH376GetFileSize( ) ); /* 读取当前文件长度 */ printf( "Locate tail\n" ); s = CH376ByteLocate( 0xFFFFFFFF ); /* 移到文件的尾部 */ mStopIfError( s ); } else if ( s == ERR_MISS_FILE ) { /* 没有找到文件,必须新建文件 */ printf( "Create\n" ); s = CH376FileCreate( NULL ); /* 新建文件并打开,如果文件已经存在则先删除后再新建,不必再提供文件名,刚才已经提供给CH376FileOpen */ mStopIfError( s ); } else mStopIfError( s ); /* 打开文件时出错 */ printf( "Write begin\n" ); s = sprintf( buf, "此前文件长度= %ld 字节\xd\xa", CH376GetFileSize( ) ); /* 注意字符串长度不能溢出buf,否则加大缓冲区或者分多次写入 */ s = CH376ByteWrite( buf, s, NULL ); /* 以字节为单位向文件写入数据 */ mStopIfError( s ); printf( "Write ADC data\n" ); printf( "Current offset ( file point ) is : "); for(i = 0;i<1000;i++){ s = sprintf( buf, "%02d.%02d.%02d.%d\xd\xa", 2012 + i/365, 1 + (i / 30) % 12, 1 + i % 30, i ); /* 将二制制数据格式为一行字符串 */ s = CH376ByteWrite( buf, s, NULL ); /* 以字节为单位向文件写入数据 */ /* 有些U盘可能会要求在写数据后等待一会才能继续操作, 所以,如果在某些U盘中发生数据丢失现象, 建议在每次写入数据后稍作延时再继续 */ for(j = 0;j<30;j++); mStopIfError( s ); printf( "\b\b\b\b\b\b" ); printf( "%6ld", CH376ReadVar32( VAR_CURRENT_OFFSET ) ); /* 读取当前文件指针 */ } printf( "Write end\n" ); strcpy( buf, "今天的ADC数据到此结束\xd\xa" ); s = CH376ByteWrite( buf, strlen( buf ), NULL ); /* 以字节为单位向文件写入数据 */ mStopIfError( s ); /* 如果实际产品中有实时时钟,可以根据需要将文件的日期和时间修改为实际值,参考EXAM10用CH376DirInfoRead/CH376DirInfoSave实现 */ printf( "Close\n" ); s = CH376FileClose( TRUE ); /* 关闭文件,自动计算文件长度,以字节为单位写文件,建议让程序库关闭文件以便自动更新文件长度 */ mStopIfError( s ); do{ for(i = 0;i<30000;i++); P1OUT_bit.P1OUT0 = 1; for(i = 0;i<30000;i++); P1OUT_bit.P1OUT0 = 0; res = CH376DiskConnect( ); /* 查询U盘是否连接,返回USB_INT_SUCCESS则说明当前已连接 */ }while(res!=ERR_DISK_DISCON); printk(" ERR_DISK_DISCON\r\n"); goto code_start; }
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 VS1003 Example. //! //! \param None. //! //! This function is to give an example of using VS10xx to play audio file. //! we use VS1003 as the decoder. This chip can decode WAV,MP3,WMA,MIDI audio //! files and also can record audio. But we don't give out the record functions //! since it relates much to file manipulation and seldom used. If you need this //! function, please refer to VS10xx datasheet on VLSI's official page. //! //! \return None. // //***************************************************************************** void VS10xxExample(void) { unsigned char i,s; unsigned short usDataCnt, usAudioDecTime = 0; unsigned int ulFileSize = 0, ulFileCursor = 0; AudioFileInfo tFLIF; xSysCtlClockSet(72000000, xSYSCTL_OSC_MAIN | xSYSCTL_XTAL_8MHZ); xSysCtlDelay( 1000000 ); PeriphInit(); VS10xxSineTest(0x24, 5000000); VS10xxSineTest(0x85, 5000000); // // Open MP3 file // s = CH376FileOpen("/01.MP3"); mStopIfError(s); // // Set start volume, balance and Bass Treble Enhancement // VS10xxSetVolume( 77 ); VS10xxSetBalance( 0 ); VS10xxBassTrebleEnhance(0, 12, 0, 10); // // Get file size. (Optional) // ulFileSize = CH376GetFileSize( ); // // Single song loop // while( 1 ) { CH376ByteRead( ucBuff, 512, &usDataCnt ); // // Send audio stream to VS1003 // for(i=0;i<16;i++) { VS10xxWriteData( ucBuff + i*32 ); } ulFileCursor += usDataCnt; if(ulFileCursor == 4096) { // // Get and print file information // VS10xxGetAudioInfo(&tFLIF); printf("\n\rFile type: %d\n\rBit rate: %d kbps\n\rSample rate: %dHz", tFLIF.eFileType, tFLIF.usBitRate, tFLIF.ucSampleRate); printf("\n\rMP3 ID: %d\n\rMP3 layer: %d\n\rOther information: %x\n\r", tFLIF.ucMp3ID, tFLIF.ucMp3Layer, tFLIF.ucOtherInfo); } if((ulFileCursor % 10240) == 0) { // // Get and print decode time // usAudioDecTime = VS10xxGetDecodeTime(); printf("\rplayed time: %2d:%2ds", usAudioDecTime/60, usAudioDecTime%60); } // // If file come to the end. // if( usDataCnt==0 ) { ulFileCursor=0; // // Move file pointer to the beginning of the file // CH376ByteLocate( 0 ); // // Reset decode time // VS10xxResetDecodeTime(); } // // Key scan // switch(JoystickKeyRead()) { case JOYSTICK_KEY_SELECT: ucKeySelFlag = !ucKeySelFlag; if(ucKeySelFlag) xGPIOSPinWrite( LED_PIN, 1 ); else xGPIOSPinWrite( LED_PIN, 0 ); break; case JOYSTICK_KEY_UP: if(!ucKeySelFlag) { // // Volume increase // if(sucVolLevel<9) sucVolLevel++; VS10xxSetVolume(ucVolumeLevel[sucVolLevel]); } else { // // Treble increase // if(ssTrebleA < 7) ssTrebleA++; VS10xxBassTrebleEnhance(ssTrebleA, 12, ssBassA, 10); } break; case JOYSTICK_KEY_DOWN: if(!ucKeySelFlag) { // // Volume decrease // if(sucVolLevel!=0) sucVolLevel--; VS10xxSetVolume(ucVolumeLevel[sucVolLevel]); } else { // // Treble decrease // if(ssTrebleA > -8) ssTrebleA--; VS10xxBassTrebleEnhance(ssTrebleA, 12, ssBassA, 10); } break; case JOYSTICK_KEY_LEFT: if(!ucKeySelFlag) { // // Balance decrease // if(ssVolBalance>-254) ssVolBalance--; VS10xxSetBalance(ssVolBalance); } else { // // bass decrease // if(ssBassA>0) ssBassA--; VS10xxBassTrebleEnhance(ssTrebleA, 12, ssBassA, 10); } break; case JOYSTICK_KEY_RIGHT: if(!ucKeySelFlag) { // // balance increase // if(ssVolBalance<254) ssVolBalance++; VS10xxSetBalance(ssVolBalance); } else { // // Bass increase // if(ssBassA<15) ssBassA++; VS10xxBassTrebleEnhance(ssTrebleA, 12, ssBassA, 10); } break; default : break; } } // CH376FileClose(FALSE); }
main( ) { UINT8 i, c, SecCount; UINT16 NewSize, count; /* 因为演示板的RAM容量只有32KB,所以NewSize限制为16位,实际上如果文件大于32256字节,应该分几次读写并且将NewSize改为UINT32以便累计 */ UINT8 code *pCodeStr; LED_OUT_INIT( ); LED_OUT_ACT( ); /* 开机后LED亮一下以示工作 */ mDelay100mS( ); /* 延时100毫秒 */ LED_OUT_INACT( ); mInitSTDIO( ); /* 为了让计算机通过串口监控演示过程 */ printf( "Start\n" ); CH375_INT_FLAG = 0; /* 清中断标志 */ i = CH375LibInit( ); /* 初始化CH375程序库和CH375芯片,操作成功返回0 */ mStopIfError( i ); CH375_INT_EN = 1; /* 允许CH375中断 */ /* 其它电路初始化 */ EA = 1; /* 初始化完成,开中断 */ while ( 1 ) { printf( "Wait Udisk\n" ); while ( CH375DiskStatus != DISK_CONNECT ); /* 等待U盘插入,单片机可以做其它事情 */ LED_OUT_ACT( ); /* LED亮 */ mDelay100mS( ); /* 延时,可选操作,有的USB存储器需要几十毫秒的延时 */ mDelay100mS( ); /* 检查U盘是否准备好,有些U盘不需要这一步,但是某些U盘必须要执行这一步才能工作 */ for ( i = 0; i < 5; i ++ ) { /* 有的U盘总是返回未准备好,不过可以被忽略 */ mDelay100mS( ); printf( "Ready ?\n" ); if ( CH375DiskReady( ) == ERR_SUCCESS ) break; /* 查询磁盘是否准备好 */ } /* 查询磁盘物理容量 */ /* printf( "DiskSize\n" ); i = CH375DiskSize( ); mStopIfError( i ); printf( "TotalSize = %u MB \n", (unsigned int)( mCmdParam.DiskSize.mDiskSizeSec >> 11 ) ); 显示为以MB为单位的容量 */ LED_RUN_ACT( ); /* 开始操作U盘 */ /* 读取原文件 */ printf( "Open\n" ); mCopyCodeStringToIRAM( mCmdParam.Open.mPathName, "\\C51\\CH375HFT.C" ); /* 文件名,该文件在C51子目录下 */ i = CH375FileOpen( ); /* 打开文件 */ if ( i == ERR_MISS_DIR || i == ERR_MISS_FILE ) { /* 没有找到文件 */ /* 列出文件 */ if ( i == ERR_MISS_DIR ) pCodeStr = "\\*"; /* C51子目录不存在则列出根目录下的文件 */ else pCodeStr = "\\C51\\CH375*"; /* CH375HFT.C文件不存在则列出\C51子目录下的以CH375开头的文件 */ printf( "List file %s\n", pCodeStr ); for ( c = 0; c < 254; c ++ ) { /* 最多搜索前254个文件,超过254需参考EXAM0使用CH375vFileSize */ i = mCopyCodeStringToIRAM( mCmdParam.Open.mPathName, pCodeStr ); /* 搜索文件名,*为通配符,适用于所有文件或者子目录 */ mCmdParam.Open.mPathName[ i ] = c; /* 根据字符串长度将结束符替换为搜索的序号,从0到254 */ i = CH375FileOpen( ); /* 打开文件,如果文件名中含有通配符*,则为搜索文件而不打开 */ if ( i == ERR_MISS_FILE ) break; /* 再也搜索不到匹配的文件,已经没有匹配的文件名 */ if ( i == ERR_FOUND_NAME ) { /* 搜索到与通配符相匹配的文件名,文件名及其完整路径在命令缓冲区中 */ printf( " match file %03d#: %s\n", (unsigned int)c, mCmdParam.Open.mPathName ); /* 显示序号和搜索到的匹配文件名或者子目录名 */ continue; /* 继续搜索下一个匹配的文件名,下次搜索时序号会加1 */ } else { /* 出错 */ mStopIfError( i ); break; } } pCodeStr = "找不到/C51/CH375HFT.C文件\xd\n"; for ( i = 0; i != 255; i ++ ) { if ( ( FILE_DATA_BUF[i] = *pCodeStr ) == 0 ) break; pCodeStr++; } NewSize = i; /* 新文件的长度 */ SecCount = 1; /* (NewSize+511)/512, 计算文件的扇区数,因为读写是以扇区为单位的 */ } else { /* 找到文件或者出错 */ mStopIfError( i ); /* printf( "Query\n" ); i = CH375FileQuery( ); 查询当前文件的信息 mStopIfError( i );*/ printf( "Read\n" ); if ( CH375vFileSize > FILE_DATA_BUF_LEN ) { /* 由于演示板用的62256只有32K字节,其中CH375子程序用512字节,所以只读取不超过63个扇区,也就是不超过32256字节 */ SecCount = FILE_DATA_BUF_LEN / 512; /* 由于演示板用的62256只有32K字节,其中CH375子程序用512字节,所以只读取不超过63个扇区,也就是不超过32256字节 */ NewSize = FILE_DATA_BUF_LEN; /* 由于RAM有限所以限制长度 */ } else { /* 如果原文件较小,那么使用原长度 */ SecCount = ( CH375vFileSize + 511 ) >> 9; /* (CH375vFileSize+511)/512, 计算文件的扇区数,因为读写是以扇区为单位的,先加511是为了读出文件尾部不足1个扇区的部分 */ NewSize = (UINT16)CH375vFileSize; /* 原文件的长度 */ } printf( "Size=%ld, Len=%d, Sec=%d\n", CH375vFileSize, NewSize, (UINT16)SecCount ); mCmdParam.Read.mSectorCount = SecCount; /* 读取全部数据,如果超过60个扇区则只读取60个扇区 */ /* current_buffer = & FILE_DATA_BUF[0]; 如果文件读写的数据的复制方式为"外部子程序",那么需要设置存放数据的缓冲区的起始地址 */ CH375vFileSize += 511; /* 默认情况下,以扇区方式读取数据时,无法读出文件尾部不足1个扇区的部分,所以必须临时加大文件长度以读取尾部零头 */ i = CH375FileRead( ); /* 从文件读取数据 */ CH375vFileSize -= 511; /* 恢复原文件长度 */ mStopIfError( i ); /* 如果文件比较大,一次读不完,可以再调用CH375FileRead继续读取,文件指针自动向后移动 while ( 1 ) { c = 32; 每次读取32个扇区 mCmdParam.Read.mSectorCount = c; 指定读取的扇区数 CH375FileRead(); 读完后文件指针自动后移 处理数据 if ( mCmdParam.Read.mSectorCount < c ) break; 实际读出的扇区数较小则说明文件已经结束 } 如果希望从指定位置开始读写,可以移动文件指针 mCmdParam.Locate.mSectorOffset = 3; 跳过文件的前3个扇区开始读写 i = CH375FileLocate( ); mCmdParam.Read.mSectorCount = 10; CH375FileRead(); 直接读取从文件的第(512*3)个字节开始的数据,前3个扇区被跳过 或者自行定义数据缓冲区的起始地址 mCmdParam.ReadX.mSectorCount = 2; mCmdParam.ReadX.mDataBuffer = 0x2000; 将读出的数据放到2000H开始的缓冲区中 CH375FileReadX(); 从文件中读取2个扇区到指定缓冲区 如果希望将新数据添加到原文件的尾部,可以移动文件指针 i = CH375FileOpen( ); mCmdParam.Locate.mSectorOffset = 0xffffffff; 移到文件的尾部,以扇区为单位,如果原文件是3字节,则从512字节开始添加 i = CH375FileLocate( ); mCmdParam.Write.mSectorCount = 10; CH375FileWrite(); 在原文件的后面添加数据 或者自行定义数据缓冲区的起始地址 mCmdParam.WiiteX.mSectorCount = 2; mCmdParam.WriteX.mDataBuffer = 0x4600; 将4600H开始的缓冲区中的数据写入 CH375FileWriteX(); 将指定缓冲区中的数据写入2个扇区到文件中 */ printf( "Close\n" ); i = CH375FileClose( ); /* 关闭文件 */ mStopIfError( i ); i = FILE_DATA_BUF[100]; FILE_DATA_BUF[100] = 0; /* 置字符串结束标志,最多显示100个字符 */ printf( "Line 1: %s\n", FILE_DATA_BUF ); FILE_DATA_BUF[100] = i; /* 恢复原字符 */ for ( count=0; count < NewSize; count ++ ) { /* 将文件中的小写字符转换为大写 */ c = FILE_DATA_BUF[ count ]; if ( c >= 'a' && c <= 'z' ) FILE_DATA_BUF[ count ] = c - ( 'a' - 'A' ); } } #ifdef EN_DISK_WRITE /* 子程序库支持写操作 */ /* 产生新文件 */ LED_WR_ACT( ); /* 写操作 */ printf( "Create\n" ); mCopyCodeStringToIRAM( mCmdParam.Create.mPathName, "\\NEWFILE.TXT" ); /* 新文件名,在根目录下 */ i = CH375FileCreate( ); /* 新建文件并打开,如果文件已经存在则先删除后再新建 */ mStopIfError( i ); printf( "Write\n" ); mCmdParam.Write.mSectorCount = SecCount; /* 写入所有扇区的数据 */ /* current_buffer = & FILE_DATA_BUF[0]; 如果文件读写的数据的复制方式为"外部子程序",那么需要设置存放数据的缓冲区的起始地址 */ i = CH375FileWrite( ); /* 向文件写入数据 */ mStopIfError( i ); printf( "Modify\n" ); mCmdParam.Modify.mFileAttr = 0xff; /* 输入参数: 新的文件属性,为0FFH则不修改 */ mCmdParam.Modify.mFileTime = 0xffff; /* 输入参数: 新的文件时间,为0FFFFH则不修改,使用新建文件产生的默认时间 */ mCmdParam.Modify.mFileDate = MAKE_FILE_DATE( 2004, 5, 18 ); /* 输入参数: 新的文件日期: 2004.05.18 */ mCmdParam.Modify.mFileSize = NewSize; /* 输入参数: 如果原文件较小,那么新的文件长度与原文件一样长,否则被RAM所限,如果文件长度大于64KB,那么NewSize必须为UINT32 */ i = CH375FileModify( ); /* 修改当前文件的信息,修改日期和长度 */ mStopIfError( i ); printf( "Close\n" ); mCmdParam.Close.mUpdateLen = 0; /* 不要自动计算文件长度,如果自动计算,那么该长度总是512的倍数 */ i = CH375FileClose( ); mStopIfError( i ); /* 删除某文件 */ /* printf( "Erase\n" ); mCopyCodeStringToIRAM( mCmdParam.Create.mPathName, "\\OLD" ); 将被删除的文件名,在根目录下 i = CH375FileErase( ); 删除文件并关闭 if ( i != ERR_SUCCESS ) printf( "Error: %02X\n", (UINT16)i ); 显示错误 */ LED_WR_INACT( ); /* 查询磁盘信息 */ /* printf( "Disk\n" ); i = CH375DiskQuery( ); mStopIfError( i ); printf( "Fat=%d, Total=%ld, Free=%ld\n", (UINT16)mCmdParam.Query.mDiskFat, mCmdParam.Query.mTotalSector, mCmdParam.Query.mFreeSector ); */ #endif LED_RUN_INACT( ); printf( "Take out\n" ); while ( CH375DiskStatus != DISK_DISCONNECT ); /* 等待U盘拔出,单片机可以做其它事情 */ LED_OUT_INACT( ); /* LED灭 */ mDelay100mS( ); mDelay100mS( ); } }