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可供使用。如果不释放映射,可能会降低系统性能。

相关问题拓展阅读:

  • 如何在64位的Linux系统上使用汇编和C语言混

如何在64位的Linux系统上使用汇编和C语言混

编译和链接的时候使用的指令:(AMD处理器,64位操作系统)

编译链接指令

1 na -f elf foo.s -o foo.o

2 gcc -c bar.c -o bar.o

3 ld -s -o foobar bar.o foo.o

汇编语言用na编写并用na编译器编译,而C语言用的是gcc编译,这些都没有问题,但是在链接的时候出错了,提示如下:

ld: i386 architecture of input file `foo.o’ is incompatible with i386:x86-64 output

google了一下,意思就是na 编译产生的是32位的目标代码,gcc 在64位平台上默认产生的是64位的目标代码,这两者在链接的时候出错,gcc在64位平台上默认以64位的方式链接。

这样在解决的时候就会有两种解决方案:

让gcc 产生32位的代码,并在链接的时候以32位的方式进行链接

在这种情况下只需要修改编译和链接指令即可,具体如下:

32位的编译链接指令

1 na -f elf foo.s -o foo.o

2 gcc -m32 -c bar.c -o bar.o

3 ld -m elf_i386 -s -o foobar foo.o bar.o

具体的-m32 和 -m elf_i386 请自行查阅gcc (man gcc)

如果你是高版本的gcc(可能是由于更新内核造成的),可能简单的使用-m32 的时候会提示以下错误(使用别人的历程,自己未曾遇到):

> In file included from /usr/include/stdio.h:28:0,

> from test.c:1:

> /usr/include/features.h:323:26: fatal error: bits/predefs.h: No such file or directory

> compilation terminated.

这应该是缺少构建32 位可执行程序缺少的包,使用以下指令安装:

sudo apt-get install libc6-dev-i386

此时应该就没有什么问题了。

编译和链接的时候使用的指令:(AMD处理器,64位操作系统)

编译链接指令

1 na -f elf foo.s -o foo.o

2 gcc -c bar.c -o bar.o

3 ld -s -o foobar bar.o foo.o

汇编语数如言用na编写并用na编译器编译,而C语言用的是gcc编译,这些都没有问题,但是在链接的时候出错了,提示如下:

ld: i386 architecture of input file `foo.o’ is incompatible with i386:x86-64 output

google了一下,意思就是na 编译产生的是32位的目标代码,gcc 在64位平台上默认产生的是64位的目标代码,这两者在链接的时候出错,gcc在64位平台上默认以64位的方式链接。

这样在解决的时候就会有两种解决方案:

让gcc 产生32位的代码,并在链接的时候以32位的方式进行链接

在这种情况下只需要修滑正改编译和链接指令即可,具体如下:

32位的编译链接指令

1 na -f elf foo.s -o foo.o

2 gcc -m32 -c bar.c -o bar.o

3 ld -m elf_i386 -s -o foobar foo.o bar.o

具体的-m32 和 -m elf_i386 请自行查阅gcc (man gcc)

如果你是高版本的gcc(可能是由于更新内核造成的),可能简单的使用-m32 的时候会提示以下错误(使用别人的历程,自己薯让启未曾遇到):

> In file included from /usr/include/stdio.h:28:0,

> from test.c:1:

> /usr/include/features.h:323:26: fatal error: bits/predefs.h: No such file or directory

> compilation terminated.

这应该是缺少构建32 位可执行程序缺少的包,使用以下指令安装:

sudo apt-get install libc6-dev-i386

此时应该就没有什么问题了。

编译和链接的时候使用的指令:(AMD处理器,64位操作系统)

编译链接指令

1 na -f elf foo.s -o foo.o

2 gcc -c bar.c -o bar.o

3 ld -s -o foobar bar.o foo.o

汇编语言用na编写并用na编译器编译,而C语言用的是gcc编译,这些都没有问题,但是在链接的时候出错了,提示如下:

