USB,是英文Universal Serial Bus(通用串行總線)、支持設(shè)備的即插即用和熱插拔功能。在1994年底由英特爾、IBM、Microsoft等公司聯(lián)合提出的,在此之前PC的接口雜亂,擴展能力差,熱拔插不支持等。USB正是為了解決速度,擴展能力,易用性等而出現(xiàn)的,本文闡述了usb 協(xié)議的特點及其四種傳輸模式。
1. usb的特點
USB 接口有4根線,兩根電源及兩根信號,Standard A類型的接口定義 如下
一般的排列方式是:紅白綠黑從左到右
紅色-USB電源:標有-VCC、Power、5V、5VSB字樣
綠色-USB數(shù)據(jù)線:(正)-DATA+、USBD+、PD+、USBDT+
白色-USB數(shù)據(jù)線:(負)-DATA-、USBD-、PD-、USBDT+
黑色-地線:GND、Ground
1.1 Usb的四層描述符
USB協(xié)議采取主從模式,從設(shè)備端沒有主動通知USB主機端的能力,從機插入后,主機控制器根據(jù)協(xié)議,獲取設(shè)備描述符及驅(qū)動匹配。 每個usb設(shè)備可以有一個或多個配置(config),不同的配置的體現(xiàn)即不同的組合接口。 接口(interface)是一個邏輯概念,接口之間通常是隔離的,互相不干擾。 端點(endpoint)是usb設(shè)備的唯一可識別部分,也是host和device之間的通信流的終點。它是一個host或device上的一個數(shù)據(jù)緩沖區(qū),用來存放和發(fā)送usb的各種數(shù)據(jù)。每個端點都是一個單一連接,支持一個方向的數(shù)據(jù)流(in: device→host)(out: host→device)。 端點0:所有的usb設(shè)備都要擁有端點0,該端點用于對一個邏輯設(shè)備進行配置。端點0支持控制傳輸,且總是在設(shè)備接入和上電時就進行配置。
一個設(shè)備通常有一個或多個配置;
一個配置通常有一個或多個接口;
一個接口通常有一個或多個端點。
》設(shè)備描述符
/* USB_DT_DEVICE: Device descriptor /
struct usb_device_descriptor {
__u8 bLength; //本結(jié)構(gòu)體大小
__u8 bDescriptorType; //描述符類型
__le16 bcdUSB; //usb版本號 200->USB2.0
__u8 bDeviceClass; //設(shè)備類
__u8 bDeviceSubClass; //設(shè)備類子類
__u8 bDeviceProtocol; //設(shè)備協(xié)議,以上三點都是USB官方定義
__u8 bMaxPacketSize0; //端點0最大包大小
__le16 idVendor; //廠家id
__le16 idProduct; //產(chǎn)品id
__le16 bcdDevice; //設(shè)備出廠編號
__u8 iManufacturer; //設(shè)備廠商字符串索引
__u8 iProduct; //產(chǎn)品描述
__u8 iSerialNumber; //設(shè)備序列號字符串索引
__u8 bNumConfigurations; //配置的個數(shù)
} attribute ((packed));
》配置描述符
/ USB_DT_CONFIG: Config descriptor /
struct usb_config_descriptor {
__u8 bLength; //自身長度
__u8 bDescriptorType;//描述符類型(0x02)
__le16 wTotalLength; //該配置下,信息的總長度
__u8 bNumInterfaces; //接口的個數(shù)
__u8 bConfigurationValue; //Set_Configuration命令所需要的參數(shù)值
__u8 iConfiguration; //描述該配置的字符串的索引值
__u8 bmAttributes;//供電模式的選擇
__u8 bMaxPower;//設(shè)備從總線提取的最大電流
} attribute ((packed));
》接口描述符
/ USB_DT_INTERFACE: Interface descriptor */
struct usb_interface_descriptor {
__u8 bLength;
__u8 bDescriptorType;//接口描述符的類型編號(0x04)
__u8 bInterfaceNumber; //該接口的編號
__u8 bAlternateSetting; //備用的接口描述符編號
__u8 bNumEndpoints; //該接口使用的端點數(shù),不包括端點0
__u8 bInterfaceClass; //接口類
__u8 bInterfaceSubClass; //子類
__u8 bInterfaceProtocol; //協(xié)議
__u8 iInterface;//描述該接口的字符串索引值
} attribute ((packed));
》端點描述符
/* USB_DT_ENDPOINT: Endpoint descriptor */
struct usb_endpoint_descriptor {
__u8 bLength;//端點描述符字節(jié)數(shù)大?。?個字節(jié))
__u8 bDescriptorType;//端點描述符類型編號(0x05)
__u8 bEndpointAddress; //端點地址及輸入輸出屬性
__u8 bmAttributes; //屬性,包含端點的傳輸類型,控制,中斷…
__le16 wMaxPacketSize; //端點收、發(fā)的最大包大小
__u8 bInterval; //主機查詢端點的時間間隔
__u8 bRefresh;
__u8 bSynchAddress;
} attribute ((packed));
1.2 USB 結(jié)構(gòu)及通信形式
USB總線是一個單主方式的實現(xiàn),是一種輪詢方式的總線,為樹形拓撲結(jié)構(gòu)。協(xié)議規(guī)定所有的數(shù)據(jù)傳輸都必須由主機發(fā)起,host controller初始化所有的數(shù)據(jù)傳輸,各種設(shè)備緊緊圍繞在主機周圍。
USB通信最基本的形式是通過USB設(shè)備里一個叫endpoint的端點(可以理解為硬件寄存器或者buff),而主機和endpoint之間的數(shù)據(jù)傳輸是通過pipe,pipe通信有兩種,一種是stream,另一種是message,協(xié)議中規(guī)定:message管道要求從它那兒過的數(shù)據(jù)必須具有一定的格式,message管道必須對應(yīng)兩個相同號碼的端點,一個用來in,一個用來out,咱們的缺省管道就是message管道,當然,與缺省管道對應(yīng)的端點0就必須是兩個具有同樣端點號0的端點。
一個USB邏輯設(shè)備就是一系列端點(endpoint)的集合,它與主機之間的通信發(fā)生在主機上的一個緩沖區(qū)和設(shè)備上的一個端點之間,通過管道來傳輸數(shù)據(jù)。管道的一端是主機上的一個緩沖區(qū),另一端是設(shè)備上的端點 ,構(gòu)成一個通信信道。
2. USB 四種通信方式
USB endpoint有四種類型,分別對應(yīng)了四種不同的數(shù)據(jù)傳輸方式。它們是控制傳輸(Control Transfers),中斷傳輸(Interrupt Data Transfers),批量傳輸(Bulk Data Transfers),等時傳輸(Isochronous Data Transfers)。其中批量傳輸、等時傳輸和中斷傳輸每傳輸一次數(shù)據(jù)都是一個事務(wù);控制傳輸包括三個過程,建立過程和狀態(tài)過程分別是一個事務(wù),數(shù)據(jù)過程則可能包含多個事務(wù)。從usb設(shè)備端來看,也可以把端點分為四種類型為控制端點、中斷端點、批量端點、等時端點。USB傳輸數(shù)據(jù)先發(fā)數(shù)據(jù)低位(LSB),再發(fā)高位數(shù)據(jù)(MSB)
2.1 控制傳輸
控制傳輸用來控制對USB設(shè)備不同部分的訪問,通常用于配置設(shè)備,獲取設(shè)備信息,發(fā)送命令到設(shè)備,或者獲取設(shè)備的狀態(tài)報告??傊褪怯脕韨魉涂刂菩畔⒌?,每個USB設(shè)備都會有一個endpoint 0的控制端點,內(nèi)核里的USB core使用它在設(shè)備插入時進行設(shè)備的配置,它會一種等待著USB core發(fā)送控制命令。
控制傳輸分為三個過程:
建立過程使用一個建立事務(wù)。建立事務(wù)是一個輸出數(shù)據(jù)包的過程,需要注意的點有:
? 首先是令牌包,建立過程使用SETUP令牌包;
? 其次是數(shù)據(jù)包類型,SETUP只能使用DATA0包;
? 最后是握手,設(shè)備只能采用ACK來應(yīng)答(錯了的情況不應(yīng)答),不能使用NAK或者STALL來應(yīng)答,即設(shè)備必須要接受建立事務(wù)的數(shù)據(jù)。
數(shù)據(jù)過程是可選的,即一個控制傳輸可能沒有數(shù)據(jù)過程。如果有,一個數(shù)據(jù)過程可以包含一個或多個數(shù)據(jù)事務(wù),需要注意的是:
? 首先所有的數(shù)據(jù)事務(wù)必須是同一個方向的(在控制讀傳輸中,數(shù)據(jù)過程中的所有數(shù)據(jù)事務(wù)都必須是輸入的;在控制寫傳輸中,數(shù)據(jù)過程中的所有數(shù)據(jù)都必須是輸出的);
? 其次,一旦數(shù)據(jù)傳輸方向改變,就會認為進入到狀態(tài)過程,狀態(tài)過程的第一個數(shù)據(jù)包必須是DATA1包;
? 最后,每次爭取傳輸一個數(shù)據(jù)包后就在DATA0和DATA1之間交換。
在這里插入圖片描述
控制傳輸?shù)腟tatus階段是序列中的最后一個事務(wù),狀態(tài)階段由前一階段的數(shù)據(jù)流方向改變描繪,并始終使用DATA1 PID。
獲取設(shè)備描述符為控制傳輸,由5個事務(wù)組成,分別是SETUP、IN、IN、IN、OUT,具體過程如下:
2.2 批量傳輸
批事務(wù)量
{In/Out Data Packet
@令牌包(In/Out token)
@數(shù)據(jù)包 (In/Out data)
@握手包 (Out/In ack)
}
批量傳輸用來傳輸大量的數(shù)據(jù),確保沒有數(shù)據(jù)丟失,并不保證在特定的時間內(nèi)完成。U盤使用的就是批量傳輸,咱們用它備份數(shù)據(jù)時需要確保數(shù)據(jù)不能丟,而且也不能指望它能在一個固定的比較快的時間內(nèi)拷貝完。
? 批量輸出(批量寫)時, Out Token→Out Data→ACK/NYET/NAK/STALL,其中ACK代表接收正常,且可以接收下次傳輸;NYET代表本次數(shù)據(jù)成功接收,但沒能力接收下一次傳輸;NAK表示沒有足夠的緩沖區(qū)來保存數(shù)據(jù);STALL表示數(shù)據(jù)接收,但是端點處于掛起狀態(tài)。
? 批量輸入(批量讀)時,In Token→In Data→ACK /NAK/STALL,與批量寫不同,主機發(fā)送In令牌包后,若設(shè)備檢測到錯誤,則不做任何響應(yīng),主機等待超時。若設(shè)備沒有檢測到地址端點等錯誤,但是設(shè)備又沒有數(shù)據(jù)需要返回,那么設(shè)備就會返回NAK直接響應(yīng)主機;若改端點處于掛起狀態(tài),返回STALL給主機;若主機正確接收到數(shù)據(jù)后,主機返回ACK應(yīng)答設(shè)備,同樣的,主機檢測到錯誤則不做出響應(yīng),設(shè)備檢測超時。USB協(xié)議規(guī)定,不允許主機使用NAK拒絕接收數(shù)據(jù)包。
2.3 中斷傳輸
中斷事務(wù)
{In/Out Data Packet
@令牌包(In/Out token)
@數(shù)據(jù)包(In/Out data)
@握手包(Out/In ack)
}
中斷傳輸是一種保證查詢頻率的傳輸,用來以一個固定的速率傳送少量的數(shù)據(jù)。中斷端點在端點描述符中要報告它的查詢時間,主機會保證在小于這個事件間隔的范圍內(nèi)安排一次傳輸,USB鍵盤和USB鼠標使用的就是這種方式,USB的觸摸屏也是,傳輸?shù)臄?shù)據(jù)包含了坐標信息。
中斷傳輸不是由設(shè)備主動的發(fā)出一個設(shè)備請求,而是由主機保證不大于某個時間間隔內(nèi)安排一次傳輸。中斷傳輸通常用在數(shù)據(jù)量不大,但是對時間要求較嚴格的設(shè)備。中斷傳輸也可以用來不停的檢查某個狀態(tài),當條件滿足后再用批量傳輸,除了對端點查詢的策略上不一樣之外,中斷傳輸和批量傳輸?shù)慕Y(jié)構(gòu)基本上是一樣的。
2.4 等時傳輸
等時事務(wù)
{In/Out Data Packet
@令牌包(In/Out token)
@數(shù)據(jù)包 (In/Out data)
}
等時傳輸使用等時事務(wù)(Isochronous Transactions)來傳輸數(shù)據(jù)。同樣用來傳輸大量的數(shù)據(jù),但并不保證數(shù)據(jù)是否到達,以穩(wěn)定的速率發(fā)送和接收實時的信息,對傳送延遲非常敏感。用在數(shù)據(jù)量大、對實時性要求高的場合,例如音頻設(shè)備、視屏設(shè)備等,這些設(shè)備對數(shù)據(jù)延時敏感,期望能夠有個比較穩(wěn)定的數(shù)據(jù)流。對音頻或者視屏設(shè)備來說,對數(shù)據(jù)的100%正確要求不高,少量數(shù)據(jù)的錯誤還是能夠容忍的,主要是要保證不能停頓;所以等時傳輸是不保證數(shù)據(jù)100%正確的。當數(shù)據(jù)錯誤時,并不能進行重傳操作。因此等時傳輸也就沒有應(yīng)答包,并不進行重傳操作。數(shù)據(jù)是否正確,可以由數(shù)據(jù)包的CRC校驗來確認。至于出錯的數(shù)據(jù)如何處理,是由軟件來決定的。
控制傳輸
{Setup Packet(建立階段)
@令牌包(setup token)-(out) @數(shù)據(jù)包(Setup data) -(out) @握手包(ack) - (in)
}
{In/Out Data Packet(可選數(shù)階段據(jù))
@(令牌包(In/Out token) @數(shù)據(jù)包(In/Out data) @握手包(Out/In ack)
}
{Out/In Stautus Packet(狀態(tài)階段)
@(令牌包(Out/In token) @數(shù)據(jù)包(Out/In data) @握手包( In/Out ack)
}
審核編輯黃昊宇
-
通信協(xié)議
+關(guān)注
關(guān)注
28文章
899瀏覽量
40349 -
usb
+關(guān)注
關(guān)注
60文章
7963瀏覽量
265250
發(fā)布評論請先 登錄
相關(guān)推薦
評論