Android Binder设计与实现 – 设计篇

Binder是Android系统进程间通讯(IPC)方式之一。Linux已经拥有管道,system V IPC,socket等IPC手段,却还要倚赖Binder来实现进程间通讯,说明Binder具备无可比拟的优点。深刻了解Binder并将之与传统 IPC作对比有助于咱们深刻领会进程间通讯的实现和性能优化。本文将对Binder的设计细节作一个全面的阐述,首先经过介绍Binder通讯模型和 Binder通讯协议了解Binder的设计需求;而后分别阐述Binder在系统不一样部分的表述方式和起的做用;最后还会解释Binder在数据接收端的设计考虑,包括线程池管理,内存映射和等待队列管理等。经过本文对Binder的详细介绍以及与其它IPC通讯方式的对比,读者将对Binder的优点和使用Binder做为Android主要IPC方式的缘由有深刻了解。
1 引言
 
基于Client-Server的通讯方式普遍应用于从互联网和数据库访问到嵌入式手持设备内部通讯等各个领域。智能手机平台特别是Android 系统中,为了向应用开发者提供丰富多样的功能,这种通讯方式更是无处不在,诸如媒体播放,视音频频捕获,到各类让手机更智能的传感器(加速度,方位,温度,光亮度等)都由不一样的Server负责管理,应用程序只需作为Client与这些Server创建链接即可以使用这些服务,花不多的时间和精力就能开发出使人眩目的功能。Client-Server方式的普遍采用对进程间通讯(IPC)机制是一个挑战。目前linux支持的IPC包括传统的管道,System V IPC,即消息队列/共享内存/信号量,以及socket中只有socket支持Client-Server的通讯方式。固然也能够在这些底层机制上架设一套协议来实现Client-Server通讯,但这样增长了系统的复杂性,在手机这种条件复杂,资源稀缺的环境下可靠性也难以保证。
 
另外一方面是传输性能。socket做为一款通用接口,其传输效率低,开销大,主要用在跨网络的进程间通讯和本机上进程间的低速通讯。消息队列和管道采用存储-转发方式,即数据先从发送方缓存区拷贝到内核开辟的缓存区中,而后再从内核缓存区拷贝到接收方缓存区,至少有两次拷贝过程。共享内存虽然无需拷贝,但控制复杂,难以使用。
 
表 1 各类IPC方式数据拷贝次数
  java


IPC
 
数据拷贝次数 node

 

共享内存
 
0 linux

 

Binder
 
1 android

 

Socket/管道/消息队列
 
2 算法

 
还有一点是出于安全性考虑。Android做为一个开放式,拥有众多开发者的的平台,应用程序的来源普遍,确保智能终端的安全是很是重要的。终端用户不但愿从网上下载的程序在不知情的状况下偷窥隐私数据,链接无线网络,长期操做底层设备致使电池很快耗尽等等。传统IPC没有任何安全措施,彻底依赖上层协议来确保。首先传统IPC的接收方没法得到对方进程可靠的UID/PID(用户ID/进程ID),从而没法鉴别对方身份。Android为每一个安装好的应用程序分配了本身的UID,故进程的UID是鉴别进程身份的重要标志。使用传统IPC只能由用户在数据包里填入UID/PID,但这样不可靠,容易被恶意程序利用。可靠的身份标记只有由IPC机制自己在内核中添加。其次传统IPC访问接入点是开放的,没法创建私有通道。好比命名管道的名称,system V的键值,socket的ip地址或文件名都是开放的,只要知道这些接入点的程序均可以和对端创建链接,无论怎样都没法阻止恶意程序经过猜想接收方地址得到链接。
 
基于以上缘由,Android须要创建一套新的IPC机制来知足系统对通讯方式,传输性能和安全性的要求,这就是Binder。Binder基于 Client-Server通讯模式,传输过程只需一次拷贝,为发送发添加UID/PID身份,既支持实名Binder也支持匿名Binder,安全性高。
2 面向对象的 Binder IPC
 
Binder使用Client-Server通讯方式:一个进程做为Server提供诸如视频/音频解码,视频捕获,地址本查询,网络链接等服务;多个进程做为Client向Server发起服务请求,得到所须要的服务。要想实现Client-Server通讯据必须实现如下两点:一是server 必须有肯定的访问接入点或者说地址来接受Client的请求,而且Client能够经过某种途径获知Server的地址;二是制定Command- Reply协议来传输数据。例如在网络通讯中Server的访问接入点就是Server主机的IP地址+端口号,传输协议为TCP协议。对Binder而言,Binder能够当作Server提供的实现某个特定服务的访问接入点, Client经过这个‘地址’向Server发送请求来使用该服务;对Client而言,Binder能够当作是通向Server的管道入口,要想和某个 Server通讯首先必须创建这个管道并得到管道入口。
 
与其它IPC不一样,Binder使用了面向对象的思想来描述做为访问接入点的Binder及其在Client中的入口:Binder是一个实体位于 Server中的对象,该对象提供了一套方法用以实现对服务的请求,就象类的成员函数。遍及于client中的入口能够当作指向这个binder对象的 ‘指针’,一旦得到了这个‘指针’就能够调用该对象的方法访问server。在Client看来,经过Binder‘指针’调用其提供的方法和经过指针调用其它任何本地对象的方法并没有区别,尽管前者的实体位于远端Server中,然后者实体位于本地内存中。‘指针’是C++的术语,而更一般的说法是引用,即Client经过Binder的引用访问Server。而软件领域另外一个术语‘句柄’也能够用来表述Binder在Client中的存在方式。从通讯的角度看,Client中的Binder也能够看做是Server Binder的‘代理’,在本地表明远端Server为Client提供服务。本文中会使用‘引用’或‘句柄’这个两普遍使用的术语。
 
