博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
Android系统匿名共享内存Ashmem(Anonymous Shared Memory)在进程间共享的原理分析
阅读量:6414 次
发布时间:2019-06-23

本文共 5115 字,大约阅读时间需要 17 分钟。

 在前面一篇文章中,我们系统地介绍了Android系统匿名共享内存的实现原理,其中着重介绍了它是如何辅助内存管理系统来有效地管理内存的,在再前面一篇文章中,我们还提到,Android系统匿名共享内存的另外一特点是通过Binder进程间通信机制来实现进程间共享的,本文中,将详细介绍Android系统匿名共享内存是如何使用Binder进程间通信机制来实现进程间共享的。

        由于Android系统匿名共享内存在进程间共享的原理涉及到Binder进程间通信机制的相关知识,所以希望读者在继续阅读本文之前,最好对Android系统的Binder进程间通信机制有一定的了解,具体可以参考这篇文章。

        在这篇文章中,我们举了一个例子来简要地介绍了Android系统的匿名共享内存机制及其使用方法,在这篇文章中,我们继续以这个实例来说明Android系统的匿名共享内存是如何使用Binder进程间通信机制来实现进程间共享的。为了方便描述,结合前面的Binder进程间通信机制知识,我们通过下面这个序列图来总结这个实例中的匿名共享内存文件的文件描述符在进程间传输的过程:

        这里, 我们需要关注的便是虚线框部分了,它在Binder驱动程序中实现了在两个进程中共享同一个打开文件的方法。我们知道,在Linux系统中,文件描述符其实就是一个整数。每一个进程在内核空间都有一个打开文件的数组,这个文件描述符的整数值就是用来索引这个数组的,而且,这个文件描述符只是在本进程内有效,也就是说,在不同的进程中,相同的文件描述符的值,代表的可能是不同的打开文件。因此,在进程间传输文件描述符时,不能简要地把一个文件描述符从一个进程传给另外一个进程,中间必须做一过转换,使得这个文件描述在目标进程中是有效的,并且它和源进程的文件描述符所对应的打开文件是一致的,这样才能保证共享。

        在一文中,我们介绍了用来传输的Binder对象的数据结构struct flat_binder_object,它定义在kernel/common/drivers/staging/android/binder.h 文件中:

 
  1. /*  
  2.  * This is the flattened representation of a Binder object for transfer  
  3.  * between processes.  The 'offsets' supplied as part of a binder transaction 
  4.  * contains offsets into the data where these structures occur.  The Binder  
  5.  * driver takes care of re-writing the structure type and data as it moves  
  6.  * between processes.  
  7.  */  
  8. struct flat_binder_object {  
  9.     /* 8 bytes for large_flat_header. */  
  10.     unsigned long       type;  
  11.     unsigned long       flags;  
  12.  
  13.     /* 8 bytes of data. */  
  14.     union {  
  15.         void        *binder;    /* local object */  
  16.         signed long handle;     /* remote object */  
  17.     };  
  18.  
  19.     /* extra data associated with local object */  
  20.     void            *cookie;  
  21. }; 

       域type是一个枚举类型,它的取值范围是:

 
  1. enum {  
  2.     BINDER_TYPE_BINDER  = B_PACK_CHARS('s''b''*', B_TYPE_LARGE),  
  3.     BINDER_TYPE_WEAK_BINDER = B_PACK_CHARS('w''b''*', B_TYPE_LARGE),  
  4.     BINDER_TYPE_HANDLE  = B_PACK_CHARS('s''h''*', B_TYPE_LARGE),  
  5.     BINDER_TYPE_WEAK_HANDLE = B_PACK_CHARS('w''h''*', B_TYPE_LARGE),  
  6.     BINDER_TYPE_FD      = B_PACK_CHARS('f''d''*', B_TYPE_LARGE),  
  7. }; 

        这里我们要介绍的Binder对象的type便是BINDER_TYPE_FD了,要传输的文件描述符的值保存在handle域中。

 

        在一文中,我们详细介绍了Binder对象在进程间通信传输的完整过程,这里就不再详述了,有兴趣的读都可以回过头去参考一下。这里,我们只关注文件描述符类型的Binder对象在Binder驱动程序中的相关处理逻辑。

        文件描述符类型的Binder对象在Binder驱动程序中的相关处理逻辑实现在binder_transact函数,这个函数定义在kernel/common/drivers/staging/android/binder.c文件中:

 
  1. static void  
  2. binder_transaction(struct binder_proc *proc, struct binder_thread *thread,  
  3. struct binder_transaction_data *tr, int reply)  
  4. {  
  5.     struct binder_transaction *t;  
  6.     struct binder_work *tcomplete;  
  7.     size_t *offp, *off_end;  
  8.     struct binder_proc *target_proc;  
  9.     struct binder_thread *target_thread = NULL;  
  10.     struct binder_node *target_node = NULL;  
  11.     struct list_head *target_list;  
  12.     wait_queue_head_t *target_wait;  
  13.     struct binder_transaction *in_reply_to = NULL;  
  14.     struct binder_transaction_log_entry *e;  
  15.     uint32_t return_error;  
  16.  
  17.     ......  
  18.       
  19.  
  20.     offp = (size_t *)(t->buffer->data + ALIGN(tr->data_size, sizeof(void *)));  
  21.  
  22.     ......  
  23.  
  24.     off_end = (void *)offp + tr->offsets_size;  
  25.     for (; offp < off_end; offp++) {  
  26.         struct flat_binder_object *fp;  
  27.         ......  
  28.         fp = (struct flat_binder_object *)(t->buffer->data + *offp);  
  29.         switch (fp->type) {  
  30.         ......  
  31.         case BINDER_TYPE_FD: {  
  32.             int target_fd;  
  33.             struct file *file;  
  34.  
  35.             if (reply) {  
  36.                 if (!(in_reply_to->flags & TF_ACCEPT_FDS)) {  
  37.                     binder_user_error("binder: %d:%d got reply with fd, %ld, but target does not allow fds\n",  
  38.                                     proc->pid, thread->pid, fp->handle);  
  39.                     return_error = BR_FAILED_REPLY;  
  40.                     goto err_fd_not_allowed;  
  41.                 }  
  42.             } else if (!target_node->accept_fds) {  
  43.                 binder_user_error("binder: %d:%d got transaction with fd, %ld, but target does not allow fds\n",  
  44.                                 proc->pid, thread->pid, fp->handle);  
  45.                 return_error = BR_FAILED_REPLY;  
  46.                 goto err_fd_not_allowed;  
  47.             }  
  48.  
  49.             file = fget(fp->handle);  
  50.             if (file == NULL) {  
  51.                 binder_user_error("binder: %d:%d got transaction with invalid fd, %ld\n",  
  52.                                     proc->pid, thread->pid, fp->handle);  
  53.                 return_error = BR_FAILED_REPLY;  
  54.                 goto err_fget_failed;  
  55.             }  
  56.             target_fd = task_get_unused_fd_flags(target_proc, O_CLOEXEC);  
  57.             if (target_fd < 0) {  
  58.                 fput(file);  
  59.                 return_error = BR_FAILED_REPLY;  
  60.                 goto err_get_unused_fd_failed;  
  61.             }  
  62.             task_fd_install(target_proc, target_fd, file);  
  63.             if (binder_debug_mask & BINDER_DEBUG_TRANSACTION)  
  64.                     printk(KERN_INFO "        fd %ld -> %d\n", fp->handle, target_fd);  
  65.             /* TODO: fput? */  
  66.             fp->handle = target_fd;  
  67.         } break;  
  68.  
  69.         ......  
  70.         }  
  71.     }  
  72.       
  73.     ......  
  74.  

        这里,我们先明确一下在这篇文章中所举的例子获取匿名共享内存文件的文件描述符的场景。匿名共享内存文件是在Server进程创建的,Client通过IMemoryService.getFileDescriptor去获取Server进程所创建的匿名共享内存文件的文件描述符,Server进程在返回这个文件描述符的过程中进入到Binder驱动程序,即这里的binder_transact函数。因此,这里的当前执行binder_transact函数的进程是Server进程,即源进程是Server进程,而目标进程是Client进程,就是这里的target_proc所表示的进程了。

 

        函数binder_transaction处理文件描述符类型的Binder对象就在中间的for循环里面。

       首先是获得Binder对象,并保存在本地变量fp中:

 
  1. fp = (struct flat_binder_object *)(t->buffer->data + *offp); 

       文件描述符的值就保存在fp->handle中,通过fget函数取回这个文件描述符所对应的打开文件结构:

 
  1. file = fget(fp->handle); 

      这里的file是一个struct file指针,它表示一个打开文件结构。注间,在Linux系统中,打开文件结构struct file是可以在进程间共享的,它与文件描述符不一样。

 

       接着在目标进程中获得一个空闲的文件描述符:

 
  1. target_fd = task_get_unused_fd_flags(target_proc, O_CLOEXEC); 

       现在,在目标进程中,打开文件结构有了,文件描述符也有了,接下来就可以把这个文件描述符和这个打开文件结构关联起来就可以了:

 
  1. task_fd_install(target_proc, target_fd, file); 

       由于这个Binder对象最终是要返回给目标进程的,所以还要修改fp->handle的值,它原来表示的是在源进程中的文件描述符,现在要改成目标进程的文件描述符:

 
  1. fp->handle = target_fd; 
      

       这样,对文件描述符类型的Binder对象的处理就完成了。目标进程拿到这个文件描述符后,就可以和源进程一起共享打开文件了。

 

       至此,Android系统匿名共享内存利用Binder进程间通信机制来实现进程间共享的学习就结束了,整个Android系统匿名共享内存机制的学习也完成了,希望对读者有所帮助,重新学习Android系统匿名共享内存机制请回到一文。

转载地址:http://hlbra.baihongyu.com/

你可能感兴趣的文章
python “双”稀疏矩阵转换为最小联通量“单”矩阵
查看>>
揭秘天猫双11背后:20万商家600万张海报,背后只有一个鹿班
查看>>
重置mysq root密码脚本
查看>>
我的友情链接
查看>>
MHA配置参数
查看>>
深入理解Lock
查看>>
vim的块选择
查看>>
HTML --块
查看>>
在DLL中获取主进程窗口句柄
查看>>
基于消息队列的双向通信
查看>>
一个不错的loading效果
查看>>
高中学渣逆袭入“大学”:如今月收入达五位数
查看>>
Debian允许root用户登录
查看>>
C++ - this指针
查看>>
Google Test and Google Mock Introduction
查看>>
linux的文件系统
查看>>
上云利器,K8S应用编排设计器之快到极致
查看>>
袋鼠云服务案例系列 | 从DB2到MySQL,某传统金融平台的互联网转型之路
查看>>
RealServer配置脚本
查看>>
九月份技术指标 华为交换机的简单配置
查看>>