Android开发:IBinder对象在进程间传递的形式[通俗易懂]

2019独角兽企业重金招聘Python工程师标准>>>…

大家好,又见面了,我是你们的朋友全栈君。

命题   

    当service经常被远程调用时,我们常常用到aidl来定一个接口供service和client来使用,这个其实就是使用Binder机制的IPC通信。当client bind service成功之后,系统AM会调用回调函数onServiceConnected将service的IBinder传递给client, client再通过调用aidl生成的asInterface()方法获得service的调用接口,此时一个bind过程结束了,我们在client端就可以远程调用service的方法了。例如
 
1.public void onServiceConnected(ComponentName className, 
 2.        IBinder service) { 
 3.    mSecondaryService = ISecondary.Stub.asInterface(service); 
 4.} 

 
    我们再看aidl生成的asInterface()的定义
 
1.public static com.example.Android.apis.app.ISecondary asInterface(android.os.IBinder obj) 
 2.{ 
 3.if ((obj==null)) { 
 4.return null; 
 5.} 
 6.android.os.IInterface iin = (android.os.IInterface)obj.queryLocalInterface(DESCRIPTOR); 
 7.if (((iin!=null)&&(iin instanceof com.example.android.apis.app.ISecondary))) { 
 8.return ((com.example.android.apis.app.ISecondary)iin); 
 9.} 
 10.return new com.example.android.apis.app.ISecondary.Stub.Proxy(obj); 
 11.} 

    首先,asInterface()会去query传入的IBinder对象是否有LocalInterface,这里的LocalInterface是指传入的IBinder是service本身的引用还是代理。
    1.当asInterface的输入的IBinder为server的引用(Binder类型)时,则直接返回该引用,那么此时调用server的方法不为IPC通信,而是直接的函数调用;
    2.当asInterface的输入的IBinder为server的代理(BinderProxy类型)时,则需要创建该server的代理并返回,此时调用server的方法为IPC通信。

 

    那么以上两种情况发生的条件是什么呢?这里我们先给出答案,然后再深入到代码中去研究2种不同的情况。
    1.当client和service处在相同进程中的话,asInterface的输入的IBinder为server的引用时;
    2.当client和service处在不同进程中的话,asInterface的输入的IBinder为server的代理。
 
在研究上述实现代码之前,我们先介绍一下IBinder作为参数使用IPC进程间传递时的状态变化,其实这个就是我们本篇文章的核心内容,理解了这个机制,我们就会很容易理解我们上述的那个命题的原理了。
 

 

    模型
 

    IBinder作为参数在IPC通信中进行传递,可能会使某些人困惑,IBinder不就是IPC通信的媒介吗?怎么还会被作为参数来传递呢,这样理解就有点狭隘了,拿native层的IPC来说,client从SM(service manager)中获取service端的Interface,这个Interface同时也是IBinder类型,当C/S两端需要双工通信时,即所谓的Service端需要反过来调用Client端的方法时,就需要client通过前述的那个Interface将Client端的IBinder传递给Service。
    拿Java应用层的Service来说更是如此,如本文的这个命题,下面我们会分析,首先来介绍原理性的知识。
    Binder IPC通信中,Binder是通信的媒介,Parcel是通信的内容。方法远程调用过程中,其参数都被打包成Parcel的形式来传递的。IBinder对象也不例外,我们看一下Parcel类中的writeStrongBinder()(由于java层和native层的方法是相对应的,java层只是native的封装,因此我们只需要看native的即可),
 
1.status_t Parcel::writeStrongBinder(const sp<IBinder>& val) 
 2.{ 
 3.    return flatten_binder(ProcessState::self(), val, this); 
 4.} 

 

 
 1.status_t flatten_binder(const sp<ProcessState>& proc, 
 2.    const sp<IBinder>& binder, Parcel* out) 
 3.{ 
 4.    flat_binder_object obj; 
 5.     
 6.    obj.flags = 0x7f | FLAT_BINDER_FLAG_ACCEPTS_FDS; 
 7.    if (binder != NULL) { 
 8.        IBinder *local = binder->localBinder(); 
 9.        if (!local) { 
 10.            BpBinder *proxy = binder->remoteBinder(); 
 11.            if (proxy == NULL) { 
 12.                LOGE(“null proxy”); 
 13.            } 
 14.            const int32_t handle = proxy ? proxy->handle() : 0; 
 15.            obj.type = BINDER_TYPE_HANDLE; 
 16.            obj.handle = handle; 
 17.            obj.cookie = NULL; 
 18.        } else { 
 19.            obj.type = BINDER_TYPE_BINDER; 
 20.            obj.binder = local->getWeakRefs(); 
 21.            obj.cookie = local; 
 22.        } 
 23.    } else { 
 24.        obj.type = BINDER_TYPE_BINDER; 
 25.        obj.binder = NULL; 
 26.        obj.cookie = NULL; 
 27.    } 
28.     
 29.    return finish_flatten_binder(binder, obj, out); 
 30.} 
