引言 通用串行總線(USB)是一種快速而靈活地連接配件與計算機工作站的接口,其應用非常廣泛。Linux中除了包含對USB主機控制器的驅動,還含有USB設備控制器,尤其是集成在StrongARM SA1110處理器上的控制器的驅動。這些控制器驅動通過使用USB可使基於Linux的嵌入式系統與主機 (運行的可以是Linux,或不是)進行通信。這裡提供三種方法給運行Linux操作系統的嵌入式系統增加USB支持,可采用其中一種與USB主機展開通信。 第一種,最復雜的設備采用專門編寫的內核模塊解析標准USB總線上通行的錯綜復雜的高層協議;相應的USB主機定制驅動和應用程序來完成連接。第二種,有些基於Linux的設備把總線當作一種簡單的運行在主機上的點對點串行連接使用;主機應用程序采用主機操作系統提供的USB編程界面,而其外在表現則仿佛是在通過一種典型的串行端口進行通信。第三種,另有一些設備把USB看作一種以太網絡,它們用主機作網關,把USB設備與辦公LAN或 Internet相連接。通常的做法是使用專門的主機驅動實現它。 最佳方案的選擇取決於研發所需時間,以及針對具體嵌入式應用,要把USB接口作成什麼樣。以下對這三種方法如何在基於Linux的USB設備上的應用逐一進行描述。本文是關於如何在基於Linux的照相機和PDA之類的USB設備上使用Linux的論述,在此,USB是指由方形連接器而非扁平矩形連接器構成的USB設備。 內核模塊 把USB加到基於Linux的設備上的第一種方法是編寫一個定制的Linux內核模塊。這種方法通常要求相應開發主機操作系統(Windows、Linux以及其它OS)的驅動。 借助定制內核模塊在設備中的安裝,可以進行文件系統仿真等,使嵌入式應用將其USB主機當作遠程存儲設備對待。這一方法的另一潛在用途是構成一種存儲轉發字符設備,從嵌入式應用程序中緩沖數據流,直到USB主機連接完成建立為止。 對於基於StrongARM的Linux設備,其USB應用內核模塊調用sa1100_usb_open(),對管理芯片的板上USB設備控制器外設的內核代碼進行初始化。然後該模塊調用sa1100_usb_get_descriptor_ptr()和 sa1100_usb_set_string_descriptor(),通過枚舉過程對USB主機的給定USB描述符進行設置。這些描述符包括設備供貨商及產品的數字標識符、正文字符串等主機可用來對設備進行識別的信息。甚至有一個序列號域,以便主機唯一地識別設備或對USB上相同設備的多個實例加以區分。 內核模塊必須在開始USB通信前完成USB描述符的建立,這是因為枚舉過程由USB設備控制器驅動,一旦USB主機連上後會自動執行。一切准備就緒後,USB設備模塊便調用sa1100_usb_start(),告訴內核接受來自主機的USB連接請求。如果模塊在USB主機連上前調用 sa1100_set_configured_ callback(),那麼內核將會在枚舉過程結束時調用所提供的回調函數。回調函數能很好地對設備完成連接狀態進行可視化指示。 如果USB通信不再需要,那麼設備的內核模塊便調用sa1100_usb_stop(),然後是 sa1100_usb_close(),關閉SA1100的USB控制器。 StrongARM USB控制器支持數據傳輸作業的bulk-in 和bulk-out。在從USB主機接收數據包時,內核模塊調用sa1100_usb_recv(),把數據緩沖區和回調函數地址傳遞給它。然後內核的底層USB設備控制代碼對來自主機的bulk-out包進行檢索,把內容放於緩沖區中,並調用回調函數。 回調函數必須從接收緩沖區提取數據並保存於其它位置或者把緩沖區空間加到一個隊列中,為下一個數據包的接收分配新的緩沖區。而後回調函數二次調用sa1100_usb_recv(),在需要時進行下一個數據包的接收。過程與對USB主機的數據傳輸相類似。在聚集起一幀的數據量後,內核模塊將數據的地址、長度和回調地址傳遞給 sa1100_usb_send()。傳輸完成時,內核調用回調函數。 主機端USB驅動的幾個例子在主流的Linux版本以及 Linux內核檔案組織分配的原始內核源中都有提供。用於Handspring Visor(drivers/usb/serial/visor.c)的模塊是編寫較為簡潔易懂的模塊之一,作為USB主機端模塊的模板 (drivers/usb/usb-skeleton.c)使用。 高速串行 對於大多數實際應用來說, 可以把USB總線當作一種高速串行端口考慮。如此在某些類型的嵌入式設備和應用中對它進行原型模擬是有意義的。StrongARM處理器的Linux內核提供現成的USB設備驅動專工於此,稱作usb-char。 在希望與USB主機通信時,Linux USB設備應用程序只是打開對其usb-char設備節點(字符型,最大10,最小240)的連接,然後開始讀寫數據即可。read()和 write()操作將一直返回錯誤值直到USB主機連上為止。一旦連接建立和枚舉完成,便開始通信,就像USB是一種點對點串行端口一樣。 由於這種USB數據傳遞方法十分直接且實用,因此usb-char設備得到高效使用。它還為其它USB通信方法的實現提供了重要的參照基准。 usb-char的實際動作從usbc_open()功能開始,部分內容示於列表1中。 列表1:打開USB上的串行連接 static int usbc_open(strUCt inode *pInode, struct file *pFile) { int retval = 0; /* start usb core */ sa1100_usb_open(_sb-char?; /* allocate memory for in-transit USB packets */ tx_buf = (char*) kmalloc(TX_PACKET_SIZE, GFP_KERNEL GFP_DMA); packet_buffer = (char*) kmalloc(RX_PACKET_SIZE, GFP_KERNEL GFP_DMA); /* allocate memory for the receive buffer; the contents of this buffer are provided during read() */ rx_ring.buf = (char*) kmalloc(RBUF_SIZE, GFP_KERNEL); /* set up USB descriptors */ twiddle_descriptors(); /* enable USB i/o */ sa1100_usb_start(); /* set up to receive a packet */ kick_start_rx(); return 0; twiddle_descriptors ()功能建立起設備的USB描述符。在描述符全部建起後,准備從USB主機枚舉並接收一個數據幀。kick_start_rx()所需的代碼大多數情況下只是一種對sa1100_usb_recv() 的調用以建立回調而已。當USB主機發送數據包時,設備的內核通過回調調用rx_done_callback_packet_buffer()函數,把數據包的內容移入usb-char 設備點上由read()返回的FIFO隊列。 主機 對於運行Linux的USB主機,usb-char相應的USB主機模塊稱為usbserial模塊。大多數Linux版本都包括Usbserial模塊,盡管通常不是自動裝入。在USB與設備的連接建立之前,usbserial 由modprobe 或 insmod載入。 一旦USB設備開始枚舉,主機上的應用程序便用usbserial設備點(字符型,最大188,最小0以上)之一與設備進行通信。這些節點通常命名為/dev/ttyUSBn。Usbserial模塊在內核報文日志記錄中報告它把哪個節點指定給USB設備使用: usbserial.c: 通用轉換器刪除 usbserial.c: 通用轉換器當前連到ttyUSB0上。連接建立後,USB主機上的應用程序便通過讀寫指定的節點與USB設備進行通信。 Linux 主機上usbserial模塊的一種替代選擇是一種稱作libusb(libusb.sourceforge.net)的庫。這種庫使用低層內核系統調用進行USB數據傳輸,而不是通過usbserial模塊,在某種程度上跨Linux內核版本建立和使用時更方便。Libusb庫還提供大量有用的調試功能,這一點在對運行在USB鏈路上的復雜通信協議進行除錯時有幫助。用libusb與采用usb-char的USB設備進行通信時,Linux主機應用程序使用usb_open()函數建立與該設備的連接。然後應用程序使用usb_bulk_read()和usb_bulk_write()與設備交換數據。 USB上的以太網 另一種選擇是把USB作為一種以太網絡來對待。Linux具有在主機和設備端均可實現這種功能的模塊。由於iPAQ硬件既沒有可接入的串行端口也沒有一種專用的網絡接口,因此,iPAQ 的Linux內核專門采用這種通信策略,在StrongARM的Linux內核中,usb-eth模塊(arch/arm/mach- sa1100/usb-eth.c)對用USB作物理媒介的虛構以太網設備進行仿真。一旦創建後,這一網絡界面便被指定一個IP地址,否則作為通常的以太網硬件對待。一旦USB主機連上後,usb-eth模塊便能使USB設備“看到” Internet(如果存在Internet的話),ping測其它IP地址,甚至“談論”DHCP, HTTP, NFS, telnet, 和e-mail。簡言之,任何在實際的以太網界面上運行的應用將不折不扣地在usb-eth接口上得到實現,因為它們不能分辨出其正在使用的不是實在的以太網硬件。 在Linux主機上,相應的Ethernet-over-USB內核模塊稱為usbnet。當usbnet模塊得到安裝且設備的USB連接建立完成時,usbnet模塊便針對主機端內核及用戶應用創建一個與實際硬件酷似的虛構以太網界面,主機端應用程序通過運行設備IP地址 ping測,可以檢查USB設備的存在。如果ping測成功,設備便加上了。 結語 Linux不再只是USB主機使用,當今它也是USB設備的合適選擇,Linux 下的USB通信是非常靈活和易用的。