android binder相关
为了解android如何在不同进程间传递fd,搜索了一下资料。
在binder内核代码中,对BINDER_TYPE_FD,会先
fget(fp->handle);
然后:
target_fd = task_get_unused_fd_flags(target_proc, O_CLOEXEC);
并
task_fd_install(target_proc, target_fd, file);
这样就相当于在新的进程理重新打开该文件,但得到的fd的值可能与原进程不一样。
http://dev.10086.cn/cmdn/wiki/index.php?doc-view-6363.html
1.文件描述符是如何在进程之间传递的?
我们知道文件描述符,就像虚拟内存的地址一样,是进程私有的资源。在一个进程中文件描 述符,在另外一个进程中,可能是无效的,也可能是对应另外一个文件。Android却可以把文件描述符从一个进程传到另外一个进程。第一次发现这种情况时,让我感到很惊奇,所以花了点时间去研究。看明白之后,发现其实现也 很简单:
Java代码:
- 复制到剪贴板 Java代码
- status_t Parcel::writeFileDescriptor(int fd)
- {
- flat_binder_object obj;
- obj.type = BINDER_TYPE_FD;
- obj.flags = 0x7f | FLAT_BINDER_FLAG_ACCEPTS_FDS;
- obj.handle = fd;
- obj.cookie = (void*)0;
- return writeObject(obj, true);
- }
在对文件描述符打包时,把对象的类型设置为BINDER_TYPE_FD。
在binder的内核模块binder_transaction函数中,我们可以看:
Java代码:
- 复制到剪贴板 Java代码
- case BINDER_TYPE_FD: {
- int target_fd;
- struct file *file;
- if (reply) {
- if (!(in_reply_to->flags & TF_ACCEPT_FDS)) {
- binder_user_error("binder: %d:%d got reply with fd, %ld, but target does not allow fds\n",
- proc->pid, thread->pid, fp->handle);
- return_error = BR_FAILED_REPLY;
- goto err_fd_not_allowed;
- }
- } else if (!target_node->accept_fds) {
- binder_user_error("binder: %d:%d got transaction with fd, %ld, but target does not allow fds\n",
- proc->pid, thread->pid, fp->handle);
- return_error = BR_FAILED_REPLY;
- goto err_fd_not_allowed;
- }
- file = fget(fp->handle);
- if (file == NULL) {
- binder_user_error("binder: %d:%d got transaction with invalid fd, %ld\n",
- proc->pid, thread->pid, fp->handle);
- return_error = BR_FAILED_REPLY;
- goto err_fget_failed;
- }
- target_fd = task_get_unused_fd_flags(target_proc, O_CLOEXEC);
- if (target_fd < 0) {
- fput(file);
- return_error = BR_FAILED_REPLY;
- goto err_get_unused_fd_failed;
- }
- task_fd_install(target_proc, target_fd, file);
- binder_debug(BINDER_DEBUG_TRANSACTION,
- " fd %ld -> %d\n", fp->handle, target_fd);
- /* TODO: fput? */
- fp->handle = target_fd;
- } break;
这里如果是文件描述符,就在目标进程中重新打开同一个文件了(虽然打开的是同一个文件,但目标进程拿到的文件描述符可能不相同)。
2.Receiver是如何工作的?
大家对Service的工作原理应该比较熟悉,先通过服务名称从 ServiceManager获取一个Binder,然后通过Binder去调用服务相应的函数。由客户端主动发起请求,这是典型是C/S模型。而 Receiver则是服务端反过来调用客户端函数,这就看起来有点让人感到迷惑了。
其实Receiver更简单,所有Broadcast都是从 ActivityManagerService发出的,所以只需要让 ActivityManagerService知道你的Receiver就行了,这是通过 ActivityManagerNative.registerReceiver完成的。实现自己的Receiver时都是实现 BroadcastReceiver接口,BroadcastReceiver本身并不是可以跨进程调用的,这是由 ActivityThread.PackageInfo.ReceiverDispatcher来包装的。
这里值得注意的是Receiver都是在ActivityThread里处理的,而不是在Binder线程里处理的,主要目的可能为了避免不必要的加锁操作吧。
搜索了几篇blog:
http://blog.csdn.net/black_berry/article/details/6618307
http://blog.csdn.net/black_berry/article/details/6621297
http://blog.csdn.net/black_berry/article/details/6621318
http://blog.csdn.net/black_berry/article/details/6621327
http://blog.csdn.net/black_berry/article/details/6621391
http://blog.csdn.net/black_berry/article/details/6621411
http://blog.csdn.net/black_berry/article/details/6621435
http://www.21876.com/project/Android/71246.html
http://blog.csdn.net/t80t90s/article/details/7704375
http://hi.baidu.com/woodyqt/item/3a5362d97982613b48e1ddab
http://www.redwolf-blog.com/?p=902
http://blog.csdn.net/lihui130135/article/details/7372282
http://w.hudong.com/3fdeadabfa514f1faf372c589632f692.html
http://www.blogjava.net/mixer-a/archive/2012/04/17/374989.html
这个人的blog也不错
http://blog.csdn.net/t80t90s/article/category/1145236/3
更多相关文章
- 没有一行代码,「2020 新冠肺炎记忆」这个项目却登上了 GitHub 中
- 一款常用的 Squid 日志分析工具
- GitHub 标星 8K+!一款开源替代 ls 的工具你值得拥有!
- RHEL 6 下 DHCP+TFTP+FTP+PXE+Kickstart 实现无人值守安装
- Linux 环境下实战 Rsync 备份工具及配置 rsync+inotify 实时同步
- 一款霸榜 GitHub 的开源 Linux 资源监视器!
- Android(安卓)Studio创建文件,自动生成类头
- Android(安卓)解决监听home键的几种方法
- Android(安卓)使用 kotlin 协程