我第一次接触Binder
是在写Service
的时候,应用层开发时不时会用到Service
,而Service
里面有一个onBind
接口,而其返回值需要一个IBinder
对象,而这个返回对象,我们通常是需要继承自Binder
后面慢慢知道了,Binder
是用来跨进程通信的,而跨进程通信,应用层开发通常使用aidl来完成,所以对Binder
的认知也就只停于此了
然而,在研究源码的时候,跨进程通信非常频繁,虽然研究了一段时间的源码了,但对Binder
一直是一知半解,于是准备系统学习下安卓的跨进程通信
首先,什么是Binder
?Binder
是一种进程间通信机制,基于开源的 OpenBinder 实现;OpenBinder 起初由 Be Inc. 开发,后由 Plam Inc. 接手。从字面上来解释 Binder
有胶水、粘合剂的意思,顾名思义就是粘和不同的进程,使之实现通信。
Android 系统是基于 Linux 内核的,Linux 已经提供了管道、消息队列、共享内存和 Socket 等 IPC 机制。那为什么 Android 还要提供 Binder
来实现 IPC 呢?主要是基于性能、稳定性和安全性几方面的原因。
首先说说性能上的优势。Socket 作为一款通用接口,其传输效率低,开销大,主要用在跨网络的进程间通信和本机上进程间的低速通信。消息队列和管道采用存储-转发方式,即数据先从发送方缓存区拷贝到内核开辟的缓存区中,然后再从内核缓存区拷贝到接收方缓存区,至少有两次拷贝过程。共享内存虽然无需拷贝,但控制复杂,难以使用。Binder 只需要一次数据拷贝,性能上仅次于共享内存。
IPC方式 | 数据拷贝次数 |
---|---|
共享内存 | 0 |
Binder | 1 |
Socket/管道/消息队列 | 2 |
再说说稳定性,Binder 基于 C/S 架构,客户端(Client)有什么需求就丢给服务端(Server)去完成,架构清晰、职责明确又相互独立,自然稳定性更好。共享内存虽然无需拷贝,但是控制负责,难以使用。从稳定性的角度讲,Binder 机制是优于内存共享的。
另一方面就是安全性。Android 作为一个开放性的平台,市场上有各类海量的应用供用户选择安装,因此安全性对于 Android 平台而言极其重要。作为用户当然不希望我们下载的 APP 偷偷读取我的通信录,上传我的隐私数据,后台偷跑流量、消耗手机电量。传统的 IPC 没有任何安全措施,完全依赖上层协议来确保。首先传统的 IPC 接收方无法获得对方可靠的进程用户ID/进程ID(UID/PID),从而无法鉴别对方身份。Android 为每个安装好的 APP 分配了自己的 UID,故而进程的 UID 是鉴别进程身份的重要标志。传统的 IPC 只能由用户在数据包中填入 UID/PID,但这样不可靠,容易被恶意程序利用。可靠的身份标识只有由 IPC 机制在内核中添加。其次传统的 IPC 访问接入点是开放的,只要知道这些接入点的程序都可以和对端建立连接,不管怎样都无法阻止恶意程序通过猜测接收方地址获得连接。同时 Binder 既支持实名 Binder,又支持匿名 Binder,安全性高。
基于上述原因,Android 需要建立一套新的 IPC 机制来满足系统对稳定性、传输性能和安全性方面的要求,这就是 Binder。
最后用一张表格来总结下 Binder 的优势:
优势 | 描述 |
---|---|
性能 | 只需要一次数据拷贝,性能上仅次于共享内存 |
安全性 | 为每个 APP 分配 UID,进程的 UID 是鉴别进程身份的重要标志 |
稳定性 | 基于 C/S 架构,职责明确、架构清晰,因此稳定性好 |
了解 Linux IPC 相关的概念和原理有助于我们理解 Binder 通信原理。因此,在介绍 Binder 跨进程通信原理之前,我们先聊聊 Linux 系统下传统的进程间通信是如何实现。
这里我们先从 Linux 中进程间通信涉及的一些基本概念开始介绍,然后逐步展开,向大家说明传统的进程间通信的原理。
上图展示了 Liunx 中跨进程通信涉及到的一些基本概念:
简单的说就是操作系统中,进程与进程间内存是不共享的。两个进程就像两个平行的世界,A 进程没法直接访问 B 进程的数据,这就是进程隔离的通俗解释。A 进程和 B 进程之间要进行数据交互就得采用特殊的通信机制:进程间通信(IPC)。
现在操作系统都是采用的虚拟存储器,对于 32 位系统而言,它的寻址空间(虚拟存储空间)就是 2 的 32 次方,也就是 4GB。操作系统的核心是内核,独立于普通的应用程序,可以访问受保护的内存空间,也可以访问底层硬件设备的权限。为了保护用户进程不能直接操作内核,保证内核的安全,操作系统从逻辑上将虚拟空间划分为用户空间(User Space)和内核空间(Kernel Space)。针对 Linux 操作系统而言,将最高的 1GB 字节供内核使用,称为内核空间;较低的 3GB 字节供各进程使用,称为用户空间。
简单的说就是,内核空间(Kernel)是系统内核运行的空间,用户空间(User Space)是用户程序运行的空间。为了保证安全性,它们之间是隔离的。
虽然从逻辑上进行了用户空间和内核空间的划分,但不可避免的用户空间需要访问内核资源,比如文件操作、访问网络等等。为了突破隔离限制,就需要借助系统调用来实现。系统调用是用户空间访问内核空间的唯一方式,保证了所有的资源访问都是在内核的控制下进行的,避免了用户程序对系统资源的越权访问,提升了系统安全性和稳定性。
Linux 使用两级保护机制:0 级供系统内核使用,3 级供用户程序使用。
当一个任务(进程)执行系统调用而陷入内核代码中执行时,称进程处于内核运行态(内核态)。此时处理器处于特权级最高的(0级)内核代码中执行。当进程处于内核态时,执行的内核代码会使用当前进程的内核栈。每个进程都有自己的内核栈。
当进程在执行用户自己的代码的时候,我们称其处于用户运行态(用户态)。此时处理器在特权级最低的(3级)用户代码中运行。
系统调用主要通过如下两个函数来实现:
copy_from_user() //将数据从用户空间拷贝到内核空间
copy_to_user() //将数据从内核空间拷贝到用户空间
理解了上面的几个概念,我们再来看看传统的 IPC 方式中,进程之间是如何实现通信的。
通常的做法是消息发送方将要发送的数据存放在内存缓存区中,通过系统调用进入内核态。然后内核程序在内核空间分配内存,开辟一块内核缓存区,调用 copy_from_user() 函数将数据从用户空间的内存缓存区拷贝到内核空间的内核缓存区中。同样的,接收方进程在接收数据时在自己的用户空间开辟一块内存缓存区,然后内核程序调用 copy_to_user() 函数将数据从内核缓存区拷贝到接收进程的内存缓存区。这样数据发送方进程和数据接收方进程就完成了一次数据传输,我们称完成了一次进程间通信。如下图:
这种传统的 IPC 通信方式有两个问题:
理解了 Linux IPC 相关概念和通信原理,接下来我们正式介绍下 Binder IPC 的原理。
正如前面所说,跨进程通信是需要内核空间做支持的。传统的 IPC 机制如管道、Socket 都是内核的一部分,因此通过内核支持来实现进程间通信自然是没问题的。但是 Binder 并不是 Linux 系统内核的一部分,那怎么办呢?这就得益于 Linux 的动态内核可加载模块(Loadable Kernel Module,LKM)的机制;模块是具有独立功能的程序,它可以被单独编译,但是不能独立运行。它在运行时被链接到内核作为内核的一部分运行。这样,Android 系统就可以通过动态添加一个内核模块运行在内核空间,用户进程之间通过这个内核模块作为桥梁来实现通信。
在 Android 系统中,这个运行在内核空间,负责各个用户进程通过 Binder 实现通信的内核模块就叫 Binder 驱动(Binder Dirver)。
那么在 Android 系统中用户进程之间是如何通过这个内核模块(Binder 驱动)来实现通信的呢?难道是和前面说的传统 IPC 机制一样,先将数据从发送方进程拷贝到内核缓存区,然后再将数据从内核缓存区拷贝到接收方进程,通过两次拷贝来实现吗?显然不是,否则也不会有开篇所说的 Binder 在性能方面的优势了。
这就不得不通道 Linux 下的另一个概念:内存映射。
Binder IPC 机制中涉及到的内存映射通过 mmap() 来实现,mmap() 是操作系统中一种内存映射的方法。内存映射简单的讲就是将用户空间的一块内存区域映射到内核空间。映射关系建立后,用户对这块内存区域的修改可以直接反应到内核空间;反之内核空间对这段区域的修改也能直接反应到用户空间。
内存映射能减少数据拷贝次数,实现用户空间和内核空间的高效互动。两个空间各自的修改能直接反映在映射的内存区域,从而被对方空间及时感知。也正因为如此,内存映射能够提供对进程间通信的支持。
Binder IPC 正是基于内存映射(mmap)来实现的,但是 mmap() 通常是用在有物理介质的文件系统上的。
比如进程中的用户区域是不能直接和物理设备打交道的,如果想要把磁盘上的数据读取到进程的用户区域,需要两次拷贝(磁盘–>内核空间–>用户空间);通常在这种场景下 mmap() 就能发挥作用,通过在物理介质和用户空间之间建立映射,减少数据的拷贝次数,用内存读写取代I/O读写,提高文件读取效率。
而 Binder 并不存在物理介质,因此 Binder 驱动使用 mmap() 并不是为了在物理介质和用户空间之间建立映射,而是用来在内核空间创建数据接收的缓存空间。
一次完整的 Binder IPC 通信过程通常是这样:
介绍完 Binder IPC 的底层通信原理,接下来我们看看实现层面是如何设计的。
一次完整的进程间通信必然至少包含两个进程,通常我们称通信的双方分别为客户端进程(Client)和服务端进程(Server),由于进程隔离机制的存在,通信双方必然需要借助 Binder 来实现。
前面我们介绍过,Binder 是基于 C/S 架构的。由一系列的组件组成,包括 Client、Server、ServiceManager、Binder 驱动。其中 Client、Server、Service Manager 运行在用户空间,Binder 驱动运行在内核空间。其中 Service Manager 和 Binder 驱动由系统提供,而 Client、Server 由应用程序来实现。Client、Server 和 ServiceManager 均是通过系统调用 open、mmap 和 ioctl 来访问设备文件 /dev/binder,从而实现与 Binder 驱动的交互来间接的实现跨进程通信。
Client、Server、ServiceManager、Binder 驱动这几个组件在通信过程中扮演的角色就如同互联网中服务器(Server)、客户端(Client)、DNS域名服务器(ServiceManager)以及路由器(Binder 驱动)之前的关系。
通常我们访问一个网页的步骤是这样的:首先在浏览器输入一个地址,如 www.google.com 然后按下回车键。但是并没有办法通过域名地址直接找到我们要访问的服务器,因此需要首先访问 DNS 域名服务器,域名服务器中保存了 www.google.com 对应的 ip 地址 10.249.23.13,然后通过这个 ip 地址才能放到到 www.google.com 对应的服务器。
至此,我们大致能总结出 Binder 通信过程:
我们看到整个通信过程都需要 Binder 驱动的接入。下图能更加直观的展现整个通信过程(为了进一步抽象通信过程以及呈现上的方便,下图我们忽略了 Binder 实体及其引用的概念):
我们已经解释清楚 Client、Server 借助 Binder 驱动完成跨进程通信的实现机制了,但是还有个问题会让我们困惑。A 进程想要 B 进程中某个对象(object)是如何实现的呢?毕竟它们分属不同的进程,A 进程 没法直接使用 B 进程中的 object。
前面我们介绍过跨进程通信的过程都有 Binder 驱动的参与,因此在数据流经 Binder 驱动的时候驱动会对数据做一层转换。当 A 进程想要获取 B 进程中的 object 时,驱动并不会真的把 object 返回给 A,而是返回了一个跟 object 看起来一模一样的代理对象 objectProxy,这个 objectProxy 具有和 object 一摸一样的方法,但是这些方法并没有 B 进程中 object 对象那些方法的能力,这些方法只需要把把请求参数交给驱动即可。对于 A 进程来说和直接调用 object 中的方法是一样的。
当 Binder 驱动接收到 A 进程的消息后,发现这是个 objectProxy 就去查询自己维护的表单,一查发现这是 B 进程 object 的代理对象。于是就会去通知 B 进程调用 object 的方法,并要求 B 进程把返回结果发给自己。当驱动拿到 B 进程的返回结果后就会转发给 A 进程,一次通信就完成了。
现在我们可以对 Binder 做个更加全面的定义了:
通常我们在做开发时,实现进程间通信用的最多的就是 AIDL。当我们定义好 AIDL 文件,在编译时编译器会帮我们生成代码实现 IPC 通信。借助 AIDL 编译以后的代码能帮助我们进一步理解 Binder IPC 的通信原理。
但是无论是从可读性还是可理解性上来看,编译器生成的代码对开发者并不友好。比如一个 BookManager.aidl 文件对应会生成一个 BookManager.java 文件,这个 java 文件包含了一个 BookManager 接口、一个 Stub 静态的抽象类和一个 Proxy 静态类。Proxy 是 Stub 的静态内部类,Stub 又是 BookManager 的静态内部类,这就造成了可读性和可理解性的问题。
Android 之所以这样设计其实是有道理的,因为当有多个 AIDL 文件的时候把 BookManager、Stub、Proxy 放在同一个文件里能有效避免 Stub 和 Proxy 重名的问题。
因此便于大家理解,下面我们来手动编写代码来实现跨进程调用。
在正式编码实现跨进程调用之前,先介绍下实现过程中用到的一些类。了解了这些类的职责,有助于我们更好的理解和实现跨进程通信。
应用层中涉及到跨进程通信的,一般在Service
组件中,这里就简单定义一个用来远程通信的Service
public class RemoteService extends Service {
@Nullable
@Override
public IBinder onBind(Intent intent) {
return new ProcComm.Stub() {
@Override
public void clientComm(CommBean bean) throws RemoteException {
System.out.println("clientComm -> process name : " + getCurProcessName());
System.out.println("bean message -> " + bean.message);
}
@Override
public void serviceComm(CommBean bean) throws RemoteException {
System.out.println("serviceComm -> process name : " + getCurProcessName());
System.out.println("bean message -> " + bean.message);
}
@Override
public void communication(CommBean bean) throws RemoteException {
System.out.println("communication -> process name : " + getCurProcessName());
System.out.println("bean message -> " + bean.message);
}
};
}
......
}
当然,这里的CommBean
是一个实现了Parcelable
的JavaBean
public class CommBean implements Parcelable {
String message;
CommBean() {
}
protected CommBean(Parcel in) {
message = in.readString();
}
public static final Creator<CommBean> CREATOR = new Creator<CommBean>() {
@Override
public CommBean createFromParcel(Parcel in) {
return new CommBean(in);
}
@Override
public CommBean[] newArray(int size) {
return new CommBean[size];
}
};
@Override
public int describeContents() {
return 0;
}
@Override
public void writeToParcel(Parcel dest, int flags) {
dest.writeString(message);
}
public void readFromParcel(Parcel dest) {
message = dest.readString();
}
}
值得注意的是,这里需要定义一个无参构造函数,否则 aidl 自动生成的 java 类会编译报错;另外,这里定义了一个readFromParcel(Parcel)
方法,这样一来,就可以在 aidl 的接口中使用out和inout修饰符了,接下来创建两个 aidl 文件
CommBean.aidl
parcelable CommBean;
ProcComm.aidl
interface ProcComm {
void clientComm(in CommBean bean);
void serviceComm(out CommBean bean);
void communication(inout CommBean bean);
}
最后,只需要绑定Service
就可以实现跨进程通信了
private ServiceConnection sc = new ServiceConnection() {
@Override
public void onServiceConnected(ComponentName name, IBinder service) {
System.out.println("onServiceConnected -> process name : " + getCurProcessName());
ProcComm pc = ProcComm.Stub.asInterface(service);
CommBean cb = new CommBean();
cb.message = "from client message";
try {
pc.clientComm(cb);
} catch (RemoteException e) {
System.out.println(e.getMessage());
}
}
@Override
public void onServiceDisconnected(ComponentName name) {
System.out.println("onServiceDisconnected -> process name : " + getCurProcessName());
}
};
最后,要在程序清单中注册Service
组件
<service
android:name=".RemoteService"
android:process=":pcProcessName" />
注意这里要定义android:process
,否则就不是跨进程而是同进程通信了,最终输出打印
System.out: onCreate -> process name : com.andova.system
System.out: onServiceConnected -> process name : com.andova.system
System.out: clientComm -> process name : com.andova.system:pcProcessName
System.out: bean message -> from client message
使用 aidl 最终会自动生成一个 java 类,这里就来看看它长成什么模样
public interface ProcComm extends android.os.IInterface {
public static abstract class Stub extends android.os.Binder implements ProcComm {
private static final java.lang.String DESCRIPTOR = "com.andova.system.ProcComm";
public Stub() {
this.attachInterface(this, DESCRIPTOR);
}
public static ProcComm asInterface(android.os.IBinder obj) {
if ((obj == null)) {
return null;
}
android.os.IInterface iin = obj.queryLocalInterface(DESCRIPTOR);
if (((iin != null) && (iin instanceof ProcComm))) {
return ((ProcComm) iin);
}
return new ProcComm.Stub.Proxy(obj);
}
@Override
public android.os.IBinder asBinder() {
return this;
}
@Override
public boolean onTransact(int code, android.os.Parcel data, android.os.Parcel reply, int flags) throws android.os.RemoteException {
java.lang.String descriptor = DESCRIPTOR;
switch (code) {
case INTERFACE_TRANSACTION: {
reply.writeString(descriptor);
return true;
}
case TRANSACTION_clientComm: {
data.enforceInterface(descriptor);
CommBean _arg0;
if ((0 != data.readInt())) {
_arg0 = CommBean.CREATOR.createFromParcel(data);
} else {
_arg0 = null;
}
this.clientComm(_arg0);
reply.writeNoException();
return true;
}
case TRANSACTION_serviceComm: {
data.enforceInterface(descriptor);
CommBean _arg0;
_arg0 = CommBean();
this.serviceComm(_arg0);
reply.writeNoException();
if ((_arg0 != null)) {
reply.writeInt(1);
_arg0.writeToParcel(reply, android.os.Parcelable.PARCELABLE_WRITE_RETURN_VALUE);
} else {
reply.writeInt(0);
}
return true;
}
case TRANSACTION_communication: {
data.enforceInterface(descriptor);
CommBean _arg0;
if ((0 != data.readInt())) {
_arg0 = CommBean.CREATOR.createFromParcel(data);
} else {
_arg0 = null;
}
this.communication(_arg0);
reply.writeNoException();
if ((_arg0 != null)) {
reply.writeInt(1);
_arg0.writeToParcel(reply, android.os.Parcelable.PARCELABLE_WRITE_RETURN_VALUE);
} else {
reply.writeInt(0);
}
return true;
}
default: {
return super.onTransact(code, data, reply, flags);
}
}
}
private static class Proxy implements ProcComm {
private android.os.IBinder mRemote;
Proxy(android.os.IBinder remote) {
mRemote = remote;
}
@Override
public android.os.IBinder asBinder() {
return mRemote;
}
public java.lang.String getInterfaceDescriptor() {
return DESCRIPTOR;
}
@Override
public void clientComm(CommBean bean) throws android.os.RemoteException {
android.os.Parcel _data = android.os.Parcel.obtain();
android.os.Parcel _reply = android.os.Parcel.obtain();
try {
_data.writeInterfaceToken(DESCRIPTOR);
if ((bean != null)) {
_data.writeInt(1);
bean.writeToParcel(_data, 0);
} else {
_data.writeInt(0);
}
mRemote.transact(Stub.TRANSACTION_clientComm, _data, _reply, 0);
_reply.readException();
} finally {
_reply.recycle();
_data.recycle();
}
}
@Override
public void serviceComm(CommBean bean) throws android.os.RemoteException {
android.os.Parcel _data = android.os.Parcel.obtain();
android.os.Parcel _reply = android.os.Parcel.obtain();
try {
_data.writeInterfaceToken(DESCRIPTOR);
mRemote.transact(Stub.TRANSACTION_serviceComm, _data, _reply, 0);
_reply.readException();
if ((0 != _reply.readInt())) {
bean.readFromParcel(_reply);
}
} finally {
_reply.recycle();
_data.recycle();
}
}
@Override
public void communication(CommBean bean) throws android.os.RemoteException {
android.os.Parcel _data = android.os.Parcel.obtain();
android.os.Parcel _reply = android.os.Parcel.obtain();
try {
_data.writeInterfaceToken(DESCRIPTOR);
if ((bean != null)) {
_data.writeInt(1);
bean.writeToParcel(_data, 0);
} else {
_data.writeInt(0);
}
mRemote.transact(Stub.TRANSACTION_communication, _data, _reply, 0);
_reply.readException();
if ((0 != _reply.readInt())) {
bean.readFromParcel(_reply);
}
} finally {
_reply.recycle();
_data.recycle();
}
}
}
static final int TRANSACTION_clientComm = (android.os.IBinder.FIRST_CALL_TRANSACTION + 0);
static final int TRANSACTION_serviceComm = (android.os.IBinder.FIRST_CALL_TRANSACTION + 1);
static final int TRANSACTION_communication = (android.os.IBinder.FIRST_CALL_TRANSACTION + 2);
}
public void clientComm(CommBean bean) throws android.os.RemoteException;
public void serviceComm(CommBean bean) throws android.os.RemoteException;
public void communication(CommBean bean) throws android.os.RemoteException;
}
这样一看,其实还是很模式化的
从上面可以很容易发现,生成的 java 文件无非就是定义接口和实现接口
public interface ProcCommInterface extends IInterface {
void clientComm(CommBean bean) throws RemoteException;
void serviceComm(CommBean bean) throws RemoteException;
void communication(CommBean bean) throws RemoteException;
}
这一步需要继承android.os.IInterface
public abstract class Stub extends Binder implements ProcCommInterface {
static final String DESCRIPTOR = "com.andova.system.aidl.ProcCommInterface";
static final int TRANSACTION_clientComm = FIRST_CALL_TRANSACTION;
static final int TRANSACTION_serviceComm = FIRST_CALL_TRANSACTION + 1;
static final int TRANSACTION_communication = FIRST_CALL_TRANSACTION + 2;
public Stub() {
attachInterface(this, DESCRIPTOR);
}
public static ProcCommInterface asInterface(IBinder obj) {
if ((obj == null)) return null;
IInterface inter = obj.queryLocalInterface(DESCRIPTOR);
if (inter instanceof ProcCommInterface) {
return ((ProcCommInterface) inter);
}
return new Proxy(obj);
}
@Override
public IBinder asBinder() {
return this;
}
@Override
protected boolean onTransact(int code, @NonNull Parcel data, @Nullable Parcel reply, int flags) throws RemoteException {
switch (code) {
case INTERFACE_TRANSACTION: {
if (reply != null) reply.writeString(DESCRIPTOR);
return true;
}
case TRANSACTION_clientComm: {
data.enforceInterface(DESCRIPTOR);
CommBean bean = null;
if (data.readInt() != 0) bean = CommBean.CREATOR.createFromParcel(data);
clientComm(bean);
if (reply != null) reply.writeNoException();
return true;
}
case TRANSACTION_serviceComm: {
data.enforceInterface(DESCRIPTOR);
CommBean bean = new CommBean();
serviceComm(bean);
if (reply != null) reply.writeNoException();
if (reply != null) reply.writeInt(1);
bean.writeToParcel(reply, Parcelable.PARCELABLE_WRITE_RETURN_VALUE);
return true;
}
case TRANSACTION_communication: {
data.enforceInterface(DESCRIPTOR);
CommBean bean = null;
if (data.readInt() != 0) bean = CommBean.CREATOR.createFromParcel(data);
communication(bean);
if (reply != null) reply.writeNoException();
if (bean != null) {
if (reply != null) reply.writeInt(1);
bean.writeToParcel(reply, Parcelable.PARCELABLE_WRITE_RETURN_VALUE);
} else {
if (reply != null) reply.writeInt(0);
}
return true;
}
default:
return super.onTransact(code, data, reply, flags);
}
}
}
public class Proxy implements ProcCommInterface {
private IBinder mRemote;
public Proxy(IBinder remote) {
mRemote = remote;
}
@Override
public IBinder asBinder() {
return mRemote;
}
@Override
public void clientComm(CommBean bean) throws RemoteException {
Parcel data = Parcel.obtain();
Parcel reply = Parcel.obtain();
try {
data.writeInterfaceToken(DESCRIPTOR);
if (bean != null) {
data.writeInt(1);
bean.writeToParcel(data, 0);
} else {
data.writeInt(0);
}
mRemote.transact(TRANSACTION_clientComm, data, reply, 0);
reply.readException();
} finally {
data.recycle();
reply.recycle();
}
}
@Override
public void serviceComm(CommBean bean) throws RemoteException {
Parcel data = Parcel.obtain();
Parcel reply = Parcel.obtain();
try {
data.writeInterfaceToken(DESCRIPTOR);
mRemote.transact(Stub.TRANSACTION_serviceComm, data, reply, 0);
reply.readException();
if (reply.readInt() != 0) bean.readFromParcel(reply);
} finally {
reply.recycle();
data.recycle();
}
}
@Override
public void communication(CommBean bean) throws RemoteException {
Parcel data = Parcel.obtain();
Parcel reply = Parcel.obtain();
try {
data.writeInterfaceToken(DESCRIPTOR);
if (bean != null) {
data.writeInt(1);
bean.writeToParcel(data, 0);
} else {
data.writeInt(0);
}
mRemote.transact(Stub.TRANSACTION_communication, data, reply, 0);
reply.readException();
if (reply.readInt() != 0 && bean != null) bean.readFromParcel(reply);
} finally {
reply.recycle();
data.recycle();
}
}
}