Android系统匿名共享内存(Anonymous Shared Memory)C++调用接口分析(2)

简介:
    回到前面MemoryHeapBase类的构造函数中,将匿名共享内存映射到本进程的地址空间去后,还看继续设置这块匿名共享内存的读写属性:

  
  
  1. if (fd >= 0) {   
  2.     if (mapfd(fd, size) == NO_ERROR) {   
  3.         if (flags & READ_ONLY) {   
  4.             ashmem_set_prot_region(fd, PROT_READ);   
  5.         }   
  6.     }   
  7. }   
      上面调用mapfd函数来映射匿名共享内存时,指定这块内存是可读写的,但是如果传进来的参数flags设置了只读属性,那么还需要调用系统运行时库存层的ashmem_set_prot_region函数来设置这块匿名共享内存为只读,这个函数定义在system/core/libcutils/ashmem-dev.c文件,有兴趣的读者可以自己去研究一下。
 
        这样,通过这个构造函数,一块匿名共享内存就建立好了,其余的三个成员函数getHeapID、getBase和getSize就简单了:

  
  
  1. int MemoryHeapBase::getHeapID() const {   
  2.     return mFD;   
  3. }   
  4.    
  5. void* MemoryHeapBase::getBase() const {   
  6.     return mBase;   
  7. }   
  8.    
  9. size_t MemoryHeapBase::getSize() const {   
  10.     return mSize;   
  11. }   
  接下来我们再来看一下MemoryHeapBase在Client端实现的类图:
         这个类图中的类也是可以划分为两部分,一部分是和业务相关的,即跟匿名共享内存操作相关的类,包括BpMemoryHeap、IMemoryBase和RefBase三个类,另一部分是和Binder机制相关的,包括IInterface、BpInterface、BpRefBase、IBinder、BpBinder、ProcessState和IPCThreadState七个类。
        在和匿名共享内存操作相关的类中,BpMemoryHeap类是前面分析的MemoryHeapBase类在Client端进程的远接接口类,当Client端进程从Service Manager或者其它途径获得了一个MemoryHeapBase对象的引用之后,就会在本地创建一个BpMemoryHeap对象来代表这个引用。BpMemoryHeap类同样是要实现IMemoryHeap接口,同时,它是从RefBase类继承下来的,因此,它可以与智能指针来结合使用。
        在和Binder机制相关的类中,和Server端实现不一样的地方是,Client端不需要实现BnInterface和BBinder两个类,但是需要实现BpInterface、BpRefBase和BpBinder三个类。BpInterface类继承于BpRefBase类,而在BpRefBase类里面,有一个成员变量mRemote,它指向一个BpBinder对象,当BpMemoryHeap类需要向Server端对象发出请求时,它就会通过这个BpBinder对象的transact函数来发出这个请求。这里的BpBinder对象是如何知道要向哪个Server对象发出请深圳市的呢?它里面有一个成员变量mHandle,它表示的是一个Server端Binder对象的引用值,BpBinder对象就是要通过这个引用值来把请求发送到相应的Server端对象去的了,这个引用值与Server端Binder对象的对应关系是在Binder驱动程序内部维护的。这里的ProcessSate类和IPCThreadState类的作用和在Server端的作用是类似的,它们都是负责和底层的Binder驱动程序进行交互,例如,BpBinder对象的transact函数就通过线程中的IPCThreadState对象来将Client端请求发送出去的。这些实现具体可以参考 Android系统进程间通信(IPC)机制Binder中的Client获得Server远程接口过程源代码分析 一文。
        这里我们主要关注BpMemoryHeap类是如何实现IMemoryHeap接口的,这个类声明和定义在frameworks/base/libs/binder/IMemory.cpp文件中:

  
  
  1. class BpMemoryHeap : public BpInterface<IMemoryHeap>   
  2. {   
  3. public:   
  4.     BpMemoryHeap(const sp<IBinder>& impl);   
  5.     ......   
  6.    
  7.     virtual int getHeapID() const;   
  8.     virtual void* getBase() const;   
  9.     virtual size_t getSize() const;   
  10.    
  11.     ......   
  12. private:   
  13.     mutable volatile int32_t mHeapId;   
  14.     mutable void*       mBase;   
  15.     mutable size_t      mSize;   
  16.    
  17.     ......   
  18. }   
 先来看构造函数BpMemoryHeap的实现:

  
  
  1. BpMemoryHeap::BpMemoryHeap(const sp<IBinder>& impl)   
  2.     : BpInterface<IMemoryHeap>(impl),   
  3.         mHeapId(-1), mBase(MAP_FAILED), mSize(0), mFlags(0), mRealHeap(false)   
  4. {   
  5. }   
        它的实现很简单,只是初始化一下各个成员变量,例如,表示匿名共享内存文件描述符的mHeapId值初化为-1、表示匿名内共享内存基地址的mBase值初始化为MAP_FAILED以及表示匿名共享内存大小的mSize初始为为0,它们都表示在Client端进程中,这个匿名共享内存还未准备就绪,要等到第一次使用时才会去创建。这里还需要注意的一点,参数impl指向的是一个BpBinder对象,它里面包含了一个指向Server端Binder对象,即MemoryHeapBase对象的引用。
 
        其余三个成员函数getHeapID、getBase和getSize的实现是类似的:

  
  
  1. int BpMemoryHeap::getHeapID() const {   
  2.     assertMapped();   
  3.     return mHeapId;   
  4. }   
  5.    
  6. void* BpMemoryHeap::getBase() const {   
  7.     assertMapped();   
  8.     return mBase;   
  9. }   
  10.    
  11. size_t BpMemoryHeap::getSize() const {   
  12.     assertMapped();   
  13.     return mSize;   
  14. }   
   即它们在使用之前,都会首先调用assertMapped函数来保证在Client端的匿名共享内存是已经准备就绪了的:

  
  
  1. void BpMemoryHeap::assertMapped() const   
  2. {   
  3.     if (mHeapId == -1) {   
  4.         sp<IBinder> binder(const_cast<BpMemoryHeap*>(this)->asBinder());   
  5.         sp<BpMemoryHeap> heap(static_cast<BpMemoryHeap*>(find_heap(binder).get()));   
  6.         heap->assertReallyMapped();   
  7.         if (heap->mBase != MAP_FAILED) {   
  8.             Mutex::Autolock _l(mLock);   
  9.             if (mHeapId == -1) {   
  10.                 mBase   = heap->mBase;   
  11.                 mSize   = heap->mSize;   
  12.                 android_atomic_write( dup( heap->mHeapId ), &mHeapId );   
  13.             }   
  14.         } else {   
  15.             // something went wrong   
  16.             free_heap(binder);   
  17.         }   
  18.     }   
  19. }   
      在解释这个函数之前,我们需要先了解一下BpMemoryHeap是如何知道自己内部维护的这块匿名共享内存有没有准备就绪的。
       在frameworks/base/libs/binder/IMemory.cpp文件中,定义了一个全局变量gHeapCache:

  
  
  1. static sp<HeapCache> gHeapCache = new HeapCache();   
     它的类型为HeapCache,这也是一个定义在frameworks/base/libs/binder/IMemory.cpp文件的类,它里面维护了本进程中所有的MemoryHeapBase对象的引用。由于在Client端进程中,可能会有多个引用,即多个BpMemoryHeap对象,对应同一个MemoryHeapBase对象(这是由于可以用同一个BpBinder对象来创建多个BpMemoryHeap对象),因此,当第一个BpMemoryHeap对象在本进程中映射好这块匿名共享内存之后,后面的BpMemoryHeap对象就可以直接使用了,不需要再映射一次,当然重新再映射一次没有害处,但是会是多此一举,Google在设计这个类时,可以说是考虑得非常周到的。      
我们来看一下HeapCache的实现:

  
  
  1. class HeapCache : public IBinder::DeathRecipient   
  2. {   
  3. public:   
  4.     HeapCache();   
  5.     virtual ~HeapCache();   
  6.    
  7.     ......   
  8.    
  9.     sp<IMemoryHeap> find_heap(const sp<IBinder>& binder);   
  10.     void free_heap(const sp<IBinder>& binder);   
  11.     sp<IMemoryHeap> get_heap(const sp<IBinder>& binder);   
  12.     ......   
  13.    
  14. private:   
  15.     // For IMemory.cpp   
  16.     struct heap_info_t {   
  17.         sp<IMemoryHeap> heap;   
  18.         int32_t         count;   
  19.     };   
  20.    
  21.     ......   
  22.    
  23.     Mutex mHeapCacheLock;   
  24.     KeyedVector< wp<IBinder>, heap_info_t > mHeapCache;   
  25. };   
  它里面定义了一个成员变量mHeapCache,用来维护本进程中的所有BpMemoryHeap对象,同时还提供了find_heap和get_heap函数来查找内部所维护的BpMemoryHeap对象的功能。函数find_heap和get_heap的区别是,在find_heap函数中,如果在mHeapCache找不到相应的BpMemoryHeap对象,就会把这个BpMemoryHeap对象加入到mHeapCache中去,而在get_heap函数中,则不会自动把这个BpMemoryHeap对象加入到mHeapCache中去。
 
       这里,我们主要看一下find_heap函数的实现:

  
  
  1. sp<IMemoryHeap> HeapCache::find_heap(const sp<IBinder>& binder)   
  2. {   
  3.     Mutex::Autolock _l(mHeapCacheLock);   
  4.     ssize_t i = mHeapCache.indexOfKey(binder);   
  5.     if (i>=0) {   
  6.         heap_info_t& info = mHeapCache.editValueAt(i);   
  7.         LOGD_IF(VERBOSE,   
  8.                 "found binder=%p, heap=%p, size=%d, fd=%d, count=%d",   
  9.                 binder.get(), info.heap.get(),   
  10.                 static_cast<BpMemoryHeap*>(info.heap.get())->mSize,   
  11.                 static_cast<BpMemoryHeap*>(info.heap.get())->mHeapId,   
  12.                 info.count);   
  13.         android_atomic_inc(&info.count);   
  14.         return info.heap;   
  15.     } else {   
  16.         heap_info_t info;   
  17.         info.heap = interface_cast<IMemoryHeap>(binder);   
  18.         info.count = 1;   
  19.         //LOGD("adding binder=%p, heap=%p, count=%d",   
  20.         //      binder.get(), info.heap.get(), info.count);   
  21.         mHeapCache.add(binder, info);   
  22.         return info.heap;   
  23.     }   
  24. }   
    这个函数很简单,首先它以传进来的参数binder为关键字,在mHeapCache中查找,看看是否有对应的heap_info对象info存在,如果有的话,就增加它的引用计数info.count值,表示这个BpBinder对象多了一个使用者;如果没有的话,那么就需要创建一个heap_info对象info,并且将它加放到mHeapCache中去了。1




