【发布时间】:2020-10-12 15:27:58
【问题描述】:
我需要访问和写入 RAM 中的一些物理地址。我在看这个answer 和mmap 的定义。
如果 addr 为 NULL,则内核选择(页面对齐的)地址 创建映射的位置;这是最便携的方法 创建一个新的映射。如果 addr 不为 NULL,则内核采用 它作为关于在哪里放置映射的提示;在 Linux 上,内核 将选择一个附近的页面边界(但总是高于或等于 /proc/sys/vm/mmap_min_addr 指定的值)并尝试创建 那里的映射。
#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;
}
为什么 mmap 函数的第一个参数是 NULL ?不应该是page_base 吗?我们希望映射从页面基础开始并延伸到offset。
我必须做类似的事情,我必须从 完全 相同的位置开始将一组值复制到 RAM 中。这不应该是对 mmap 的调用吗:
unsigned char *mem = mmap(page_base, page_offset + len, PROT_READ | PROT_WRITE, MAP_SHARED, fd, page_base);
【问题讨论】: