c - 访问 PCIe 设备的内部寄存器

标签 c linux memory-management linux-kernel linux-device-driver

我在这里发现了一些关于它的话题,但没有一个能解释我遇到的问题。 我只是想通过将其映射到 Linux 中的用户内存空间来访问 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 并且对系统可见。我尝试访问内存 Controller 偏移量为 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

关于c - 访问 PCIe 设备的内部寄存器,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/52954461/

相关文章:

linux - Shell脚本查找从今天起12个工作日(忽略节假日)的日期

c++ - 关于 union 和内存管理的问题

c - 从 *char 解析 int 的快速简便方法

linux - 通过加载项管理器对 monodevelop 的 Fsharp 绑定(bind)无法解析依赖项

c++ - EvtSubscribe 拉与推模型

regex - awk 模式匹配

c++ - 为转发链链接新的运营商

ios - 协议(protocol)上的内存管理

c - fgets 函数不保存字符串中的第一个字母

c - 指针指向链表中错误的元素 (C)