- 浏览: 75451 次
- 性别:
- 来自: 北京
序言:
前面我们提到,设备驱动程序的主要功能操作设备,更准确的说就是如何操作设备寄存器或设备内存。不同的计算机体系结构提供了不同的设备操作接口,主要就是 IO端口映射(Ports)或IO内存映射(Memory-Map )。例如X86平台,它对设备的访问就同时提供了IO端口映射方式或IO内存映射方式,这个在大学的汇编语言课程里有详细的介绍,当然还有一些平台紧提供 IO内存映射。IO端口映射方式是CPU提供了独立的地址空间给设备IO,并且使用特定的汇编指令操作IO端口。IO内存映射方式提供了统一的内存编址方 式来访问设备IO,就像你访问系统内存一样。
通常对于一个给定的硬件平台电路板,它的设备寄存器或内存的物理地址就是确定的了,或者是相对确定的了(它们具有自己的IO地址空间)。但对于向 Linux这样的操作系统,驱动程序是不能直接访问设备的物理地址的,它必须把设备的物理地址映射到Linux内核的虚拟地址空间,这样驱动程序才能通过 虚拟地址操作设备。
IO区域:
Linux中使用IO区域(IO Region)来管理设备IO无论它是IO端口映射还是IO内存映射。IO区域是基于IO资源(Resource)来实现的,我们首先来看看IO资源在Linux里的定义:
struct resource {
resource_size_t start;
resource_size_t end;
const char *name;
unsigned long flags;
struct resource *parent, *sibling, *child;
};
#define IORESOURCE_IO 0x00000100
#define IORESOURCE_MEM 0x00000200
#define IORESOURCE_IRQ 0x00000400
#define IORESOURCE_DMA 0x00000800
extern int request_resource(struct resource *root, struct resource *new);
extern int release_resource(struct resource *new);
很明显,它是一个树结构。Linux里将IO资源分成不同的类型,如IO(Port)、MEM、IRQ、DMA,同时内核提供了IO Resource的操作函数,用于分配、请求、释放IO资源。
如果管理的IO资源有多个,直接使用IO资源函数就显得有些麻烦,还好Linux可以使用IO区域来管理这些资源,具体来说,就是Linux定义了一些宏管理IO资源,定义在头文件中,如下:
#define request_region(start,n,name) __request_region(&ioport_resource,(start), (n),(name))
#define request_mem_region(start,n,name)__request_region(&iomem_resource,(start), (n),(name))
#define release_region(start,n) __release_region(&ioport_resource, (start), (n))
#define check_mem_region(start,n) __check_region(&iomem_resource, (start), (n))
#define release_mem_region(start,n) __release_region(&iomem_resource, (start), (n))
在实际的编程中,我们基本上是使用这些宏来操作IO资源,即使你只有一个IO资源,这样可以保证程序的可扩展性和跨平台的兼容性。当然,你必须获取到IO 资源后才可以在Linux内核中操作IO设备。因此,一般来说,你需要在驱动的初始化函数在调用IO区域请求函数来获取IO区域。
最后要说明一点,就是这些宏操作的IO资源有两类,分别是ioport_resource和iomem_resource,他们定义在中:
struct resource ioport_resource = {
.name = "PCI IO",
.start = 0,
.end = IO_SPACE_LIMIT,
.flags = IORESOURCE_IO,
};
struct resource iomem_resource = {
.name = "PCI mem",
.start = 0,
.end = -1,
.flags = IORESOURCE_MEM,
};
IO 端口映射:
在一些平台,特别是X86平台,外设通常具有一个独立的地址空间,叫IO地址空间,对IO地址空间的访问必须使用特定的IO指令(如x86的IN/OUT 指令)。还有一些平台并没有IO地址空间,所有的IO都是内存映射(memory-mapped)的,为了提供程序的跨平台及兼容性,Linux为那些并 不支持IO地址空间的平台提供了IO端口操作函数,他们实际上还是通过访问IO内存映射地址来访问的。因此,不管你的程序是使用IO端口映射还是IO内存 映射,它都可以很好的运行到各种平台上。
回到我们的主题-IO端口映射。前面我们提到,在使用IO设备之前我们必须向Linux内核申请使用的资源,因此通常在我们的设备初始化函数或探测函数之中会有如下的代码:
if (!request_region(io_addr, IO_NUM, DRV_NAME))
return -ENODEV;
如果成功申请了IO端口资源,那么我们就可以调用IO端口访问函数来访问IO端口了,它们通常定义在头文件中(每个平台的定义都有所不同,但类似于下表)具体请参考头文件:
inb(unsigned port)
outb(u8 v, unsigned port)
inw(unsigned port)
oubw(u16 v, unsigned port)
inl(unsigned port)
outl(u32 v, unsigned port)
IO内存映射(memory-mapped)
一些新的驱动程序都会使用IO内存映射方式来访问IO设备,因为有些平台仅仅支持IO内存映射,如ARM平台。通常来说,使用IO内存就象使用系统RAM 内存一样的简单,确实有些平台是支持这样的访问的,但还是有些平台不能象访问内存那样直接使用IO内存地址来访问外设IO(Register和RAM), 因此内核提供了一组通用的API来支持程序的跨平台及可移植性。
Linux内核提供了两种操作IO内存的函数,一组类似于IO端口函数用于读取1、2、4个字节数据,定义在头文件中:
ioread8(p)
ioread16(p)
ioread32(p)
iowrite8(v,p)
iowrite16(v,p)
iowrite32(v,p)
这些是新的IO内存操作函数,我们推荐你使用这些函数。如果你浏览Linux内核,你会发现还有其他一些函数接口,它们是老的IO内存操作函 数,Linux内核会慢慢舍弃这些函数接口,因此尽量不要使用这些函数,但有必要在这里把这些函数列出来,因为确实还是有一些新的驱动仍然使用它们(参 考):
readb()
readw()
readl()
writeb()
writew()
writel()
如果你想象操作内存那样成块的操作IO内存,内核提供了另外的方法,它们类似于内存操作函数。推荐你使用这些函数操作IO内存而不是直接使用IO内存地址,这样你的程序可以移植到不同的平台上,它们定义在头文件中。
extern void _memcpy_fromio(void *, const volatile void __iomem *, size_t);
extern void _memcpy_toio(volatile void __iomem *, const void *, size_t);
extern void _memset_io(volatile void __iomem *, int, size_t);
同样在使用IO内存之前,你需要向Linux内核申请IO区域:
if (!request_mem_region(mapbase, size, DRVNAME)) {
ret = -EBUSY;
break;
}
申请完IO区域后,你还不能直接使用它们,你必须把这个地址映射到Linux内核的虚拟地址空间中来,这个操作是通过ioremap函数来实现的,请参考头文件:
membase = ioremap(mapbase, size);
通过这两步操作后,你就可以调用IO内存函数来访问设备IO了。
IO端口重映射
这里说的IO端口重映射不是ioremap的功能,ioremap是将IO内存映射到Linux内核的虚拟地址空间中。我们说的IO端口重映射是将IO端 口映射为IO内存,这样就可以象操作IO内存一样操作IO端口了。这样做的好处是我们可以统一驱动程序的接口(都使用IO内存映射),避免为同一个设备提 供不同的驱动接口。这个函数同样定义在头文件中:
extern void __iomem *ioport_map(unsigned long port, unsigned int nr);
extern void ioport_unmap(void __iomem *addr);
有一点要注意,在调用完IO端口重映射后,还是需要调用ioremap函数把它映射到Linux内核的虚拟地址空间中来。
后记
设备IO的操作就是这些了,其实在我们的编程中只要调用几个简单的函数或宏就可以完成IO端口的操作了。这里有个问题没有说明,就是设备的访问是需要同步 的或着需要延时等待一段时间才能进行下一步的操作。我们在《内核同步技术》一章有个简单的介绍,后面我们将补充几个例子来进一步说明如何进行IO操作。
发表评论
-
排序算法---计数排序
2011-11-27 14:57 609#include <stdio.h> vo ... -
排序算法---归并排序
2011-11-26 19:33 748#include <stdio.h> vo ... -
排序算法---交换排序(冒泡排序、快速排序)
2011-11-26 19:32 702#include <stdio.h> vo ... -
排序算法---选择排序(简单插入排序、堆排序)
2011-11-26 19:31 649#include <stdio.h> vo ... -
排序算法---插入排序(简单排序、shell排序)
2011-11-26 19:29 650#include <stdio.h> vo ... -
删除字符串中的特定字符和重复字符
2011-11-26 13:45 666#include <stdio.h> vo ... -
Linux编程-多线程、同步和互斥(转载)
2011-11-14 15:27 1209http://www.cnblogs.com/skynet/a ... -
寻找字符串中的最大数字子串
2011-09-22 17:17 1523#include <stdio.h> int f ... -
删除子字符串
2011-09-21 15:27 604#include <stdio.h> #incl ... -
c语言随机数
2011-09-18 17:15 687#include <stdio.h> #i ... -
带头结点有序单链表的合并
2011-09-08 14:21 1184typedef int Item; typedef s ... -
链表逆序的递归/非递归算法
2011-09-01 23:37 1412/** *链表逆序的递归/非递归算法 */ # ... -
递归算法---字符串---全/部分组合和全排列
2011-08-30 23:01 1222#include <stdio.h> #i ... -
递归算法---0-1背包问题(面试宝典)
2011-08-28 21:11 1901/** *正整数n,m,从数列1、2、3、...、n中随 ... -
递归算法---字符串全组合(面试宝典)
2011-08-28 17:24 1257/** *求一字符串所有字串的组合 */ #i ... -
递归算法---求解多元一次方程
2011-08-28 10:38 1898/** * 求解x1+x2+x3+...+x10 = ... -
(zz)关于类的sizeof
2011-08-27 18:16 587http://blog.sina.com.cn/s/blog_ ... -
(zz)结构体字节对齐原则
2011-08-27 17:53 1579结构体默认的字节对齐一般满足三个准则: 结构体变量的首 ... -
list.h from linux-2.4
2011-08-25 09:59 607#ifndef _LIST_H_ #define _L ... -
The C Programming Lang (K&R) hash table
2011-08-25 09:52 936hash.h #include <stdio.h ...
相关推荐
对于更复杂的设备,`file_operations`结构体可能会包含更多的函数,如`ioctl`用于处理设备特有的控制命令,`llseek`用于设置文件指针的位置,`poll`用于支持非阻塞I/O等。驱动程序开发者需要根据实际设备的需求实现...
在Linux操作系统中,`select`系统调用是一个非常重要的I/O多路复用机制,它允许程序同时监控多个文件描述符(file descriptor,简称句柄)的状态变化,以便在任何句柄准备好进行读写操作时能够及时响应。这篇文章将...
5. **I/O模型**:讨论阻塞I/O、非阻塞I/O、多路复用(如select、poll、epoll)和异步I/O,帮助开发者设计高效的I/O处理程序。 6. **标准I/O库**:介绍stdio.h头文件中的函数,如printf、scanf系列,以及文件位置...
10. **系统监控**:了解资源监控工具,如htop、iostat、vmstat,监控CPU、内存、磁盘I/O和网络状态。 11. **防火墙与安全**:理解iptables防火墙规则,设置端口开放和关闭,了解SSH安全实践,如禁用密码登录、设置...
20. i节点:i节点是一个固定长度的表,通常为64字节,包含文件的相关信息,如文件大小、权限、所有权等。 21. 文件权限标志:在Linux中,文件权限的标志符号依次是`rwx`,代表读、写、执行。 22. Linux系统中,更...
另外,要转载请保持本文件的完整性,请尊重别人的劳动果实. 修改日志: 08/3/29 修改了一下Makefile,旧的Makefile在某些文件更新后还会重新编译。 修改了Trans.cpp中的一个dug,该dug在translate MinGW gcc编译的程序...
### 新手入门 Linux 菜鸟必学的60个命令 #### 登录与退出 (Login and Logout) **登录命令 (login):** - **功能:** `login` 命令用于用户登录系统。 - **用法:** `login [name] [参数]` - `[name]`: 用户名 - `...
Linux核心对名字空间的支持完全隔离了工作环境中应用程序的视野,包括进程树、网络、用户ID与挂载文件系统,而核心的cgroup提供资源隔离,包括CPU、存储器、block I/O与网络。从0.9版本起,Dockers在使用抽象虚拟是...
它采用事件驱动模型,如epoll(Linux上的高效I/O多路复用技术)。 2. **线程模型**:muduo库实现了主-从线程模型,主线程负责接收连接请求,而工作线程处理实际的业务逻辑。这种设计有助于避免锁竞争,提高并发性能...
* 主机一配置要求包括:CPU类型、处理器个数、TPCC值要求、内存、内置磁盘容量、系统带宽、I/O带宽、实际配置系统I/O插槽、内置DVD-ROM、快速以太网口、多模光纤网卡、FC光纤通道卡等 * 操作系统:UNIX,支持中文...
另外,要转载请保持本文件的完整性,请尊重别人的劳动果实. 修改日志: 08/3/29 修改了一下Makefile,旧的Makefile在某些文件更新后还会重新编译。 修改了Trans.cpp中的一个dug,该dug在translate MinGW gcc编译的程序...
然而,在没有操作系统的情况下,Newlib的一些功能,如标准I/O流(stdio),是无法直接使用的,因为它们依赖于OS的stdio子系统。 在标题和描述中提到的问题,主要聚焦在两个方面:如何在裸奔程序中使用C库的数学运算...
- **启用按任务延迟计费**:记录进程等待系统资源(如CPU时间、I/O操作)的时间。 - **UTS命名空间支持**:提供对UTS命名空间的支持,允许进程拥有不同的主机名和域名等信息。 - **审计支持**:某些安全模块(如...
"o=josua 0 0 IN IP4 %s\r\n" "s=conversation\r\n" "c=IN IP4 %s\r\n" "t=0 0\r\n" "m=audio %s RTP/AVP 0 8 101\r\n" "a=rtpmap:0 PCMU/8000\r\n" "a=rtpmap:8 PCMA/8000\r\n" "a=rtpmap:101 telephone-...