访问 PCIe 设备的内部寄存器
Getting access to internal register of PCIe device
我在这里发现的主题很少,但其中 none 解释了我遇到的问题。
我只是想通过将其映射到 linux 中的用户内存 space 来访问 PCIe 设备的内部状态寄存器。
这是我的系统配置:
# uname -a
Linux localhost.localdomain 4.18.13-200.fc28.x86_64 #1 SMP Wed Oct 10 17:29:59 UTC 2018 x86_64 x86_64 x86_64 GNU/Linux
# lspci -tv
-[0000:00]-+-00.0 Intel Corporation Device 1980
+-04.0 Intel Corporation Device 19a1
<...>
+-1f.2 Intel Corporation Device 19de
# cat /proc/iomem
df570000-df573fff : 0000:00:1f.2
# lspci -s 00:1f.2 -x
00:1f.2 Memory controller: Intel Corporation Device 19de (rev 11)
00: 86 80 de 19 00 00 00 00 11 00 80 05 00 00 80 00
10: 00 00 57 df 00 00 00 00 00 00 00 00 00 00 00 00
20: 00 00 00 00 00 00 00 00 00 00 00 00 d9 15 69 09
30: 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00
因此,设备位于 00:1f.2 上并且对系统可见。我尝试访问内存控制器偏移量为 0x110 的内部 "ERRCORSTS" 寄存器,它显示了 PCI Express 设备上各个可纠正错误源的错误状态 page1673(这里是manual for my SoC)。我从程序中得到的输出是:
数据=ffffffff
PCI BAR0 0x0000 = 0xffff
似乎我在理解 linux 内存映射时遗漏了一些东西,或者他们可能只是在 4.18 内核中更改了一些东西,所以它不像以前那么容易了。
有人可以帮我解决一下吗?
这是我的代码:
#include <sys/types.h>
#include <unistd.h>
#include <sys/stat.h>
#include <sys/mman.h>
#include <fcntl.h>
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#define BASE_ADDR 0xdf570000
#define DATA_OFFSET 0x110
extern int errno;
int main()
{
int i;
int fd = open("/dev/mem",O_RDWR|O_SYNC);
if(fd < 0) {
printf("Can't open /dev/mem\n");
return 1;
}
u_int32_t* mapped_base = (u_int32_t *) mmap(0, 4096UL, PROT_READ|PROT_WRITE, MAP_SHARED, fd, BASE_ADDR);
// Trying to get access to device memory
if(mapped_base == NULL) {
printf("Can't mmap\n");
return 1;
} else {
unsigned int status_register0 = *(int *)(mapped_base + DATA_OFFSET );
printf("data = %lx \n",status_register0);
}
// Trying to get access to DevID
int fb = open("/sys/devices/pci0000:00/0000:00:1f.2/resource0", O_RDWR | O_SYNC);
u_int32_t* ptr = mmap(0, 4096, PROT_READ | PROT_WRITE, MAP_SHARED, fb, 0);
printf("PCI BAR0 0x0000 = 0x%4x\n", *((unsigned short *) ptr) );
return 0;
}
这执行指针运算。
mapped_base + DATA_OFFSET
偏移量自动乘以元素大小,根据
大概是4
u_int32_t* mapped_base
但是,您的文档似乎以字节为单位指定了偏移量。
因此,您需要阅读 0xdf570110
但实际上正在阅读 0xdf570440
好像是我在试验的时候把内存弄乱了。所以在我重启机器后一切正常。
我在这里发现的主题很少,但其中 none 解释了我遇到的问题。 我只是想通过将其映射到 linux 中的用户内存 space 来访问 PCIe 设备的内部状态寄存器。 这是我的系统配置:
# uname -a
Linux localhost.localdomain 4.18.13-200.fc28.x86_64 #1 SMP Wed Oct 10 17:29:59 UTC 2018 x86_64 x86_64 x86_64 GNU/Linux
# lspci -tv
-[0000:00]-+-00.0 Intel Corporation Device 1980
+-04.0 Intel Corporation Device 19a1
<...>
+-1f.2 Intel Corporation Device 19de
# cat /proc/iomem
df570000-df573fff : 0000:00:1f.2
# lspci -s 00:1f.2 -x
00:1f.2 Memory controller: Intel Corporation Device 19de (rev 11)
00: 86 80 de 19 00 00 00 00 11 00 80 05 00 00 80 00
10: 00 00 57 df 00 00 00 00 00 00 00 00 00 00 00 00
20: 00 00 00 00 00 00 00 00 00 00 00 00 d9 15 69 09
30: 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00
因此,设备位于 00:1f.2 上并且对系统可见。我尝试访问内存控制器偏移量为 0x110 的内部 "ERRCORSTS" 寄存器,它显示了 PCI Express 设备上各个可纠正错误源的错误状态 page1673(这里是manual for my SoC)。我从程序中得到的输出是:
数据=ffffffff
PCI BAR0 0x0000 = 0xffff
似乎我在理解 linux 内存映射时遗漏了一些东西,或者他们可能只是在 4.18 内核中更改了一些东西,所以它不像以前那么容易了。
有人可以帮我解决一下吗?
这是我的代码:
#include <sys/types.h>
#include <unistd.h>
#include <sys/stat.h>
#include <sys/mman.h>
#include <fcntl.h>
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#define BASE_ADDR 0xdf570000
#define DATA_OFFSET 0x110
extern int errno;
int main()
{
int i;
int fd = open("/dev/mem",O_RDWR|O_SYNC);
if(fd < 0) {
printf("Can't open /dev/mem\n");
return 1;
}
u_int32_t* mapped_base = (u_int32_t *) mmap(0, 4096UL, PROT_READ|PROT_WRITE, MAP_SHARED, fd, BASE_ADDR);
// Trying to get access to device memory
if(mapped_base == NULL) {
printf("Can't mmap\n");
return 1;
} else {
unsigned int status_register0 = *(int *)(mapped_base + DATA_OFFSET );
printf("data = %lx \n",status_register0);
}
// Trying to get access to DevID
int fb = open("/sys/devices/pci0000:00/0000:00:1f.2/resource0", O_RDWR | O_SYNC);
u_int32_t* ptr = mmap(0, 4096, PROT_READ | PROT_WRITE, MAP_SHARED, fb, 0);
printf("PCI BAR0 0x0000 = 0x%4x\n", *((unsigned short *) ptr) );
return 0;
}
这执行指针运算。
mapped_base + DATA_OFFSET
偏移量自动乘以元素大小,根据
大概是4u_int32_t* mapped_base
但是,您的文档似乎以字节为单位指定了偏移量。
因此,您需要阅读 0xdf570110
但实际上正在阅读 0xdf570440
好像是我在试验的时候把内存弄乱了。所以在我重启机器后一切正常。