162306a36Sopenharmony_ci.. SPDX-License-Identifier: GPL-2.0 262306a36Sopenharmony_ci.. include:: <isonum.txt> 362306a36Sopenharmony_ci.. include:: ../disclaimer-zh_CN.rst 462306a36Sopenharmony_ci 562306a36Sopenharmony_ci:Original: Documentation/PCI/pciebus-howto.rst 662306a36Sopenharmony_ci 762306a36Sopenharmony_ci:翻译: 862306a36Sopenharmony_ci 962306a36Sopenharmony_ci 司延腾 Yanteng Si <siyanteng@loongson.cn> 1062306a36Sopenharmony_ci 1162306a36Sopenharmony_ci:校译: 1262306a36Sopenharmony_ci 1362306a36Sopenharmony_ci 1462306a36Sopenharmony_ci 1562306a36Sopenharmony_ci.. _cn_pciebus-howto: 1662306a36Sopenharmony_ci 1762306a36Sopenharmony_ci=========================== 1862306a36Sopenharmony_ciPCI Express端口总线驱动指南 1962306a36Sopenharmony_ci=========================== 2062306a36Sopenharmony_ci 2162306a36Sopenharmony_ci:作者: Tom L Nguyen tom.l.nguyen@intel.com 11/03/2004 2262306a36Sopenharmony_ci:版权: |copy| 2004 Intel Corporation 2362306a36Sopenharmony_ci 2462306a36Sopenharmony_ci关于本指南 2562306a36Sopenharmony_ci========== 2662306a36Sopenharmony_ci 2762306a36Sopenharmony_ci本指南介绍了PCI Express端口总线驱动程序的基本知识,并提供了如何使服务驱 2862306a36Sopenharmony_ci动程序在PCI Express端口总线驱动程序中注册/取消注册的介绍。 2962306a36Sopenharmony_ci 3062306a36Sopenharmony_ci 3162306a36Sopenharmony_ci什么是PCI Express端口总线驱动程序 3262306a36Sopenharmony_ci================================= 3362306a36Sopenharmony_ci 3462306a36Sopenharmony_ci一个PCI Express端口是一个逻辑的PCI-PCI桥结构。有两种类型的PCI Express端 3562306a36Sopenharmony_ci口:根端口和交换端口。根端口从PCI Express根综合体发起一个PCI Express链接, 3662306a36Sopenharmony_ci交换端口将PCI Express链接连接到内部逻辑PCI总线。交换机端口,其二级总线代表 3762306a36Sopenharmony_ci交换机的内部路由逻辑,被称为交换机的上行端口。交换机的下行端口是从交换机的内部 3862306a36Sopenharmony_ci路由总线桥接到代表来自PCI Express交换机的下游PCI Express链接的总线。 3962306a36Sopenharmony_ci 4062306a36Sopenharmony_ci一个PCI Express端口可以提供多达四个不同的功能,在本文中被称为服务,这取决于 4162306a36Sopenharmony_ci其端口类型。PCI Express端口的服务包括本地热拔插支持(HP)、电源管理事件支持(PME)、 4262306a36Sopenharmony_ci高级错误报告支持(AER)和虚拟通道支持(VC)。这些服务可以由一个复杂的驱动程序 4362306a36Sopenharmony_ci处理,也可以单独分布并由相应的服务驱动程序处理。 4462306a36Sopenharmony_ci 4562306a36Sopenharmony_ci为什么要使用PCI Express端口总线驱动程序? 4662306a36Sopenharmony_ci========================================= 4762306a36Sopenharmony_ci 4862306a36Sopenharmony_ci在现有的Linux内核中,Linux设备驱动模型允许一个物理设备只由一个驱动处理。 4962306a36Sopenharmony_ciPCI Express端口是一个具有多个不同服务的PCI-PCI桥设备。为了保持一个干净和简 5062306a36Sopenharmony_ci单的解决方案,每个服务都可以有自己的软件服务驱动。在这种情况下,几个服务驱动将 5162306a36Sopenharmony_ci竞争一个PCI-PCI桥设备。例如,如果PCI Express根端口的本机热拔插服务驱动程序 5262306a36Sopenharmony_ci首先被加载,它就会要求一个PCI-PCI桥根端口。因此,内核不会为该根端口加载其他服 5362306a36Sopenharmony_ci务驱动。换句话说,使用当前的驱动模型,不可能让多个服务驱动同时加载并运行在 5462306a36Sopenharmony_ciPCI-PCI桥设备上。 5562306a36Sopenharmony_ci 5662306a36Sopenharmony_ci为了使多个服务驱动程序同时运行,需要有一个PCI Express端口总线驱动程序,它管 5762306a36Sopenharmony_ci理所有填充的PCI Express端口,并根据需要将所有提供的服务请求分配给相应的服务 5862306a36Sopenharmony_ci驱动程序。下面列出了使用PCI Express端口总线驱动程序的一些关键优势: 5962306a36Sopenharmony_ci 6062306a36Sopenharmony_ci - 允许在一个PCI-PCI桥接端口设备上同时运行多个服务驱动。 6162306a36Sopenharmony_ci 6262306a36Sopenharmony_ci - 允许以独立的分阶段方式实施服务驱动程序。 6362306a36Sopenharmony_ci 6462306a36Sopenharmony_ci - 允许一个服务驱动程序在多个PCI-PCI桥接端口设备上运行。 6562306a36Sopenharmony_ci 6662306a36Sopenharmony_ci - 管理和分配PCI-PCI桥接端口设备的资源给要求的服务驱动程序。 6762306a36Sopenharmony_ci 6862306a36Sopenharmony_ci配置PCI Express端口总线驱动程序与服务驱动程序 6962306a36Sopenharmony_ci============================================= 7062306a36Sopenharmony_ci 7162306a36Sopenharmony_ci将PCI Express端口总线驱动支持纳入内核 7262306a36Sopenharmony_ci------------------------------------- 7362306a36Sopenharmony_ci 7462306a36Sopenharmony_ci包括PCI Express端口总线驱动程序取决于内核配置中是否包含PCI Express支持。当内核 7562306a36Sopenharmony_ci中的PCI Express支持被启用时,内核将自动包含PCI Express端口总线驱动程序作为内核 7662306a36Sopenharmony_ci驱动程序。 7762306a36Sopenharmony_ci 7862306a36Sopenharmony_ci启用服务驱动支持 7962306a36Sopenharmony_ci---------------- 8062306a36Sopenharmony_ci 8162306a36Sopenharmony_ciPCI设备驱动是基于Linux设备驱动模型实现的。所有的服务驱动都是PCI设备驱动。如上所述, 8262306a36Sopenharmony_ci一旦内核加载了PCI Express端口总线驱动程序,就不可能再加载任何服务驱动程序。为了满 8362306a36Sopenharmony_ci足PCI Express端口总线驱动程序模型,需要对现有的服务驱动程序进行一些最小的改变,其 8462306a36Sopenharmony_ci对现有的服务驱动程序的功能没有影响。 8562306a36Sopenharmony_ci 8662306a36Sopenharmony_ci服务驱动程序需要使用下面所示的两个API,将其服务注册到PCI Express端口总线驱动程 8762306a36Sopenharmony_ci序中(见第5.2.1和5.2.2节)。在调用这些API之前,服务驱动程序必须初始化头文件 8862306a36Sopenharmony_ci/include/linux/pcieport_if.h中的pcie_port_service_driver数据结构。如果不这 8962306a36Sopenharmony_ci样做,将导致身份不匹配,从而使PCI Express端口总线驱动程序无法加载服务驱动程序。 9062306a36Sopenharmony_ci 9162306a36Sopenharmony_cipcie_port_service_register 9262306a36Sopenharmony_ci~~~~~~~~~~~~~~~~~~~~~~~~~~ 9362306a36Sopenharmony_ci:: 9462306a36Sopenharmony_ci 9562306a36Sopenharmony_ci int pcie_port_service_register(struct pcie_port_service_driver *new) 9662306a36Sopenharmony_ci 9762306a36Sopenharmony_ci这个API取代了Linux驱动模型的 pci_register_driver API。一个服务驱动应该总是在模 9862306a36Sopenharmony_ci块启动时调用 pcie_port_service_register。请注意,在服务驱动被加载后,诸如 9962306a36Sopenharmony_cipci_enable_device(dev) 和 pci_set_master(dev) 的调用不再需要,因为这些调用由 10062306a36Sopenharmony_ciPCI端口总线驱动执行。 10162306a36Sopenharmony_ci 10262306a36Sopenharmony_cipcie_port_service_unregister 10362306a36Sopenharmony_ci~~~~~~~~~~~~~~~~~~~~~~~~~~~~ 10462306a36Sopenharmony_ci:: 10562306a36Sopenharmony_ci 10662306a36Sopenharmony_ci void pcie_port_service_unregister(struct pcie_port_service_driver *new) 10762306a36Sopenharmony_ci 10862306a36Sopenharmony_cipcie_port_service_unregister取代了Linux驱动模型的pci_unregister_driver。当一 10962306a36Sopenharmony_ci个模块退出时,它总是被服务驱动调用。 11062306a36Sopenharmony_ci 11162306a36Sopenharmony_ci示例代码 11262306a36Sopenharmony_ci~~~~~~~~ 11362306a36Sopenharmony_ci 11462306a36Sopenharmony_ci下面是服务驱动代码示例,用于初始化端口服务的驱动程序数据结构。 11562306a36Sopenharmony_ci:: 11662306a36Sopenharmony_ci 11762306a36Sopenharmony_ci static struct pcie_port_service_id service_id[] = { { 11862306a36Sopenharmony_ci .vendor = PCI_ANY_ID, 11962306a36Sopenharmony_ci .device = PCI_ANY_ID, 12062306a36Sopenharmony_ci .port_type = PCIE_RC_PORT, 12162306a36Sopenharmony_ci .service_type = PCIE_PORT_SERVICE_AER, 12262306a36Sopenharmony_ci }, { /* end: all zeroes */ } 12362306a36Sopenharmony_ci }; 12462306a36Sopenharmony_ci 12562306a36Sopenharmony_ci static struct pcie_port_service_driver root_aerdrv = { 12662306a36Sopenharmony_ci .name = (char *)device_name, 12762306a36Sopenharmony_ci .id_table = &service_id[0], 12862306a36Sopenharmony_ci 12962306a36Sopenharmony_ci .probe = aerdrv_load, 13062306a36Sopenharmony_ci .remove = aerdrv_unload, 13162306a36Sopenharmony_ci 13262306a36Sopenharmony_ci .suspend = aerdrv_suspend, 13362306a36Sopenharmony_ci .resume = aerdrv_resume, 13462306a36Sopenharmony_ci }; 13562306a36Sopenharmony_ci 13662306a36Sopenharmony_ci下面是一个注册/取消注册服务驱动的示例代码。 13762306a36Sopenharmony_ci:: 13862306a36Sopenharmony_ci 13962306a36Sopenharmony_ci static int __init aerdrv_service_init(void) 14062306a36Sopenharmony_ci { 14162306a36Sopenharmony_ci int retval = 0; 14262306a36Sopenharmony_ci 14362306a36Sopenharmony_ci retval = pcie_port_service_register(&root_aerdrv); 14462306a36Sopenharmony_ci if (!retval) { 14562306a36Sopenharmony_ci /* 14662306a36Sopenharmony_ci * FIX ME 14762306a36Sopenharmony_ci */ 14862306a36Sopenharmony_ci } 14962306a36Sopenharmony_ci return retval; 15062306a36Sopenharmony_ci } 15162306a36Sopenharmony_ci 15262306a36Sopenharmony_ci static void __exit aerdrv_service_exit(void) 15362306a36Sopenharmony_ci { 15462306a36Sopenharmony_ci pcie_port_service_unregister(&root_aerdrv); 15562306a36Sopenharmony_ci } 15662306a36Sopenharmony_ci 15762306a36Sopenharmony_ci module_init(aerdrv_service_init); 15862306a36Sopenharmony_ci module_exit(aerdrv_service_exit); 15962306a36Sopenharmony_ci 16062306a36Sopenharmony_ci可能的资源冲突 16162306a36Sopenharmony_ci============== 16262306a36Sopenharmony_ci 16362306a36Sopenharmony_ci由于PCI-PCI桥接端口设备的所有服务驱动被允许同时运行,下面列出了一些可能的资源冲突和 16462306a36Sopenharmony_ci建议的解决方案。 16562306a36Sopenharmony_ci 16662306a36Sopenharmony_ciMSI 和 MSI-X 向量资源 16762306a36Sopenharmony_ci--------------------- 16862306a36Sopenharmony_ci 16962306a36Sopenharmony_ci一旦设备上的MSI或MSI-X中断被启用,它就会一直保持这种模式,直到它们再次被禁用。由于同 17062306a36Sopenharmony_ci一个PCI-PCI桥接端口的服务驱动程序共享同一个物理设备,如果一个单独的服务驱动程序启用或 17162306a36Sopenharmony_ci禁用MSI/MSI-X模式,可能会导致不可预知的行为。 17262306a36Sopenharmony_ci 17362306a36Sopenharmony_ci为了避免这种情况,所有的服务驱动程序都不允许在其设备上切换中断模式。PCI Express端口 17462306a36Sopenharmony_ci总线驱动程序负责确定中断模式,这对服务驱动程序来说应该是透明的。服务驱动程序只需要知道 17562306a36Sopenharmony_ci分配给结构体pcie_device的字段irq的向量IRQ,当PCI Express端口总线驱动程序探测每 17662306a36Sopenharmony_ci个服务驱动程序时,它被传入。服务驱动应该使用(struct pcie_device*)dev->irq来调用 17762306a36Sopenharmony_cirequest_irq/free_irq。此外,中断模式被存储在struct pcie_device的interrupt_mode 17862306a36Sopenharmony_ci字段中。 17962306a36Sopenharmony_ci 18062306a36Sopenharmony_ciPCI内存/IO映射的区域 18162306a36Sopenharmony_ci-------------------- 18262306a36Sopenharmony_ci 18362306a36Sopenharmony_ciPCI Express电源管理(PME)、高级错误报告(AER)、热插拔(HP)和虚拟通道(VC)的服务 18462306a36Sopenharmony_ci驱动程序访问PCI Express端口的PCI配置空间。在所有情况下,访问的寄存器是相互独立的。这 18562306a36Sopenharmony_ci个补丁假定所有的服务驱动程序都会表现良好,不会覆盖其他服务驱动程序的配置设置。 18662306a36Sopenharmony_ci 18762306a36Sopenharmony_ciPCI配置寄存器 18862306a36Sopenharmony_ci------------- 18962306a36Sopenharmony_ci 19062306a36Sopenharmony_ci每个服务驱动都在自己的功能结构体上运行PCI配置操作,除了PCI Express功能结构体,其中根控制 19162306a36Sopenharmony_ci寄存器和设备控制寄存器是在PME和AER之间共享。这个补丁假定所有的服务驱动都会表现良好,不会 19262306a36Sopenharmony_ci覆盖其他服务驱动的配置设置。 193