面向对象思想的引入将进程间通讯转化为经过对某个Binder对象的引用调用该对象的方法,而其独特之处在于Binder对象是一个能够跨进程引用的对象,它的实体位于一个进程中,而它的引用却遍及于系统的各个进程之中。最诱人的是,这个引用和java里引用同样既能够是强类型,也能够是弱类型,并且能够从一个进程传给其它进程,让你们都能访问同一Server,就象将一个对象或引用赋值给另外一个引用同样。Binder模糊了进程边界,淡化了进程间通讯过程,整个系统仿佛运行于同一个面向对象的程序之中。形形色色的Binder对象以及星罗棋布的引用仿佛粘接各个应用程序的胶水,这也是Binder 在英文里的原意。
 
固然面向对象只是针对应用程序而言,对于Binder驱动和内核其它模块同样使用C语言实现,没有类和对象的概念。Binder驱动为面向对象的进程间通讯提供底层支持。
3 Binder 通讯模型
 
Binder框架定义了四个角色:Server,Client,ServiceManager(之后简称SMgr)以及驱动。其中 Server,Client,SMgr运行于用户空间,驱动运行于内核空间。这四个角色的关系和互联网相似:Server是服务器,Client是客户终端,SMgr是域名服务器(DNS),驱动是路由器。
3.1 Binder 驱动
 
和路由器同样,Binder驱动虽然默默无闻,倒是通讯的核心。尽管名叫‘驱动’,实际上和硬件设备没有任何关系,只是实现方式和设备驱动程序是同样的:它工做于内核态,提供open(),mmap(),poll(),ioctl()等标准文件操做,以字符驱动设备中的misc设备注册在设备目录 /dev下,用户经过/dev/binder访问该它。驱动负责进程之间Binder通讯的创建,Binder在进程之间的传递,Binder引用计数管理,数据包在进程之间的传递和交互等一系列底层支持。驱动和应用程序之间定义了一套接口协议,主要功能由ioctl()接口实现,不提供 read(),write()接口,由于ioctl()灵活方便,且可以一次调用实现先写后读以知足同步交互,而没必要分别调用write()和 read()。
3.2 ServiceManager 与实名Binder
 
和DNS相似,SMgr的做用是将字符形式的Binder名字转化成Client中对该Binder的引用,使得Client可以经过Binder 名字得到对Server中Binder实体的引用。注册了名字的Binder叫实名Binder,就象每一个网站除了有IP地址外都有本身的网址。 Server建立了Binder实体,为其取一个字符形式,可读易记的名字,将这个Binder连同名字以数据包的形式经过Binder驱动发送给 SMgr,通知SMgr注册一个名叫张三的Binder,它位于某个Server中。驱动为这个穿过进程边界的Binder建立位于内核中的实体节点以及 SMgr对实体的引用,将名字及新建的引用传递给SMgr。SMgr收数据包后,从中取出名字和引用填入一张查找表中。
 
细心的读者可能会发现其中的蹊跷:SMgr是一个进程,Server是另外一个进程,Server向SMgr注册Binder必然会涉及进程间通讯。当前实现的是进程间通讯却又要用到进程间通讯,这就好象蛋能够孵出鸡前提倒是要找只鸡来孵蛋。Binder的实现比较巧妙:预先创造一只鸡来孵蛋。 SMgr和其它进程一样采用Binder通讯,SMgr是Server端,有本身的Binder实体,其它进程都是Client,须要经过这个 Binder的引用来实现Binder的注册,查询和获取。SMgr提供的Binder比较特殊,它没有名字也不须要注册,当一个进程使用 BINDER_SET_CONTEXT_MGR命令将本身注册成SMgr时Binder驱动会自动为它建立Binder实体(这就是那只预先造好的鸡)。其次这个Binder的引用在全部Client中都固定为0而无须经过其它手段得到。也就是说,一个Server若要向SMgr注册本身Binder就必需经过0这个引用和SMgr的Binder通讯。类比网络通讯,0号引用就比如域名服务器的地址,你必须手工或动态配置好。要注意这里说的Client是相对SMgr而言的,一个应用程序是个提供服务的Server,但对SMgr来讲它仍然是个Client。
3.3 Client 得到实名Binder的引用
 
Server向SMgr注册了Binder实体及其名字后,Client就能够经过名字得到该Binder的引用了。Client也利用保留的0号引用向SMgr请求访问某个Binder:我申请得到名字叫张三的Binder的引用。SMgr收到这个链接请求,从请求数据包里得到Binder的名字,在查找表里找到该名字对应的条目,从条目中取出Binder的引用,将该引用做为回复发送给发起请求的Client。从面向对象的角度,这个 Binder对象如今有了两个引用:一个位于SMgr中,一个位于发起请求的Client中。若是接下来有更多的Client请求该Binder,系统中就会有更多的引用指向该Binder,就象java里一个对象存在多个引用同样。并且相似的这些指向Binder的引用是强类型,从而确保只要有引用 Binder实体就不会被释放掉。经过以上过程能够看出,SMgr象个火车票代售点,收集了全部火车的车票,能够经过它购买到乘坐各趟火车的票,即获得某个Binder的引用。
3.4 匿名 Binder
 
并非全部Binder都须要注册给SMgr广而告之的。Server端能够经过已经创建的Binder链接将建立的Binder实体传给 Client,固然这条已经创建的Binder链接必须是经过实名Binder实现。因为这个Binder没有向SMgr注册名字,因此是个匿名 Binder。Client将会收到这个匿名Binder的引用,经过这个引用向位于Server中的实体发送请求。匿名Binder为通讯双方创建一条私密通道,只要Server没有把匿名Binder发给别的进程,别的进程就没法经过穷举或猜想等任何方式得到该Binder的引用,向该Binder发送请求。
 
