?? usbcore.c
字號:
0x74, 0x00, //t
0x74, 0x00, //t
0x70, 0x00, //p
0x3a, 0x00, //:
0x2f, 0x00, ///
0x2f, 0x00, ///
0x67, 0x00, //g
0x72, 0x00, //r
0x6f, 0x00, //o
0x75, 0x00, //u
0x70, 0x00, //p
0x2e, 0x00, //.
0x65, 0x00, //e
0x64, 0x00, //d
0x6e, 0x00, //n
0x63, 0x00, //c
0x68, 0x00, //h
0x69, 0x00, //i
0x6e, 0x00, //n
0x61, 0x00, //a
0x2e, 0x00, //.
0x63, 0x00, //c
0x6f, 0x00, //o
0x6d, 0x00, //m
0x2f, 0x00, ///
0x39, 0x00, //9
0x33, 0x00, //3
0x2f, 0x00 ///
};
/////////////////////////廠商字符串結束/////////////////////////////
//字符串“21IC DIY U盤學習板 之用戶自定義HID設備”的Unicode編碼”的Unicode編碼
//8位小端格式
const uint8 ProductStringDescriptor[54]={
54, //該描述符的長度為54字節
0x03, //字符串描述符的類型編碼為0x03
0x32, 0x00, //2
0x31, 0x00, //1
0x49, 0x00, //I
0x43, 0x00, //C
0x20, 0x00, //
0x44, 0x00, //D
0x49, 0x00, //I
0x59, 0x00, //Y
0x20, 0x00, //
0x55, 0x00, //U
0xd8, 0x76, //盤
0x66, 0x5b, //學
0x60, 0x4e, //習
0x7f, 0x67, //板
0x20, 0x00, //
0x4b, 0x4e, //之
0x28, 0x75, //用
0x37, 0x62, //戶
0xea, 0x81, //自
0x9a, 0x5b, //定
0x49, 0x4e, //義
0x48, 0x00, //H
0x49, 0x00, //I
0x44, 0x00, //D
0xbe, 0x8b, //設
0x07, 0x59 //備
};
////////////////////////產品字符串結束////////////////////////////
//字符串“2009-03-12”的Unicode編碼
//8位小端格式
const uint8 SerialNumberStringDescriptor[22]={
22, //該描述符的長度為22字節
0x03, //字符串描述符的類型編碼為0x03
0x32, 0x00, //2
0x30, 0x00, //0
0x30, 0x00, //0
0x39, 0x00, //9
0x2d, 0x00, //-
0x30, 0x00, //0
0x33, 0x00, //3
0x2d, 0x00, //-
0x31, 0x00, //1
0x32, 0x00 //2
};
//////////////////////產品序列號字符串結束/////////////////////////
/********************************************************************
函數功能:總線掛起中斷處理函數。
入口參數:無。
返 回:無。
備 注:無。
********************************************************************/
void UsbBusSuspend(void)
{
#ifdef DEBUG0
Prints("USB總線掛起。\r\n");
#endif
}
////////////////////////End of function//////////////////////////////
/********************************************************************
函數功能:總線復位中斷處理函數。
入口參數:無。
返 回:無。
備 注:無。
********************************************************************/
void UsbBusReset(void)
{
#ifdef DEBUG0
Prints("USB總線復位。\r\n");
#endif
UsbChipResetEndpoint(); //復位端點
ConfigValue=0; //配置值初始化為0
UsbChipSetConfig(0); //設置芯片的配置值為0
Ep1InIsBusy=0; //復位后端點1輸入緩沖區空閑。
Ep3InIsBusy=0; //復位后端點3輸入緩沖區空閑。
}
////////////////////////End of function//////////////////////////////
/********************************************************************
函數功能:根據pData和SendLength將數據發送到端點0的函數。
入口參數:無。
返 回:無。
備 注:無。
********************************************************************/
void UsbEp0SendData(void)
{
//將數據寫到端點中去準備發送
//寫之前要先判斷一下需要發送的數據是否比端點0
//最大長度大,如果超過端點大小,則一次只能發送
//最大包長的數據。端點0的最大包長在DeviceDescriptor[7]
if(SendLength>DeviceDescriptor[7])
{
//按最大包長度發送
UsbChipWriteEndpointBuffer(0,DeviceDescriptor[7],pSendData);
//發送后剩余字節數減少最大包長
SendLength-=DeviceDescriptor[7];
//發送一次后指針位置要調整
pSendData+= DeviceDescriptor[7];
}
else
{
if(SendLength!=0)
{
//不夠最大包長,可以直接發送
UsbChipWriteEndpointBuffer(0,SendLength,pSendData);
//發送完畢后,SendLength長度變為0
SendLength=0;
}
else //如果要發送的數據包長度為0
{
if(NeedZeroPacket==1) //如果需要發送0長度數據
{
UsbChipWriteEndpointBuffer(0,0,pSendData); //發送0長度數據包
NeedZeroPacket=0; //清需要發送0長度數據包標志
}
}
}
}
////////////////////////End of function//////////////////////////////
/********************************************************************
函數功能:USB端點0數據過程數據處理函數。
入口參數:無。
返 回:無。
備 注:該函數用來處理0端點控制傳輸的數據或狀態過程。
********************************************************************/
void UsbEp0DataOut(void)
{
UsbChipReadEndpointBuffer(0,16,Buffer);
UsbChipClearBuffer(0);
}
////////////////////////End of function//////////////////////////////
/********************************************************************
函數功能:端點0輸出中斷處理函數。
入口參數:無。
返 回:無。
備 注:無。
********************************************************************/
void UsbEp0Out(void)
{
int32 L;
#ifdef DEBUG0
Prints("USB端點0輸出中斷。\r\n");
#endif
//判斷是否是建立包
if(UsbChipIsSetup(0))
{
L=UsbChipReadEndpointBuffer(0,16,Buffer); //讀建立過程數據
UsbChipAcknowledgeSetup(0); //應答建立包
UsbChipClearBuffer(0); //清緩沖區
if(L!=8) //不是8字節的標準請求,直接返回
return;
//將緩沖數據填到設備請求的各字段中
bmRequestType=Buffer[0];
bRequest=Buffer[1];
wValue=Buffer[2]+(((uint16)Buffer[3])<<8);
wIndex=Buffer[4]+(((uint16)Buffer[5])<<8);
wLength=Buffer[6]+(((uint16)Buffer[7])<<8);
//下面的代碼判斷具體的請求,并根據不同的請求進行相關操作
//如果D7位為1,則說明是輸入請求
if((bmRequestType&0x80)==0x80)
{
//根據bmRequestType的D6~5位散轉,D6~5位表示請求的類型
//0為標準請求,1為類請求,2為廠商請求。
switch((bmRequestType>>5)&0x03)
{
case 0: //標準請求
#ifdef DEBUG0
Prints("USB標準輸入請求:");
#endif
//USB協議定義了幾個標準輸入請求,我們實現這些標準請求即可
//請求的代碼在bRequest中,對不同的請求代碼進行散轉
//事實上,我們還需要對接收者進行散轉,因為不同的請求接收者
//是不一樣的。接收者在bmRequestType的D4~D0位中定義。
//我們這里為了簡化操作,有些就省略了對接收者的判斷。
//例如獲取描述符的請求,只根據描述符的類型來區別。
switch(bRequest)
{
case GET_CONFIGURATION: //獲取配置
#ifdef DEBUG0
Prints("獲取配置。\r\n");
#endif
break;
case GET_DESCRIPTOR: //獲取描述符
#ifdef DEBUG0
Prints("獲取描述符——");
#endif
//對描述符類型進行散轉,對于全速設備,
//標準請求只支持發送到設備的設備、配置、字符串三種描述符
switch((wValue>>8)&0xFF)
{
case DEVICE_DESCRIPTOR: //設備描述符
#ifdef DEBUG0
Prints("設備描述符。\r\n");
#endif
pSendData=(uint8)DeviceDescriptor; //需要發送的數據
//判斷請求的字節數是否比實際需要發送的字節數多
//這里請求的是設備描述符,因此數據長度就是
//DeviceDescriptor[0]。如果請求的比實際的長,
//那么只返回實際長度的數據
if(wLength>DeviceDescriptor[0])
{
SendLength=DeviceDescriptor[0];
if(SendLength%DeviceDescriptor[7]==0) //并且剛好是整數個數據包時
{
NeedZeroPacket=1; //需要返回0長度的數據包
}
}
else
{
SendLength=wLength;
}
//將數據通過EP0返回
UsbEp0SendData();
break;
case CONFIGURATION_DESCRIPTOR: //配置描述符
#ifdef DEBUG0
Prints("配置描述符。\r\n");
#endif
pSendData=(uint8 *)ConfigurationDescriptor; //需要發送的數據為配置描述符
//判斷請求的字節數是否比實際需要發送的字節數多
//這里請求的是配置描述符集合,因此數據長度就是
//ConfigurationDescriptor[3]*256+ConfigurationDescriptor[2]。
//如果請求的比實際的長,那么只返回實際長度的數據
SendLength=ConfigurationDescriptor[3];
SendLength=SendLength*256+ConfigurationDescriptor[2];
if(wLength>SendLength)
{
if(SendLength%DeviceDescriptor[7]==0) //并且剛好是整數個數據包時
{
NeedZeroPacket=1; //需要返回0長度的數據包
}
}
else
{
SendLength=wLength;
}
//將數據通過EP0返回
UsbEp0SendData();
break;
case STRING_DESCRIPTOR: //字符串描述符
#ifdef DEBUG0
Prints("字符串描述符");
#endif
switch(wValue&0xFF) //根據wValue的低字節(索引值)散轉
{
case 0: //獲取語言ID
#ifdef DEBUG0
Prints("(語言ID)。\r\n");
#endif
pSendData=(uint8 *)LanguageId;
SendLength=LanguageId[0];
break;
case 1: //廠商字符串的索引值為1,所以這里為廠商字符串
#ifdef DEBUG0
Prints("(廠商描述)。\r\n");
#endif
pSendData=(uint8 *)ManufacturerStringDescriptor;
SendLength=ManufacturerStringDescriptor[0];
break;
case 2: //產品字符串的索引值為2,所以這里為產品字符串
#ifdef DEBUG0
Prints("(產品描述)。\r\n");
#endif
pSendData=(uint8 *)ProductStringDescriptor;
SendLength=ProductStringDescriptor[0];
break;
case 3: //產品序列號的索引值為3,所以這里為序列號
#ifdef DEBUG0
Prints("(產品序列號)。\r\n");
#endif
pSendData=(uint8 *)SerialNumberStringDescriptor;
SendLength=SerialNumberStringDescriptor[0];
break;
default :
#ifdef DEBUG0
Prints("(未知的索引值)。\r\n");
#endif
//對于未知索引值的請求,返回一個0長度的包
SendLength=0;
NeedZeroPacket=1;
break;
}
//判斷請求的字節數是否比實際需要發送的字節數多
//如果請求的比實際的長,那么只返回實際長度的數據
if(wLength>SendLength)
{
if(SendLength%DeviceDescriptor[7]==0) //并且剛好是整數個數據包時
{
NeedZeroPacket=1; //需要返回0長度的數據包
}
}
else
{
SendLength=wLength;
}
//將數據通過EP0返回
UsbEp0SendData();
break;
case REPORT_DESCRIPTOR: //報告描述符
#ifdef DEBUG0
Prints("報告描述符。\r\n");
#endif
pSendData=(uint8 *)ReportDescriptor; //需要發送的數據為報告描述符
SendLength=sizeof(ReportDescriptor); //需要返回的數據長度
//判斷請求的字節數是否比實際需要發送的字節數多
//如果請求的比實際的長,那么只返回實際長度的數據
if(wLength>SendLength)
{
if(SendLength%DeviceDescriptor[7]==0) //并且剛好是整數個數據包時
{
?? 快捷鍵說明
復制代碼
Ctrl + C
搜索代碼
Ctrl + F
全屏模式
F11
切換主題
Ctrl + Shift + D
顯示快捷鍵
?
增大字號
Ctrl + =
減小字號
Ctrl + -