linux系统的硬件驱动程序编写原理

上传人:xzh****18 文档编号:34632052 上传时间:2018-02-26 格式:DOC 页数:10 大小:175.50KB
返回 下载 相关 举报
linux系统的硬件驱动程序编写原理_第1页
第1页 / 共10页
linux系统的硬件驱动程序编写原理_第2页
第2页 / 共10页
linux系统的硬件驱动程序编写原理_第3页
第3页 / 共10页
linux系统的硬件驱动程序编写原理_第4页
第4页 / 共10页
linux系统的硬件驱动程序编写原理_第5页
第5页 / 共10页
点击查看更多>>
资源描述

《linux系统的硬件驱动程序编写原理》由会员分享,可在线阅读,更多相关《linux系统的硬件驱动程序编写原理(10页珍藏版)》请在金锄头文库上搜索。

1、Linux 系统的硬件驱动程序编写原理 *本文详细地介绍如何 Linux 系统的硬件驱动程序的编写原理,指出哪些内核例程将会被调用、如何初始化驱动程序及如何分配内存等等。大家一定对 Linux 操作系统有所了解了,在此本人也不再赘述了。好吧,下面简单地介绍一下设备驱动程序。顾名思义,驱动程序是用来控制计算机外围设备的,Linux 系统将所有的外围设备都高度地抽象成一些字节的序列,并且以文件的形式来表示这些设备。我们可以来看一下 Linux 的 I/O 子系统(图 1)。图 1 Linux 的 I/O 子系统从图上我们可以看出,内核紧紧地包围在硬件周围,内核是一些软件包的组合,它们可以直接访问系

2、统的硬件,包括处理器、内存和 I/O 设备。而用户进程则通过内核提供的用户服务来和内核通讯,从而间接地控制系统硬件。我们可以通过图 2 来了解这些动作的具体情况。 图 2 用户级、内核级和硬件级三者之间的通讯图上显示了用户级的程序使用内核提供的标准系统调用来与内核通讯,这些系统调用有:open(), read(), write(), ioctl(), close() 等等。Linux 的内核是一个有机的整体。每一个用户进程运行时都好像有一份内核的拷贝,每当用户进程使用系统调用时,都自动地将运行模式从用户级转为内核级,此时进程在内核的地址空间中运行。图 3 Linux 的 I/O 子系统Linu

3、x 内核使用 设备无关 的 I/O 子系统来为所有的设备服务。每个设备都提供标准接口给内核,从而尽可能地隐藏了自己的特性。图 3 展示了用户程序使用一些基本的系统调用从设备读取数据并且将它们存入缓冲的例子。我们可以看到,每当一个系统调用被使用时,内核就转到相应的设备驱动例程来操纵硬件。每个设备在 Linux 系统上看起来都像一个文件,它们存放在/dev 目录中并被称为特殊文件或是 设备节点 。大家可以使用 ls -l /dev/lp* 来得到以下的输出: crw-rw-rw 1 root root 6, 0 April 23 1994 /dev/lp0 这行输出表示 lp0 是一个字符设备(属

4、性字段的第一个字符是c),主设备号是 6,次设备号是 0。主设备号用来向内核表明这一设备节点所代表的驱动程序的类型(比如:主设备号是 3 的块设备是 IDE 磁盘驱动程序,而主设备号为 8 的块设备是 SCSI 磁盘驱动程序);每个驱动程序负责管理它所驱动的几个硬件实例,这些硬件实例则由次设备号来表示(例如:次设备号为 0 的 SCSI 磁盘代表整个也可以说是 第一个SCSI 磁盘,而次设备号为 1 到 15 的磁盘代表此 SCSI 磁盘上的 15 个分区)。到此大家应该对 Linux 的设备有所了解了吧,下面就可以开始我们的正题 设备驱动程序。设备驱动程序是一组由内核中的相关子例程和数据组成

