非常感謝!
UD+怎么和地短接了?CH375部分問題不大,把UD+修改過來.
用串口操作ch375的時候,串口線要交叉相接. 單片機TXD--ch375 RXD 單片機RXD--ch375 TXD
太謝謝了!
請幫我看看這個程序缺什么呢?能不能和圖堆起來啊? 這個程序調(diào)試到一半就進行不下去了!到等待U盤連接就調(diào)不下去了! #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數(shù)據(jù)端口的I/O地址 */ #define CH375_INT_WIRE INT0 /* P3.2, 連接CH375的INT#引腳,用于查詢中斷狀態(tài) */ UINT8X DISK_BUFFER[512*32] _at_ 0x0000; /* 外部RAM數(shù)據(jù)緩沖區(qū)的起始地址 */
UINT32 DiskStart; /* 邏輯盤的起始絕對扇區(qū)號LBA */ UINT8 SecPerClus; /* 邏輯盤的每簇扇區(qū)數(shù) */ UINT8 RsvdSecCnt; /* 邏輯盤的保留扇區(qū)數(shù) */ UINT16 FATSz16; /* FAT16邏輯盤的FAT表占用的扇區(qū)數(shù) */
/* ********** 硬件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 -- ); /* 發(fā)出命令碼前后應(yīng)該各延時2uS */ } void CH375_WR_DAT_PORT( UINT8 dat ) { /* 向CH375的數(shù)據(jù)端口寫入數(shù)據(jù) */ CH375_DAT_PORT=dat; /* 因為MCS51單片機較慢所以實際上無需延時 */ } UINT8 CH375_RD_DAT_PORT( void ) { /* 從CH375的數(shù)據(jù)端口讀出數(shù)據(jù) */ return( CH375_DAT_PORT ); /* 因為MCS51單片機較慢所以實際上無需延時 */ } UINT8 mWaitInterrupt( void ) { /* 等待CH375中斷并獲取狀態(tài),返回操作狀態(tài) */ while( CH375_INT_WIRE ); /* 查詢等待CH375操作完成中斷(INT#低電平) */ CH375_WR_CMD_PORT( CMD_GET_STATUS ); /* 產(chǎn)生操作完成中斷,獲取中斷狀態(tài) */ return( CH375_RD_DAT_PORT( ) ); }
/* ********** BulkOnly傳輸協(xié)議層,被CH375內(nèi)置了,無需編寫單片機程序 */
/* ********** RBC/SCSI命令層,雖然被CH375內(nèi)置了,但是要寫程序發(fā)出命令及收發(fā)數(shù)據(jù) */
UINT8 mInitDisk( void ) { /* 初始化磁盤 */ UINT8 Status; CH375_WR_CMD_PORT( CMD_GET_STATUS ); /* 產(chǎn)生操作完成中斷, 獲取中斷狀態(tài) */ Status = CH375_RD_DAT_PORT( ); if ( Status == USB_INT_DISCONNECT ) return( Status ); /* USB設(shè)備斷開 */ CH375_WR_CMD_PORT( CMD_DISK_INIT ); /* 初始化USB存儲器 */ Status = mWaitInterrupt( ); /* 等待中斷并獲取狀態(tài) */ if ( Status != USB_INT_SUCCESS ) return( Status ); /* 出現(xiàn)錯誤 */ CH375_WR_CMD_PORT( CMD_DISK_SIZE ); /* 獲取USB存儲器的容量 */ Status = mWaitInterrupt( ); /* 等待中斷并獲取狀態(tài) */ if ( Status != USB_INT_SUCCESS ) { /* 出錯重試 */ /* 對于CH375A芯片,建議在此執(zhí)行一次CMD_DISK_R_SENSE命令 */ mDelaymS( 250 ); CH375_WR_CMD_PORT( CMD_DISK_SIZE ); /* 獲取USB存儲器的容量 */ Status = mWaitInterrupt( ); /* 等待中斷并獲取狀態(tài) */ } if ( Status != USB_INT_SUCCESS ) return( Status ); /* 出現(xiàn)錯誤 */ return( 0 ); /* U盤已經(jīng)成功初始化 */ }
UINT8 mReadSector( UINT32 iLbaStart, UINT8 iSectorCount, UINT8X *oDataBuffer ) { UINT16 mBlockCount; UINT8 c; CH375_WR_CMD_PORT( CMD_DISK_READ ); /* 從USB存儲器讀數(shù)據(jù)塊 */ 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 ); /* 扇區(qū)數(shù) */ for ( mBlockCount = iSectorCount * 8; mBlockCount != 0; mBlockCount -- ) { c = mWaitInterrupt( ); /* 等待中斷并獲取狀態(tài) */ if ( c == USB_INT_DISK_READ ) { /* 等待中斷并獲取狀態(tài),請求數(shù)據(jù)讀出 */ CH375_WR_CMD_PORT( CMD_RD_USB_DATA ); /* 從CH375緩沖區(qū)讀取數(shù)據(jù)塊 */ c = CH375_RD_DAT_PORT( ); /* 后續(xù)數(shù)據(jù)的長度 */ while ( c -- ) *oDataBuffer++ = CH375_RD_DAT_PORT( ); CH375_WR_CMD_PORT( CMD_DISK_RD_GO ); /* 繼續(xù)執(zhí)行USB存儲器的讀操作 */ } else break; /* 返回錯誤狀態(tài) */ } if ( mBlockCount == 0 ) { c = mWaitInterrupt( ); /* 等待中斷并獲取狀態(tài) */ if ( c== USB_INT_SUCCESS ) return( 0 ); /* 操作成功 */ } return( c ); /* 操作失敗 */ }
UINT16 mGetPointWord( UINT8X *iAddr ) { /* 獲取字數(shù)據(jù),因為MCS51是大端格式 */ return( iAddr[0]|(UINT16)iAddr[1] << 8 ); }
UINT8 mIdenDisk( void ) { /* 識別分析當前邏輯盤 */ UINT8 Status; DiskStart = 0; /* 以下是非常簡單的FAT文件系統(tǒng)的分析,正式應(yīng)用絕對不應(yīng)該如此簡單 */ Status = mReadSector( 0, 1, DISK_BUFFER ); /* 讀取邏輯盤引導信息 */ if ( Status != 0 ) return( Status ); if ( DISK_BUFFER[0] != 0xEB && DISK_BUFFER[0] != 0xE9 ) { /* 不是邏輯引導扇區(qū) */ 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]; /* 每簇扇區(qū)數(shù) */ RsvdSecCnt = DISK_BUFFER[0x0E]; /* 邏輯盤的保留扇區(qū)數(shù) */ FATSz16 = mGetPointWord( &DISK_BUFFER[0x16] ); /* FAT表占用扇區(qū)數(shù) */ 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 ) { /* 將簇號轉(zhuǎn)換為絕對LBA扇區(qū)地址 */ return( DiskStart + RsvdSecCnt + FATSz16 * 2 + 32 + ( iCluster - 2 ) * SecPerClus ); }
void mInitSTDIO( void ) { /* 僅用于調(diào)試用途及顯示內(nèi)容到PC機,與該程序功能完全無關(guān) */ SCON = 0x50; PCON = 0x80; TMOD = 0x20; TH1 = 0xf3; TR1=1; TI=1; /* 24MHz, 9600bps */ } void mStopIfError( UINT8 iErrCode ) { /* 如果錯誤則停止運行并顯示錯誤狀態(tài) */ 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,設(shè)置USB工作模式 */ CH375_WR_DAT_PORT( 6 ); /* 模式代碼,自動檢測USB設(shè)備連接 */ while ( 1 ) { printf( "Insert USB disk\n" ); while ( mWaitInterrupt( ) != USB_INT_CONNECT ); /* 等待U盤連接 */ mDelaymS( 250 ); /* 延時等待U盤進入正常工作狀態(tài) */ Status = mInitDisk( ); /* 初始化U盤,實際是識別U盤的類型,必須進行此步驟 */ mStopIfError( Status ); Status = mIdenDisk( ); /* 識別分析U盤文件系統(tǒng),必要操作 */ mStopIfError( Status ); Status = mReadSector( DiskStart + RsvdSecCnt + FATSz16 * 2, 32, DISK_BUFFER ); mStopIfError( Status ); /* 讀取FAT16邏輯盤的根目錄,通常根目錄占用32個扇區(qū) */ for ( CurrentDir = DISK_BUFFER; CurrentDir[0] != 0; CurrentDir += 32 ) { if ( ( CurrentDir[0x0B] & 0x08 ) == 0 && CurrentDir[0] != 0xE5 ) { CurrentDir[0x0B] = 0; /* 為了便于顯示,設(shè)置文件名或者目錄名的結(jié)束標志 */ 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 ) { /* 文件簇未結(jié)束 */ if ( Cluster == 0 ) mStopIfError( 0x8F ); /* 對于首簇,可能是0長度文件 */ Status = mReadSector( mClusterToLba( Cluster ), SecPerClus, DISK_BUFFER ); mStopIfError( Status ); /* 讀取首簇到緩沖區(qū) */ DISK_BUFFER[30] = 0; printf( "Data: %s\n", DISK_BUFFER ); /* 顯示首行 */ Cluster = mLinkCluster( Cluster ); /* 獲取鏈接簇,返回0說明錯誤 */ } } while ( mWaitInterrupt( ) != USB_INT_DISCONNECT ); /* 等待U盤拔出 */ mDelaymS( 250
你的程序是采用并口總線方式操作375的 串口方式操作CH375的請參考 CH375LIB\MCS51\FILELIB5\EXAM7 下面的例程.
你用的是串口, printf( "Insert USB disk\n" ); 這種類似的語句都要去掉,串口只能和375來通訊了. UINT8VX CH375_CMD_PORT _at_ 0xBDF1; /* CH375命令端口的I/O地址 */ UINT8VX CH375_DAT_PORT _at_ 0xBCF0; /* CH375數(shù)據(jù)端口的I/O地址 */ 這是用來并口操作的,串口的不可以這樣操作,你好好看看參考程序吧
好的噢!謝謝!
請幫我看看畫的并口的電路圖有什么問題?[Emot]1[/Emot]
并口操作的時候,CH375的TXD引腳要接地
就這個問題嗎?
再問一個比較愚蠢的問題,并口通信用于什么情況???
一般使用并口通信的話都是在要求速度比較快,同時硬件主控單片機上面帶總線口的情況下.
什么類型的單片機有總線口啊?51的有嗎?我用的是w78e516b40的,有總線口嗎?
p0是不是總線口?