天天看點

3Linux記憶體映射,mmap()函數



1mmap()依賴的頭檔案

#include <sys/mman.h>

2函數聲明:

void *mmap(void *addr, size_t length, intprot, int flags,

intfd, off_t offset);

int munmap(void *addr, size_t length);

函數說明:

mmap可以把磁盤檔案的一部分直接映射到記憶體,這樣檔案中的位置直接就有對應的記憶體

位址,對檔案的讀寫可以直接用指針來做而不需要read/write函數。

3.記憶體映射圖:

3Linux記憶體映射,mmap()函數

說明:

如果addr參數為null,核心會自己在程序位址空間中選擇合适的位址建立映射。如果

addr不是null,則給核心一個提示,應該從什麼位址開始映射,核心會選擇addr之上的某個合适的位址開始映射。建立映射後,真正的映射首位址通過傳回值可以得到。len參數是需要映射的那一部分檔案的長度。off參數是從檔案的什麼位置開始映射,必須是頁大小的整數倍(在32位體系統結構上通常是4k)。filedes是代表該檔案的描述符。

prot參數有四種取值:

* prot_exec表示映射的這一段可執行,例如映射共享庫

* prot_read表示映射的這一段可讀

* prot_write表示映射的這一段可寫

* prot_none表示映射的這一段不可通路

flag參數有很多種取值,這裡隻講兩種,其它取值可檢視mmap(2)

* map_shared多個程序對同一個檔案的映射是共享的,一個程序對映射的記憶體做了修

改,另一個程序也會看到這種變化。

* map_private多個程序對同一個檔案的映射不是共享的,一個程序對映射的記憶體做了修改,另一個程序并不會看到這種變化,也不會真的寫到檔案中去。

如果mmap成功則傳回映射首位址,如果出錯則傳回常數map_failed。當程序終止時,該程序的映射記憶體會自動解除,也可以調用munmap解除映射。munmap成功傳回0,出錯傳回-1。

下面做一個簡單的實驗

xingwenpeng@ubuntu:~$ vi hello

xingwenpeng@ubuntu:~$ cat hello

helloworld

xingwenpeng@ubuntu:~$ od -tx1 -tc hello

0000000 68 65 6c 6c 6f 77 6f 72 6c 64 0a

h

 e  l  l o w o

 r  l  d \n

0000013

使用mmap映射

#include <stdlib.h>

#include <fcntl.h>

int main(void)

{

int *p;

int fd = open("hello", o_rdwr);

if (fd < 0) {

perror("open hello");

exit(1);

}

p = mmap(null, 6, prot_write, map_shared, fd, 0);

if (p == map_failed) {

perror("mmap");

close(fd);

p[0] = 0x30313233;

munmap(p, 6);

return 0;

案例2

寫端案例(注意要傳遞進去一個已經存在的檔案名):

#include<stdio.h>

#include<sys/types.h>

#include<sys/stat.h>

#include<sys/mman.h>

#include<unistd.h>

#include<stdlib.h>

#include<fcntl.h>

#define maplen 0x1000

void sys_err(char *str,int exitno)

   perror(str);

   exit(exitno);

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

   char *mm;

   int fd,i = 0;

   if(argc < 2)

       printf("./a.out filename\n");

       exit(1);

   fd = open(argv[1],o_rdwr|o_creat,0777);

   if(fd < 0)

       sys_err("open",1);

   if(lseek(fd,maplen-1,seek_set) < 0)

       sys_err("lseek",3);

   if(write(fd,"\0",1) < 0)

       sys_err("write",4);

//使用map_shared共享的方式,也就是說:記憶體或磁盤檔案中的一者發生内容變化

   mm = mmap(null,maplen,prot_read|prot_write,map_shared,fd,0);

   if(mm == map_failed){

       sys_err("mmap",2);

   close(fd);

   while(1)

       sprintf(mm,"hello%d",i++);

       sleep(1);

   munmap(mm,maplen);

   return 0;

讀端的案例:

       printf("./a.out filname\n");

   fd = open(argv[1],o_rdwr);

   if(mm == map_failed)

       printf("%s\n",mm);

munmap(mm,maplen);