Android MemoryFile内存文件 + Ashmem匿名共享内存

文章详细解释了JNI中的Ashmem(匿名共享内存)在android_os_MemoryFile中的应用,包括pin_region和get_size_region方法的使用,以及ashmem_create_region、ashmem_set_prot_region等Ashmem核心功能。讨论了Ashmem在进程间高效共享数据的优势和实现原理。
摘要由CSDN通过智能技术生成

throws IOException {

beginAccess();

try {

mMapping.position(destOffset);

mMapping.put(buffer, srcOffset, count);

} finally {

endAccess();

}

}

JNI:./frameworks/base/core/jni/android_os_MemoryFile.cpp

namespace android {

static jboolean android_os_MemoryFile_pin(JNIEnv* env, jobject clazz, jobject fileDescriptor,

jboolean pin) {

int fd = jniGetFDFromFileDescriptor(env, fileDescriptor);

int result = (pin ? ashmem_pin_region(fd, 0, 0) : ashmem_unpin_region(fd, 0, 0));

if (result < 0) {

jniThrowException(env, “java/io/IOException”, NULL);

}

return result == ASHMEM_WAS_PURGED;

}

static jint android_os_MemoryFile_get_size(JNIEnv* env, jobject clazz,

jobject fileDescriptor) {

int fd = jniGetFDFromFileDescriptor(env, fileDescriptor);

// Use ASHMEM_GET_SIZE to find out if the fd refers to an ashmem region.

// ASHMEM_GET_SIZE should succeed for all ashmem regions, and the kernel

// should return ENOTTY for all other valid file descriptors

int result = ashmem_get_size_region(fd);

if (result < 0) {

if (errno == ENOTTY) {

// ENOTTY means that the ioctl does not apply to this object,

// i.e., it is not an ashmem region.

return (jint) -1;

}

// Some other error, throw exception

jniThrowIOException(env, errno);

return (jint) -1;

}

return (jint) result;

}

static const JNINativeMethod methods[] = {

{“native_pin”, “(Ljava/io/FileDescriptor;Z)Z”, (void*)android_os_MemoryFile_pin},

{“native_get_size”, “(Ljava/io/FileDescriptor;)I”,

(void*)android_os_MemoryFile_get_size}

};

int register_android_os_MemoryFile(JNIEnv* env) {

return RegisterMethodsOrDie(env, “android/os/MemoryFile”, methods, NELEM(methods));

}

}

可以看到使用了ashmem, ashmem又是什么?

Ashmem


Ashmem (Anonymous Shared Memroy,匿名共享内存)是一种共享内存的机制,它利用了Linux的mmap系统调用,将不同进程中的同一段物理内存映射到进程各自的虚拟地址空间,从而实现高效的进程间共享。

优势
  • 在dev目录下对应的设备是/dev/ashmem,相比于传统的内存分配机制,如malloc、anonymous/named mmap,其好处是提供了辅助内核内存回收算法的pin/unpin机制。

  • 我们知道Binder 每个进程 mmap接收数据的大小有限制,超过 1M 就会报错。所以我们可以用Ashmem在 IPC 中传递大型数据。

源码位置:./system/core/libcutils/ashmem-dev.cpp

主要方法
  • ashmem_create_region :向ashmem驱动程序请求为应用程序创建一块匿名共享内存,并且放回它的文件描述符。

  • ashmem_set_prot_region:用来设置匿名共享内存的访问保护位。

  • ashmem_pin_region:向fd描述的文件发送io控制命令ASHMEM_PIN,用来锁定一小块内存区域。offset表示在匿名内存中的偏移位置,len表示长度。

  • ashmem_unpin_region:和 ashmem_pin_region 基本一样,只是发送了ASHMEM_UNPIN命令用来解锁一块内存区域。

  • ashmem_get_size_region:用来获取匿名共享内存大小。

其中 MemoryFile用到了两个:pin_region和get_size_region.即锁定一块内存区域和获取匿名共享内存的大小。

/*

  • ashmem_create_region - creates a new ashmem region and returns the file

  • descriptor, or <0 on error

  • `name’ is an optional label to give the region (visible in /proc/pid/maps)

  • `size’ is the size of the region, in page-aligned bytes

*/

int ashmem_create_region(const char *name, size_t size)

{

int ret, save_errno;

int fd = __ashmem_open();

if (fd < 0) {

return fd;

}

if (name) {

char buf[ASHMEM_NAME_LEN] = {0};

strlcpy(buf, name, sizeof(buf));

ret = TEMP_FAILURE_RETRY(ioctl(fd, ASHMEM_SET_NAME, buf));

if (ret < 0) {

goto error;

}

}

ret = TEMP_FAILURE_RETRY(ioctl(fd, ASHMEM_SET_SIZE, size));

if (ret < 0) {

goto error;

}

return fd;

error:

save_errno = errno;

close(fd);

errno = save_errno;

return ret;

}

