基于Linux的USB主、从设备之间的三种通信方式
基于Linux的USB主/从设备之间的三种通信方式
随着简单易用的USB接口日益流行,在嵌入式系统中添加对USB接口的支持已成为大势所趋。本文通过介绍Linux中支持USB的各种模块和库,分析了在Linux上利用USB实现高速串口和以太网连接等通信方式的具体方法。
通用串行总线(USB,Universal Serial Bus)是一种非常实用的通信接口,其应用日益广泛。有三种方法可以使运行Linux操作系统的嵌入式系统支持USB接口,本文将对这三种方法逐一进行介绍。
基于Linux的USB设备与USB主机一般有以下三种通信方式:1.一些功能最完备结构也最复杂的设备采用用户定制核模块来实现在标准USB总线上运行复杂的高级协议,而由USB主机上相应的用户驱动程序和应用来完成连接。2.另一些基于Linux的USB设备则利用USB总线来实现与主机上所运行的某个应用的简单的点对点串行连接。主机上的应用虽然利用了主操作系统所提供的USB编程接口,但表面看来却似乎是在通过一个典型的串口进行通信。3.最后,还有些设备以主计算机作为网关,将USB设备连接到办公局域网或互联网上,从而使USB设备看起仿佛构成了一个以太网。这种方法专业性较强,但通常可行,是主机驱动程序
使该方法成为可能。
分组图标在这三种方法中,您可以根据预留给开发的时间长短和期望USB接口在嵌入式应用中所扮演的角来决定选用那一种方法比较恰当。为了帮助您做出正确的选择,下一节将向您介绍这三种方法分别应用于基于Linux的USB设备时的情况,但首先让我们对USB接口做一个大致介绍。
适合中秋节发朋友圈的句子
USB概述
USB是一种方便快捷的接口,可用于为计算机工作站连接一些小配件。根据USB规的定义,鼠标、键盘、音频播放和录音设备、照相机、大容量存储设备以及许多其他设备均可以通过USB接口,以高达480Mbps的速度连接到一台主计算机。协议定制者对USB上运行的这种复杂的主从式协议做出了仔细的说明,这就帮助保证了所有这些设备之间具备互操作性和兼容性。例如,该协议规定,USB设备只有在被询问时才可以回答,并且USB主机会根据所连接的USB设备类型的不同,采用某些特定的格式,在某些特定的时间段从不同的设备获取数据。 花海
USB设备和主机之间通常通过专用的总线控制芯片建立连接。在USB主机上,名为UHCI或OHCI等的控制芯片通过插卡形式加入主机或直接集成到工作站的主板上。在主机一端的总线控制驱动程序管理着主机控制芯片,它同时还跟踪监视着主机目前连接的是哪些USB设备,从而决定应如何与它们通信。
可用于连接照相机和鼠标之类USB设备的总线控制器有很多种。其中的一种就在一块芯片上同时集成了USB接口以及另一端的串口、I2C接口或并口。USB控制器(包括主机上的和USB设备上的控制器)也可能集成到英特尔StrongARM或 Hitachi H8之类的微控制器中去。这些芯片及其外围部件有点类似以太网和CAN控制器,不同的是他们用于连接USB设备,并运行USB协议。
很多人都知道Linux操作系统中包含了USB主机控制器的驱动程序,因而USB键盘、数码相机以及其他一些USB设备都可以在一个运行Linux操作系统的桌面工作站上使用。但很少有人知道Linux中还包含了一组USB设备控制器的驱动程序,尤其是集成到StrongARM SA1110处理器中的控制器。有了这些控制器驱动程序,基于Linux的嵌入式系统就能利用USB接口来与主计算机(运行Linux或其他操作系统)通信。大多数USB通信的实现过程都是
双端的。主机利用一个核模块或驱动程序来与USB设备通信,而USB设备则通过其自身的驱动程序来与主机通信。根据主机和USB设备所采用的通信风格的不同,驱动程序可以很简单明白,也可以很复杂,很具挑战性。本文主要关注USB设备端的通信过程,但也在适当的地方包含了关于主机端通信过程实现的信息。
以下讨论的技术应当引起读者的注意。本文的目的是介绍如何在数码相机和PDA等基于Linux的USB设备上使用Linux。此处所指的USB设备是严格意义上的USB设备,即带正方形连接器的完整的设备,而不是哪些连接器形状为扁平矩形的设备。此外,USB连接的另一端(通常是一台PC工作站),应该是一台USB主机。
关于USB信息包的格式和通信参数的详细信息,见本文的参考文献。
买房合同注意事项
通过编写核模块添加USB接口
1. USB设备端通信过程
向一个基于Linux的设备中添加USB接口的第一种方法是编写一个用户定制的Linux核模块,这也是可实现最完备功能的一种做法。采用这种方法时通常需要针对主机的操作系统(
Windows, Linux等)开发相应的驱动程序。
清华同方驱动下载
一旦在设备中实现了用户定制的核模块,就可以使该设备完成相当复杂的功能,例如仿真一个文件系统,从而允许嵌入式应用将其USB主机当作一个远程存储设备。除此以外,采用这种方法之后,设备还可以具备存储转发(store-and-forward)的功能,因而能够在与USB主机的连接建立之前对来自嵌入式应用的数据流进行缓冲。
在基于StrongARM的Linux设备中,核代码用于管理芯片所携带的USB设备控制器外设,通过调用函数sa1100_usb_open()来初始化。在初始化之后,核模块还会调用函数sa1100_usb_get_descriptor_ptr() 和sa1100_usb_set_string_descriptor()来设置在设备查询期间传送给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()函数。接着,在数据传送结束之后,核模块会调用回调函数。
在bedded./code.htm(arch/arm/mach-sa1100/usb-char.c)可以到一个叫做usb-char的模块,这是一个很好的设备端SA1110 Linux USB模块的例子。该模块将USB设备与USB 主机之间的连接变成一种高速串行。此外, usb-eth( arch/arm/mach-sa1100/usb-eth.c)模块也是个不错的例子,该模块将USB变成了一种虚拟的以太型网络。后面会深入探
讨这两种模块。
2. USB主机端通信过程
有些很好的主机端USB驱动程序的例子是随主流Linux操作系统的发布而提供的,位于The Linux Kernel Archives ()发布的原始核源代码中。其中,Handspring Visor 模块(drivers/usb/serial/visor.c)是一个编写得更清晰,也更易理解的模块,它同时也是USB 主机端模块(drivers/usb/usb-skeleton.c)的模板。
利用USB实现高速串行通信
1. USB设备端通信过程
为了达到最实用的效果,我们可以将USB总线简单地看作一个高速串口,然后,在一些嵌入式设备和应用中,我们就可以用USB接口来模拟串口。StrongARM处理器的Linux核就提供了一个名为usb-char的USB设备驱动程序,它所完成的恰好就是用USB模拟串口的功能。
当需要与USB 主机通信时,Linux操作系统中的USB设备应用只是简单地打开一个与其usb-char设备节点的连接(连接类型为字符型,major number 为10, minor 为240),然后就开始读写数据。在与USB 主机的连接建立之前,read()和write()操作均返回一个错误信息。一旦连接建立好,并且设备查询完成之后,USB接口就开始象一个点对点的串口一样与主机进行通信。
这种进行USB数据传送的方法非常简单有效,因而usb-char设备模块发布之后一直很受欢迎。而且,该模块还为通过其他方法进行USB通信提供了一个参考。
在usb-char中,真正的操作开始于usbc_open()函数,列表1给出了函数的一部分代码。笔者由于临时的兴趣,对该代码做了一点修改,取消了错误和超时句柄。在此向代码的原作者Brad Parker、Nicolas Pitre 和Ward Willats致歉。
非主流游戏名字
twiddle_descriptors()函数用于设置设备的USB描述符。在描述符设置好之后,我们就可以开始进行设备查询,并从USB 主机接收一帧数据。kick_start_rx()函数段的代码主要用于调用sa1100_usb_recv(),建立回调。
在USB主机发送一个数据包时,设备的核模块会通过回调方式调用rx_done_callback_packet_buffer()函数,将数据包的容送入一个FIFO队列,以便能通过read()函数将该数据包返回给usb-char设备节点。

版权声明:本站内容均来自互联网,仅供演示用,请勿用于商业和其他非法用途。如果侵犯了您的权益请与我们联系QQ:729038198,我们将在24小时内删除。