下图展现了参与Binder通讯的全部角色,将在之后章节中一一提到。 数据库

 


本篇文章来源于 Linux公社网站(www.linuxidc.com)  原文连接:http://www.linuxidc.com/Linux/2011-07/39271.htm 设计模式

 

4 Binder 协议
 
Binder协议基本格式是(命令+数据),使用ioctl(fd, cmd, arg)函数实现交互。命令由参数cmd承载,数据由参数arg承载,随cmd不一样而不一样。下表列举了全部命令及其所对应的数据:
 
表 2 Binder通讯命令字
  数组


命令 缓存

含义 安全

arg

 

BINDER_WRITE_READ
 
该命令向Binder写入或读取数据。参数分为两段:写部分和读部分。若是write_size不为0就先将write_buffer里的数据写入 Binder;若是read_size不为0再从Binder中读取数据存入read_buffer中。write_consumed和 read_consumed表示操做完成时Binder驱动实际写入或读出的数据个数。
 
struct binder_write_read {

signed long write_size;
 
signed long write_consumed;
 
unsigned long write_buffer;
 
signed long read_size;
 
signed long read_consumed;
 
unsigned long read_buffer;
 
};
 


BINDER_SET_MAX_THREADS
 
该命令告知Binder驱动接收方(一般是Server端)线程池中最大的线程数。因为Client是并发向Server端发送请求 的,Server端必须开辟线程池为这些并发请求提供服务。告知驱动线程池的最大值是为了让驱动在线程达到该值时不要再命令接收端启动新的线程。
 
int max_threads;

 

BINDER_SET_CONTEXT_MGR
 
将当前进程注册为SMgr。系统中同时只能存在一个SMgr。只要当前的SMgr没有调用close()关闭Binder驱动就不能有别的进程能够 成为SMgr。
 

 

BINDER_THREAD_EXIT
 
通知Binder驱动当前线程退出了。Binder会为全部参与Binder通讯的线程(包括Server线程池中的线程和Client发出请求的 线程)创建相应的数据结构。这些线程在退出时必须通知驱动释放相应的数据结构。
 

 

BINDER_VERSION
 
得到Binder驱动的版本号。
 

 
这其中最经常使用的命令是BINDER_WRITE_READ。该命令的参数包括两部分数据:一部分是向Binder写入的数据,一部分是要从 Binder读出的数据,驱动程序先处理写部分再处理读部分。这样安排的好处是应用程序能够很灵活地处理命令的同步或异步。例如若要发送异步命令能够只填 入写部分而将read_size置成0;若要只从Binder得到数据能够将写部分置空即write_size置成0;若要发送请求并同步等待返回数据可 以将两部分都置上。
 
4.1 BINDER_WRITE_READ 之写操做
 
Binder写操做的数据时格式一样也是(命令+数据)。这时候命令和数据都存放在binder_write_read 结构write_buffer域指向的内存空间里,多条命令能够连续存放。数据紧接着存放在命令后面,格式根据命令不一样而不一样。下表列举了Binder写 操做支持的命令:
 
表 3 Binder写操做命令字
 


cmd
 
含义

arg

 

BC_TRANSACTION
BC_REPLY
 
BC_TRANSACTION用于写入请求数据;BC_REPLY用于写入回复数据。其后面紧接着一个 binder_transaction_data结构体代表要写入的数据。
 
struct binder_transaction_data

 

BC_ACQUIRE_RESULT
BC_ATTEMPT_ACQUIRE
 
暂未实现
 

 

BC_FREE_BUFFER
 
释放一块映射的内存。Binder接收方经过mmap()映射一块较大的内存空间,Binder驱动基于这片内存采用最佳匹配算法实现接收数据缓存 的动态分配和释放,知足并发请求对接收缓存区的需求。应用程序处理完这片数据后必须尽快使用该命令释放缓存区,不然会由于缓存区耗尽而没法接收新数据。
 
指向须要释放的缓存区的指针;该指针位于收到的Binder数据包中

 

BC_INCREFS
BC_ACQUIRE
BC_RELEASE
BC_DECREFS
 
这组命令增长或减小Binder的引用计数,用以实现强指针或弱指针的功能。
 
32位Binder引用号

 

BC_INCREFS_DONE
BC_ACQUIRE_DONE
 
第一次增长Binder实体引用计数时,驱动向Binder实体所在的进程发送BR_INCREFS, BR_ACQUIRE消息;Binder实体所在的进程处理完毕回馈BC_INCREFS_DONE,BC_ACQUIRE_DONE
 
void *ptr:Binder实体在用户空间中的指针

void *cookie:与该实体相关的附加数据
 


BC_REGISTER_LOOPER
BC_ENTER_LOOPER
BC_EXIT_LOOPER
 
这组命令同BINDER_SET_MAX_THREADS一道实现Binder驱动对接收方线程池管理。BC_REGISTER_LOOPER通知 驱动线程池中一个线程已经建立了;BC_ENTER_LOOPER通知驱动该线程已经进入主循环,能够接收数据;BC_EXIT_LOOPER通知驱动该 线程退出主循环,再也不接收数据。
 

 

BC_REQUEST_DEATH_NOTIFICATION
 
得到Binder引用的进程经过该命令要求驱动在Binder实体销毁获得通知。虽然说强指针能够确保只要有引用就不会销毁实体,但这毕竟是个跨进程 的引用,谁也没法保证明体因为所在的Server关闭Binder驱动或异常退出而消失,引用者能作的是要求Server在此刻给出通知。
 
uint32 *ptr; 须要获得死亡通知的Binder引用

void **cookie: 与死亡通知相关的信息,驱动会在发出死亡通知时返回给发出请求的进程。
 


BC_DEAD_BINDER_DONE
 