int ashmem_set_prot_region(int fd, int prot)

{

int ret = __ashmem_is_ashmem(fd, 1);

if (ret < 0) {

return ret;

}

return TEMP_FAILURE_RETRY(ioctl(fd, ASHMEM_SET_PROT_MASK, prot));

}

int ashmem_pin_region(int fd, size_t offset, size_t len)

{

// TODO: should LP64 reject too-large offset/len?

ashmem_pin pin = { static_cast<uint32_t>(offset), static_cast<uint32_t>(len) };

int ret = __ashmem_is_ashmem(fd, 1);

if (ret < 0) {

return ret;

}

return TEMP_FAILURE_RETRY(ioctl(fd, ASHMEM_PIN, &pin));

}

int ashmem_unpin_region(int fd, size_t offset, size_t len)

{

// TODO: should LP64 reject too-large offset/len?

ashmem_pin pin = { static_cast<uint32_t>(offset), static_cast<uint32_t>(len) };

int ret = __ashmem_is_ashmem(fd, 1);

if (ret < 0) {

return ret;

}

return TEMP_FAILURE_RETRY(ioctl(fd, ASHMEM_UNPIN, &pin));

}

int ashmem_get_size_region(int fd)

{

int ret = __ashmem_is_ashmem(fd, 1);

if (ret < 0) {

return ret;

}

return TEMP_FAILURE_RETRY(ioctl(fd, ASHMEM_GET_SIZE, NULL));

}

实现

ashmme的典型用法是先打开设备文件,然后做mmap映射。

第一步通过调用ashmem_create_region函数,这个函数完成这几件事:

  1. fd = open(“/dev/ashmem”, O_RDWR);

  2. ioctl(fd, ASHMEM_SET_NAME, region_name); // 这一步可选

  3. ioctl(fd, ASHMEM_SET_SIZE, region_size);

第二步,应用程序一般会调用mmap来把ashmem分配的空间映射到进程空间:

mapAddr = mmap(NULL, pHdr->mapLength, PROT_READ | PROT_WRITE, MAP_PRIVATE, fd, 0);

自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。

深知大多数初中级Android工程师,想要提升技能,往往是自己摸索成长或者是报班学习,但对于培训机构动则近万的学费,着实压力不小。自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!

因此收集整理了一份《2024年Android移动开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。

img

img

img

img

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Android开发知识点,真正体系化!

由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!

如果你觉得这些内容对你有帮助,可以扫码获取!!(备注:Android)

小福利:

在当下这个碎片化信息环境的时代,很多资源都可以在网络上找到,只取决于你愿不愿意找或是找的方法对不对了

很多朋友不是没有资料,大多都是有几十上百个G,但是杂乱无章,不知道怎么看从哪看起,甚至是看后就忘

如果大家觉得自己在网上找的资料非常杂乱、不成体系的话,我也分享一套给大家,比较系统,我平常自己也会经常研读。

2021大厂最新Android面试真题解析

Android大厂面试真题解析

各个模块学习视频:如数据结构与算法

算法与数据结构资料图

只有系统,有方向的学习,才能在段时间内迅速提高自己的技术。
一线互联网架构师

这份体系学习笔记,适应人群:**第一,**学习知识比较碎片化,没有合理的学习路线与进阶方向。**第二,**开发几年,不知道如何进阶更进一步,比较迷茫。第三,到了合适的年纪,后续不知道该如何发展,转型管理,还是加强技术研究。如果你有需要,我这里恰好有为什么,不来领取!说不定能改变你现在的状态呢!点赞+评论即可获得!

《Android学习笔记总结+移动架构视频+大厂面试真题+项目实战源码》,点击传送门即可获取!

…(img-WRa4kF6a-1712142790513)]

只有系统,有方向的学习,才能在段时间内迅速提高自己的技术。
[外链图片转存中…(img-vwWpYPb2-1712142790513)]

[外链图片转存中…(img-H15Fg5TV-1712142790514)]

这份体系学习笔记,适应人群:**第一,**学习知识比较碎片化,没有合理的学习路线与进阶方向。**第二,**开发几年,不知道如何进阶更进一步,比较迷茫。第三,到了合适的年纪,后续不知道该如何发展,转型管理,还是加强技术研究。如果你有需要,我这里恰好有为什么,不来领取!说不定能改变你现在的状态呢!点赞+评论即可获得!

《Android学习笔记总结+移动架构视频+大厂面试真题+项目实战源码》,点击传送门即可获取!
  • 21
    点赞
  • 18
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值