IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> 网络协议 -> MemoryFile 共享内存原理分析 -> 正文阅读

[网络协议]MemoryFile 共享内存原理分析

Android 上层提供了一些内存共享工具类,比如 MemoryFile。你使用过吗?知道它的实现原理吗?

MemoryFile 是 Java 层对 Ashmem 的一个封装,下面来一起学习 MemoryFile,掌握它的使用姿势和底层原理。

MemoryFile 使用方法大致如下:

「进程 A 中申请一块共享内存写入数据,并准备好文件描述符:」

MemoryFile?memoryFile?=?new?MemoryFile(name,?size);
memoryFile.getOutputStream().write(data);
Method?method?=?MemoryFile.class.getDeclaredMethod("getFileDescriptor");
FileDescriptor?des?=?(FileDescriptor)?method.invoke(memoryFile);
ParcelFileDescriptor?pfd?=?ParcelFileDescriptor.dup(des);

「进程 B 中通过 binder 拿到 A 进程中准备好的文件描述符,然后直接读取数据:」

FileDescriptor?descriptor?=?pfd.getFileDescriptor();
FileInputStream?fileInputStream?=?new?FileInputStream(descriptor);
fileInputStream.read(data);

如上代码所示,使用起来和文件读写一样很简单,但我们不能只停留在仅会使用的浅显层面。接着来看 MemoryFile 是怎么从 Java API 调用到 Ashmem 驱动函数的,先来看 MemoryFile 的构造函数:

public?MemoryFile(String?name,?int?length)?throws?IOException?{????
try?{?
mSharedMemory?=?SharedMemory.create(name,?length);????????
mMapping?=?mSharedMemory.mapReadWrite();????
}?catch?(ErrnoException?ex)?{????????
ex.rethrowAsIOException();???
?}
}

可以看到构造 MemoryFile 时通过 SharedMemory create 方法申请了一块匿名共享内存,SharedMemory create 方法中调用了 nCreate native 方法:

private?static?native?FileDescriptor?nCreate(String?name,?int?size)?throws?ErrnoException;

对应的 native 实现在 android_os_SharedMemory.cpp 中,实现如下:

static?jobject?SharedMemory_create(JNIEnv*?env,?jobject,?jstring?jname,?jint?size)?{????
const?char*?name?=?jname???envGetStringUTFChars(jname,?nullptr)?:?nullptr;???
int?fd?=?ashmem_create_region(name,?size);?//创建匿名共享内存????
...???
?return?jniCreateFileDescriptor(env,?fd);
}

接着来看 ashmem_create_region 方法,它的对应实现在 ashmem-dev.cpp 中,如下:

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));?????
???//?设置 Ashmem 名字
ret?=?TEMP_FAILURE_RETRY(ioctl(fd,?ASHMEM_SET_NAME,?buf));??
???????if?(ret?<?0)?{goto?error;?}???
?}
}

如上设置 Ashmem 名字执行了 ioctl 系统调用, 它会进一步调用到?「ashmem_ioctl」?驱动函数。接着来看 __ashmem_open 方法:

static?int?__ashmem_open(){
int?fd;???
?pthread_mutex_lock(&__ashmem_lock);????
fd?=?__ashmem_open_locked();?//创建匿名共享内存
?pthread_mutex_unlock(&__ashmem_lock);????
return?fd;
}

进一步调用到 __ashmem_open_locked 方法,如下:

#define?ASHMEM_DEVICE?"/dev/ashmem"?//Ashmem?设备驱动
static?int?__ashmem_open_locked(){
int?ret;????
struct?stat?st;???
?int?fd?=?TEMP_FAILURE_RETRY(??????????
??????open(ASHMEM_DEVICE,?O_RDWR?|?O_CLOEXEC));?//创建匿名共享内存??
??...????return?fd;
}

在 __ashmem_open_locked 方法中调用了 open() 系统调用,和 ioctl 一样,对应到 Ashmem 设备驱动函数了,即?「ashmem_open」?驱动函数。

通过上面的分析知道 Ashmem 驱动的 ashmem_open 函数是由 SharedMemory 的 create 方法触发一步一步调用到的,期间还调用了 ashmem_ioctl 函数。

而 ashmem_mmap 驱动函数是通过 SharedMemory 的 mapReadWrite 方法触发,下面来分析这个过程:

//android.os.SharedMemory.java
public?@NonNull?ByteBuffer?mapReadWrite()?throws?ErrnoException?{????
return?map(OsConstants.PROT_READ?|?OsConstants.PROT_WRITE,?0,?mSize);
}

public?@NonNull?ByteBuffer?map(int?prot,?int?offset,?int?length)?throws?ErrnoException?{????
...???
?long?address?=?Os.mmap(0,?length,?prot,?OsConstants.MAP_SHARED,?mFileDescriptor,?offset);????
...????return?new?DirectByteBuffer(length,?address,?mFileDescriptor,?unmapper,?readOnly);}

比较关键的是 mFileDescriptor,它是执行 SharedMemory create 方法申请匿名共享内存后,返回的文件描述符。继续来跟踪 mmap 调用:

//android.system.Os.java
public?static?long?mmap(long?address,?long?byteCount,?int?prot,?int?flags,?FileDescriptor?fd,?long?offset)?throws?ErrnoException?{????
return?Libcore.os.mmap(address,?byteCount,?prot,?flags,?fd,?offset);
}

//libcore.io.Libcore.java
public?final?class?Libcore?{????
private?Libcore()?{?}????
public?static?Os?rawOs?=?new?Linux();??
??public?static?Os?os?=?new?BlockGuardOs(rawOs);
}

//libcore.io.Linux.java
public?native?long?mmap(long?address,?long?byteCount,?int?prot,?int?flags,?FileDescriptor?fd,?long?offset)?throws?ErrnoException;

Libcore 中使用 BlockGuardOs 对 Linux 进行了一层包装,但实际还是通过 Linux 来执行的,最后调用到 Linux 中的 native mmap 方法,native 中对应的实现是 mmap.cpp:

void*?mmap(void*?addr,?size_t?size,?int?prot,?int?flags,?int?fd,?off_t?offset)?{???
?return?mmap64(addr,?size,?prot,?flags,?fd,?static_cast<off64_t>(offset));
}

到此为止,由 SharedMemory 的 mapReadWrite 方法调用到 native mmap 函数,传递的关键参数是文件描述符,后续它将这样调用到 ashmem_mmap:

  1. 通过 fd 可以找到所属设备,也就是 Ashmem 设备
  2. 调用 Ashmem 设备的 ashmem_mmap 驱动函数

关键代码如下:

if(file){??
??...
????error?=?file->f_op->mmap(file,vma);???
?...
}

file 代表文件或设备驱动,这里指的就是 Ashmem 设备,f_op 就是 Ashmem 设备驱动函数集,也就是注册的 Ashmem 设备描述,至此便是 ashmem_mmap 驱动函数的调用过程。

关注我,每天分享知识干货~

  网络协议 最新文章
使用Easyswoole 搭建简单的Websoket服务
常见的数据通信方式有哪些?
Openssl 1024bit RSA算法---公私钥获取和处
HTTPS协议的密钥交换流程
《小白WEB安全入门》03. 漏洞篇
HttpRunner4.x 安装与使用
2021-07-04
手写RPC学习笔记
K8S高可用版本部署
mySQL计算IP地址范围
上一篇文章      下一篇文章      查看所有文章
加:2021-12-04 13:46:48  更:2021-12-04 13:47:57 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2024年7日历 -2024/7/6 7:40:55-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码