收到实体死亡通知书的进程在删除引用后用本命令告知驱动。
 
void **cookie

 
在这些命令中,最经常使用的是BC_TRANSACTION/BC_REPLY命令对,Binder数据经过这对命令发送给接收方。这对命令所承载的数 据包由结构体struct binder_transaction_data定义。Binder交互有同步和异步之分,利用binder_transaction_data中 flag域区分。若是flag域的TF_ONE_WAY位为1则为异步交互,即Client端发送完请求交互即结束, Server端再也不返回BC_REPLY数据包;不然Server会返回BC_REPLY数据包,Client端必须等待接收完该数据包方才完成一次交 互。
 
4.2 BINDER_WRITE_READ :从Binder读出数据
 
从Binder里读出的数据格式和向Binder中写入的数据格式同样,采用(消息ID+数据)形式,而且多条消息能够连续存放。下表列举了从 Binder读出的命令字及其相应的参数:
 
表 4 Binder读操做消息ID
 


消息

含义

参数

 

BR_ERROR
 
发生内部错误(如内存分配失败)
 

 

BR_OK
BR_NOOP
 
操做完成
 

 

BR_SPAWN_LOOPER
 
该消息用于接收方线程池管理。当驱动发现接收方全部线程都处于忙碌状态且线程池里的线程总数没有超过BINDER_SET_MAX_THREADS 设置的最大线程数时,向接收方发送该命令要求建立更多线程以备接收数据。
 

 

BR_TRANSACTION
BR_REPLY
 
这两条消息分别对应发送方的BC_TRANSACTION和BC_REPLY,表示当前接收的数据是请求或是回复。
 
binder_transaction_data

 

BR_ACQUIRE_RESULT
BR_ATTEMPT_ACQUIRE
BR_FINISHED
 
还没有实现
 

 

BR_DEAD_REPLY
 
交互过程当中若是发现对方进程或线程已经死亡则返回该消息
 

 

BR_TRANSACTION_COMPLETE
 
发送方经过BC_TRANSACTION或BC_REPLY发送完一个数据包后,都能收到该消息作为成功发送的反馈。这和BR_REPLY不同, 是驱动告知发送方已经发送成功,而不是接收方返回请求数据。因此无论同步仍是异步交互接收方都能得到本消息。
 

 

BR_INCREFS
BR_ACQUIRE
BR_RELEASE
BR_DECREFS
 
这一组消息用于管理强/弱指针的引用计数。只有提供Binder实体的进程才能收到这组消息。
 
void *ptr:Binder实体在用户空间中的指针

void *cookie:与该实体相关的附加数据
 


BR_DEAD_BINDER
BR_CLEAR_DEATH_NOTIFICATION_DONE
 
向得到Binder引用的进程发送Binder实体死亡通知书;收到死亡通知书的进程接下来会返回BC_DEAD_BINDER_DONE作确认。
 
void **cookie:在使用BC_REQUEST_DEATH_NOTIFICATION注册死亡通知时的附加参数。

 

BR_FAILED_REPLY
 
若是发送非法引用号则返回该消息
 

 
和写数据同样,其中最重要的消息是BR_TRANSACTION 或BR_REPLY,代表收到了一个格式为binder_transaction_data的请求数据包(BR_TRANSACTION)或返回数据包 (BR_REPLY)。
 
4.3 struct binder_transaction_data :收发数据包结构
 
该结构是Binder接收/发送数据包的标准格式,每一个成员定义以下:
 
表 5 Binder收发数据包结构:binder_transaction_data
 


成员

含义

 

union {

size_t handle;
 
void *ptr;
 
} target;
 
对于发送数据包的一方,该成员指明发送目的地。因为目的是在远端,因此这里填入的是对Binder实体的引用,存放在target.handle 中。如前述,Binder的引用在代码中也叫句柄(handle)。
 
当数据包到达接收方时,驱动已将该成员修改为Binder实体,即指向Binder对象内存的指针,使用target.ptr来得到。该指针是接收 方在将Binder实体传输给其它进程时提交给驱动的,驱动程序可以自动将发送方填入的引用转换成接收方Binder对象的指针,故接收方能够直接将其当 作对象指针来使用(一般是将其reinterpret_cast成相应类)。
 


void *cookie;
 
发送方忽略该成员;接收方收到数据包时,该成员存放的是建立Binder实体时由该接收方自定义的任意数值,作为与Binder指针相关的额外信息 存放在驱动中。驱动基本上不关心该成员。

 

unsigned int code;
 
该成员存放收发双方约定的命令码,驱动彻底不关心该成员的内容。一般是Server端定义的公共接口函数的编号。

 

unsigned int flags;
 
与交互相关的标志位,其中最重要的是TF_ONE_WAY位。若是该位置上代表此次交互是异步的,接收方不会返回任何数据。驱动利用该位来决定是否 构建与返回有关的数据结构。另一位TF_ACCEPT_FDS是出于安全考虑,若是发起请求的一方不但愿在收到的回复中接收文件形式的Binder能够 将该位置上。由于收到一个文件形式的Binder会自动为接收方打开一个文件,使用该位能够防止打开文件过多。

 

pid_t sender_pid;

uid_t sender_euid;
 
该成员存放发送方的进程ID和用户ID,由驱动负责填入,接收方能够读取该成员获知发送方的身份。

 

size_t data_size;
 
该成员表示data.buffer指向的缓冲区存放的数据长度。发送数据时由发送方填入,表示即将发送的数据长度;在接收方用来告知接收到数据的长 度。

 

size_t offsets_size;
 
驱动通常状况下不关心data.buffer里存放什么数据,但若是有Binder在其中传输则须要将其相对data.buffer的偏移位置指出 来让驱动知道。有可能存在多个Binder同时在数据中传递,因此须用数组表示全部偏移位置。本成员表示该数组的大小。

 