5、的 I/O 设备软件接口。每当内核意识到要对某个设备进行特殊的操作时,它就调用相应的驱动例程。这就使得控制从用户进程转移到了驱动例程,当驱动例程完成后,控制又被返回至用户进程。图 5 就显示了以上的过程。图 5 设备驱动程序的作用每个设备驱动程序都具有以下几个特性:l 具有一整套的和硬件设备通讯的例程,并且提供给操作系统一套标准的软件接口;l 具有一个可以被操作系统动态地调用和移除的自包含组件;l 可以控制和管理用户程序和物理设备之间的数据流。接下来我们来了解一下字符设备和块设备,它们是 Linux 系统中两种主要的外围设备。我们常见的磁盘是块设备,而终端和打印机是字符设备。块设备被用户程序通

6、过系统缓冲来访问。特别是系统内存分配和管理进程就没有必要来充当从外设读写的数据传输者了。正好与之相反的是,字符设备直接与用户程序进行通讯,而且两者似乎没有缓冲区。Linux的传输控制机制会根据用户程序的需要来正确地操纵内存和磁盘等外设来取得数据。在Linux 系统中字符设备驱动器被保存为/usr/src/linux/drivers/char 目录中。下面我们重点介绍字符设备驱动程序的开发方法。首先了解一下 Linux 的内核编程环境。我们知道每个 Linux 用户进程都在一个独立的系统空间中运行着,与系统区和其他用户进程相隔离。这样就保护了一个用户进程的运行环境,以免被其他用户进程所破坏。与这

7、种情况正相反的是,设备驱动程序运行在内核模式,它们具有很大的自由度。这些设备驱动程序都是被假设为正确和可靠的,它们是内核的一部分,可以处理系统中断请求和访问外围设备,同时它们有效地处理中断请求以便系统调度程序保持系统需求的平衡。所以设备驱动程序可以脱离系统的限制来使用系统区,比如系统的缓冲区等等。一个设备驱动程序同时包括中断和同步区域。其中中断区域处理实时事件并且被设备的中断所驱动;而同步区域则组成了设备的剩余部分,处理进程的同步事件。所以,当一个设备需要一些软件服务时,就发出一个中断 ,然后中断处理器得到产生中断的原因同时进行相应的动作。一个 Linux 进程可能会在事件发生之前一直等待下去

8、。例如,一个进程可能会在运行中等待一些写入硬件设备的信息的到来。其中一种方式是进程可以使用 sleep()和 wakeup()这两个系统调用,进程先使自己处于睡眠状态,等待事件的到来,一旦事件发生,进程即可被唤醒。举个例子来说:interruptible_sleep_on(&dev_wait_queue)函数使进程睡眠并且将此进程的进程号加到进程睡眠列表 dev_wait_queue 中,一旦设备准备好后,设备发出一个中断,从而导致设备驱动程序中相应的例程被调用,这个驱动程序例程处理完一些设备要求的事宜后会发出一个唤醒进程的信号,通常使用wake_up_interruptible(&dev_w

9、ait_queue)函数,它可以唤醒 dev_wait_queue 所示列表中的所有进程。特别要注意的是,如果两个和两个以上的进程共享一些公共数据区时,我们必须将之视为临界区,临界区保证了进程间互斥地访问公共数据。在 Linux 系统中我们可以使用 cli()和sti()两个内核例程来处理这种互斥,当一个进程在访问临界区时可以使用 cli()来关闭中断,离开时则使用 sti()再将中断打开,就像下面的写法:cli()临界区sti()除了以上这些,我们还得了解一下虚拟文件系统交换(VFS)的概念。图 6 虚拟文件系统交换图 6 中的文件操作结构在/usr/include/linux/fs.h 文

10、件中定义,此结构包含了驱动程序中的函数列表。图上的初始化例程 xxx_init()根据 VFS 和设备的主设备号来注册 文件操作结构。下面是一些设备驱动程序的支撑函数(具体使用方法详见 Linux 编程手册,使用 man 命令):add_timer() 定时间一过,可以引发函数的执行;cli() 关闭中断,阻止中断的捕获;end_request() 当一个请求被完成或被撤销时被执行;free_irq() 释放一个先前被 request_irq()和 irqaction()捕获的的中断请求;get_fs*() 允许一个设备驱动程序访问用户区数据(一块不属于内核的内存区);inb(), inb_p

