#include #include #include
#ifndef TRUE #define TRUE 1 #define FALSE 0 #endif
/* 定義CH375命令代碼及返回狀態(tài) */ #include "CH375INC.H"
/* 以下定義適用于MCS-51單片機(jī),其它單片機(jī)參照修改,為了提供C語言的速度需要對本程序進(jìn)行優(yōu)化 */ //#include unsigned char volatile xdata CH375_CMD_PORT _at_ 0xBDF1; /* CH375命令端口的I/O地址 */ unsigned char volatile xdata CH375_DAT_PORT _at_ 0xBCF0; /* CH375數(shù)據(jù)端口的I/O地址 */ unsigned char xdata FILE_DATA_BUF[512] _at_ 0x0000; /* 外部RAM數(shù)據(jù)緩沖區(qū)的起始地址,長度不少于一次讀寫的數(shù)據(jù)長度 */
/* 在P1.4連接一個(gè)LED用于監(jiān)控演示程序的進(jìn)度,低電平LED亮,當(dāng)U盤插入后亮 */ sbit P1_4 = P1^4; #define LED_OUT_ACT( ) { P1_4 = 0; } /* P1.4 低電平驅(qū)動(dòng)LED顯示 */ #define LED_OUT_INACT( ) { P1_4 = 1; } /* P1.4 低電平驅(qū)動(dòng)LED顯示 */
unsigned char volatile UdiskStatus; /* 當(dāng)前U盤狀態(tài),定義如下 */ #define STATUS_DISCONNECT 0 /* U盤尚未連接或者已經(jīng)斷開 */ #define STATUS_CONNECT 1 /* U盤剛剛連接 */ #define STATUS_ERROR 2 /* U盤操作錯(cuò)誤或者不支持 */ #define STATUS_WAIT 3 /* U盤正在操作 */ #define STATUS_READY 4 /* U盤準(zhǔn)備好接受操作 */
unsigned char *mBufferPoint;
/* 延時(shí)2微秒,不精確 */ void delay2us( ) { unsigned char i; for ( i = 2; i != 0; i -- ); }
/* 延時(shí)1微秒,不精確 */ void delay1us( ) { unsigned char i; for ( i = 1; i != 0; i -- ); }
/* 延時(shí)100毫秒,不精確 */ void mDelay100mS( ) { unsigned char i, j, c; for ( i = 200; i != 0; i -- ) for ( j = 200; j != 0; j -- ) c+=3; }
/* 基本操作 */
void CH375_WR_CMD_PORT( unsigned char cmd ) { /* 向CH375的命令端口寫入命令,周期不小于4uS,如果單片機(jī)較快則延時(shí) */ delay2us(); CH375_CMD_PORT=cmd; delay2us(); }
void CH375_WR_DAT_PORT( unsigned char dat ) { /* 向CH375的數(shù)據(jù)端口寫入數(shù)據(jù),周期不小于1.5uS,如果單片機(jī)較快則延時(shí) */ CH375_DAT_PORT=dat; delay1us(); /* 因?yàn)镸CS51單片機(jī)較慢所以實(shí)際上無需延時(shí) */ }
unsigned char CH375_RD_DAT_PORT() { /* 從CH375的數(shù)據(jù)端口讀出數(shù)據(jù),周期不小于1.5uS,如果單片機(jī)較快則延時(shí) */ delay1us(); /* 因?yàn)镸CS51單片機(jī)較慢所以實(shí)際上無需延時(shí) */ return( CH375_DAT_PORT ); }
/* CH375的INT#引腳連接51單片機(jī)的INT0引腳, 采用中斷方式 */ /* CH375中斷服務(wù)程序,使用寄存器組1 */ void CH375Interrupt( ) interrupt 0 using 1 { unsigned char i, s, len; CH375_WR_CMD_PORT( CMD_GET_STATUS ); /* 獲取中斷狀態(tài)并取消中斷請求 */ for ( i = 2; i != 0; i -- ); /* 至少延時(shí)2uS */ s = CH375_RD_DAT_PORT( ); /* 獲取中斷狀態(tài) */ if ( s == USB_INT_SUCCESS ) UdiskStatus = STATUS_READY; /* 操作成功 */ else if ( s == USB_INT_DISCONNECT ) { UdiskStatus = STATUS_DISCONNECT; /* 檢測到USB設(shè)備斷開事件 */ LED_OUT_INACT( ); } else if ( s == USB_INT_CONNECT ) { UdiskStatus = STATUS_CONNECT; /* 檢測到USB設(shè)備連接事件 */ LED_OUT_ACT( ); } else if ( s == USB_INT_DISK_READ ) { /* USB存儲器讀數(shù)據(jù)塊,請求數(shù)據(jù)讀出 */ CH375_WR_CMD_PORT( CMD_RD_USB_DATA ); /* 從CH375緩沖區(qū)讀取數(shù)據(jù)塊 */ for ( i = 2; i != 0; i -- ); /* 至少延時(shí)2uS */ len = CH375_RD_DAT_PORT( ); /* 后續(xù)數(shù)據(jù)的長度 */ while ( len ) { /* 根據(jù)長度讀取數(shù)據(jù) */ *mBufferPoint = CH375_RD_DAT_PORT( ); /* 讀出數(shù)據(jù)并保存 */ mBufferPoint ++; len --; } CH375_WR_CMD_PORT( CMD_DISK_RD_GO ); /* 繼續(xù)執(zhí)行USB存儲器的讀操作 */ } else if ( s == USB_INT_DISK_WRITE ) { /* USB存儲器寫數(shù)據(jù)塊,請求數(shù)據(jù)寫入 */ CH375_WR_CMD_PORT( CMD_WR_USB_DATA7 ); /* 向CH375緩沖區(qū)寫入數(shù)據(jù)塊 */ for ( i = 2; i != 0; i -- ); /* 至少延時(shí)2uS */ len = CH375_MAX_DATA_LEN; CH375_WR_DAT_PORT( len ); /* 后續(xù)數(shù)據(jù)的長度 */ do { /* 對于C51,這個(gè)DO+WHILE結(jié)構(gòu)比上面的WHILE效率高,速度快 */ CH375_WR_DAT_PORT( *mBufferPoint ); mBufferPoint ++; } while ( -- len ); CH375_WR_CMD_PORT( CMD_DISK_WR_GO ); /* 繼續(xù)執(zhí)行USB存儲器的寫操作 */ } else { /* 操作失敗 */ UdiskStatus = STATUS_ERROR; } /* CH375_INT_FLAG = 0; 清中斷標(biāo)志 */ }
/* 設(shè)置CH375為USB主機(jī)方式 */ unsigned char mCH375Init( ) { unsigned char i; UdiskStatus = STATUS_DISCONNECT; CH375_WR_CMD_PORT( CMD_SET_USB_MODE ); /* 設(shè)置USB工作模式 */ CH375_WR_DAT_PORT( 6 ); /* 模式代碼,自動(dòng)檢測USB設(shè)備連接 */ for ( i = 0xff; i != 0; i -- ) { /* 等待操作成功,通常需要等待10uS-20uS */ if ( CH375_RD_DAT_PORT( ) == CMD_RET_SUCCESS ) break; /* 操作成功 */ } if ( i != 0 ) return( TRUE ); /* 操作成功 */ else return( FALSE ); /* CH375出錯(cuò),例如芯片型號錯(cuò)或者處于串口方式或者不支持 */ }
/* 等待U盤準(zhǔn)備好或者等待上次操作結(jié)束 */ unsigned char mWaitReady( ) { while( UdiskStatus == STATUS_WAIT ); /* 等待上次操作結(jié)束 */ if ( UdiskStatus == STATUS_READY ) return( TRUE ); /* U盤已經(jīng)準(zhǔn)備好操作,上次操作成功 */ else if ( UdiskStatus == STATUS_ERROR ) { /* 上次操作失敗 */ UdiskStatus = STATUS_WAIT; CH375_WR_CMD_PORT( CMD_DISK_RESET ); /* 復(fù)位U盤 */ while( UdiskStatus == STATUS_WAIT ); /* 等待中斷狀態(tài) */ if ( UdiskStatus == STATUS_READY ) return( TRUE ); /* 等待中斷狀態(tài),操作成功 */ return( FALSE ); /* UdiskStatus = STATUS_ERROR */ } else if ( UdiskStatus == STATUS_DISCONNECT ) { /* U盤已經(jīng)斷開 */ /* while ( UdiskStatus == STATUS_DISCONNECT );*/ return( FALSE ); } else if ( UdiskStatus == STATUS_CONNECT ) { /* U盤已經(jīng)連接 */ mDelay100mS( ); /* 如果U盤剛插入那么最好稍等一下再操作 */ mDelay100mS( ); CH375_WR_CMD_PORT( CMD_DISK_INIT ); /* 初始化USB存儲器 */ while ( UdiskStatus == STATUS_CONNECT ); /* 等待中斷狀態(tài) */ if ( UdiskStatus == STATUS_READY ) { /* 操作成功,磁盤已經(jīng)初始化并安裝到系統(tǒng)中 */ /* 檢查U盤是否準(zhǔn)備好,大多數(shù)U盤不需要這一步,但是某些U盤必須要執(zhí)行這一步才能工作 */ // do { // mDelay100mS( ); // printf( "Disk Ready ?\n" ); // i = CH375DiskReady( ); /* 查詢磁盤是否準(zhǔn)備好,如果省掉這個(gè)子程序可以節(jié)約將近1KB的程序代碼 */ // } while ( i != ERR_SUCCESS ); /* CH375DiskReady 在CH375的U盤文件子程序庫中,因?yàn)榇a較多,所以此處省去 */ return( TRUE ); } else return( FALSE ); } return( FALSE ); }
/* 從U盤讀取多個(gè)扇區(qū)的數(shù)據(jù)塊到緩沖區(qū) */ unsigned char mReadSector( unsigned long iLbaStart, unsigned char iSectorCount, unsigned char *iBuffer ) /* iLbaStart 是準(zhǔn)備讀取的線性起始扇區(qū)號, iSectorCount 是準(zhǔn)備讀取的扇區(qū)數(shù), iBuffer 是存放返回?cái)?shù)據(jù)的緩沖區(qū)的起址 */ { if ( mWaitReady( ) == FALSE ) return( FALSE ); mBufferPoint = iBuffer; /* 指向緩沖區(qū)起始地址 */ UdiskStatus = STATUS_WAIT; CH375_WR_CMD_PORT( CMD_DISK_READ ); /* 從USB存儲器讀數(shù)據(jù)塊 */ CH375_WR_DAT_PORT( (unsigned char)iLbaStart ); /* LBA的最低8位 */ CH375_WR_DAT_PORT( (unsigned char)( iLbaStart >> 8 ) ); CH375_WR_DAT_PORT( (unsigned char)( iLbaStart >> 16 ) ); CH375_WR_DAT_PORT( (unsigned char)( iLbaStart >> 24 ) ); /* LBA的最高8位 */ CH375_WR_DAT_PORT( iSectorCount ); /* 扇區(qū)數(shù) */ /* 如果是純中斷方式,那么可以做其它事情,中斷服務(wù)程序會(huì)讀取數(shù)據(jù) */ return( mWaitReady( ) ); }
struct _HD_MBR_DPT { unsigned char PartState; unsigned char StartHead; unsigned int StartSec; unsigned char PartType; unsigned char EndHead; unsigned int EndSec; unsigned long StartSector; unsigned long TotalSector; };
/* 為printf和getkey輸入輸出初始化串口 */ void mInitSTDIO( ) { SCON = 0x58; PCON = 0x80; TMOD = 0x21; TH1 = 0xff; /* 22MHz晶振, 9600bps */ TR1 = 1; TI = 1; }
void COM_send(unsigned char c) { SBUF=c; while(TI==0); TI=0; }
main( ) { unsigned char c; unsigned char i; LED_OUT_ACT( ); /* 開機(jī)后LED亮一下以示工作 */ mDelay100mS( ); /* 延時(shí)100毫秒 */ LED_OUT_INACT( ); mInitSTDIO( ); c = mCH375Init( ); /* 初始化CH375 */ //if ( c == FALSE ) printf( "Error @CH375Init\n" ); //printf( "Insert USB disk\n" ); while ( 1 ) { /* 主程序 */ mDelay100mS( ); mDelay100mS( ); mDelay100mS( ); mDelay100mS( ); mDelay100mS( ); /* 如果打算從U盤讀取數(shù)據(jù) */ //printf( "Read\n" ); c = mReadSector( 0, 1, FILE_DATA_BUF ); //if ( c == FALSE ) printf( "Error @ReadSector\n" ); for(i=0;i!=512;i++) COM_send(FILE_DATA_BUF); mDelay100mS( ); }