union {

struct {
 
const void *buffer;
 
const void *offsets;
 
} ptr;
 
uint8_t buf[8];
 
} data;
 
data.bufer存放要发送或接收到的数据;data.offsets指向Binder偏移位置数组,该数组能够位于data.buffer 中,也能够在另外的内存空间中,并没有限制。buf[8]是为了不管保证32位仍是64位平台,成员data的大小都是8个字节。

 
这里有必要再强调一下offsets_size和data.offsets两个成员,这是Binder通讯有别于其它IPC的地方。如前 述,Binder采用面向对象的设计思想,一个Binder实体能够发送给其它进程从而创建许多跨进程的引用;另外这些引用也能够在进程之间传递,就象 java里将一个引用赋给另外一个引用同样。为Binder在不一样进程中创建引用必须有驱动的参与,由驱动在内核建立并注册相关的数据结构后接收方才能使用 该引用。并且这些引用能够是强类型,须要驱动为其维护引用计数。然而这些跨进程传递的Binder混杂在引用程序发送的数据包里,数据格式彻底由用户定 义,若是不把它们一一标记出来告知驱动,驱动将没法从数据中将它们提取出来。因而就使用数组data.offsets存放用户数据中每一个Binder相对 data.buffer的偏移量,用offsets_size表示这个数组的大小。驱动在发送数据包时会根据data.offsets和 offset_size将散落于data.buffer中的Binder找出来并一一为它们建立相关的数据结构。在数据包中传输的Binder是类型为 struct flat_binder_object的结构体,详见后文。
 
对于接收方来讲,该结构只至关于一个定长的消息头,真正的用户数据存放在data.buffer所指向的缓存区中。若是发送方在数据中内嵌了一个或 多个Binder,接收到的数据包中一样会用data.offsets和offset_size指出每一个Binder的位置和总个数。不过一般接收方能够 忽略这些信息,由于接收方是知道数据格式的,参考双方约定的格式定义就能知道这些Binder在什么位置。

 

 

本篇文章来源于 Linux公社网站(www.linuxidc.com)  原文连接:http://www.linuxidc.com/Linux/2011-07/39271p2.htm

 

 


5 Binder 的表述
 
考察一次Binder通讯的全过程会发现,Binder存在于系统如下几个部分中:
 
· 应用程序进程:又分为Server进程和Client进程
 
· Binder驱动:Server和Client有不一样表述形式
 
· 传输数据:因为Binder能够跨进程传递,须要在传输数据中予以表述
 
在系统不一样部分,Binder实现的功能不一样,表现形式也不同的。接下来逐一探讨Binder在各部分所扮演的角色和使用的数据结构。
 
5.1 Binder 在应用程序中的表述
 
虽然Binder用到了面向对象的思想,但并不限制应用程序必定要使用面向对象的语言,不管是C语言仍是C++语言均可以很容易的使用Binder 来通讯。例如尽管Android主要使用java或C++,象SMgr这么重要的进程就是用C语言实现的。不过面向对象的方式表述起来更方便,因此本文假 设应用程序是用面向对象语言实现的。
 
Binder本质上只是一种底层通讯方式,和具体服务没有关系。为了提供具体服务,Server必须提供一套接口函数以便Client经过远程访问 使用各类服务。这时一般采用Proxy设计模式:将接口函数定义在一个抽象类中,Server和Client都会以该抽象类为基类实现全部接口函数,所不 同的是Server端是真正的功能实现,而Client端是对这些函数远程调用请求的包装。如何将Binder和Proxy设计模式结合起来是应用程序实 现面向对象Binder通讯的根本问题。
 
5.1.1 Binder 在Server端的表述 – Binder实体
 
作为Proxy设计模式的基础,首先定义一个抽象接口类封装Server全部功能,其中包含一系列纯虚函数留待Server和Proxy各自实现。 因为这些函数须要跨进程调用,须为其一一编号,从而Server能够根据收到的编号决定调用哪一个函数。其次就要引入Binder了。Server端定义另 一个Binder抽象类处理来自Client的Binder请求数据包,其中最重要的成员是虚函数onTransact()。该函数分析收到的数据包,调 用相应的接口函数处理请求。
 
接下来采用继承方式以接口类和Binder抽象类为基类构建Binder在Server中的实体,实现基类里全部的虚函数,包括公共接口函数以及数 据包处理函数:onTransact()。这个函数的输入是来自Client的binder_transaction_data结构的数据包。前面提到, 该结构里有个成员code,包含此次请求的接口函数编号。onTransact()将case-by-case地解析code值,从数据包里取出函数参 数,调用接口类中相应的,已经实现的公共接口函数。函数执行完毕,若是须要返回数据就再构建一个binder_transaction_data包将返回 数据包填入其中。
 
那么各个Binder实体的onTransact()又是何时调用呢?这就须要驱动参与了。前面说过,Binder实体需要以Binde传输结 构flat_binder_object形式发送给其它进程才能创建Binder通讯,而Binder实体指针就存放在该结构的handle域中。驱动根 据Binder位置数组从传输数据中获取该Binder的传输结构,为它建立位于内核中的Binder节点,将Binder实体指针记录在该节点中。若是 接下来有其它进程向该Binder发送数据,驱动会根据节点中记录的信息将Binder实体指针填入binder_transaction_data的 target.ptr中返回给接收线程。接收线程从数据包中取出该指针,reinterpret_cast成Binder抽象类并调用 onTransact()函数。因为这是个虚函数,不一样的Binder实体中有各自的实现,从而能够调用到不一样Binder实体提供的 onTransact()。
 
5.1.2 Binder 在Client端的表述 – Binder引用
 
