本文介绍了 PCI 的基本概念,并从 Linux 内核的角度出发,介绍了 PCI 设备的初始化以及配置。
PCI 介绍
随着计算机应用的不断更新和发展(比如百兆网卡、视屏流等),计算机内数据传输的带宽要求越来越高,传统内部总线带宽已经远远不能满足这些应用的需要,因此人们推出了 PCI 总线标准
PCI 是 Peripheral Component Interconnect 的缩写,它因为高性能、低成本以及良好的扩展性而在计算机系统中被广泛使用。上至服务器,下至嵌入式设备都能找到它的身影。图 1 显示了一个标准 PCI 总线的组织结构图。
图 1. 标准 PCI 总线的组织结构图
从图中我们可以看出 PCI 总线架构主要被分成三部分:
- PCI 设备。符合 PCI 总线标准的设备就被称为 PCI 设备,PCI 总线架构中可以包含多个 PCI 设备。图中的 Audio、LAN 都是一个 PCI 设备。PCI 设备同时也分为主设备和目标设备两种,主设备是一次访问操作的发起者,而目标设备则是被访问者。
- PCI 总线。PCI 总线在系统中可以有多条,类似于树状结构进行扩展,每条 PCI 总线都可以连接多个 PCI 设备/桥。上图中有两条 PCI 总线。
- PCI 桥。当一条 PCI 总线的承载量不够时,可以用新的 PCI 总线进行扩展,而 PCI 桥则是连接 PCI 总线之间的纽带。图中的 PCI 桥有两个,一个桥用来连接处理器、内存以及 PCI 总线,而另外一条则用来连接另一条 PCI 总线。
PCI 总线操作
PCI 总线操作表示主设备向目标设备所发起的操作请求,最多有16种类型。主要类型有:IO 方式读/写,Memory 方式读/写,Configuration 方式读/写等。
PCI 配制空间
对于软件开发者来说,该如何对 PCI 设备进行编程呢?PCI 总线标准中定义了一套配置空间寄存器用于读取或者设置 PCI 设备的信息。每个 PCI 设备/桥都有自己的配置空间寄存器。
配置空间共有256字节,设备类型不同,其配置空间的布局也不尽相同。设备类型的区分可以通过配置空间内的 Header Type 寄存器(0Eh)进行,该寄存器值为 00h 表示当前设备是一个 PCI 设备,01h 表示当前设备是一个 PCI 桥。
配置空间的前64字节是配置空间起始段,它对于每种类型的设备都是相同的。显示了 PCI 设备的配置空间起始段。
图 2. PCI 设备的配置空间起始段
图 3 显示了 PCI 桥的配置空间起始段。
图 3. PCI 桥的配置空间起始段
配置空间寄存器有些是只读的,有些是可写的,下面介绍几个在编程时会用到的寄存器。
Device ID 和 Vendor ID 寄存器
这两个寄存器分别存放了设备信息和厂商信息(值在 0x0000 和 0xFFFF 之间,但不能取 0xFFFF),因此软件开发者可以通过读取这两个寄存器的值,并与 0xFFFF 比较,从而判断当前设备是否有效。
Command 和 Status 寄存器
Command 寄存器存放了设备的配置信息,比如是否允许 Memory/IO 方式的总线操作、是否为主设备等。Status 寄存器存放了设备的状态信息,比如中断状态、错误状态等。
Header Type 寄存器
这个寄存器前面曾经提过,它定义了设备类型,比如 PCI 设备、PCI 桥等。
Base Address 寄存器
这个寄存器有三个作用。
- 该寄存器存放了 Memory/IO 访问空间的起始地址。
- 该寄存器存放了 Memory/IO 访问空间的大小,这个数据可以通过下面的方式读出:
- 往寄存器里写 0xFFFFFFFF;
- 读出寄存器的值,并取反;
- 将上一步的值加上1后就是该空间的大小。
- 该寄存器定义了这段地址空间的访问类型(Memory 方式还是 IO 方式)。
PCI 设备最多有6个 Base Address 寄存器,而 PCI 桥最多有2个 Base Address 寄存器。
Subordinate Bus Number,Secondary Bus Number 和 Primary Bus Number 寄存器
这三个寄存器只在 PCI 桥配置空间中存在,因为 PCI 桥会连接两条 PCI 总线,上行的总线被称为 Primary Bus,下行的总线被称为 Secondary Bus,Primary Bus Number 和 Secondary Bus Number 寄存器分别存储了上行和下行总线的编号,而 Subordinate Bus Number 寄存器则是存储了当前桥所能直接或者间接访问到的总线的最大编号。
PPC 对于 PCI 的支持
通常 PPC 会提供一个(或更多的)PCI 控制器来连接 PCI 总线,通过 PCI 控制器,CPU 可以发起 Configuration 读写操作来访问所连接的所有 PCI 设备/桥的配置空间。每个 PCI 设备/桥都会用(总线号,设备号,功能号)这一组合来进行编号,因此在 PCI 控制器中输入设备对应的(总线号,设备号,功能号)就能寻址到具体的 PCI 设备/桥。以 PPC8548 为例,它提供了两个寄存器来实现 Configuration 操作,分别是 CFG_ADDR 和 CFG_DATA 寄存器,如果想对某个设备发起读/写操作,则首先将该设备的(总线号,设备号,功能号)写入 CFG_ADDR 中,这代表寻址一个具体的 PCI 设备,同时在 CFG_ADDR 中写入需要操作的配置空间寄存器的编号,最后从 CFG_DATA 中读取/写入相应的数据即可。
Linux 内核对 PCI 的支持
Linux 内核(2.6 版本)在初始化之初就对所有 PCI 设备进行了扫描并且配制,具体操作分为下面几个步骤。
编译时的 PCI 配制
如果想要 Linux 内核支持 PCI,首先需要对其配制文件进行相应的修改,在 config 文件中需要配置下面的宏参数。
- Linux 提供了 PCI 配制全局控制宏参数 CONFIG_PCI,它控制着 PCI 控制器和设备是否能够被配制的流程,因此该宏值需要被设置成“Y”。
- 对于某些处理器来说,比如 PPC85XX、PPC83XX 等,它们提供了两个或者更多的 PCI 控制器,因此在 config 文件中专门提供了诸如 CONFIG_MPC85xx_PCI2、CONFIG_MPC83xx_PCI2 等宏参数,它们控制着第二个(或更多的)PCI 控制器的配置流程,因此如果需要使用第二个(或更多的)PCI 控制器时,需要将相应的宏值设置成“Y”。
在编译内核之前,如果在 config 文件中提供了以上宏参数的设置,则编译出来的内核映像提供了对于 PCI 支持的功能。
PCI 相关数据结构
Linux 提供了三类数据结构用以描述 PCI 控制器、PCI 设备以及 PCI 总线。
PCI 控制器
PCI 控制器用 pci_controller 结构来描述,它有以下几个主要的属性:
- index:该属性标志 PCI 控制器的编号。
- next:该属性指向下一个 PCI 控制器,通过 next 属性,PCI 控制器可以形成一个单向链表。
- first_busno:该属性标志了连接在该控制器上第一条总线的编号。
- last_busno:该属性标志了连接在该控制器上最后一条总线的编号。
- ops:该属性标志了当前 PCI 控制器所对应的 PCI 配制空间读写操作函数。
- io_space:该属性标志了当前 PCI 控制器所支持的 IO 地址空间。
- mem_space:该属性标志了当前 PCI 控制器所支持的 Memory 地址区间。
- cfg_addr:该属性标志了当前 PCI 控制器发起 Configuration 访问方式时所需要写入的地址空间。
- cfg_data:该属性标志了当前 PCI 控制器发起 Configuration 访问方式时所需要读写的数据空间。
- bus:该属性标志了当前 PCI 控制器所连接的 PCI 总线,它对应的数据结构是 pci_bus。
PCI 总线
PCI 总线用 pci_bus 结构来描述,它有以下几个主要的属性:
- parent:可通过该属性索引到上层 PCI 总线。
- self:该属性标志了连接的上行 PCI 桥(对应的数据结构是 pci_dev)。
- children:该属性标志了总线连接的所有 PCI 子总线链表。
- devices:该属性标志了总线连接的所有 PCI 设备链表。
- ops:该属性标志了总线上所有 PCI 设备的配制空间读写操作函数。
- number:该属性标志了当前 PCI 总线的编号。
- primary:该属性标志了 PCI 上行总线编号。
- secondary:该属性标志了 PCI 下行总线编号。
- subordinate:该属性标志了能够访问到的最大总线编号。
- resource:该属性标志了 Memory/IO 地址空间。
PCI 设备
PCI 设备用 pci_dev 结构来描述,它有以下几个主要的属性:
- global_list:Linux 定义了一个全局列表来索引所有的 PCI 设备,该属性标志了这个全局列表的首指针。
- bus:该属性标志了当前设备所在的 PCI 总线(对应的数据结构是 pci_bus)。
- devfn:该属性标志了设备编号和功能编号。
- vendor:该属性标志了供应商编号。
- device:该属性标志了设备编号。
- driver:该属性标志了设备对应的驱动代码(对应的数据结构是 pci_driver)。
- irq:该属性标志了中断号。
- resource:该属性标志了 Memory/IO 地址区间。
内核里的 PCI 数据结构图
当 Linux 内核在做 PCI 初始化工作时,它会根据图 4 建立一个由 pci_controller、pci_bus 和 pci_dev 三者组成的一个组织结构图。根据这个结构,软件开发者可以很方便的通过 PCI 控制器索引到每个 PCI 设备或者 PCI 总线。
图 4. 组织结构图
PCI 控制器初始化
当一个支持 PCI 的内核映像开始运行时,它会在系统初始化时对 PCI 进行配置。函数调用链如下所示(以 PPC85XX 为例)。
图 5. 函数调用链
内核从 start_kernel() 函数处开始进行系统初始化,一直执行到 mpc85xx_setup_hose() 函数处便是配制 PCI 控制器以及连接在该控制器上所有设备的过程。
所有这些函数的定义处都加上了 __init 的符号类型,由 __init 修饰的函数表明在链接最终的内核映像时,这些函数将被放在一个特殊的初始化代码段中(.init.text,可以在链接文件 vmlinux.lds.S 中找到相关的段描述),这个初始化代码段会随着内核初始化完成而被释放。
在这一步骤中,内核会对它所支持的所有 PCI 控制器进行初始化工作,每个 PCI 控制器都对应一个 pci_controller 属性的变量,初始化工作会在这些变量中设置 Memory/IO 访问空间的起始地址以及结束地址、设置当前 PCI 控制器所连接的第一条和最后一条 PCI 总线编号等等。
PCI 自动扫描
系统如何知道当前连接了多少 PCI 设备?有多少根 PCI 总线?每个 PCI 设备的访问空间如何配置?等等。这些都得靠 PCI 自动扫描来完成。PCI 自动扫描主要做下面的工作: