我们可以在内核模块中使用virt_to_phys来存储用户空间吗?

val*_*iki 6 memory malloc memory-management linux-device-driver linux-kernel

我将使用分配用户应用程序中的内存,malloc并将malloc返回的地址通过字符驱动程序接口发送到内核​​模块。

我将get_user_pages_fast在内核模块中使用该内存的页面。

我可以virt_to_phys用来获取所返回的地址吗malloc?有效吗 如果没有,我如何获得正确的物理地址?

我的目的是获取用户空间分配内存的物理地址。我将传输大小限制为页面大小(4KB)。

小智 6

不,你不能,virt_to_phys 将内核虚拟地址转换为物理地址。Linux 中存在 3(或 4)种地址类型:

  • 内核虚拟地址:物理地址+/-偏移量(PAGE_OFFSET)。
  • 内核物理地址:实际物理地址(通过__pa或virt_to_phys函数获得)。
  • 用户虚拟地址:到物理地址的转换位于进程的页表内。

请注意,页表“布局”取决于处理器的体系结构,因此您需要实现与您所使用的体系结构相对应的软件页表遍历。

最后一句话,存在的第四种地址是:

  • 总线地址:这是设备看到的地址。


Aru*_*vel 0

该程序将帮助您直接访问物理内存。想法是mmap/dev/mem(RAM)

#include <stdio.h>
#include <stdlib.h>
#include <fcntl.h>
#include <sys/mman.h>
#include <unistd.h>

int main(int argc, char *argv[]) {
    if (argc < 3) {
        printf("Usage: %s <phys_addr> <offset>\n", argv[0]);
        return 0;
    }

    off_t offset = strtoul(argv[1], NULL, 0);
    size_t len = strtoul(argv[2], NULL, 0);

    // Truncate offset to a multiple of the page size, or mmap will fail.
    size_t pagesize = sysconf(_SC_PAGE_SIZE);
    off_t page_base = (offset / pagesize) * pagesize;
    off_t page_offset = offset - page_base;

    int fd = open("/dev/mem", O_SYNC);
    unsigned char *mem = mmap(NULL, page_offset + len, PROT_READ | PROT_WRITE, MAP_PRIVATE, fd, page_base);
    if (mem == MAP_FAILED) {
        perror("Can't map memory");
        return -1;
    }

    size_t i;
    for (i = 0; i < len; ++i)
        printf("%02x ", (int)mem[page_offset + i]);

    return 0;
}
Run Code Online (Sandbox Code Playgroud)

归功于从用户空间访问物理地址