作为Proxy设计模式的一部分,Client端的Binder一样要继承Server提供的公共接口类并实现公共函数。但这不是真正的实现,而是 对远程函数调用的包装:将函数参数打包,经过Binder向Server发送申请并等待返回值。为此Client端的Binder还要知道Binder实 体的相关信息,即对Binder实体的引用。该引用或是由SMgr转发过来的,对实名Binder的引用或是由另外一个进程直接发送过来的,匿名 Binder的引用。
 
因为继承了一样的公共接口类,Client Binder提供了与Server Binder同样的函数原型,使用户感受不出Server是运行在本地仍是远端。Client Binder中,公共接口函数的实现方式是:建立一个binder_transaction_data数据包,将其对应的编码填入code域,将调用该函 数所需的参数填入data.buffer指向的缓存中,并指明数据包的目的地,那就是已经得到的对Binder实体的引用,填入数据包的 target.handle中。注意这里和Server的区别:实际上target域是个联合体,包括ptr和handle两个成员,前者用于做为响应方 的Server,指向 Binder实体对应的内存空间;后者用于做为请求方的Client,存放Binder实体的引用,告知驱动数据包将路由给哪一个实体。数据包准备好后,通 过驱动接口发送出去。通过BC_TRANSACTION/BC_REPLY回合完成函数的远程调用并获得返回值。
 
5.2 Binder 在传输数据中的表述
 
Binder能够塞在数据包的有效数据中越进程边界从一个进程传递给另外一个进程,这些传输中的Binder用结构 flat_binder_object表示,以下表所示:
 
表 6 Binder传输结构:flat_binder_object
 


成员

含义

 

unsigned long type
 
代表该Binder的类型,包括如下几种:

BINDER_TYPE_BINDER:表示传递的是Binder实体,而且指向该实体的引用都是强类型;
 
BINDER_TYPE_WEAK_BINDER:表示传递的是Binder实体,而且指向该实体的引用都是弱类型;
 
BINDER_TYPE_HANDLE:表示传递的是Binder强类型的引用
 
BINDER_TYPE_WEAK_HANDLE:表示传递的是Binder弱类型的引用
 
BINDER_TYPE_FD:表示传递的是文件形式的Binder,详见下节
 


unsigned long flags
 
该域只对第一次传递Binder实体时有效,由于此刻驱动须要在内核中建立相应的实体节点,有些参数须要从该域取出:

第0-7位:代码中用FLAT_BINDER_FLAG_PRIORITY_MASK取得,表示处理本实体请求数据包的线程的最低优先级。当一个应 用程序提供多个实体时,能够经过该参数调整分配给各个实体的处理能力。
 
第8位:代码中用FLAT_BINDER_FLAG_ACCEPTS_FDS取得,置1表示该实体能够接收其它进程发过来的文件形式的 Binder。因为接收文件形式的Binder会在本进程中自动打开文件,有些Server能够用该标志禁止该功能,以防打开过多文件。
 


union {

void *binder;
 
signed long handle;
 
};
 
当传递的是Binder实体时使用binder域,指向Binder实体在应用程序中的地址。

当传递的是Binder引用时使用handle域,存放Binder在进程中的引用号。
 


void *cookie;
 
该域只对Binder实体有效,存放与该Binder有关的附加信息。

 
不管是Binder实体仍是对实体的引用都从属与某个进程,因此该结构不能透明地在进程之间传输,必须有驱动的参与。例如当Server把 Binder实体传递给Client时,在发送数据中,flat_binder_object中的type是 BINDER_TYPE_BINDER,binder指向Server进程用户空间地址。若是透传给接收端将毫无用处,驱动必须对数据流中的这个 Binder作修改:将type该成BINDER_TYPE_HANDLE;为这个Binder在接收进程中建立位于内核中的引用并将引用号填入 handle中。对于发生数据流中引用类型的Binder也要作一样转换。通过处理后接收进程从数据流中取得的Binder引用才是有效的,才能够将其填 入数据包binder_transaction_data的target.handle域,向Binder实体发送请求。
 
这样作也是出于安全性考虑:应用程序不能随便猜想一个引用号填入target.handle中就能够向Server请求服务了,由于驱动并无为你 在内核中建立该引用,一定会驱动被拒绝。惟有通过身份认证确认合法后,由‘权威机构’经过数据流授予你的Binder才能使用,由于这时驱动已经在内核中 为你创建了引用,交给你的引用号是合法的。
 
下表总结了当flat_binder_object结构穿过驱动时驱动所作的操做:
 
表 7 驱动对flat_binder_object的操做
 


Binder 类型( type 域)

在发送方的操做

在接收方的操做

 

BINDER_TYPE_BINDER

BINDER_TYPE_WEAK_BINDER
 
只有实体所在的进程能发送该类型的Binder。若是是第一次发送驱动将建立实体在内核中的节点,并保存binder,cookie,flag域。
 
若是是第一次接收该Binder则建立实体在内核中的引用;将handle域替换为新建的引用号;将type域替换为 BINDER_TYPE_(WEAK_)HANDLE

 

BINDER_TYPE_HANDLE

BINDER_TYPE_WEAK_HANDLE
 
得到Binder引用的进程都能发送该类型Binder。驱动根据handle域提供的引用号查找创建在内核的引用。若是找到说明引用号合法,不然 拒绝该发送请求。
 
若是收到的Binder实体位于接收进程中:将ptr域替换为保存在节点中的binder值;cookie替换为保存在节点中的cookie 值;type替换为BINDER_TYPE_(WEAK_)BINDER。
 
若是收到的Binder实体不在接收进程中:若是是第一次接收则建立实体在内核中的引用;将handle域替换为新建的引用号
 


BINDER_TYPE_FD
 
