#include <sys/mman.h>
void *mmap(void *addr, size_t length, int prot, int flags, int fd, off_t offset);
int munmap(void *addr, size_t length);
addr | 所要映射的虚拟内存地址,映射成功后返回该地址。 如果为NULL或者无效地址,由系统会自动分配映射的虚拟地址。 |
length | 需要映射的空间大小 |
prot | 描述映射区域内存保护方式,包括:PROT_EXEC、PROT_READ、PROT_WRITE、PROT_NONE. |
flags | 描述映射区域的特性,比如是否对其他进程共享,是否建立匿名映射,是否创建私有的cow. |
fd | 要映射到内存中的文件描述符,一般是由open()函数返回,其值也可以设置为-1,此时需要指定flags参数中的MAP_ANON,表明进行的是匿名映射 |
offset | 文件映射的偏移量,需要页对齐(4096字节) |
返回值 | 映射区的虚拟地址 |
参考:
Linux系统“/dev/mem”设备使用详解(Hi3520D)_只要思想不滑坡,想法总比问题多。-CSDN博客_linux mem文件
#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 FATAL do { fprintf(stderr, "Error at line %d, file %s (%d) [%s]n",\
__LINE__, __FILE__, errno, strerror(errno)); exit(1); } while(0)
#define MAP_SIZE 4096UL
#define MAP_MASK (MAP_SIZE - 1)
int main(int argc, char **argv)
{
int fd;
void *map_base, *virt_addr;
unsigned long read_result, writeval;
off_t target;
int access_type = 'w';
if(argc < 2) {//若参数个数少于两个则打印此工具的使用方法
fprintf(stderr, "nUsage:t%s { address } [ type [ data ] ]n"
"taddress : memory address to act uponn"
"ttype : access operation type : [b]yte, [h]alfword, [w]ordn"
"tdata : data to be writtennn",
argv[0]);
exit(1);
}
target = strtoul(argv[1], 0, 0);
if(argc > 2)
access_type = tolower(argv[2][0]);
if((fd = open("/dev/mem", O_RDWR | O_SYNC)) == -1) FATAL;
printf("/dev/mem opened\n");
printf("目标物理地址:0x%x\n",target);
printf("页对齐后的地址:0x%x\n",target & ~MAP_MASK);
printf("对齐偏移量:0x%x\n",target & MAP_MASK);
fflush(stdout);
//将内核空间映射到用户空间
//第一个参数,所要映射到的虚拟地址,为NULL时,由系统分配
//最后一个参数,所要映射的物理地址,需要进行对齐
map_base = mmap(NULL, MAP_SIZE, PROT_READ | PROT_WRITE, MAP_SHARED, fd, target & ~MAP_MASK);
if(map_base == (void *) -1) FATAL;
printf("physic address mapped at vir address 0x%x <=> %p\n",target & ~MAP_MASK,map_base);
fflush(stdout);
virt_addr = map_base + (target & MAP_MASK);
//针对不同的参数获取不同类型内存数据
switch(access_type) {
case 'b':
read_result = *((unsigned char *) virt_addr);
break;
case 'h':
read_result = *((unsigned short *) virt_addr);
break;
case 'w':
read_result = *((unsigned long *) virt_addr);
break;
default:
fprintf(stderr, "Illegal data type '%c'\n", access_type);
exit(2);
}
printf("phyAddr:0x%x ,virAddr:%p: value:0x%x\n", target, virt_addr, read_result);
fflush(stdout);
//若参数大于3个,则说明为写入操作,针对不同参数写入不同类型的数据
if(argc > 3) {
writeval = strtoul(argv[3], 0, 0);
switch(access_type) {
case 'b':
*((unsigned char *) virt_addr) = writeval;
read_result = *((unsigned char *) virt_addr);
break;
case 'h':
*((unsigned short *) virt_addr) = writeval;
read_result = *((unsigned short *) virt_addr);
break;
case 'w':
*((unsigned long *) virt_addr) = writeval;
read_result = *((unsigned long *) virt_addr);
break;
}
printf("write val:0x%x,readback:0x%x\n", writeval, read_result);
fflush(stdout);
}
if(munmap(map_base, MAP_SIZE) == -1) FATAL;
close(fd);
return 0;
}