您的位置:首页 > 运维架构 > Linux

Linux的uio驱动机制的由来和优缺点(一)

2015-10-05 15:46 916 查看
【摘要】linux用户态的设备驱动开发:并不是所有的设备驱动程序都要在内核编写,有些情况下,在用户空间编写驱动程序能够更好地解决遇到的问题。本文对用户态驱动优缺点进行分析。


1、用户空间驱动程序的优点

1、可以和整个C库链接。

2、在驱动中可以使用浮点数,在某些特殊的硬件中,可能需要使用浮点数,而linux内核并不提供浮点数的支持。如果能在用户态实现驱动,就可以轻松解决这一问题。

3、驱动问题不会导致整个系统挂起。内核态驱动的一些错误常常导致整个系统挂起。

4、用户态的驱动调试方便。

5、可以给出封闭源码的驱动程序,不必采用GPL,更为灵活


2、用户空间驱动程序的缺点

1、无法使用中断。中断在用户空间不可用,最新的UIO接口已经解决了这一问题。

有些硬件厂商只提供和某些linux开发版(常常早就过时了)相匹配的用户空间驱动。尽管对用户空间驱动存在争议,但内核还是选择对其进行支持。最新的接口称为UIO(以前的内核也有,但新版本做了很多修改),是在2.6.22版本的一个补丁中出现的,并且在2.6.23中正式合并到了内核的代码树中。和以前相比,该接口有了一些改变。和以前的版本一样,UIO并没有完全取消内核空间代码。在内核中有一个很小的模块用于建立连接到PCI总线的设备(device)或者接口(interface),并提供中断处理程序。这一点(中断处理程序)很重要,尽管有更多的事情可以在用户空间完成,但还是需要有一个内核中的中断处理程序来通知设备停止发送中断。

该内核模块需要包括< linux/uio_driver.h>。如果他是一个PCI设备的驱动,需要按照统一设备模型的要求注册PCI驱动。当需要连接设备的时候(可能在PCI的probe()函数中),该驱动需要填写一个uio_info结构体:
<code class="hljs d has-numbering" style="display: block; padding: 0px; background-color: transparent; color: inherit; box-sizing: border-box; font-family: 'Source Code Pro', monospace;font-size:undefined; white-space: pre; border-top-left-radius: 0px; border-top-right-radius: 0px; border-bottom-right-radius: 0px; border-bottom-left-radius: 0px; word-wrap: normal; background-position: initial initial; background-repeat: initial initial;"><span class="hljs-keyword" style="color: rgb(0, 0, 136); box-sizing: border-box;">struct</span> uio_info {
<span class="hljs-built_in" style="color: rgb(102, 0, 102); box-sizing: border-box;">char</span> *name;
<span class="hljs-built_in" style="color: rgb(102, 0, 102); box-sizing: border-box;">char</span>  *<span class="hljs-keyword" style="color: rgb(0, 0, 136); box-sizing: border-box;">version</span>;
<span class="hljs-keyword" style="color: rgb(0, 0, 136); box-sizing: border-box;">struct</span> uio_mem  mem[MAX_UIO_MAPS];
<span class="hljs-built_in" style="color: rgb(102, 0, 102); box-sizing: border-box;">long</span> irq;
unsigned <span class="hljs-built_in" style="color: rgb(102, 0, 102); box-sizing: border-box;">long</span> irq_flags;
<span class="hljs-keyword" style="color: rgb(0, 0, 136); box-sizing: border-box;">void</span>  *priv;
irqreturn_t (*handler)(<span class="hljs-keyword" style="color: rgb(0, 0, 136); box-sizing: border-box;">int</span> irq, <span class="hljs-keyword" style="color: rgb(0, 0, 136); box-sizing: border-box;">struct</span> uio_info *dev_info);
<span class="hljs-keyword" style="color: rgb(0, 0, 136); box-sizing: border-box;">int</span> (*mmap)(<span class="hljs-keyword" style="color: rgb(0, 0, 136); box-sizing: border-box;">struct</span> uio_info *info, <span class="hljs-keyword" style="color: rgb(0, 0, 136); box-sizing: border-box;">struct</span> vm_area_struct *vma);
<span class="hljs-keyword" style="color: rgb(0, 0, 136); box-sizing: border-box;">int</span> (*open)(<span class="hljs-keyword" style="color: rgb(0, 0, 136); box-sizing: border-box;">struct</span> uio_info *info, <span class="hljs-keyword" style="color: rgb(0, 0, 136); box-sizing: border-box;">struct</span> inode *inode);
<span class="hljs-keyword" style="color: rgb(0, 0, 136); box-sizing: border-box;">int</span> (*release)(<span class="hljs-keyword" style="color: rgb(0, 0, 136); box-sizing: border-box;">struct</span> uio_info *info, <span class="hljs-keyword" style="color: rgb(0, 0, 136); box-sizing: border-box;">struct</span> inode *inode);
<span class="hljs-comment" style="color: rgb(136, 0, 0); box-sizing: border-box;">/* Internal stuff omitted */</span>
};</code><ul class="pre-numbering" style="box-sizing: border-box; position: absolute; width: 50px; background-color: rgb(238, 238, 238); top: 0px; left: 0px; margin: 0px; padding: 6px 0px 40px; border-right-width: 1px; border-right-style: solid; border-right-color: rgb(221, 221, 221); list-style: none; text-align: right;"><li style="box-sizing: border-box; padding: 0px 5px;">1</li><li style="box-sizing: border-box; padding: 0px 5px;">2</li><li style="box-sizing: border-box; padding: 0px 5px;">3</li><li style="box-sizing: border-box; padding: 0px 5px;">4</li><li style="box-sizing: border-box; padding: 0px 5px;">5</li><li style="box-sizing: border-box; padding: 0px 5px;">6</li><li style="box-sizing: border-box; padding: 0px 5px;">7</li><li style="box-sizing: border-box; padding: 0px 5px;">8</li><li style="box-sizing: border-box; padding: 0px 5px;">9</li><li style="box-sizing: border-box; padding: 0px 5px;">10</li><li style="box-sizing: border-box; padding: 0px 5px;">11</li><li style="box-sizing: border-box; padding: 0px 5px;">12</li><li style="box-sizing: border-box; padding: 0px 5px;">13</li></ul>


