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可供使用。如果不释放映射,可能会降低系统性能。
相关问题拓展阅读:
编译和链接的时候使用的指令:(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的介绍到此就结束了,不知道你从中找到你需要的信息了吗 ?如果你还想了解更多这方面的信息,记得收藏关注本站。
创新互联网络推广网站建设,网站设计,网站建设公司,网站制作,网页设计,1500元定制网站优化全包,先排名后付费,已为上千家服务,联系电话:13518219792
名称栏目:Linux读取PCIeBAR(linuxreadpciebar)
文章URL:http://www.gawzjz.com/qtweb/news49/167599.html
网站建设、网络推广公司-创新互联,是专注品牌与效果的网站制作,网络营销seo公司;服务项目有等
声明:本网站发布的内容(图片、视频和文字)以用户投稿、用户转载内容为主,如果涉及侵权请尽快告知,我们将会在第一时间删除。文章观点不代表本网站立场,如需处理请联系客服。电话:028-86922220;邮箱:631063699@qq.com。内容未经允许不得转载,或转载时需注明来源: 创新互联