11、() 从一个端口读取一个字节,其中 inb_p() 会一直阻塞直到从端口得到字节为止;irqaction() 注册一个中断;IS_*(inode) 测试 inode 是否在一个被 mount 了的文件系统上;kfree*() 放先前被 kmalloc()分配的内存区;kmalloc() 分配大于 4096 个字节的大块内存区;MAJOR() 返回设备的主设备号;MINOR() 返回设备的次设备号;memcpy_*fs() 在用户区和内核区之间复制大块的内存;outb(), outb_p() 向一个端口写一个字节,其中 outb_p()一直阻塞直到写字节成功为止;printk() 内核使用的 p

12、rintf()版本;put_fs*() 允许设备驱动程序将数据写入用户区;register_*dev() 在内核中注册一个设备;request_irq() 向内核申请一个中断请求 IRQ,如果成功则安装一个中断请求处理器;select_wait() 将一个进程加到相应 select_wait 队列中;*sleep_on() 使进程睡眠以等待事件的到来,并且将 wait_queue 入口点加到列表中以便事件到来时将进程唤醒;sti() 和 cti()相对应,恢复中断捕获;sys_get*() 系统调用,得到进程的有关信息;wake_up*() 唤醒先前被*sleep_on() 睡眠的进程;Lin

13、ux 的用户进程不能直接访问系统物理内存。每个用户进程都有自己的内存空间(用户虚拟地址空间,开始于虚拟 0 地址)。同样内核也具有自己特定的内存空间-系统虚拟地址空间。每当用户使用系统调用 read()或 write()时,设备驱动程序就在内核地址空间和用户程序地址空间之间拷贝数据。许多 Linux 例程,比如 memcpy_*fs() 和 put_fs*()可以使设备驱动程序穿越用户系统 边界来传输数据。而且数据可以是字节、字或任意长度的数据块。例如,memcpy_fromfs()可以从用户内存空间传输任意长度的数据块到设备,而get_fs_byte()则只从用户内存空间传输一个字节;相同的

14、 memcpy_tofs()和 put_fs_byte()也是如此,只不过它们是写数据到用户内存空间。然而,在内核可访问内存空间和设备本身之间传输数据则要视不同的计算机而定。一些计算机需要使用一些特殊 CPU 输入输出指令来完成这项工作,这通常被称为 DMA(直接内存访问)。而另一种方案则是使用内存映射 I/O 来解决,通常使用系统提供的 I/O 函数,比如 inb()和 outb()来分别地从 I/O 地址(即端口)读取和向 I/O 地址输出一单字节,可以使用以下的语句:unsigned char inb(int port)outb(char data, int port)好,下面就可以来看

15、看字符设备驱动程序的基本结构。如图 6 所示 xxx_write()例程轮询设备是否已经准备好接收数据,如果准备好了,则将指定长度的字符串从用户内存空间发送到字符设备。另外还可以使用中断来通知设备是否准备好,这样就不需要程序为了轮询而等待,从而提高 CPU 的利用率。xxx_table是一个结构的数组,它包含很多成员变量,包括 xxx_wait_queue 和 bytes_xfered(两者都被用于读写操作)。xxx_open() 使用request_irq()或 irqaction()来调用 xxx_interrupt()例程。为了使设备驱动程序被正确地初始化,每当系统启动时,xxx_init() 例程必须被调用。为了确保这一操作,需要将语句 mem_start = xxx_init(mem_start); 加到/usr/src/linux/driver/char/mem.c 文件的 chr_drv_init()函数末。接下来的工作就是将驱动设备安装到内核中去了(注意:字符设备驱动程序只能被安装在/usr/src/linux/drivers/char/char.a 库文件中)。

展开阅读全文
相关资源
正为您匹配相似的精品文档
相关搜索

最新文档


当前位置:首页 > 行业资料 > 其它行业文档

电脑版 |金锄头文库版权所有
经营许可证:蜀ICP备13022795号 | 川公网安备 51140202000112号