`
weiyuhu
  • 浏览: 235888 次
  • 性别: Icon_minigender_1
  • 来自: 南京
社区版块
存档分类
最新评论

linux进程通信(一)--共享内存--mmap()

阅读更多
mmap()及其相关系统调用

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

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

1、mmap()系统调用形式如下:

void* mmap ( void * addr , size_t len , int prot , int flags , int fd , off_t offset )
参数fd为即将映射到进程空间的文件描述字,一般由open()返回,同时,fd可以指定为-1,此时须指定flags参数中的MAP_ANON,表明进行的是匿名映射(不涉及具体的文件名,避免了文件的创建及打开,很显然只能用于具有亲缘关系的进程间通信)。len是映射到调用进程地址空间的字节数,它从被映射文件开头offset个字节开始算起。prot 参数指定共享内存的访问权限。可取如下几个值的或:PROT_READ(可读) , PROT_WRITE (可写), PROT_EXEC (可执行), PROT_NONE(不可访问)。flags由以下几个常值指定:MAP_SHARED , MAP_PRIVATE , MAP_FIXED,其中,MAP_SHARED , MAP_PRIVATE必选其一,而MAP_FIXED则不推荐使用。offset参数一般设为0,表示从文件头开始映射。参数addr指定文件应被映射到进程空间的起始地址,一般被指定一个空指针,此时选择起始地址的任务留给内核来完成。函数的返回值为最后文件映射到进程空间的地址,进程可直接操作起始地址为该值的有效地址。这里不再详细介绍mmap()的参数,读者可参考mmap()手册页获得进一步的信息。

2、系统调用mmap()用于共享内存的两种方式:

(1)使用普通文件提供的内存映射:适用于任何进程之间;此时,需要打开或创建一个文件,然后再调用mmap();典型调用代码如下:

fd=open(name, flag, mode);
if(fd<0)
...



ptr=mmap(NULL, len , PROT_READ|PROT_WRITE, MAP_SHARED , fd , 0); 通过mmap()实现共享内存的通信方式有许多特点和要注意的地方,我们将在范例中进行具体说明。

(2)使用特殊文件提供匿名内存映射:适用于具有亲缘关系的进程之间;由于父子进程特殊的亲缘关系,在父进程中先调用mmap(),然后调用 fork()。那么在调用fork()之后,子进程继承父进程匿名映射后的地址空间,同样也继承mmap()返回的地址,这样,父子进程就可以通过映射区域进行通信了。注意,这里不是一般的继承关系。一般来说,子进程单独维护从父进程继承下来的一些变量。而mmap()返回的地址,却由父子进程共同维护。
对于具有亲缘关系的进程实现共享内存最好的方式应该是采用匿名内存映射的方式。此时,不必指定具体的文件,只要设置相应的标志即可,参见范例2。

3、系统调用munmap()

int munmap( void * addr, size_t len )
该调用在进程地址空间中解除一个映射关系,addr是调用mmap()时返回的地址,len是映射区的大小。当映射关系解除后,对原来映射地址的访问将导致段错误发生。

4、系统调用msync()

int msync ( void * addr , size_t len, int flags)
一般说来,进程在映射空间的对共享内容的改变并不直接写回到磁盘文件中,往往在调用munmap()后才执行该操作。可以通过调用msync()实现磁盘上文件内容与共享内存区的内容一致。



三、mmap()范例

下面将给出使用mmap()的两个范例:范例1给出两个进程通过映射普通文件实现共享内存通信;范例2给出父子进程通过匿名映射实现共享内存。系统调用mmap()有许多有趣的地方,下面是通过mmap()映射普通文件实现进程间的通信的范例,我们通过该范例来说明mmap()实现共享内存的特点及注意事项。

范例1:两个进程通过映射普通文件实现共享内存通信

范例1包含两个子程序:map_normalfile1.c及map_normalfile2.c。编译两个程序,可执行文件分别为 map_normalfile1及map_normalfile2。两个程序通过命令行参数指定同一个文件来实现共享内存方式的进程间通信。 map_normalfile2试图打开命令行参数指定的一个普通文件,把该文件映射到进程的地址空间,并对映射后的地址空间进行写操作。 map_normalfile1把命令行参数指定的文件映射到进程地址空间,然后对映射后的地址空间执行读操作。这样,两个进程通过命令行参数指定同一个文件来实现共享内存方式的进程间通信。

下面是两个程序代码:


C代码

   1. /*-------------map_normalfile1.c-----------*/ 
   2. #include <sys/mman.h> 
   3. #include <sys/types.h> 
   4. #include <fcntl.h> 
   5. #include <unistd.h> 
   6. typedef struct{ 
   7.     char name[4]; 
   8.     int  age; 
   9. }people; 
  10. main(int argc, char** argv) // map a normal file as shared mem: 
  11. { 
  12.     int fd,i; 
  13.     people *p_map; 
  14.     char temp; 
  15.      
  16.     fd=open(argv[1],O_CREAT|O_RDWR|O_TRUNC,00777); 
  17.     lseek(fd,sizeof(people)*5-1,SEEK_SET); 
  18.     write(fd,"",1); 
  19.      
  20.     p_map = (people*) mmap( NULL,sizeof(people)*10,PROT_READ|PROT_WRITE,MAP_SHARED,fd,0 ); 
  21.     close( fd ); 
  22.     temp = 'a'; 
  23.     for(i=0; i<10; i++) 
  24.     { 
  25.         temp += 1; 
  26.         memcpy( ( *(p_map+i) ).name, &temp,2 ); 
  27.         ( *(p_map+i) ).age = 20+i; 
  28.     } 
  29.     printf(" initialize over \n "); 
  30.     sleep(10); 
  31.     munmap( p_map, sizeof(people)*10 ); 
  32.     printf( "umap ok \n" ); 
  33. } 
  34. /*-------------map_normalfile2.c-----------*/ 
  35. #include <sys/mman.h> 
  36. #include <sys/types.h> 
  37. #include <fcntl.h> 
  38. #include <unistd.h> 
  39. typedef struct{ 
  40.     char name[4]; 
  41.     int  age; 
  42. }people; 
  43. main(int argc, char** argv) // map a normal file as shared mem: 
  44. { 
  45.     int fd,i; 
  46.     people *p_map; 
  47.     fd=open( argv[1],O_CREAT|O_RDWR,00777 ); 
  48.     p_map = (people*)mmap(NULL,sizeof(people)*10,PROT_READ|PROT_WRITE,MAP_SHARED,fd,0); 
  49.     for(i = 0;i<10;i++) 
  50.     { 
  51.     printf( "name: %s age %d;\n",(*(p_map+i)).name, (*(p_map+i)).age ); 
  52.     } 
  53.     munmap( p_map,sizeof(people)*10 ); 
  54. } 

/*-------------map_normalfile1.c-----------*/
#include <sys/mman.h>
#include <sys/types.h>
#include <fcntl.h>
#include <unistd.h>
typedef struct{
char name[4];
int  age;
}people;
main(int argc, char** argv) // map a normal file as shared mem:
{
int fd,i;
people *p_map;
char temp;

fd=open(argv[1],O_CREAT|O_RDWR|O_TRUNC,00777);
lseek(fd,sizeof(people)*5-1,SEEK_SET);
write(fd,"",1);

p_map = (people*) mmap( NULL,sizeof(people)*10,PROT_READ|PROT_WRITE,MAP_SHARED,fd,0 );
close( fd );
temp = 'a';
for(i=0; i<10; i++)
{
temp += 1;
memcpy( ( *(p_map+i) ).name, &temp,2 );
( *(p_map+i) ).age = 20+i;
}
printf(" initialize over \n ");
sleep(10);
munmap( p_map, sizeof(people)*10 );
printf( "umap ok \n" );
}
/*-------------map_normalfile2.c-----------*/
#include <sys/mman.h>
#include <sys/types.h>
#include <fcntl.h>
#include <unistd.h>
typedef struct{
char name[4];
int  age;
}people;
main(int argc, char** argv) // map a normal file as shared mem:
{
int fd,i;
people *p_map;
fd=open( argv[1],O_CREAT|O_RDWR,00777 );
p_map = (people*)mmap(NULL,sizeof(people)*10,PROT_READ|PROT_WRITE,MAP_SHARED,fd,0);
for(i = 0;i<10;i++)
{
printf( "name: %s age %d;\n",(*(p_map+i)).name, (*(p_map+i)).age );
}
munmap( p_map,sizeof(people)*10 );
}



map_normalfile1.c首先定义了一个people数据结构,(在这里采用数据结构的方式是因为,共享内存区的数据往往是有固定格式的,这由通信的各个进程决定,采用结构的方式有普遍代表性)。map_normfile1首先打开或创建一个文件,并把文件的长度设置为5个people 结构大小。然后从mmap()的返回地址开始,设置了10个people结构。然后,进程睡眠10秒钟,等待其他进程映射同一个文件,最后解除映射。

map_normfile2.c只是简单的映射一个文件,并以people数据结构的格式从mmap()返回的地址处读取10个people结构,并输出读取的值,然后解除映射。

分别把两个程序编译成可执行文件map_normalfile1和map_normalfile2后,在一个终端上先运行./map_normalfile2 /tmp/test_shm,程序输出结果如下:

initialize over
umap ok

在map_normalfile1输出initialize over 之后,输出umap ok之前,在另一个终端上运行map_normalfile2 /tmp/test_shm,将会产生如下输出(为了节省空间,输出结果为稍作整理后的结果):

name: b age 20; name: c age 21; name: d age 22; name: e age 23; name: f age 24;
name: g age 25; name: h age 26; name: I age 27; name: j age 28; name: k age 29;

在map_normalfile1 输出umap ok后,运行map_normalfile2则输出如下结果:

name: b age 20; name: c age 21; name: d age 22; name: e age 23; name: f age 24;
name: age 0; name: age 0; name: age 0; name: age 0; name: age 0;

从程序的运行结果中可以得出的结论

1、 最终被映射文件的内容的长度不会超过文件本身的初始大小,即映射不能改变文件的大小;

2、可以用于进程通信的有效地址空间大小大体上受限于被映射文件的大小,但不完全受限于文件大小。打开文件被截短为5个people结构大小,而在 map_normalfile1中初始化了10个people数据结构,在恰当时候(map_normalfile1输出initialize over 之后,输出umap ok之前)调用map_normalfile2会发现map_normalfile2将输出全部10个people结构的值,后面将给出详细讨论。
注:在linux中,内存的保护是以页为基本单位的,即使被映射文件只有一个字节大小,内核也会为映射分配一个页面大小的内存。当被映射文件小于一个页面大小时,进程可以对从mmap()返回地址开始的一个页面大小进行访问,而不会出错;但是,如果对一个页面以外的地址空间进行访问,则导致错误发生,后面将进一步描述。因此,可用于进程间通信的有效地址空间大小不会超过文件大小及一个页面大小的和。

3、文件一旦被映射后,调用mmap()的进程对返回地址的访问是对某一内存区域的访问,暂时脱离了磁盘上文件的影响。所有对mmap()返回地址空间的操作只在内存中有意义,只有在调用了munmap()后或者msync()时,才把内存中的相应内容写回磁盘文件,所写内容仍然不能超过文件的大小。



范例2:父子进程通过匿名映射实现共享内存
C代码

   1. #include <sys/mman.h> 
   2. #include <sys/types.h> 
   3. #include <fcntl.h> 
   4. #include <unistd.h> 
   5. typedef struct{ 
   6.     char name[4]; 
   7.     int  age; 
   8. }people; 
   9. main(int argc, char** argv) 
  10. { 
  11.     int i; 
  12.     people *p_map; 
  13.     char temp; 
  14.     p_map=(people*)mmap(NULL,sizeof(people)*10,PROT_READ|PROT_WRITE,MAP_SHARED|MAP_ANONYMOUS,-1,0); 
  15.     if(fork() == 0) 
  16.     { 
  17.         sleep(2); 
  18.         for(i = 0;i<5;i++) 
  19.             printf("child read: the %d people's age is %d\n",i+1,(*(p_map+i)).age); 
  20.         (*p_map).age = 100; 
  21.         munmap(p_map,sizeof(people)*10); //实际上,进程终止时,会自动解除映射。 
  22.         exit(); 
  23.     } 
  24.     temp = 'a'; 
  25.     for(i = 0;i<5;i++) 
  26.     { 
  27.         temp += 1; 
  28.         memcpy((*(p_map+i)).name, &temp,2); 
  29.         (*(p_map+i)).age=20+i; 
  30.     } 
  31.     sleep(5); 
  32.     printf( "parent read: the first people,s age is %d\n",(*p_map).age ); 
  33.     printf("umap\n"); 
  34.     munmap( p_map,sizeof(people)*10 ); 
  35.     printf( "umap ok\n" ); 
  36. } 

#include <sys/mman.h>
#include <sys/types.h>
#include <fcntl.h>
#include <unistd.h>
typedef struct{
char name[4];
int  age;
}people;
main(int argc, char** argv)
{
int i;
people *p_map;
char temp;
p_map=(people*)mmap(NULL,sizeof(people)*10,PROT_READ|PROT_WRITE,MAP_SHARED|MAP_ANONYMOUS,-1,0);
if(fork() == 0)
{
sleep(2);
for(i = 0;i<5;i++)
printf("child read: the %d people's age is %d\n",i+1,(*(p_map+i)).age);
(*p_map).age = 100;
munmap(p_map,sizeof(people)*10); //实际上,进程终止时,会自动解除映射。
exit();
}
temp = 'a';
for(i = 0;i<5;i++)
{
temp += 1;
memcpy((*(p_map+i)).name, &temp,2);
(*(p_map+i)).age=20+i;
}
sleep(5);
printf( "parent read: the first people,s age is %d\n",(*p_map).age );
printf("umap\n");
munmap( p_map,sizeof(people)*10 );
printf( "umap ok\n" );
}





考察程序的输出结果,体会父子进程匿名共享内存:

child read: the 1 people's age is 20
child read: the 2 people's age is 21
child read: the 3 people's age is 22
child read: the 4 people's age is 23
child read: the 5 people's age is 24
parent read: the first people,s age is 100
umap
umap ok
分享到:
评论

相关推荐

    进程间通讯---共享内存的使用方法

    共享内存是一种高效的进程间通信(IPC,Inter-Process Communication)方式,它允许不同的进程直接读写同一块内存区域,无需通过任何中介。这种方式避免了数据复制,因此在速度上具有显著优势。本教程将深入探讨如何...

    Linux-code.rar_linux进程通信

    这些代码实例是深入理解和掌握Linux进程通信的关键,对于提升你的Linux编程技能和解决实际问题的能力大有裨益。通过实践,你将能够灵活地运用这些通信机制,编写出高效、可靠的多进程应用程序。

    linux多进程生产消费

    共享内存是Linux提供的一种进程间通信(IPC)机制,允许不同进程共享同一块内存空间。在Linux中,可以使用`mmap()`系统调用来创建和映射共享内存区域。生产者将生成的数据写入共享内存,而消费者则读取并处理这些...

    Linux 进程通信(IPC)深刻理解

    在深入探讨Android IPC机制之前,我们首先来了解一下基础且关键的Linux进程间通信(IPC)技术。Linux作为一款强大的操作系统,提供了多种进程间通信的方式,这些机制使得不同进程之间能够有效地交换数据和同步执行。...

    C语言 Linux进程间通信

    共享内存是进程间通信的一种方式,它允许多个进程访问同一块内存区域。这种方式非常高效,因为数据不需要在进程之间复制,直接通过内存地址访问即可。但是,由于多个进程可以同时访问这块内存,因此必须采取一定的...

    商业编程-源码-共享内存封装类.zip

    共享内存是一种在多进程间高效通信的技术,它允许不同的进程直接读写同一块内存空间,而无需通过传统的I/O操作来传递数据。在商业编程中,这种技术常常被用来优化性能,尤其是在高并发和大数据量的场景下。下面将...

    电信设备-基于Linux共享内存实现家庭网关数据通信的系统及方法.zip

    共享内存是进程间通信(IPC,Inter-Process Communication)的一种方式,允许不同的进程访问同一块内存区域,从而实现数据的快速交换。在Linux中,通过`mmap()`函数或`shmget()`、`shmat()`、`shmdt()`等系统调用来...

    Linux下得mmap内存映射机制分析

    Linux操作系统中,mmap内存映射机制是一种高效的进程间通信方式。通过mmap机制,多个进程可以共享同一个文件的内存区域,从而实现高效的数据交换。下面是mmap机制的详细分析: 共享内存 共享内存是一种高效的进程...

    Linux下进程通信的八种方法.docx

    在Linux操作系统中,进程通信是实现多进程协作和数据交换的关键技术。本文将详细阐述Linux下的八种进程通信方法,帮助读者理解并掌握这些技术。 1. **管道(Pipe)**: 管道是一种半双工通信方式,允许一个进程...

    深刻理解Linux进程间通信.pdf

    Linux进程间通信(IPC)是操作系统中非常重要的部分,它包括多种机制用于进程间的数据交换。了解这些机制对于进行系统编程和软件开发非常重要。本文档详细介绍了管道、信号、消息队列、信号量和共享内存等通信方法。...

    linux下进程通信实例

    在本实例中,我们将探讨几种常见的Linux进程通信机制,并通过"ADOS_进程通信实验报告_余萝_2011103308.docx"文档提供的案例来进一步阐述。 1. **管道(Pipe)**:管道是一种半双工的通信方式,数据只能单向流动。它...

    linux-c-api-ref.zip_linux api_linux-c-api-ref

    以上只是Linux C API中的一部分核心知识点,实际的API包含更多内容,如进程间通信(IPC)、定时器、共享内存、管道等。深入理解和熟练使用这些API是开发高效、可靠的Linux应用程序的关键。通过阅读“linux-c-api-ref...

    9. Linux驱动开发-视频广告机开发、Linux进程编程介绍.pdf

    - **内存映射(mmap)**:通过内存映射文件,使得多个进程可以共享同一份文件内容。 - **信号(Signal)**:用于进程间同步和异常处理,如发送中断请求或通知事件。 4. **实用函数** - **execl函数族**:如`...

    linux mmap文件内存映射机制

    它不仅提高了程序对文件数据的访问效率,还为进程间通信(IPC)提供了一种高效的共享内存解决方案。 #### 二、mmap机制概述 `mmap`系统调用允许进程将一个文件或其他对象映射到进程的地址空间中,使得文件的内容...

    深刻理解Linux进程间通信

    在Linux操作系统中,进程可以通过多种方式进行通信,包括管道、信号、消息队列、信号灯、共享内存和套接字等。下面将详细阐述这些通信方式。 1. **管道**: - **普通管道(Pipe)**:是一种半双工的通信方式,数据...

    codesys如何通过共享内存的方式跟第三方程序通信

    共享内存是一种多进程间通信(IPC)的方法,允许不同进程访问同一块内存空间,从而实现实时的数据交换。在Codesys中,这通常通过创建和管理特定的内存区域来完成,这些内存区域可以被PLC程序和第三方应用程序(如C++...

    两进程间共享内存通讯

    在描述中提到的“内存映射文件进行进程通信”,这是一种更高级的共享内存实现方式。它利用了文件映射功能,将文件内容映射到进程的地址空间,使得进程可以直接读写文件,如同操作普通内存一样。这种方式简化了共享...

    linux内存共享源代码

    在Linux操作系统中,内存共享是一种进程间通信(IPC, Inter-Process Communication)的方式,它允许多个进程访问同一块内存区域,从而实现数据的高效交换。本知识点将深入探讨如何在Linux环境下通过共享内存实现两个...

Global site tag (gtag.js) - Google Analytics