Memory 使用mmap进行虚拟到物理寻址

Memory 使用mmap进行虚拟到物理寻址,memory,Memory,我想写一个物理地址,用ARM板改变一个引脚的电压,但是为了写一个物理地址,我需要一个虚拟地址,然后用mmap把它映射到物理地址。 所以我这样做了: #include <stdio.h> #include <stdlib.h> #include <unistd.h> #include <string.h> #include <errno.h> #include <signal.h> #include <fcntl.h&g

我想写一个物理地址,用ARM板改变一个引脚的电压,但是为了写一个物理地址,我需要一个虚拟地址,然后用mmap把它映射到物理地址。 所以我这样做了:

#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <string.h>
#include <errno.h>
#include <signal.h>
#include <fcntl.h>
#include <ctype.h>
#include <termios.h>
#include <sys/types.h>
#include <sys/mman.h>

#define MAP_SIZE 4096UL
#define MAP_MASK (MAP_SIZE - 1)

int main(void) {
int fd;
int *map_base_c,*map_base_d, *map_base_p, *virt_addr;
off_t target,control,data,pullup;

control=0x56000050;

if((fd = open("/dev/mem", O_RDWR | O_SYNC)) == -1) FATAL;

map_base_d = mmap(0, MAP_SIZE, PROT_READ | PROT_WRITE, MAP_SHARED, fd,data & ~MAP_MASK);

printf("Memory mapped at address %p.\n", map_base_d);

virt_addr = map_base_d; //+ (data & MAP_MASK)
*virt_addr = 0x00;  //This is where it goes off. find out why!!!
printf("Value at address 0x%X (%p): 0x%X\n", data, virt_addr,(*virt_addr));

close(fd);
return 0;
#包括
#包括
#包括
#包括
#包括
#包括
#包括
#包括
#包括
#包括
#包括
#定义地图尺寸4096UL
#定义映射掩码(映射大小-1)
内部主(空){
int-fd;
int*map\u base\u c、*map\u base\u d、*map\u base\u p、*virt\u addr;
偏离目标、控制、数据、上拉;
控制=0x56000050;
如果((fd=open(“/dev/mem”,O|RDWR | O|u SYNC))==-1)致命;
map_base_d=mmap(0,map_大小,PROT_读取| PROT_写入,map_共享,fd,数据&~map_掩码);
printf(“在地址%p处映射的内存。\n”,映射\u base\u d);
virt\u addr=map\u base\u d;/+(数据和映射掩码)
*virt_addr=0x00;//这就是它发生的地方。找出原因!!!
printf(“地址0x%X(%p):0x%X\n处的值”,数据,virt_addr,(*virt_addr));
关闭(fd);
返回0;
}

但是,引脚没有像我预期的那样获得高电压。我改变地址的方式有问题吗? 另外,是否有办法查看映射到虚拟地址的物理地址?
谢谢

在对
mmap
的调用中,
offset
参数应该是您想要访问的最低物理地址。在代码中,传递
数据&~MAP\u MASK
,数据尚未初始化(或已默认初始化为0)

我相信你想要类似于以下的东西:

uintptr_t control = 0x56000050;
uintptr_t base = control & ~MAP_MASK;
int fd;
void *map_base_d;
int *virt_addr;

if((fd = open("/dev/mem", O_RDWR | O_SYNC)) == -1) FATAL;

map_base_d = mmap(0, MAP_SIZE, PROT_READ | PROT_WRITE, MAP_SHARED, fd, base);
//now *map_base_d corresponds to the physical address of 0x56000000

virt_addr = map_base_d + (control - base);
*virt_addr = 0x00; //Make sure virt_addr is a pointer of the right width (int*, char*, etc), so that you don't accidentally write a dword when you really only want to write a single word.