日本黄色一级经典视频|伊人久久精品视频|亚洲黄色色周成人视频九九九|av免费网址黄色小短片|黄色Av无码亚洲成年人|亚洲1区2区3区无码|真人黄片免费观看|无码一级小说欧美日免费三级|日韩中文字幕91在线看|精品久久久无码中文字幕边打电话

當前位置:首頁 > > 充電吧
[導讀]說的usb子系統(tǒng)的IO操作,不得不說usb requestblock,簡稱urb。事實上,可以打一個這樣的比喻,usb總線就像一條高速公路,貨物、人流之類的可以看成是系統(tǒng)與設(shè)備交互的數(shù)據(jù),而urb就可

說的usb子系統(tǒng)的IO操作,不得不說usb requestblock,簡稱urb。事實上,可以打一個這樣的比喻,usb總線就像一條高速公路,貨物、人流之類的可以看成是系統(tǒng)與設(shè)備交互的數(shù)據(jù),而urb就可以看成是交通工具。在一開始對USB規(guī)范細節(jié)的介紹,我們就說過USB的endpoint有4種不同類型,于是能在這條高速公路上流動的數(shù)據(jù)也就有四種。但對車是沒有要求的,urb可以運載四種數(shù)據(jù),不過你要先告訴司機你要運什么,目的地是什么。我們現(xiàn)在就看看structurb的具體內(nèi)容。它的內(nèi)容很多,為了不讓我的理解誤導各位,大家最好還是看一看內(nèi)核源碼的注釋,具體內(nèi)容參見源碼樹下include/linux/usb.h。

在這里我們重點介紹程序中出現(xiàn)的幾個關(guān)鍵字段:

struct usb_device *dev

urb所發(fā)送的目標設(shè)備。

unsigned int pipe

???? 一個管道號碼,該管道記錄了目標設(shè)備的端點以及管道的類型。每個管道只有一種類型和一個方向,它與他的目標設(shè)備的端點向?qū)?yīng),我們可以通過以下幾個函數(shù)來獲得管道號并設(shè)置管道類型:

???? unsigned int usb_sndctrlpipe(struct usb_device *dev, unsigned int endpoint)

???????? 把指定USB設(shè)備指定端點設(shè)置為一個控制OUT端點。

???? unsigned int usb_rcvctrlpipe(struct usb_device *dev, unsigned int endpoint)

???????? 把指定USB設(shè)備指定端點設(shè)置為一個控制IN端點。

???? unsigned int usb_sndbulkpipe(struct usb_device *dev, unsigned int endpoint)

???????? 把指定USB設(shè)備指定端點設(shè)置為一個批量OUT端點。

???? unsigned int usb_rcvbulkpipe(struct usb_device *dev, unsigned int endpoint)

???????? 把指定USB設(shè)備指定端點設(shè)置為一個批量OUT端點。

???? unsigned int usb_sndintpipe(struct usb_device *dev, unsigned int endpoint)

???????? 把指定USB設(shè)備指定端點設(shè)置為一個中斷OUT端點。

???? unsigned int usb_rcvintpipe(struct usb_device *dev, unsigned int endpoint)

???????? 把指定USB設(shè)備指定端點設(shè)置為一個中斷OUT端點。

???? unsigned int usb_sndisocpipe(struct usb_device *dev, unsigned int endpoint)

???????? 把指定USB設(shè)備指定端點設(shè)置為一個等時OUT端點。

???? unsigned int usb_rcvisocpipe(struct usb_device *dev, unsigned int endpoint)

???????? 把指定USB設(shè)備指定端點設(shè)置為一個等時OUT端點。

unsigned int transfer_flags

當不使用DMA時,應(yīng)該transfer_flags |= URB_NO_TRANSFER_DMA_MAP(按照代碼的理解,希望沒有錯)。

Int status

???? 當一個urb把數(shù)據(jù)送到設(shè)備時,這個urb會由系統(tǒng)返回給驅(qū)動程序,并調(diào)用驅(qū)動程序的urb完成回調(diào)函數(shù)處理。這時,status記錄了這次數(shù)據(jù)傳輸?shù)挠嘘P(guān)狀態(tài),例如傳送成功與否。成功的話會是0。

???? 要能夠運貨當然首先要有車,所以第一步當然要創(chuàng)建urb:

???? struct urb *usb_alloc_urb(int isoc_packets, int mem_flags);

???? 第一個參數(shù)是等時包的數(shù)量,如果不是乘載等時包,應(yīng)該為0,第二個參數(shù)與kmalloc的標志相同。

???? 要釋放一個urb可以用:

???? void usb_free_urb(struct urb *urb);

???? 要承載數(shù)據(jù),還要告訴司機目的地信息跟要運的貨物,對于不同的數(shù)據(jù),系統(tǒng)提供了不同的函數(shù),對于中斷urb,我們用

void usb_fill_int_urb(struct urb *urb, struct usb_device *dev, unsigned int pipe,

void *transfer_buffer, int buffer_length,

usb_complete_t complete, void *context, int interval);

???? 這里要解釋一下,transfer_buffer是一要送/收的數(shù)據(jù)的緩沖,buffer_length是它的長度,complete是urb完成回調(diào)函數(shù)的入口,context有用戶定義,可能會在回調(diào)函數(shù)中使用的數(shù)據(jù),interval就是urb被調(diào)度的間隔。

???? 對于批量urb和控制urb,我們用:

???? void usb_fill_bulk_urb(struct urb *urb, struct usb_device *dev, unsigned int pipe,

??????????????????????????? void *transfer_buffer, int buffer_length, usb_complete_t complete,

??????????????????????????? void *context);

void usb_fill_bulk_urb(struct urb *urb, struct usb_device *dev, unsigned int pipe,

??????????????????????????? unsigned char* setup_packet,void *transfer_buffer,

int buffer_length, usb_complete_t complete,void *context);

控制包有一個特殊參數(shù)setup_packet,它指向即將被發(fā)送到端點的設(shè)置數(shù)據(jù)報的數(shù)據(jù)。

對于等時urb,系統(tǒng)沒有專門的fill函數(shù),只能對各urb字段顯示賦值。

有了汽車,有了司機,下一步就是要開始運貨了,我們可以用下面的函數(shù)來提交urb

???? int usb_submit_urb(struct urb *urb, int mem_flags);

mem_flags有幾種:GFP_ATOMIC、GFP_NOIO、GFP_KERNEL,通常在中斷上下文環(huán)境我們會用GFP_ATOMIC。

當我們的卡車運貨之后,系統(tǒng)會把它調(diào)回來,并調(diào)用urb完成回調(diào)函數(shù),并把這輛車作為函數(shù)傳遞給驅(qū)動程序。我們應(yīng)該在回調(diào)函數(shù)里面檢查status字段,以確定數(shù)據(jù)的成功傳輸與否。下面是用urb來傳送數(shù)據(jù)的細節(jié)。

???? /* initialize the urb properly */

???? usb_fill_bulk_urb(urb, dev->udev,

????????????? ? usb_sndbulkpipe(dev->udev, dev->bulk_out_endpointAddr),

????????????? ? buf, writesize, skel_write_bulk_callback, dev);

???? urb->transfer_flags |= URB_NO_TRANSFER_DMA_MAP;

?

???? /* send the data out the bulk port */

???? retval = usb_submit_urb(urb, GFP_KERNEL);

這里skel_write_bulk_callback就是一個完成回調(diào)函數(shù),而他做的主要事情就是檢查數(shù)據(jù)傳輸狀態(tài)和釋放urb:

???? dev = (struct usb_skel *)urb->context;

???? /* sync/async unlink faults aren't errors */

???? if (urb->status &&

???? ??? !(urb->status == -ENOENT ||

???? ????? urb->status == -ECONNRESET ||

???? ????? urb->status == -ESHUTDOWN)) {

???????? dbg("%s - nonzero write bulk status received: %d",

???????? ??? __FUNCTION__, urb->status);

???? }

???? /* free up our allocated buffer */

???? usb_buffer_free(urb->dev, urb->transfer_buffer_length,

????????????? urb->transfer_buffer, urb->transfer_dma);

事實上,如果數(shù)據(jù)的量不大,那么可以不一定用卡車來運貨,系統(tǒng)還提供了一種不用urb的傳輸方式,而usb-skeleton的讀操作正是采用這種方式實現(xiàn):

???? /* do a blocking bulk read to get data from the device */

???? retval = usb_bulk_msg(dev->udev,

????????????? ????? usb_rcvbulkpipe(dev->udev, dev->bulk_in_endpointAddr),

????????????? ????? dev->bulk_in_buffer,

????????????? ????? min(dev->bulk_in_size, count),

????????????? ????? &bytes_read, 10000);

?

???? /* if the read was successful, copy the data to userspace */

???? if (!retval) {

???????? if (copy_to_user(buffer, dev->bulk_in_buffer, bytes_read))

????????????? retval = -EFAULT;

???????? else

????????????? retval = bytes_read;

???? }

程序使用了usb_bulk_msg來傳送數(shù)據(jù),它的原型如下:

???? int usb_bulk_msg(struct usb_device *usb_dev, unsigned int pipe,void *data,

int len, int *actual length, int timeout)

???? 這個函數(shù)會阻塞等待數(shù)據(jù)傳輸完成或者等到超時,data是輸入/輸出緩沖,len是它的大小,actual length是實際傳送的數(shù)據(jù)大小,timeout是阻塞超時。

???? 對于控制數(shù)據(jù),系統(tǒng)提供了另外一個函數(shù),他的原型是:

???????? Int usb_contrl_msg(struct usb_device *dev, unsigned int pipe, __u8 request,

??????????????????????????? __u8 requesttype, __u16 value, __u16 index, void *data,

??????????????????????????? __u16 size, int timeout);

Request是控制消息的USB請求值、requesttype是控制消息的USB請求類型,value是控制消息的USB消息值,index是控制消息的USB消息索引。具體是什么,暫時不是很清楚,希望大家提供說明。

至此,Linux下的USB驅(qū)動框架分析基本完成了。


本站聲明: 本文章由作者或相關(guān)機構(gòu)授權(quán)發(fā)布,目的在于傳遞更多信息,并不代表本站贊同其觀點,本站亦不保證或承諾內(nèi)容真實性等。需要轉(zhuǎn)載請聯(lián)系該專欄作者,如若文章內(nèi)容侵犯您的權(quán)益,請及時聯(lián)系本站刪除。
換一批
延伸閱讀

LED驅(qū)動電源的輸入包括高壓工頻交流(即市電)、低壓直流、高壓直流、低壓高頻交流(如電子變壓器的輸出)等。

關(guān)鍵字: 驅(qū)動電源

在工業(yè)自動化蓬勃發(fā)展的當下,工業(yè)電機作為核心動力設(shè)備,其驅(qū)動電源的性能直接關(guān)系到整個系統(tǒng)的穩(wěn)定性和可靠性。其中,反電動勢抑制與過流保護是驅(qū)動電源設(shè)計中至關(guān)重要的兩個環(huán)節(jié),集成化方案的設(shè)計成為提升電機驅(qū)動性能的關(guān)鍵。

關(guān)鍵字: 工業(yè)電機 驅(qū)動電源

LED 驅(qū)動電源作為 LED 照明系統(tǒng)的 “心臟”,其穩(wěn)定性直接決定了整個照明設(shè)備的使用壽命。然而,在實際應(yīng)用中,LED 驅(qū)動電源易損壞的問題卻十分常見,不僅增加了維護成本,還影響了用戶體驗。要解決這一問題,需從設(shè)計、生...

關(guān)鍵字: 驅(qū)動電源 照明系統(tǒng) 散熱

根據(jù)LED驅(qū)動電源的公式,電感內(nèi)電流波動大小和電感值成反比,輸出紋波和輸出電容值成反比。所以加大電感值和輸出電容值可以減小紋波。

關(guān)鍵字: LED 設(shè)計 驅(qū)動電源

電動汽車(EV)作為新能源汽車的重要代表,正逐漸成為全球汽車產(chǎn)業(yè)的重要發(fā)展方向。電動汽車的核心技術(shù)之一是電機驅(qū)動控制系統(tǒng),而絕緣柵雙極型晶體管(IGBT)作為電機驅(qū)動系統(tǒng)中的關(guān)鍵元件,其性能直接影響到電動汽車的動力性能和...

關(guān)鍵字: 電動汽車 新能源 驅(qū)動電源

在現(xiàn)代城市建設(shè)中,街道及停車場照明作為基礎(chǔ)設(shè)施的重要組成部分,其質(zhì)量和效率直接關(guān)系到城市的公共安全、居民生活質(zhì)量和能源利用效率。隨著科技的進步,高亮度白光發(fā)光二極管(LED)因其獨特的優(yōu)勢逐漸取代傳統(tǒng)光源,成為大功率區(qū)域...

關(guān)鍵字: 發(fā)光二極管 驅(qū)動電源 LED

LED通用照明設(shè)計工程師會遇到許多挑戰(zhàn),如功率密度、功率因數(shù)校正(PFC)、空間受限和可靠性等。

關(guān)鍵字: LED 驅(qū)動電源 功率因數(shù)校正

在LED照明技術(shù)日益普及的今天,LED驅(qū)動電源的電磁干擾(EMI)問題成為了一個不可忽視的挑戰(zhàn)。電磁干擾不僅會影響LED燈具的正常工作,還可能對周圍電子設(shè)備造成不利影響,甚至引發(fā)系統(tǒng)故障。因此,采取有效的硬件措施來解決L...

關(guān)鍵字: LED照明技術(shù) 電磁干擾 驅(qū)動電源

開關(guān)電源具有效率高的特性,而且開關(guān)電源的變壓器體積比串聯(lián)穩(wěn)壓型電源的要小得多,電源電路比較整潔,整機重量也有所下降,所以,現(xiàn)在的LED驅(qū)動電源

關(guān)鍵字: LED 驅(qū)動電源 開關(guān)電源

LED驅(qū)動電源是把電源供應(yīng)轉(zhuǎn)換為特定的電壓電流以驅(qū)動LED發(fā)光的電壓轉(zhuǎn)換器,通常情況下:LED驅(qū)動電源的輸入包括高壓工頻交流(即市電)、低壓直流、高壓直流、低壓高頻交流(如電子變壓器的輸出)等。

關(guān)鍵字: LED 隧道燈 驅(qū)動電源
關(guān)閉