ld: i386 architecture of input file `foo.o’ is incompatible with i386:x86-64 output

google了一下,意思就是na 编译产生的是32位的目标代码,gcc 在64位平台上默认产生的是64位的目标代码,这两者在链接的时候出错,gcc在64位平台上默认以64位的方式链银缺蠢接。

这样在解决的时候就会有两种解决方案:

让gcc 产生32位的代码,并在链接的时候以32位的方式进行链接

在这种情况下只需要修改编译和链接指令即可,具体如下:

32位的编译链接指令

1 na -f elf foo.s -o foo.o

2 gcc -m32 -c bar.c -o bar.o

3 ld -m elf_i386 -s -o foobar foo.o bar.o

具体的-m32 和 -m elf_i386 请自行查阅gcc (man gcc)

如果你是高版本的gcc(可能是由于更新内核造成的扮枯),可能简单的使用-m32 的时候会提示以下错误(使用别人的历程,自己未曾遇到):

> In file included from /usr/include/stdio.h:28:0,

> from test.c:1:

> /usr/include/锋陪features.h:323:26: fatal error: bits/predefs.h: No such file or directory

> compilation terminated.

这应该是缺少构建32 位可执行程序缺少的包,使用以下指令安装:

sudo apt-get install libc6-dev-i386

关于linux read pcie bar的介绍到此就结束了,不知道你从中找到你需要的信息了吗 ?如果你还想了解更多这方面的信息,记得收藏关注本站。

香港服务器首选树叶云,2H2G首月10元开通。
树叶云(shuyeidc.com)提供简单好用,价格厚道的香港/美国云服务器和独立服务器。IDC+ISP+ICP资质。ARIN和APNIC会员。成熟技术团队15年行业经验。

文章来源网络,作者:运维,如若转载,请注明出处:https://shuyeidc.com/wp/191445.html<

(0)
运维的头像运维
上一篇2025-04-01 03:21
下一篇 2025-04-01 03:23

相关推荐

  • 个人主题怎么制作?

    制作个人主题是一个将个人风格、兴趣或专业领域转化为视觉化或结构化内容的过程,无论是用于个人博客、作品集、社交媒体账号还是品牌形象,核心都是围绕“个人特色”展开,以下从定位、内容规划、视觉设计、技术实现四个维度,详细拆解制作个人主题的完整流程,明确主题定位:找到个人特色的核心主题定位是所有工作的起点,需要先回答……

    2025-11-20
    0
  • 社群营销管理关键是什么?

    社群营销的核心在于通过建立有温度、有价值、有归属感的社群,实现用户留存、转化和品牌传播,其管理需贯穿“目标定位-内容运营-用户互动-数据驱动-风险控制”全流程,以下从五个维度展开详细说明:明确社群定位与目标社群管理的首要任务是精准定位,需明确社群的核心价值(如行业交流、产品使用指导、兴趣分享等)、目标用户画像……

    2025-11-20
    0
  • 香港公司网站备案需要什么材料?

    香港公司进行网站备案是一个涉及多部门协调、流程相对严谨的过程,尤其需兼顾中国内地与香港两地的监管要求,由于香港公司注册地与中国内地不同,其网站若主要服务内地用户或使用内地服务器,需根据服务器位置、网站内容性质等,选择对应的备案路径(如工信部ICP备案或公安备案),以下从备案主体资格、流程步骤、材料准备、注意事项……

    2025-11-20
    0
  • 如何企业上云推广

    企业上云已成为数字化转型的核心战略,但推广过程中需结合行业特性、企业痛点与市场需求,构建系统性、多维度的推广体系,以下从市场定位、策略设计、执行落地及效果优化四个维度,详细拆解企业上云推广的实践路径,精准定位:明确目标企业与核心价值企业上云并非“一刀切”的方案,需先锁定目标客户群体,提炼差异化价值主张,客户分层……

    2025-11-20
    0
  • PS设计搜索框的实用技巧有哪些?

    在PS中设计一个美观且功能性的搜索框需要结合创意构思、视觉设计和用户体验考量,以下从设计思路、制作步骤、细节优化及交互预览等方面详细说明,帮助打造符合需求的搜索框,设计前的规划明确使用场景:根据网站或APP的整体风格确定搜索框的调性,例如极简风适合细线条和纯色,科技感适合渐变和发光效果,电商类则可能需要突出搜索……

    2025-11-20
    0

发表回复

您的邮箱地址不会被公开。必填项已用 * 标注