上面代码分下面2种情况
1. 如果传递的IBinder为service的本地IBinder对象,那么该IBinder对象为BBinder类型的,因此上面的local不为NULL,故binder type为BINDER_TYPE_BINDER。
2. 如果传递的IBinder对象代理IBinder对象,那么binder type则为BINDER_TYPE_HANDLE。

client端将方法调用参数打包成Parcel之后,会发送到内核的Binder模块,因此下面我们将分析一下内核的Binder模块的处理。
 
kernel/drivers/staging/android/Binder.c中的函数binder_transaction()
 

 

 
 1.switch (fp->type) { 
 2.        case BINDER_TYPE_BINDER: 
 3.        case BINDER_TYPE_WEAK_BINDER: { 
 4.            struct binder_ref *ref; 
 5.            struct binder_node *node = binder_get_node(proc, fp->binder); 
 6.            if (node == NULL) { 
 7.                node = binder_new_node(proc, fp->binder, fp->cookie); 
 8.                if (node == NULL) { 
 9.                    return_error = BR_FAILED_REPLY; 
 10.                    goto err_binder_new_node_failed; 
 11.                } 
 12.                node->min_priority = fp->flags & FLAT_BINDER_FLAG_PRIORITY_MASK; 
 13.                node->accept_fds = !!(fp->flags & FLAT_BINDER_FLAG_ACCEPTS_FDS); 
 14.            } 
 15.            if (fp->cookie != node->cookie) { 
 16.                binder_user_error(“binder: %d:%d sending u%p ” 
 17.                    “node %d, cookie mismatch %p != %p\n”, 
 18.                    proc->pid, thread->pid, 
 19.                    fp->binder, node->debug_id, 
 20.                    fp->cookie, node->cookie); 
 21.                goto err_binder_get_ref_for_node_failed; 
 22.            } 
 23.            ref = binder_get_ref_for_node(target_proc, node); 
 24.            if (ref == NULL) { 
 25.                return_error = BR_FAILED_REPLY; 
 26.                goto err_binder_get_ref_for_node_failed; 
 27.            } 
 28.            if (fp->type == BINDER_TYPE_BINDER) 
 29.                fp->type = BINDER_TYPE_HANDLE; 
 30.            else 
 31.                fp->type = BINDER_TYPE_WEAK_HANDLE; 
 32.            fp->handle = ref->desc; 
 33.            binder_inc_ref(ref, fp->type == BINDER_TYPE_HANDLE, 
 34.                       &thread->todo); 
 35. 
 36.            binder_debug(BINDER_DEBUG_TRANSACTION, 
 37.                     ”        node %d u%p -> ref %d desc %d\n”, 
 38.                     node->debug_id, node->ptr, ref->debug_id, 
 39.                     ref->desc); 
 40.        } break; 
 41.        case BINDER_TYPE_HANDLE: 
 42.        case BINDER_TYPE_WEAK_HANDLE: { 
 43.            struct binder_ref *ref = binder_get_ref(proc, fp->handle); 
 44.            if (ref == NULL) { 
 45.                binder_user_error(“binder: %d:%d got ” 
 46.                    “transaction with invalid ” 
 47.                    “handle, %ld\n”, proc->pid, 
 48.                    thread->pid, fp->handle); 
 49.                return_error = BR_FAILED_REPLY; 
 50.                goto err_binder_get_ref_failed; 
 51.            } 
 52.            if (ref->node->proc == target_proc) { 
 53.                if (fp->type == BINDER_TYPE_HANDLE) 
 54.                    fp->type = BINDER_TYPE_BINDER; 
 55.                else 
 56.                    fp->type = BINDER_TYPE_WEAK_BINDER; 
 57.                fp->binder = ref->node->ptr; 
 58.                fp->cookie = ref->node->cookie; 
 59.                binder_inc_node(ref->node, fp->type == BINDER_TYPE_BINDER, 0, NULL); 
 60.                binder_debug(BINDER_DEBUG_TRANSACTION, 
 61.                         ”        ref %d desc %d -> node %d u%p\n”, 
 62.                         ref->debug_id, ref->desc, ref->node->debug_id, 
 63.                         ref->node->ptr); 
 64.            } else { 
 65.                struct binder_ref *new_ref; 
 66.                new_ref = binder_get_ref_for_node(target_proc, ref->node); 
 67.                if (new_ref == NULL) { 
 68.                    return_error = BR_FAILED_REPLY; 
 69.                    goto err_binder_get_ref_for_node_failed; 
 70.                } 
 71.                fp->handle = new_ref->desc; 
 72.                binder_inc_ref(new_ref, fp->type == BINDER_TYPE_HANDLE, NULL); 
 73.                binder_debug(BINDER_DEBUG_TRANSACTION, 
 74.                         ”        ref %d desc %d -> ref %d desc %d (node %d)\n”, 
 75.                         ref->debug_id, ref->desc, new_ref->debug_id, 
 76.                         new_ref->desc, ref->node->debug_id); 
 77.            } 
 78.        } break; 
    上面代码也分为了2种不同的分支:
    1. 传来的IBinder类型为BINDER_TYPE_BINDER时,会将binder type值为BINDER_TYPE_HANDLE;
    2. 传来的IBinder类型为BINDER_TYPE_HANDLE时,会判断该IBinder的实体被定义的进程(也就是该IBinder代表的server被定义的进程)与目标进程(也即IBinder被传递的目标进程)是否相同,如果相同,则将该IBinder type转化为BINDER_TYPE_BINDER,同时使其变为IBinder本地对象的引用。
本篇文章来源于 Linux公社网站(www.linuxidc.com)  原文链接:http://www.linuxidc.com/Linux/2011-08/40720.htm

转载于:https://my.oschina.net/zhuzihasablog/blog/114337

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 举报,一经查实,本站将立刻删除。

发布者:全栈程序员-用户IM,转载请注明出处:https://javaforall.cn/128774.html原文链接:https://javaforall.cn

【正版授权,激活自己账号】: Jetbrains全家桶Ide使用,1年售后保障,每天仅需1毛

【官方授权 正版激活】: 官方授权 正版激活 支持Jetbrains家族下所有IDE 使用个人JB账号...

(0)
blank

相关推荐

  • BitNami一键安装Redmine

    BitNami一键安装Redmine

    2021年12月15日
  • bgp多线是什么宽带_双线制

    bgp多线是什么宽带_双线制众所周知,南方带宽主要是以电信为主,北方带宽主要以联通为主,要想实现南北互联互通,就要选择双线服务器托管,目前双线服务器托管一般有两种,即双线双ip服务器托管和bgp双线服务器托管,这两者有什么区别呢?    双IP双线路实现方式是指在一台服务器上安装两块网卡,分别接入电信网线与网通网线并设置一个网通IP与一个电信IP,这样一台服务器上就有了两个IP地址,需要在服务器上添加网通或

  • Logout_logoutAll

    Logout_logoutAll现在到注销。早些时候我们离开这个函数一片空白,现在是时候来填补它。在UsersController:注销()添加以下:$this->Session->setFlash(‘Good-Bye’);$this->redirect($this->Auth->logout());这个设置一个会话flash信息和日志使用身份验证用户的注销方法。身份验证的注销方法基本上删除身份验证会话密钥并返回一个可

    2022年10月21日
  • ODT 学习笔记「建议收藏」

    ODT 学习笔记「建议收藏」珂朵莉,要一直幸福下去哟!warning:本文在大白天书写,脑子可能不大好用。目前代码选自题解,等有时间自己写一下。简介ODT(OldDriverTree(中文译名张舟树),又称ChthollyTree,即众人皆知的珂朵莉树)是一种非常暴力的思想或者做法(注意我没有说是数据结构)简单来说,其核心思想是把一段区间推平(这也是其适用的地方——区间赋值),推平之后,原数列变成一段一段的了(每段的数值相同),然后就可以搞事了。ODT在随机数据下,复杂度近似O(mlogn)O(mlog

  • Windows连接树莓派_树莓派4 win10专业版

    Windows连接树莓派_树莓派4 win10专业版一、准备工作树莓派3B+主板一个(系统已烧制完毕)win10电脑一台网线一条二、win10设置1.将网线与树莓派连接2.右击网络连接图标,选择打开“网络和Internet”设置3.打开网络和共享中心4.打开更改适配器设置5.选择以太网2,右击属性进入,对共享选项卡进行设置,勾选允许,并选择以太网三、查看树莓派IP通过快捷键win+R进入命令行界面…

    2022年10月14日
  • pythonrandom函数用法_python之random模块函数的使用

    pythonrandom函数用法_python之random模块函数的使用1)random.random()#用于生成一个0到1的随机浮点数,(0,1】2)random.randint(a,b)#用于生成一个指定范围内的整数,【a,b】3)random.randrange([start],stop[,step])#从指定范围内,按指定基数递增的集合中获取一个随机数。注意参数是整数,且不包括stop。random.randrange(10,30,2),结果相当…

发表回复

您的电子邮箱地址不会被公开。

关注全栈程序员社区公众号