验证handle域中提供的打开文件号是否有效,无效则拒绝该发送请求。
 
在接收方建立新的打开文件号并将其与提供的打开文件描述结构绑定。

 
5.2.1 文件形式的 Binder
 
除了一般意义上用来通讯的Binder,还有一种特殊的Binder:文件Binder。这种Binder的基本思想是:将文件当作Binder实 体,进程打开的文件号当作Binder的引用。一个进程能够将它打开文件的文件号传递给另外一个进程,从而另外一个进程也打开了同一个文件,就象Binder 的引用在进程之间传递同样。
 
一个进程打开一个文件,就得到与该文件绑定的打开文件号。从Binder的角度,linux在内核建立的打开文件描述结构struct file是Binder的实体,打开文件号是该进程对该实体的引用。既然是Binder那么就能够在进程之间传递,故也能够用 flat_binder_object结构将文件Binder经过数据包发送至其它进程,只是结构中type域的值为BINDER_TYPE_FD,代表 该Binder是文件Binder。而结构中的handle域则存放文件在发送方进程中的打开文件号。咱们知道打开文件号是个局限于某个进程的值,一旦跨 进程就没有意义了。这一点和Binder实体用户指针或Binder引用号是同样的,若要跨进程一样须要驱动作转换。驱动在接收Binder的进程空间创 建一个新的打开文件号,将它与已有的打开文件描述结构struct file勾连上,今后该Binder实体又多了一个引用。新建的打开文件号覆盖flat_binder_object中原来的文件号交给接收进程。接收进 程利用它能够执行read(),write()等文件操做。
 
传个文件为啥要这么麻烦,直接将文件名用Binder传过去,接收方用open()打开不就好了吗?其实这仍是有区别的。首先对同一个打开文件共享 的层次不一样:使用文件Binder打开的文件共享linux VFS中的struct file,struct dentry,struct inode结构,这意味着一个进程使用read()/write()/seek()改变了文件指针另外一个进程的文件指针也会改变;而若是两个进程分别使用 文件名打开同一文件则有各自的struct file结构,从而各自独立维护文件指针,互不干扰。其次是一些特殊设备文件要求在struct file一级共享才能使用,例如android的另外一个驱动ashmem,它和Binder同样也是misc设备,用以实现进程间的共享内存。一个进程打 开的ashmem文件只有经过文件Binder发送到另外一个进程才能实现内存共享,这大大提升了内存共享的安全性,道理和Binder加强了IPC的安全 性是同样的。
 
5.3 Binder 在驱动中的表述
 
驱动是Binder通讯的核心,系统中全部的Binder实体以及每一个实体在各个进程中的引用都登记在驱动中;驱动须要记录Binder引用 ->实体之间多对一的关系;为引用找到对应的实体;在某个进程中为实体建立或查找到对应的引用;记录Binder的归属地(位于哪一个进程中);经过 管理Binder的强/弱引用建立/销毁Binder实体等等。
 
驱动里的Binder是何时建立的呢?前面提到过,为了实现实名Binder的注册,系统必须建立第一只鸡 – 为SMgr建立的,注册实名Binder专用的Binder实体,负责实名Binder注册过程当中的进程间通讯。既然建立了实体也要有对应的引用:驱动将 全部进程中的0号引用都预留给该Binder实体,即一开始全部进程的0号引用都指注册实名Binder专用的Binder,无须特殊操做任何进程经过0 号引用均可以注册实名Binder。接下来随着应用程序经过不断地注册实名Binder,不断向SMgr索要Binder的引用,不断将Binder从一 个进程传递给另外一个进程,愈来愈多的Binder以传输结构 – flat_binder_object的形式穿越驱动作跨进程的迁徙。因为binder_transaction_data中data.offset数组 的存在,全部流经驱动的Binder都逃不过驱动的眼睛。Binder将对每一个穿越进程边界的Binder作以下操做:检查传输结构的type域,若是是 BINDER_TYPE_BINDER或BINDER_TYPE_WEAK_BINDER则建立Binder的实体;若是是 BINDER_TYPE_HANDLE或BINDER_TYPE_WEAK_HANDLE则建立Binder的引用;若是是 BINDER_TYPE_HANDLE则为进程打开文件,无须建立任何数据结构。详细过程可参考表7。随着愈来愈多的Binder实体或引用穿过驱动在进 程间传递,驱动会在内核里建立愈来愈多的节点或引用,固然这个过程对用户来讲是透明的。
 
5.3.1 Binder 实体在驱动中的表述
 
驱动中的Binder实体也叫‘节点’,隶属于提供实体的进程,由struct binder_node结构来表示:
 
表 8 Binder节点描述结构:binder_node
 


成员

含义

 

int debug_id;
 
用于调试

 

struct binder_work work;
 
当本节点引用计数发生改变,须要通知所属进程时,经过该成员挂入所属进程的to-do队列里,唤醒所属进程执行Binder实体引用计数的修改。

 

union {

struct rb_node rb_node;
 
struct hlist_node dead_node;
 
};
 
每一个进程都维护一棵红黑树,以Binder实体在用户空间的指针,即本结构的ptr成员为索引存放该进程全部的Binder实体。这样驱动能够根据 Binder实体在用户空间的指针很快找到其位于内核的节点。rb_node用于将本节点链入该红黑树中。
 
销毁节点时须将rb_node从红黑树中摘除,但若是本节点还有引用没有切断,就用dead_node将节点隔离到另外一个链表中,直到通知全部进程 切断与该节点的引用后,该节点才可能被销毁。
 


struct binder_proc *proc;
 
本成员指向节点所属的进程,即提供该节点的进程。

 

struct hlist_head refs;
 
本成员是队列头,全部指向本节点的引用都连接在该队列里。这些引用可能隶属于不一样的进程。经过该队列能够遍历指向该节点的全部引用。

 

