Chinaunix首页 | 论坛 | 博客
  • 博客访问: 705009
  • 博文数量: 152
  • 博客积分: 0
  • 博客等级: 民兵
  • 技术积分: 1793
  • 用 户 组: 普通用户
  • 注册时间: 2013-09-12 12:26
个人简介

相信自己,只有不想做的,没有做不到的。

文章分类

全部博文(152)

文章存档

2021年(1)

2015年(2)

2014年(74)

2013年(75)

分类: LINUX

2014-07-31 15:51:29

  

头文件:#include 

原型: void *mmap(void *addr, size_t length, int prot, int flags, int fd, off_t offsize);

返回值成功则返回映射区起始地址失败则返回MAP_FAILED(-1).

参数:

addr: 指定映射的起始地址通常设为NULL, 由系统指定.

length: 将文件的多大长度映射到内存.

prot: 映射区的保护方式可以是:

PROT_EXEC: 映射区可被执行.

PROT_READ: 映射区可被读取.

PROT_WRITE: 映射区可被写入.

PROT_NONE: 映射区不能存取.

flags: 映射区的特性可以是:

MAP_SHARED: 对映射区域的写入数据会复制回文件且允许其他映射该文件的进程共享.

MAP_PRIVATE: 对映射区域的写入操作会产生一个映射的复制(copy-on-write), 对此区域所做的修改不会写回原文件.

此外还有其他几个flags不很常用具体查看linux C函数说明.

fd: open返回的文件描述符代表要映射的文件.

offset: 以文件开始处的偏移量必须是分页大小的整数倍通常为0, 表示从文件头开始映射.

具体实现代码如下:

 

#include 

#include 

#include 

#include 

#include 

#include 

#include 

 

#define DAVINCI_SYSTEM_MODULE_BASE      (0x10000000)

#define SYSTEM_LEN    0x8000

#define DEV_MEM_CTL   "/dev/mem"

static int dev_mem = -1;  

static unsigned int *pMem_map = NULL;

 

static int mem_open(void);

static int mmap_init(void); 

static int mmap_free(void);  

static void mem_close(void);  

void mem_free_all(void);

int  ReadRegister(unsigned long phy_addr);  

 

static unsigned int * data_buf = NULL;

 

int main(int argc, char *argv[]) 

{

data_buf = (int *)malloc(0x8000);

int ret = -1;     

unsigned long phr_add = 1;  

if((ret = mem_open()) < 0)  

{  

return -1;  

}  

ret = mmap_init();  

if(ret < 0)  

{  

return -1;  

}  

ret = ReadRegister(1);  

 

if(ret < 0)  

{  

return -1;  

}  

mem_free_all();  

 

return 0;

}

 

 

 

int  ReadRegister(unsigned long phy_addr)  

{  

 

int i = 0;

 

int ret = -1;  

unsigned int *WB_WGN_B = NULL;  

unsigned int *WB_WGN_GB = NULL;  

unsigned int *WB_WGN_GR = NULL;  

unsigned int *WB_WGN_R = NULL;  

 

if(dev_mem < 0) {  

ret = mem_open();  

if(ret < 0){  

return -1;  

}  

}  

 

if(pMem_map == NULL){  

if(mmap_init() < 0)  

return -1;  

}  

 

memcpy(data_buf,pMem_map,0x8000);

 

for(i = 0;i < 0x8000 /4 ;)

{

printf("%08x ",*((unsigned int *)((char *)data_buf+i)));

i+= 0x4;

}

printf("\n");

/*

WB_WGN_R = (unsigned int *)((char *)pMem_map + 0x0);    

WB_WGN_GR = (unsigned int *)((char *)pMem_map + 0x4);   

WB_WGN_GB = (unsigned int *)((char *)pMem_map + 0x8);   

WB_WGN_B = (unsigned int *)((char *)pMem_map + 0xc);   

printf("%x %x %x  %x/n",  

*WB_WGN_R,  

*WB_WGN_GR,  

*WB_WGN_GB,  

 

*WB_WGN_B);  

*/

return 0;  

}  

 

 

static int mmap_init(void)  

{  

    unsigned long phyaddr = DAVINCI_SYSTEM_MODULE_BASE;  

if(pMem_map == NULL){  

        pMem_map = mmap((void *)phyaddr, SYSTEM_LEN, PROT_READ|PROT_WRITE|PROT_EXEC, MAP_SHARED,  

                            dev_mem, phyaddr);  

if(pMem_map != (void *)phyaddr){  

            printf("mem failed/n");  

return -1;  

        }  

    }  

else{  

return 0;  

    }  

}  

 

static int mem_open(void)  

{  

if(dev_mem < 0){  

        dev_mem = open(DEV_MEM_CTL, O_RDWR|O_SYNC);  

if(dev_mem < 0){  

            printf("open %s error/n", DEV_MEM_CTL);  

return -1;  

        }  

    }  

return 0;  

}  

 

static int mmap_free(void)  

{  

if(pMem_map){  

        munmap(pMem_map, SYSTEM_LEN);  

        pMem_map = NULL;  

    }  

}  

static void mem_close(void)  

{  

if(dev_mem > 0) {  

        close(dev_mem);  

        dev_mem = -1;  

    }  

}  

 

void mem_free_all(void)  

{  

    mmap_free();      

    mem_close();  

}  

 

mmap()的特点:mmap系统调用并不是完全为了用于共享内存而设计的。它本身提供了不同于一般对普通文件的访问方式,进程可以像读写内存一样对普通文件的操作。而 Posix或系统V的共享内存IPC则纯粹用于共享目的,当然mmap()实现共享内存也是其主要应用之一。

mmap系统调用使得进程之间通过映射同一个普通文件实现共享内存。普通文件被映射到进程地址空间后,进程可以像访问普通内存一样对文件进行访问,不必再调用read()write()等操作。

我们的程序中大量运用了mmap,用到的正是mmap的这种“像访问普通内存一样对文件进行访问”的功能。实践证明,当要对一个文件频繁的进行访问,并且指针来回移动时,调用mmap比用常规的方法快很多。

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

 

阅读(3673) | 评论(0) | 转发(0) |
给主人留下些什么吧!~~