CH372 HID類鍵盤的連續(xù)上傳數(shù)據(jù)的問題

/*;CH372/CH375 USB device mode & external firmware ; U2(AT89C51) Program ; ; Website: http://winchiphead.com ; Email: 個人信息保護(hù),已隱藏 ; Author: W.ch 2003.12, 2005.03 ; ;**************************************************************************** 關(guān)于程序的說明: 模擬鍵盤程序,主機(jī)枚舉好后,發(fā)送K,可以在TXT文檔中顯示出來。作為演示,需要其他鍵值可以自己修改,如果不是連續(xù)發(fā)數(shù)據(jù)的 話,一般發(fā)送完畢鍵值后發(fā)送一包00,主機(jī)認(rèn)為發(fā)送結(jié)束 */

/* MCS-51單片機(jī)C語言的示例程序 */ #include #include "CH375INC.H"

unsigned char idata buf[8]; typedef union _REQUEST_PACK{ unsigned char buffer[8]; struct{ unsigned char bmReuestType; //標(biāo)準(zhǔn)請求字 unsigned char bRequest; //請求代碼 unsigned int wValue; //特性選擇高 unsigned int wIndx; //索引 unsigned int wLength; //數(shù)據(jù)長度 }r; } mREQUEST_PACKET, *mpREQUEST_PACKET;

//設(shè)備描述符 unsigned char code DevDes[]={ 0x12 //描述符大小 , 0x01 //常數(shù)DEVICE , 0x10 //USB規(guī)范版本信息 , 0x01 , 0x00 //類別碼, , 0x00 //子類別碼 , 0x00 //協(xié)議碼 , 0x08 //端點(diǎn)0的最大信息包大小 , 0x3c //廠商ID , 0x41 , 0x03 //產(chǎn)品ID , 0x20 , 0x00 //設(shè)備版本信息 , 0x02 , 0x01 //索引值 , 0x02 , 0x00 , 0x01 //可能配置的數(shù)目 }; //配置描述符 unsigned char code ConDes[]={ //配置描述符 0x09, 0x02, 0x22, 0x00, 0x01, 0x01, 0x04, 0xa0, 0x23,//配置描述符 0x09, 0x04, 0x00, 0x00, 0x01, 0x03, 0x01, 0x01, 0x05,//接口描述符 0x09, 0x21, 0x10, 0x01, 0x00, 0x01, 0x22, 0x41, 0x00,//HID類描述符 0x07, 0x05, 0x81, 0x03, 0x08, 0x00, 0x18 //端點(diǎn)描述符,只配置了輸入端點(diǎn),輸出端點(diǎn)用0端點(diǎn) }; //配置描述符

/*報表描述符*/ unsigned char code Hid_des[]={0x05, 0x01, 0x09, 0x06, 0xa1, 0x01, 0x05, 0x07, 0x19, 0xe0, 0x29, 0xe7, 0x15, 0x00, 0x25, 0x01, 0x75, 0x01, 0x95, 0x08, 0x81, 0x02, 0x95, 0x01, 0x75, 0x08, 0x81, 0x01, 0x95, 0x03, 0x75, 0x01, 0x05, 0x08, 0x19, 0x01, 0x29, 0x03, 0x91, 0x02, 0x95, 0x01, 0x75, 0x05, 0x91, 0x01, 0x95, 0x06, 0x75, 0x08, 0x15, 0x00, 0x26, 0xff, 0x00, 0x05, 0x07, 0x19, 0x00, 0x2a, 0xff, 0x00, 0x81, 0x00, 0xc0 }; /*關(guān)于全局變量的定義*/ unsigned char mVarSetupRequest; // ;USB請求碼 unsigned char mVarSetupLength; // 后續(xù)數(shù)據(jù)長度 unsigned char code * VarSetupDescr; //描述符偏移地址 unsigned char VarUsbAddress ; sbit CH375_INT_WIRE = 0xB0^2; //P3.2, INT0, 連接CH375的INT#引腳,用于查詢中斷狀態(tài) */ sbit CH375ACT = P1^4; bit CH375FLAGERR; //錯誤清0 bit CH375CONFLAG; unsigned char idata UPDATA_FLAG; mREQUEST_PACKET request;

/*硬件定義,根據(jù)硬件修改*/ 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地址 */

/* 延時2微秒,不精確 */ void Delay1us() { ; }

void Delay2us( ) { unsigned char i; #define DELAY_START_VALUE 1 /* 根據(jù)單片機(jī)的時鐘選擇初值,20MHz以下為0,30MHz以上為2 */ for ( i=DELAY_START_VALUE; i!=0; i-- ); }

/* 延時50毫秒,不精確 */ void Delay50ms( ) { unsigned char i, j; for ( i=200; i!=0; i-- ) for ( j=250; j!=0; j-- ); }

void CH375_WR_CMD_PORT( unsigned char cmd ) { /* 向CH375的命令端口寫入命令,周期不小于4uS,如果單片機(jī)較快則延時 */ CH375_CMD_PORT=cmd; Delay2us( ); }

void CH375_WR_DAT_PORT( unsigned char dat ) { /* 向CH375的數(shù)據(jù)端口寫入數(shù)據(jù),周期不小于1.5uS,如果單片機(jī)較快則延時 */ CH375_DAT_PORT=dat; Delay1us(); /* 因?yàn)镸CS51單片機(jī)較慢所以實(shí)際上無需延時 */ }

unsigned char CH375_RD_DAT_PORT() { /* 從CH375的數(shù)據(jù)端口讀出數(shù)據(jù),周期不小于1.5uS,如果單片機(jī)較快則延時 */ Delay1us( ); /* 因?yàn)镸CS51單片機(jī)較慢所以實(shí)際上無需延時 */ return( CH375_DAT_PORT );

} /* CH375初始化子程序 */ void CH375_Init( ) { /* 設(shè)置USB工作模式, 必要操作 */ CH375_WR_CMD_PORT( CMD_SET_USB_MODE ); CH375_WR_DAT_PORT( 1 ); /* 設(shè)置為使用內(nèi)置固件的USB設(shè)備方式 */ for ( ;; ) { /* 等待操作成功,通常需要等待10uS-20uS */ if ( CH375_RD_DAT_PORT( )==CMD_RET_SUCCESS ) break; } }

//端點(diǎn)0數(shù)據(jù)上傳 void mCh375Ep0Up() { unsigned char i,len; if(mVarSetupLength) { //長度不為0傳輸具體長度的數(shù)據(jù) if(mVarSetupLength<=8) { len=mVarSetupLength; mVarSetupLength=0; } //長度小于8則長輸要求的長度 else { len=8; mVarSetupLength-=8; } //長度大于8則傳輸8個,切總長度減8 CH375_WR_CMD_PORT(CMD_WR_USB_DATA3); //發(fā)出寫端點(diǎn)0的命令 CH375_WR_DAT_PORT(len); //寫入長度 for(i=0;i!=len;i++) CH375_WR_DAT_PORT(request.buffer[i]); //循環(huán)寫入數(shù)據(jù) } else { CH375_WR_CMD_PORT(CMD_WR_USB_DATA3); //發(fā)出寫端點(diǎn)0的命令 CH375_WR_DAT_PORT(0); //上傳0長度數(shù)據(jù),這是一個狀態(tài)階段 } }

//復(fù)制描述符以便上傳 void mCh375DesUp() { unsigned char k; for (k=0; k!=8; k++ ) { request.buffer[k]=*VarSetupDescr; //依次復(fù)制8個描述符, VarSetupDescr++; } }

/*中斷函數(shù),但是本例用的是查詢方式,只處理了輸入端點(diǎn)1和端點(diǎn)0的事件,其余不處理*/ void mCH375Interrupt( ) { unsigned char InterruptStatus; unsigned char length, c1, len; CH375_WR_CMD_PORT(CMD_GET_STATUS); /* 獲取中斷狀態(tài)并取消中斷請求 */ InterruptStatus =CH375_RD_DAT_PORT(); /* 獲取中斷狀態(tài) */ switch(InterruptStatus){ // 分析中斷狀態(tài) case USB_INT_EP1_OUT: //中斷端點(diǎn)下傳成功,未處理 CH375_WR_CMD_PORT(CMD_RD_USB_DATA); //發(fā)出讀數(shù)據(jù)命令 if( length=CH375_RD_DAT_PORT( ) ) { //長度為0跳出 for(len=0;len!=length;len++)c1=CH375_RD_DAT_PORT(); //取出下傳數(shù)據(jù) } break; case USB_INT_EP0_SETUP: //控制端點(diǎn)建立成功 CH375_WR_CMD_PORT(CMD_RD_USB_DATA); length=CH375_RD_DAT_PORT(); for(len=0;len!=length;len++)request.buffer[len]=CH375_RD_DAT_PORT(); // 取出數(shù)據(jù) if(length==0x08){ mVarSetupLength=request.buffer[6]; //控制傳輸數(shù)據(jù)長度最大設(shè)置為128 if((c1=request.r.bmReuestType)&0x40){ //廠商請求,未處理 } if((c1=request.r.bmReuestType)&0x20) { //類請求,未處理 if(request.buffer[1]==0x0a) { } //SET_IDLE else if(request.buffer[1]==0x09) { UPDATA_FLAG=1; } } if(!((c1=request.r.bmReuestType)&0x60)) { //標(biāo)準(zhǔn)請求 mVarSetupRequest=request.r.bRequest; //暫存標(biāo)準(zhǔn)請求碼 switch(request.r.bRequest){ // 分析標(biāo)準(zhǔn)請求 case DEF_USB_CLR_FEATURE: //清除特性 if((c1=request.r.bmReuestType&0x1F)==0X02) { //不是端點(diǎn)不支持 switch(request.buffer[4]) { case 0x82: CH375_WR_CMD_PORT(CMD_SET_ENDP7); //清除端點(diǎn)2上傳 CH375_WR_DAT_PORT(0x8E); //發(fā)命令清除端點(diǎn) break; case 0x02: CH375_WR_CMD_PORT(CMD_SET_ENDP6); CH375_WR_DAT_PORT(0x80); //清除端點(diǎn)2下傳 break; case 0x81: CH375_WR_CMD_PORT(CMD_SET_ENDP5); //清除端點(diǎn)1上傳 CH375_WR_DAT_PORT(0x8E); break; case 0x01: CH375_WR_CMD_PORT(CMD_SET_ENDP4); //清除端點(diǎn)1下傳 CH375_WR_DAT_PORT(0x80); break; default: break; } } else { CH375FLAGERR=1; //不支持的清除特性,置錯誤標(biāo)志 } break; case DEF_USB_GET_STATUS: //獲得狀態(tài) request.buffer[0]=0; request.buffer[1]=0; //上傳狀態(tài) break; case DEF_USB_SET_ADDRESS: //設(shè)置地址 Var

//連續(xù)發(fā)送11個數(shù) Up_Data( 0x62 );//0 Up_Data( 0x5a );//1 Up_Data( 0x5b );//2 Up_Data( 0x5c );//3 Up_Data( 0x5b );//4 Up_Data( 0x5d );//5 Up_Data( 0x5e );//6 Up_Data( 0x5f );//7 Up_Data( 0x60 );//8 Up_Data( 0x61 );//9 Up_Data( 0x58 );//回車 //Up_Data1( 0 ); 這樣寫是不對的。比如: Up_Data( 0x62 ); 那你必須要等待PC把這個數(shù)據(jù)取走后你才能發(fā)送Up_Data( 0x5a ); 當(dāng)PC取走后,會產(chǎn)生USB_INT_EP1_IN這個中斷,所以發(fā)送完畢后,等待USB_INT_EP1_IN中斷產(chǎn)生。


按照你的程序來說的話,應(yīng)該只有回車,因?yàn)槟阕詈笠淮螌懙臄?shù)據(jù)就是回車鍵的鍵值。 正確做法是每次寫你要寫的鍵值,等CH372產(chǎn)生中斷之后,讀取中斷狀態(tài)為“中斷端點(diǎn)上傳成功”中斷之后,在發(fā)送下一個鍵值給中斷端點(diǎn)。


case USB_INT_EP1_IN: //PC機(jī)取走鍵值標(biāo)記, USB_INT_EP1_IN_FLAG=1;//PC機(jī)取走數(shù)據(jù)標(biāo)志,1表示數(shù)據(jù)已經(jīng)取走,可以發(fā)送下一個鍵值 break; default: if( (InterruptStatus&0x03)==0x03 ) { //總線復(fù)位 CH375FLAGERR=0; //錯誤清0 CH375CONFLAG=0; //配置清0 mVarSetupLength=0; CH375ACT=1; //清配置完成輸出 }

while(1) { /* 主程序 */ if(CH375_INT_WIRE == 0) { mCH375Interrupt( ); if(UPDATA_FLAG==0x01) {

//Up_Data( 0x0e );//鍵k //連續(xù)發(fā)送11個數(shù) Up_Data( 0x62 );//0

if (USB_INT_EP1_IN_FLAG =1) { USB_INT_EP1_IN_FLAG=0; Up_Data( 0x5a );//1 }

/* Up_Data( 0x5b );//2 Up_Data( 0x5c );//3 Up_Data( 0x5b );//4 Up_Data( 0x5d );//5 Up_Data( 0x5e );//6 Up_Data( 0x5f );//7 Up_Data( 0x60 );//8 Up_Data( 0x61 );//9 Up_Data( 0x58 );//回車 */ } } }

等待了也還是不行,還是一直發(fā)送最后一個鍵值


UPDATA_FLAG==0x01一直成立,也就是說一直發(fā)送數(shù)據(jù)(Up_Data( 0x62 )),流程還是不對 另外你的程序看不出來是傳不同的數(shù)據(jù)。


Up_Data( 0x62 );//0 1。先傳一個數(shù)字0

if (USB_INT_EP1_IN_FLAG ==1) //等待中斷 { USB_INT_EP1_IN_FLAG=0; //標(biāo)記置0 Up_Data( 0x5a );//1 發(fā)送數(shù)字1 }

UPDATA_FLAG==0x01 一直成立,應(yīng)該一直發(fā)送 01 啊


case USB_INT_EP1_IN: //PC機(jī)取走鍵值標(biāo)記, USB_INT_EP1_IN_FLAG=1;//PC機(jī)取走數(shù)據(jù)標(biāo)志,1表示數(shù)據(jù)已經(jīng)取走,可以發(fā)送下一個鍵值 break; 上面還需要修改,在發(fā)送8個0X00的數(shù)據(jù)上去告訴計(jì)算機(jī)此按鍵釋放。完了之后在上傳下一個按鍵。


我也還是沒改來這程序 郁悶ING 哪位大俠 指點(diǎn)下嘛


樓主labor能不能加你qq,有幾個問題想請教你。我也在做這個,也是再弄這個程序,543131667.


我知道怎么連續(xù)發(fā)送數(shù)據(jù)了。把中斷查詢方式改為中斷方式,然后像CH375上傳一個數(shù)據(jù)后,產(chǎn)生中斷,進(jìn)入中斷函數(shù),我們在上傳端點(diǎn)0下面設(shè)一個中斷標(biāo)志flag,并改變flag的值。而主函數(shù)中發(fā)送完一個數(shù)據(jù)后馬上查詢flag,如果flag的值改變了就證明一個數(shù)已經(jīng)發(fā)送完畢。不斷重復(fù)就能連續(xù)發(fā)送不同的數(shù)據(jù)了。哈哈。。。

還不明白就看一下程序吧: http://d.download.csdn.net/down/3518670/xiangiscoming


只有登錄才能回復(fù),可以選擇微信賬號登錄

国产91精品新入口,国产成人综合网在线播放,九热这里只有精品,本道在线观看,美女视频a美女视频,韩国美女激情视频,日本美女pvp视频