mmap函数用法,实现一个devmem工具

#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 对内存的直接读写(devmem) - 尚码园

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;
}

  • 0
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值