arm_linux_device_mem内存映射

/dev/mem: 物理内存的全镜像。可以用来访问物理内存。

/dev/kmem: kernel看到的虚拟内存的全镜像。可以用来访问kernel的内容。

/dev/mem 用来访问物理IO设备比如X用来访问显卡的物理内存或嵌入式中访问GPIO。用

法一般就是open然后mmap接着可以使用map之后的地址来访问物理内存。这其实就是实现

用户空间驱动的一种方法。

/dev/kmem 一般可以用来查看kernel的变量或者用作rootkit之类的。 通过/dev/mem设备文件和mmap系统调用可以将线性地址描述的物理内存映射到进程 

的地址空间然后就可以直接访问这段内存了。  比如标准VGA 16色模式的实模式地址是A000:0000而线性地址则是A0000。设定显 

存大小为0x10000则可以如下操作      mem_fd  = open( "/dev/mem", O_RDWR ); 

    vga_mem = mmap( 0, 0x10000, PROT_READ | PROT_WRITE, MAP_SHARED, mem_fd, 0xA0000 ); 

    close( mem_fd );  然后直接对vga_mem进行访问就可以了。当然如果是操作VGA显卡还要获得I/O 

端口的访问权限以便进行直接的I/O操作用来设置模式/调色板/选择位面等等  在工控领域中还有一种常用的方法用来在内核和应用程序之间高效传递数据:  1) 假定系统有64M物理内存则可以通过lilo通知内核只使用63M而保留1M物理内    存作为数据交换使用(使用 mem=63M 标记)。 

2) 然后打开/dev/mem设备并将63M开始的1M地址空间映射到进程的地址空间。 使用/dev/kmem查看kernel变量 从lwn.net学到的

实例代码如下

#include <stdio.h>

#include <stdlib.h>

#include <string.h>

#include <stdarg.h>

#include <fcntl.h>

#include <unistd.h>

#include <errno.h> #include <sys/types.h>

#include <sys/stat.h>

#include <sys/poll.h>

#include <sys/mman.h> int page_size;

#define PAGE_SIZE page_size

#define PAGE_MASK (~(PAGE_SIZE-1)) void get_var (unsigned long addr) {

        off_t ptr = addr & ~(PAGE_MASK);

        off_t offset = addr & PAGE_MASK;

        int i = 0;

        char *map;

        static int kfd = -1;         kfd = open("/dev/kmem",O_RDONLY);

        if (kfd < 0) {

                perror("open");                 exit(0);

        }         map = mmap(NULL,PAGE_SIZE,PROT_READ,MAP_SHARED,kfd,offset);         if (map == MAP_FAILED) {

                perror("mmap");

                exit(-1);

        }

        /* 假定这里是字符串 */

        printf("%s\n",map+ptr);         return;

} int main(int argc, char **argv) {

        FILE *fp;

        char addr_str[11]="0x";

        char var[51];

        unsigned long addr;

        char ch;

        int r;

        if (argc != 2) {

                fprintf(stderr,"usage: %s System.map\n",argv[0]);

                exit(-1);

        }         if ((fp = fopen(argv[1],"r")) == NULL) {

                perror("fopen");

                exit(-1);

        }         do {

                r = fscanf(fp,"%8s %c %50s\n",&addr_str[2],&ch,var);

                if (strcmp(var,"modprobe_path")==0)

                        break;

        } while(r > 0);

        if (r < 0) {

                printf("could not find modprobe_path\n");

                exit(-1);

        }

        page_size = getpagesize();

        addr = strtoul(addr_str,NULL,16);

        printf("found modprobe_path at (%s) %08lx\n",addr_str,addr);

        get_var(addr);

} 运行 # ./tmap /boot/System.map

found modprobe_path at (0xc03aa900) c03aa900 /sbin/modprobe

区别

1. /dev/mem: 物理内存的全镜像。可以用来访问物理内存。 

2. /dev/kmem: kernel看到的虚拟内存的全镜像。可以用来访问kernel的

内容。

作用

1. 前者用来访问物理IO设备比如X用来访问显卡的物理内存或嵌入式

中访问GPIO。用法一般就是open然后mmap接着可以使用map之后的

地址来访问物理内存。这其实就是实现用户空间驱动的一种方法。 

2. 后者一般可以用来查看kernel的变量或者用作rootkit之类的。参考

1和2描述了用来查看kernel变量这个问题。

参考

1. http://lwn.net/Articles/147902/ 

2. http://lkml.org/lkml/2005/8/11/301

LINUX 

在2.4可以直接打开/dev/mem然后读取。

在2.6直接打开/dev/mem后只可以读取前0x101000部分的内容(ubuntu)。

大约是1MB加4KB。读取后面的内容将出现"open not permitted"错误。

解决的方法是使用mmap()。routine如下

f = open("/dev/mem", O_RDONLY);

my_mem = mmap (0, 0x1000, PROT_READ, MAP_SHARED, f, 0x34f000);

if (my_mem == MAP_FAILED)

printf("map failed %s\n",strerror(errno));

通过my_mem就可以得到0x101000之后的内存内容了 

时间: 2024-09-12 05:57:54

arm_linux_device_mem内存映射的相关文章

我用了共享内存映射,但是,串口来的数写到共享内存里,就是更新不到磁盘文件里去咋办??

问题描述 我用了共享内存映射,但是,串口来的数写到共享内存里,就是更新不到磁盘文件里去咋办?? 共享内存映射,flushviewoffile,更新磁盘文件(那个flushviewoffile 返回都是对的可是文件里就是没有东西) 解决方案 先看i数据是否正确接收了,有没有写入共享内存.

c++-求教!通过内存映射,实现生成的数组存储到文件中,显示出来头100个数?

问题描述 求教!通过内存映射,实现生成的数组存储到文件中,显示出来头100个数? 我目前就了解个大概,现在我不明白const DWORD mmf_size = 512*1024;//?这个地方不懂 ,这个原来的代码部分,我拿别人的代码改的,增加了自己的double的动态数组SigNRead,这个怎么弄 我就想实现,通过内存映射 实现写入 读取 double数组 数组有 1列 90000行个double数组成. #include #include #include #include #define

建立对象数据库-内存映射范式,需要中间层容器的支持

对象|数据|数据库|中间层 在<设想使用XML和关系数据库形成一个对象数据库>一文中,已经阐述了关系数据库和对象数据库的主要区别:对外键的使用是通过直接记录,还是遍历外键子表来获得.而这里,是面对另一个重要的区别:如何释放对象的资源.对于对象数据库而言,它等同于是一批直接存于数据库中的串行化对象,它的实际存储形式到底是不是按关系结构存储并不是最重要的,重要的是调用程序读入内存的就是一个对象,而不是游标映射:这样,使用对象数据库就必须存在着一个如何清除用完的对象资源的问题:同时由于对象数据库必然

内存映射修改大文件

本文介绍利用内存映射文件修改大文件:在大文件内存前加入一段数据,若要使用内存映射文件,必须执行下列操作步骤: 创建或打开一个文件内核对象,该对象用于标识磁盘上你想用作内存映射文件的文件: 创建一个文件映射内核对象,告诉系统该文件的大小和你打算如何访问该文件: 让系统将文件映射对象的全部或一部分映射到你的进程地址空间中: 当完成对内存映射文件的使用时,必须执行下面这些步骤将它清除: 告诉系统从你的进程的地址空间中撤消文件映射内核对象的映像: 关闭文件映射内核对象: 关闭文件内核对象: 下面将用一个

汇编教程之内存映射文件

本课中我们将要讲解内存映射文件并且演示如何运用它.您将会发现使用内存映射文件是非常简单的. 理论: 如果您仔细地研究了前一课的例子, 就会发现它有一个严重的缺陷:如果您想读的内容大于系统分配的内存块怎么办?如果您想搜索的字符串刚好超过内存块的边界又该如何处理?对于第一个问题,您也许会说,只要不断地读就不解决了吗.至于第二个问题,您又会说在内存块的边界处做一些特别的处理,譬如放上一些标志位就可以了.原理上确实是行得通,但是这随问题复杂程度加深而显得非常难以处理.其中的第二个问题是有名的边界判断问题

内存映射文件原理及实例

本课中我们将要讲解内存映射文件并且演示如何运用它.您将会发现使用内存映射文件是非常简单的. 理论:如果您仔细地研究了前一课的例子, 就会发现它有一个严重的缺陷:如果您想读的内容大于系统分配的内存块怎么办?如果您想搜索的字符串刚好超过内存块的边界又该如何处理?对于第一个问题,您也许会说,只要不断地读就不解决了吗.至于第二个问题,您又会说在内存块的边界处做一些特别的处理,譬如放上一些标志位就可以了.原理上确实是行得通,但是这随问题复杂程度加深而显得非常难以处理.其中的第二个问题是有名的边界判断问题,

VC++中使用内存映射文件处理大文件

摘要:本文给出了一种方便实用的解决大文件的读取.存储等处理的方法,并结合相关程序代码对具体的实现过程进行了介绍. 引言 文件操作是应用程序最为基本的功能之一,Win32 API和MFC均提供有支持文件处理的函数和类,常用的有Win32 API的CreateFile().WriteFile().ReadFile()和MFC提供的CFile类等.一般来说,以上这些函数可以满足大多数场合的要求,但是对于某些特殊应用领域所需要的动辄几十GB.几百GB.乃至几TB的海量存储,再以通常的文件处理方法进行处理

Linux中共享内存及内存映射技术研究

Linux给我们提供了丰富的内部进程通信机制,包括共享内存.内存映射文件.先入先出(FIFO).接口(sockets)以及多种用于同步的标识.在本文中,我们主要讨论一下共享内存和内存映射文件技术. 一般来说,内部进程通信(interprocess communication)也就是IPC,是指两个或两个以上进程以及两个或者两个以上线程之间进行通信联系.每个IPC机制都有不同的强项或者弱点,不过没有一个IPC机制包含内建的同步方法.因此程序员不但需要自己在程序中实现同步,而且还需要为了利用IPC机

Linux的共享内存及内存映射

一.POSIX共享内存的实现 共享内存是在进程间共享某一块内存.是最快一种ipc通信机构.其中posix共享内存机制 它主要是通过内存映射(mmap)机制来实现的. 在进程间共享内存使用如下固定步骤: 1.创建一个共享内存 int shm_open(const char *name, int oflag, mode_t mode); name是共享内存名字,各个进程通过名字来找到同一块内存. oflag,是这个内存属性.类似于文件属性.使用O_RDWR/O_RDONLY/O_CREAT,第一次创