本文转自 Luoshengyang 51CTO博客,原文链接:http://blog.51cto.com/shyluo/966900,如需转载请自行联系原作者
目录
相关文章
|
开发框架 前端开发 Android开发
Flutter 与原生模块(Android 和 iOS)之间的通信机制,包括方法调用、事件传递等,分析了通信的必要性、主要方式、数据传递、性能优化及错误处理,并通过实际案例展示了其应用效果,展望了未来的发展趋势
本文深入探讨了 Flutter 与原生模块(Android 和 iOS)之间的通信机制,包括方法调用、事件传递等,分析了通信的必要性、主要方式、数据传递、性能优化及错误处理,并通过实际案例展示了其应用效果,展望了未来的发展趋势。这对于实现高效的跨平台移动应用开发具有重要指导意义。
1221 4
|
7月前
|
缓存 编解码 Android开发
Android内存优化之图片优化
本文主要探讨Android开发中的图片优化问题,包括图片优化的重要性、OOM错误的成因及解决方法、Android支持的图片格式及其特点。同时介绍了图片储存优化的三种方式:尺寸优化、质量压缩和内存重用,并详细讲解了相关的实现方法与属性。此外,还分析了图片加载优化策略,如异步加载、缓存机制、懒加载等,并结合多级缓存流程提升性能。最后对比了几大主流图片加载框架(Universal ImageLoader、Picasso、Glide、Fresco)的特点与适用场景,重点推荐Fresco在处理大图、动图时的优异表现。这些内容为开发者提供了全面的图片优化解决方案。
286 1
|
10月前
|
存储 监控 算法
员工屏幕监控系统之 C++ 图像差分算法
在现代企业管理中,员工屏幕监控系统至关重要。本文探讨了其中常用的图像差分算法,该算法通过比较相邻两帧图像的像素差异,检测屏幕内容变化,如应用程序切换等。文中提供了C++实现代码,并介绍了其在实时监控、异常行为检测和数据压缩等方面的应用,展示了其实现简单、效率高的特点。
301 15
|
安全 Android开发 数据安全/隐私保护
深入探索Android与iOS系统安全性的对比分析
在当今数字化时代,移动操作系统的安全已成为用户和开发者共同关注的重点。本文旨在通过比较Android与iOS两大主流操作系统在安全性方面的差异,揭示两者在设计理念、权限管理、应用审核机制等方面的不同之处。我们将探讨这些差异如何影响用户的安全体验以及可能带来的风险。
748 21
|
监控 Java Android开发
深入探讨Android系统的内存管理机制
本文将深入分析Android系统的内存管理机制,包括其内存分配、回收策略以及常见的内存泄漏问题。通过对这些方面的详细讨论,读者可以更好地理解Android系统如何高效地管理内存资源,从而提高应用程序的性能和稳定性。
527 16
|
Java 开发工具 Android开发
安卓与iOS开发环境对比分析
在移动应用开发的广阔天地中,安卓和iOS两大平台各自占据半壁江山。本文深入探讨了这两个平台的开发环境,从编程语言、开发工具到用户界面设计等多个角度进行比较。通过实际案例分析和代码示例,我们旨在为开发者提供一个清晰的指南,帮助他们根据项目需求和个人偏好做出明智的选择。无论你是初涉移动开发领域的新手,还是寻求跨平台解决方案的资深开发者,这篇文章都将为你提供宝贵的信息和启示。
214 8
|
监控 Java Android开发
深入探索Android系统的内存管理机制
本文旨在全面解析Android系统的内存管理机制,包括其工作原理、常见问题及其解决方案。通过对Android内存模型的深入分析,本文将帮助开发者更好地理解内存分配、回收以及优化策略,从而提高应用性能和用户体验。
|
5月前
|
存储
阿里云轻量应用服务器收费标准价格表:200Mbps带宽、CPU内存及存储配置详解
阿里云香港轻量应用服务器,200Mbps带宽,免备案,支持多IP及国际线路,月租25元起,年付享8.5折优惠,适用于网站、应用等多种场景。
1819 0
|
5月前
|
存储 缓存 NoSQL
内存管理基础:数据结构的存储方式
数据结构在内存中的存储方式主要包括连续存储、链式存储、索引存储和散列存储。连续存储如数组,数据元素按顺序连续存放,访问速度快但扩展性差;链式存储如链表,通过指针连接分散的节点,便于插入删除但访问效率低;索引存储通过索引表提高查找效率,常用于数据库系统;散列存储如哈希表,通过哈希函数实现快速存取,但需处理冲突。不同场景下应根据访问模式、数据规模和操作频率选择合适的存储结构,甚至结合多种方式以达到最优性能。掌握这些存储机制是构建高效程序和理解高级数据结构的基础。
526 1