What is a Binder?
- A Binder is a process communication mechanism
- Binder is a virtual physical device driver
- Binder is a class that initiates communication
Why do you need multiple processes
- Each process is independent of each other, and the crash of the child process does not affect the stability of the main process (wechat applets, plugins…).
- Exceeding the memory limit The VM memory size is limited for each process, so that some components run in an independent process, reducing OOM occurrence probability and killing probability (loading image…).
- Business needs some communication or service process needs to be kept alive, plug-in access plug-in development needs (message push process…)
Why Binder is used
Linux existing IPC mechanism
Before we look at Binder mechanisms, let’s briefly talk about all the existing interprocess IPC mechanisms in Linux:
- Pipes: A page size memory is allocated at creation time, and the cache size is limited
- Message queuing: Information is copied twice, extra CPU consumption; Not suitable for frequent or informative communication
- Shared memory: no replication, shared buffer directly attached to the process virtual address space, fast; However, the synchronization problem between processes cannot be realized by the operating system and must be solved by each process using the synchronization tool
- Socket: As a more general interface, the transmission efficiency is low, mainly used for communication between different machines or across networks
- Semaphore: Often used as a locking mechanism to prevent other processes from accessing a shared resource while one process is accessing it. Therefore, it is mainly used as a means of synchronization between processes and between different threads within the same process
- Signal: not suitable for information exchange, more suitable for process interrupt control, such as illegal memory access, killing a process, etc
contrast
- From the perspective of performance, data copy times: Binder data copy only needs one time, while pipes, message queues and sockets need two times, but shared memory does not need one memory copy. Binders are second only to shared memory in terms of performance.
- From the perspective of stability, Binder is based on C/S architecture. If the Client side has any requirements, it is directly sent to the Server side for completion. The architecture is clear and clear, the Server side is relatively independent from the Client side, and the stability is good. However, the implementation of shared memory is complex, and there is no distinction between client and server. Therefore, the concurrent synchronization of access to critical resources should be fully considered, otherwise deadlocks and other problems may occur. The Binder architecture is superior to shared memory from a stability perspective.
- The receiver of traditional Linux IPC cannot obtain the reliable UID/PID of the process of the other party, so it cannot identify the identity of the other party. The security protection measures are completely guaranteed by the upper layer protocol. Android assigns its own UID to each installed application, so the PROCESS UID is an important symbol to identify the process identity. As mentioned above in the C/S architecture, only the Client is exposed in the Android system. The Client sends tasks to the Server, and the Server will follow the permission control policy. Check whether the UID/PID meets the access permission. Binder architectures are superior to traditional IPC from a security perspective.
- Linux is based on C (procedural) and Android is based on Java (object-oriented), and Binder is exactly object-oriented, translating interprocess communication into methods that call a Binder object through a reference to that object. Binder objects are unique in that they can be referenced across processes. Their entities reside in one process, but their references are distributed throughout the system. You can pass from one process to another so that everyone can access the same Server, just like assigning an object or reference to another reference. Binder blurs process boundaries and plays down interprocess communication, as if the whole system were running in the same object-oriented program. From a language perspective, Binder is better suited to Android, which is based on object-oriented languages, than Linux, which may be a bit awkward.
Binder | The Shared memory | Socket | |
---|---|---|---|
performance | Copy a | Don’t need to copy | Copy twice |
The characteristics of | Based on C/S architecture, easy to use | Complex control and poor ease of use | Based on C/S architecture, universal interface, low transmission efficiency, high overhead |
security | Each App is assigned a UID and supports both real and anonymous names | Depending on the upper-layer protocol, the access point is open | Depending on the upper-layer protocol, the access point is open |
Principle of Binder
Binder communication uses the C/S architecture and consists of Client, Server, ServiceManager, and Binder drivers from a component perspective. The ServiceManager manages various services in the system. As shown in the figure below.
Both the process of registering the Service and obtaining the Service require a ServiceManager. Note that the ServiceManager here refers to the Native layer ServiceManager (C++). Does not refer to the Framework layer ServiceManager(Java).
The ServiceManager is the master of the Binder communication mechanism and the daemon of the Android interprocess communication mechanism Binder. After the Service Manager is started, the Client and Server must obtain the Service Manager interface before communicating with each other.
Figure in the mutual communication between a Client/Server/ServiceManage are based on the mechanism of Binder. Since communication is based on Binder mechanism and also C/S architecture, the three steps in the figure have corresponding Client side and Server side.
- Register Service (addService) : The Server process registers Service with the ServiceManager first. This process: The Server is the client, and the ServiceManager is the Server.
- GetService: The Client process obtains a Service from the ServiceManager before using a Service. The Client is the Client, and the ServiceManager is the server.
- Using a Service: The Client establishes a path to communicate with the Server process of the Service based on the obtained Service information, and then directly interacts with the Service. The process: Client is the client and server is the server.
The interactions among Client, Server and Service Manager in the figure are all dotted lines, because they do not directly interact with each other, but with Binder drivers to realize IPC communication. Binder drivers are located in kernel space, and Client, Server, and Service Manager are located in user space.
Binder drivers and Service Manager can be regarded as the infrastructure of Android platform, while Client and Server are the application layer of Android. Developers only need to customize the implementation of Client and Server. The basic Android platform architecture enables direct IPC communication.
Application of Binder: AIDL
AIDL = Android Interface Definition Language, the main purpose is to simplify the process of calling Binder.
To use AIDL in Android, you need to first write an xxx. AIDL file, and Android Studio automatically generates an xxx. Java file from that
- Manually written: xxx.aidl
interface IDemoAidlInterface {
void functionA(int arg);
void functionB(String arg);
}
Copy the code
- Automatically generated: xxx.java
* This file is auto-generated. DO NOT MODIFY.
*/
package com.example.serverdemo;
public interface IDemoAidlInterface extends android.os.IInterface {
public void functionA(int arg) throws android.os.RemoteException;
public void functionB(java.lang.String arg) throws android.os.RemoteException;
/** * Default implementation for IDemoAidlInterface. */
public static class Default implements com.example.serverdemo.IDemoAidlInterface {
@Override
public void functionA(int arg) throws android.os.RemoteException {}@Override
public void functionB(java.lang.String arg) throws android.os.RemoteException {}@Override
public android.os.IBinder asBinder(a) {
return null; }}/** * Local-side IPC implementation stub class. */
public static abstract class Stub extends android.os.Binder implements com.example.serverdemo.IDemoAidlInterface {
static final int TRANSACTION_functionA = (android.os.IBinder.FIRST_CALL_TRANSACTION + 0);
static final int TRANSACTION_functionB = (android.os.IBinder.FIRST_CALL_TRANSACTION + 1);
private static final java.lang.String DESCRIPTOR = "com.example.serverdemo.IDemoAidlInterface";
/** * Construct the stub at attach it to the interface. */
public Stub(a) {
this.attachInterface(this, DESCRIPTOR);
}
/** * Cast an IBinder object into an com.example.serverdemo.IDemoAidlInterface interface, * generating a proxy if needed. */
public static com.example.serverdemo.IDemoAidlInterface asInterface(android.os.IBinder obj) {
if ((obj == null)) {
return null;
}
android.os.IInterface iin = obj.queryLocalInterface(DESCRIPTOR);
if(((iin ! =null) && (iin instanceof com.example.serverdemo.IDemoAidlInterface))) {
return ((com.example.serverdemo.IDemoAidlInterface) iin);
}
return new com.example.serverdemo.IDemoAidlInterface.Stub.Proxy(obj);
}
public static boolean setDefaultImpl(com.example.serverdemo.IDemoAidlInterface impl) {
// Only one user of this interface can use this function
// at a time. This is a heuristic to detect if two different
// users in the same process use this function.
if(Stub.Proxy.sDefaultImpl ! =null) {
throw new IllegalStateException("setDefaultImpl() called twice");
}
if(impl ! =null) {
Stub.Proxy.sDefaultImpl = impl;
return true;
}
return false;
}
public static com.example.serverdemo.IDemoAidlInterface getDefaultImpl(a) {
return Stub.Proxy.sDefaultImpl;
}
@Override
public android.os.IBinder asBinder(a) {
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_functionA: {
data.enforceInterface(descriptor);
int _arg0;
_arg0 = data.readInt();
this.functionA(_arg0);
reply.writeNoException();
return true;
}
case TRANSACTION_functionB: {
data.enforceInterface(descriptor);
java.lang.String _arg0;
_arg0 = data.readString();
this.functionB(_arg0);
reply.writeNoException();
return true;
}
default: {
return super.onTransact(code, data, reply, flags); }}}private static class Proxy implements com.example.serverdemo.IDemoAidlInterface {
public static com.example.serverdemo.IDemoAidlInterface sDefaultImpl;
private android.os.IBinder mRemote;
Proxy(android.os.IBinder remote) {
mRemote = remote;
}
@Override
public android.os.IBinder asBinder(a) {
return mRemote;
}
public java.lang.String getInterfaceDescriptor(a) {
return DESCRIPTOR;
}
@Override
public void functionA(int arg) throws android.os.RemoteException {
android.os.Parcel _data = android.os.Parcel.obtain();
android.os.Parcel _reply = android.os.Parcel.obtain();
try {
_data.writeInterfaceToken(DESCRIPTOR);
_data.writeInt(arg);
boolean _status = mRemote.transact(Stub.TRANSACTION_functionA, _data, _reply, 0);
if(! _status && getDefaultImpl() ! =null) {
getDefaultImpl().functionA(arg);
return;
}
_reply.readException();
} finally{ _reply.recycle(); _data.recycle(); }}@Override
public void functionB(java.lang.String arg) throws android.os.RemoteException {
android.os.Parcel _data = android.os.Parcel.obtain();
android.os.Parcel _reply = android.os.Parcel.obtain();
try {
_data.writeInterfaceToken(DESCRIPTOR);
_data.writeString(arg);
boolean _status = mRemote.transact(Stub.TRANSACTION_functionB, _data, _reply, 0);
if(! _status && getDefaultImpl() ! =null) {
getDefaultImpl().functionB(arg);
return;
}
_reply.readException();
} finally {
_reply.recycle();
_data.recycle();
}
}
}
}
}
Copy the code
The key to process communication with Binder is automatically generated xxx.java files, rather than xxx.aidl files, which make it easier for developers to write and generate the former.
After the file is written and generated, there are three key actions to make remote calls using AIDL in Android:
- The client is bound to the server
- The client obtains the IBinder object from the server
- The client remotely invokes the server interface
The client is bound to the server
The client obtains the IBinder object from the server
com.example.serverdemo.IDemoAidlInterface.Stub#asInterface</pre>
Copy the code
If the client and server are in the same process:
return((com.example.serverdemo.IDemoAidlInterface) iin); </pre>Copy the code
If the client and server are in different processes:
return newcom.example.serverdemo.IDemoAidlInterface.Stub.Proxy(obj); </pre>Copy the code
The client remotely invokes the server interface
References and Recommendations
Gityuan.com/2015/10/31….
The last
Because the level is limited, have wrong place unavoidable, rather misdirect others, welcome big guy to point out! Code word is not easy, thank you for your attention! 🙏 If you are studying with me, you can follow my official account — ❤️ Program ape Development Center ❤️. Every week, we will share the technology of Android regularly.