int internal_strong_refs;
 
用以实现强指针的计数器:产生一个指向本节点的强引用该计数就会加1。

 

int local_weak_refs;
 
驱动为传输中的Binder设置的弱引用计数。若是一个Binder打包在数据包中从一个进程发送到另外一个进程,驱动会为该Binder增长引用计 数,直到接收进程经过BC_FREE_BUFFER通知驱动释放该数据包的数据区为止。

 

int local_strong_refs;
 
驱动为传输中的Binder设置的强引用计数。同上。

 

void __user *ptr;
 
指向用户空间Binder实体的指针,来自于flat_binder_object的binder成员

 

void __user *cookie;
 
指向用户空间的附加指针,来自于flat_binder_object的cookie成员

 

unsigned has_strong_ref;

unsigned pending_strong_ref;
 
unsigned has_weak_ref;
 
unsigned pending_weak_ref
 
这一组标志用于控制驱动与Binder实体所在进程交互式修改引用计数

 

unsigned has_async_transaction;
 
该成员代表该节点在to-do队列中有异步交互还没有完成。驱动将全部发送往接收端的数据包暂存在接收进程或线程开辟的to-do队列里。对于异步交 互,驱动作了适当流控:若是to-do队列里有异步交互尚待处理则该成员置1,这将致使新到的异步交互存放在本结构成员 – asynch_todo队列中,而不直接送到to-do队列里。目的是为同步交互让路,避免长时间阻塞发送端。

 

unsigned accept_fds
 
代表节点是否赞成接受文件方式的Binder,来自flat_binder_object中flags成员的 FLAT_BINDER_FLAG_ACCEPTS_FDS位。因为接收文件Binder会为进程自动打开一个文件,占用有限的文件描述符,节点能够设置 该位拒绝这种行为。

 

int min_priority
 
设置处理Binder请求的线程的最低优先级。发送线程将数据提交给接收线程处理时,驱动会将发送线程的优先级也赋予接收线程,使得数据即便跨了进 程也能以一样优先级获得处理。不过若是发送线程优先级太低,接收线程将以预设的最小值运行。
 
该域的值来自于flat_binder_object中flags成员。
 


struct list_head async_todo
 
异步交互等待队列;用于分流发往本节点的异步交互包

 
每一个进程都有一棵红黑树用于存放建立好的节点,以Binder在用户空间的指针做为索引。每当在传输数据中侦测到一个表明Binder实体的 flat_binder_object,先以该结构的binder指针为索引搜索红黑树;若是没找到就建立一个新节点添加到树中。因为对于同一个进程来讲 内存地址是惟一的,因此不会重复建设形成混乱。
 
5.3.2 Binder 引用在驱动中的表述
 
和实体同样,Binder的引用也是驱动根据传输数据中的flat_binder_object建立的,隶属于得到该引用的进程,用struct binder_ref结构体表示:
 
表 9 Binder引用描述结构:binder_ref
 


成员

含义

 

int debug_id;
 
调试用

 

struct rb_node rb_node_desc;
 
每一个进程有一棵红黑树,进程全部引用以引用号(即本结构的desc域)为索引添入该树中。本成员用作连接到该树的一个节点。

 

struct rb_node rb_node_node;
 
每一个进程又有一棵红黑树,进程全部引用以节点实体在驱动中的内存地址(即本结构的node域)为所引添入该树中。本成员用作连接到该树的一个节点。

 

struct hlist_node node_entry;
 
该域将本引用作为节点链入所指向的Binder实体结构binder_node中的refs队列

 

struct binder_proc *proc;
 
本引用所属的进程

 

struct binder_node *node;
 
本引用所指向的节点(Binder实体)

 

uint32_t desc;
 
本结构的引用号

 

int strong;
 
强引用计数

 

int weak;
 
弱引用计数

 

struct binder_ref_death *death;
 
应用程序向驱动发送BC_REQUEST_DEATH_NOTIFICATION或BC_CLEAR_DEATH_NOTIFICATION命令从 而当Binder实体销毁时可以收到来自驱动的提醒。该域不为空代表用户订阅了对应实体销毁的‘噩耗’。

 
就象一个对象有不少指针同样,同一个Binder实体可能有不少引用,不一样的是这些引用可能分布在不一样的进程中。和实体同样,每一个进程使用红黑树存 放全部该进程正在使用的引用。但Binder的引用能够经过两个键值索引:
 
· 对应实体在内核中的地址。注意这里指的是驱动建立于内核中的binder_node结构的地址,而不是Binder实体在用户进程中的地址。实体在内核中 的地址是惟一的,用作索引不会产生二义性;但实体可能来自不一样用户进程,而实体在不一样用户进程中的地址可能重合,不能用来作索引。驱动利用该红黑树在一个 进程中快速查找某个Binder实体所对应的引用(一个实体在一个进程中只创建一个引用)。
 
· 引用号。引用号是驱动为引用分配的一个32位标识,在一个进程内是惟一的,而在不一样进程中可能会有一样的值,这和进程的打开文件号很相似。引用号将返回给 应用程序,能够看做Binder引用在用户进程中的句柄。除了0号引用在全部进程里都保留给SMgr,其它值由驱动在建立引用时动态分配。向Binder 发送数据包时,应用程序经过将引用号填入binder_transaction_data结构的target.handle域中代表该数据包的目的 Binder。驱动根据该引用号在红黑树中找到引用的binder_ref结构,进而经过其node域知道目标Binder实体所在的进程及其它相关信 息,实现数据包的路由。
 

 

本篇文章来源于 Linux公社网站(www.linuxidc.com)  原文连接:http://www.linuxidc.com/Linux/2011-07/39271p3.htm

相关文章
相关标签/搜索