PCI是外围设备互连(Peripheral Component Interconnect)的简称,作为一种通用的总线接口标准,它在目前的计算机系统中得到了非常广泛的应用。PCI提供了一组完整的总线接口规范,其目的是描述如何将计算机系统中的外围设备以一种结构化和可控化的方式连接在一起。
PCI将计算机系统中的总线子系统与存储子系统完全地分开,CPU通过一块称为PCI桥(PCI-Bridge)的设备来完成同总线子系统的交互,尽管目前PCI设备大多采用32位数据总线,但PCI规范中已经给出了64位的扩展实现。
(1)关键数据结构
PCI设备上有三种地址空间:PCI的I/O空间、PCI的存储空间和PCI的配置空间。CPU可以访问PCI设备上的所有地址空间,其中I/O空间和存储空间提供给设备驱动程序使用,而配置空间则由Linux内核中的PCI初始化代码使用。内核在启动时负责对所有PCI设备进行初始化,配置好所有的PCI设备,包括中断号以及I/O基址,并在文件/proc/pci中列出所有找到的PCI设备,以及这些设备的参数和属性。
A,pci_driver:这个数据结构在文件include/linux/pci.h里,这是Linux内核版本2.4之后为新型的PCI设备驱动程序所添加的,其中最主要的是用于识别设备的id_table结构,以及用于检测设备的函数probe( )和卸载设备的函数remove( )。
B,pci_dev:这个数据结构也在文件include/linux/pci.h里,它详细描述了一个PCI设备几乎所有的硬件信息,包括厂商ID、设备ID、各种资源等。
(2) 基本框架
在用模块方式实现PCI设备驱动程序时,通常至少要实现以下几个部分:初始化设备模块、设备打开模块、数据读写和控制模块、中断处理模块、设备释放模块、设备卸载模块。下面给出一个典型的PCI设备驱动程序的基本框架(只列出主要引用)。
static struct pci_device_id demo_pci_tbl [] __initdata = {
{PCI_VENDOR_ID_DEMO, PCI_DEVICE_ID_DEMO, PCI_ANY_ID, PCI_ANY_ID, 0, 0, DEMO}, {0,} }; struct demo_card { unsigned int magic; struct demo_card *next; } static void demo_interrupt(int irq, void *dev_id, struct pt_regs *regs) { } static struct file_operations demo_fops = { owner: THIS_MODULE, read: demo_read, write: demo_write, ioctl: demo_ioctl, mmap: demo_mmap, open: demo_open, release: demo_release }; static struct pci_driver demo_pci_driver = { name: demo_MODULE_NAME, id_table: demo_pci_tbl, probe: demo_probe, remove: demo_remove }; static int __init demo_init_module (void) { } static void __exit demo_cleanup_module (void) { pci_unregister_driver(&demo_pci_driver); } module_init(demo_init_module); module_exit(demo_cleanup_module); 上面这段代码给出了一个典型的PCI设备驱动程序的框架,是一种相对固定的模式。需要注意的是,同加载和卸载模块相关的函数或数据结构都要在前面加上__init、__exit等标志符,以使同普通函数区分开来。接下去的工作就是如何完成框架内的各个功能模块了。 (3)初始化设备
在Linux系统下,想要完成对一个PCI设备的初始化,需要完成以下工作:检查PCI总线是否被Linux内核支持;检查设备是否插在总线插槽上,如果在的话则保存它所占用的插槽的位置等信息;读出配置头中的信息提供给驱动程序使用。 当Linux内核启动并完成对所有PCI设备进行扫描、登录和分配资源等初始化操作的同时,会建立起系统中所有PCI设备的拓扑结构,此后当PCI驱动程序需要对设备进行初始化时,一般都会调用如下的代码:
static int __init demo_init_module (void)
{ if (!pci_present()) return -ENODEV; if (!pci_register_driver(&demo_pci_driver)) { pci_unregister_driver(&demo_pci_driver); return -ENODEV; }