Linux读取PCIe BAR (linux read pcie bar)

PCIe(Peripheral Component Interconnect-Express)是一种高速串行接口标准,它被用于连接计算机主板和各种外部设备,例如显卡、网卡、存储设备等。为了提高系统性能,PCIe设备通常将一部分内存映射到PCIe总线上,以便在主机和设备之间进行数据交换。在Linux系统中,为了正确使用PCIe设备,我们需要首先了解如何读取PCIe BAR(Base Address Register)。

PCIe BAR是标志PCIe设备内存地址的寄存器,它告诉主机操作系统设备内存的基地址和大小。每个PCIe设备通常都有多个BAR,因为它们可能需要映射多个内存区域。在Linux内核中,我们可以使用设备树和pci_resource_start()函数来访问PCIe设备的BAR。下面是如何使用这些工具读取设备之一个BAR的基地址的示例代码:

“`

#include

#include

static void __iomem *dev_mem;

static int probe(struct pci_dev *pdev, const struct pci_device_id *ent)

{

struct resource *r;

u64 mem_start, mem_len;

r = &pdev->resource[0];

mem_start = r->start;

mem_len = resource_size(r);

dev_mem = pci_iomap(pdev, 0, 0);

if (!dev_mem) {

dev_err(&pdev->dev, “pci_iomap fled\n”);

return -ENODEV;

}

printk(KERN_INFO “BAR0 start %#llx, size %#llx\n”, mem_start, mem_len);

return 0;

}

static void remove(struct pci_dev *pdev)

{

pci_iounmap(pdev, dev_mem);

}

static struct pci_device_id ids[] = {

{PCI_DEVICE(0x1234, 0x5678)},

{0},

}

MODULE_DEVICE_TABLE(pci, ids);

static struct pci_driver drv = {

.name = “mydriver”,

.id_table = ids,

.probe = probe,

.remove = remove,

}

module_pci_driver(drv);

“`

此代码包含一个设备树的片段,该片段定义了一个PCIe设备。节点包含厂商ID、设备ID和之一个BAR的基地址和大小。在probe()函数中,我们使用pci_iomap()函数将BAR映射到内核中的虚拟地址。然后,我们可以使用dev_mem指针来读取和写入该内存区域。

PCIe设备的BAR可能是32位或64位。如果是64位,我们需要使用pci_resource_len()和pci_resource_flags()函数来访问设备的高32位BAR。设备树中的reg属性可以指定高32位BAR的值。

在移除设备驱动程序时,我们使用pci_iounmap()函数释放BAR映射。这很重要,因为每个PCIe设备只有一定数量的BAR可供使用。如果不释放映射,可能会降低系统性能。


数据运维技术 » Linux读取PCIe BAR (linux read pcie bar)