上面结构体中,name是设备的名字;version是驱动的版本号(将显示在sysfs中);irp是设备所使用的IRQ号;irq_flags是中断调用标志,将传递给request_irq();handler()是中断处理程序,除了负责应答硬件中断,一般不再做别的工作;mmap()/open()/release()由file_operations中的对应成员调用

结构体中的mem数组用于描述任何可以被映射到用户空间的内存区域。uio_mem结构体的主要成员如下:
<code class="hljs r has-numbering" style="display: block; padding: 0px; background-color: transparent; color: inherit; box-sizing: border-box; font-family: 'Source Code Pro', monospace;font-size:undefined; white-space: pre; border-top-left-radius: 0px; border-top-right-radius: 0px; border-bottom-right-radius: 0px; border-bottom-left-radius: 0px; word-wrap: normal; background-position: initial initial; background-repeat: initial initial;">struct uio_mem {
unsigned long addr;
unsigned long size;
int memtype;
void __iomem *internal_addr;
/* <span class="hljs-keyword" style="color: rgb(0, 0, 136); box-sizing: border-box;">...</span> */
};</code><ul class="pre-numbering" style="box-sizing: border-box; position: absolute; width: 50px; background-color: rgb(238, 238, 238); top: 0px; left: 0px; margin: 0px; padding: 6px 0px 40px; border-right-width: 1px; border-right-style: solid; border-right-color: rgb(221, 221, 221); list-style: none; text-align: right;"><li style="box-sizing: border-box; padding: 0px 5px;">1</li><li style="box-sizing: border-box; padding: 0px 5px;">2</li><li style="box-sizing: border-box; padding: 0px 5px;">3</li><li style="box-sizing: border-box; padding: 0px 5px;">4</li><li style="box-sizing: border-box; padding: 0px 5px;">5</li><li style="box-sizing: border-box; padding: 0px 5px;">6</li><li style="box-sizing: border-box; padding: 0px 5px;">7</li></ul>


对于每个可以映射的区域,addr是该区域的地址(物理地址);size是该区域的大小;internal_addr是由ioremap()返回的该区域地址(虚拟地址)。

memtype用于描述该区域的属性,包括:

*UIO_MEM_PHYS

表明addr是一个物理地址,通常用于I/O内存区

*UIO_MEM_LOGICAL

该区域处于内核的逻辑地址空间,例如那些通过kmalloc()获得的空间

*UIO_MEM_VIRTUAL

该区域处于内核的虚拟地址空间,这些区域由vmalloc_user()使用

一旦uio_info结构体填写完毕,驱动会将其传给如下函数:

int uio_register_device(struct device *parent, struct uio_info *info);

parent指针告诉内核该UIO设备是和哪个”real”的设备相关联,如果驱动是针对PCI设备,则parent指向pci_dev->dev。

内核空间的UIO基本上就这些API了,当设备被拔除,驱动需要调用:

void uio_unregister_device(struct uio_info *info);

最后一个和通知(note)相关的函数是:

void uio_event_notify(struct uio_info *info);

该函数的目的是通知UIO核心,发生了一个事件(典型情况是一个中断)。当真正的中断发生时,stub驱动不必调用uio_event_notify(),但这个函数可以在其他的情况下模拟中断。

在用户空间,第一个UIO处理的设备将显示为/dev/uio0(假设udev已启动)。用户空间的驱动将会打开该设备。对该设备的读操作返回一个int值,该值保存了事件计数(发生了多少次中断)。如果从上次读设备以来还没有产生中断,则读操作会阻塞直到一个中断发生(当然,支持非阻塞的操作)。文件描述符可以被发送给poll()。

在内核空间驱动中所描述的内存区域可以通过mmap()调用映射到用户空间中。传递给mmap()的参数有一些奇怪,内核的第N个区域,其offset参数的值将是当前页大小的第N倍。也就是说,如果当前系统的页大小是4096个字节,映射第一个内存区的offset值为0,而第二个内存区就是4096,第三个就是8192,以此类推。

当然了,使用UIO会受到一些限制。首先,UIO驱动是字符型的驱动,目前没有提供创建用户空间块设备或网络设备驱动的接口。另外,在用户空间也不可能建立DMA操作。但是,对于那些只包括I/O内存访问以及简单的中断处理程序的驱动,UIO接口完全可以胜任。

在UIO接口的patch中,除了UIO核心程序,还有一个示例程序。根据作者ThomasGleixner的实验,如果采用完全装入内核的方式实现驱动,则需要实现68个不同的ioctl()命令,整个驱动的长度超过5000行,对应的用户空间代码超过3000行;而如果采用UIO模式的驱动,内核代码只有156行,而用户空间代码也下降到3000行以内。

不过,linux内核的重要维护者Andrew Morton也对UIO的使用表达了一些保留意见:

“我对UIO的整个想法有一些不确定,我感觉我们应该更鼓励人们在GPL许可证下开发内核驱动,而不是鼓励他们开发不公开源代码的用户空间驱动。这些驱动往往更慢,缺乏可靠性,并且无法跨平台使用。”

在2.6.23中,和UIO相关的主要有以下文件:

(1)/drivers/uio/uio.c

uio子系统的核心文件

(2)/drivers/uio/uio_cif.c

Hilscher CIF DeviceNet以及Profibus card的驱动。


3、从dpdk分析uio驱动

uio提供了用户态驱动开发的框架,主要是由于驱动依赖的内核函数和宏因内核版本变化,导致驱动可能也需要改,所以改为用户态驱动来完成任务,这也算是类似windows的HAL的机制,当然HAL是硬件抽象层。

dpdk是如何与硬件交互的?

dpdk中的驱动需要定期去poll/select /dev/uioX 检查设备是否有中断产生,并不是用来收发数据,另外通过mmap来操作设备的设备内存。uio框架本身要处理设备的中断,中断只能在内核态处理,uio的中断处理函数也只是增加中断的计数而已。

mmap可以处理物理内存映射,逻辑内存,内核虚拟内存映射。 uio也是通过mmap将设备的内存映射到用户空间,当然用户态也可以通过/sys/class/uio/uioX/maps/mapX来实现对设备内存的访问,所以发送和接受数据是通过操作设备的内存来完成的。
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: