--- Service 组件的启动过程 --- #### Service 组件在新进程中的启动过程 在我们调用 Context 的 startService 时,其实是调用到 ContextImpl 的 startService: ```java class ContextImpl extends Context { @Override public ComponentName startService(Intent service) { ComponentName cn = ActivityManagerNative.getDefault().startService( mMainThread.getApplicationThread(), service, service.resolveTypeIfNeeded(getContextResolver())); return cn; } } ``` 获取 AMS 的代理对象去处理 startService 的请求: ```java // ActivityManagerProxy public ComponentName startService(IApplicationThread caller, Intent service, String resolvedType) { Parcel data = Parcel.obtain(); data.writeInterfaceToken(IActivityManager.descriptor); data.writeStrongBinder(caller); mRemote.transact(START_SERVICE_TRANSACTION, data, ...); } ``` 接着再通过 ActivityManagerProxy 类内部的一个 Binder 代理对象 mRemote 向 AMS 发送一个类型为 START_SERVICE_TRANSACTION 的进程间通信请求。 接下来就是在 AMS 中去处理 Client 组件发送来的类型为 START_SERVICE_TRANSACTION 的进程间通信请求。 ```java // AMS public ComponentName startService(IApplicationThread caller, Intent service, String resolvedType) { ComponentName res = startServiceLocked(caller, service, ...); return res; } ComponentName startServiceLocked(IApplicationThread caller, Intent service, ...) { ServiceLookupResult res = retrieveServiceLocked(service, ...); ServiceRecord r = res.record; bringUpServiceLocked(r, service.getFlags()); return r.name; } ``` 在 AMS 中,每一个 Service 组件都使用一个 ServiceRecord 对象来描述,就像每一个 Activity 组件都使用一个 ActivityRecord 对象来描述一样。 首先调用 retrieveServiceLocked 在 AMS 中查找是否存在与参数 service 对应的一个 ServiceRecord 对象。如果不存在,AMS 就会到 PKMS 中去获取与参数 service 对应的一个 Service 组件的信息,然后再将这些信息封装成一个 ServiceRecord 对象,最后调用 bringUpServiceLocked 来启东 ServiceRecord 对象 r 所描述的一个 Service 组件。 ```java // AMS private final boolean bringUpServiceLocked(ServiceRecord r, ...) { final String appName = r.processName; ProcessRecord app = getProcessRecordLocked(appName, r.appInfo.uid); if (app != null && app.thread != null) { realStartServiceLocked(r, app); return true; } if (startProcessLocked(appName, r.appInfom, "service", ...)) { return false; } if (!mPendingServices.contains(r)) { mPendingServices.add(r); } return true; } ``` 首先判断该 Service 组件所在的进程是否已经创建了,如果没有创建,就会去执行 startProcessLocked 去创建一个应用程序进程,并且把该 Service 添加到 mPendingService 列表中。 startProcessLocked 就是调用 Process 类的 start 来创建一个新的应用程序进程,创建完成之后,新的应用程序进程会把 ApplicationThread 传递给 AMS,以便 AMS 可以和这个新创建的应用程序进程通信。新创建的应用程序进程会向 AMS 发送一个 ATTACH_APPLICATION_TRANSACTION 的进程间通信请求。 ```java // AMS private final boolean attachApplicationLocked(IApplicationThread thread, int pid) { ProcessRecord app = mPidsSlefLocked.get(pid); app.thread = thread; if (mPendingServices.size() > 0) { ServiceRecord sr = null; for (int i=0;i map = mService.get(context); if (map != null) { sd = map.get(c); } if (sd == null) { sd = new ServiceDispatcher(c, context, handler, flags); if (map == null) { map = new HashMap<>(); mServices.put(context, map); } map.put(c, sd); } return sd.getIServiceConnection(); } ``` 每一个绑定过 Service 组件的 Activity 组件在 LoadedApk 类中都有一个对应的 ServiceDispatcher 对象,它负责将这个被绑定的 Service 与绑定它的 Activity 组件关联在一起。这些 ServiceDispatcher 对象保存在一个 HashMap 中,并且以它们所关联的 ServiceConnection 对象为关键字。最后,用来保存这些 ServiceDispatcher 对象的 HashMap 又以它们所关联的 Activity 组件的 Context 接口为关键字保存在 LoadedApk 类的成员变量 mServices 中。 首先在 mServices 中查找是否存在一个以 ServiceConnection 为 key 的 ServiceDispatcher 对象 sd。如果不存在就先创建 sd 然后存入 map 中。最后通过 ServiceDispatcher 的 getIServiceConnection 来获得一个实现了 IServiceConnection 接口的 Binder 本地对象,它的实现如下: ```java // LoadedApk static final class ServiceDispatcher { private final ServiceDispatcher.InnerConnection mIServiceConnection; private final ServiceConnection mConnection; private final Handler mActivityThread; private final Context mContext; private static class InnerConnection extends IServiceConnection.Stub { fianl WeakReference mDispatcher; InnerConnection(LoadedApk.ServiceDispatcher sd) { mDispatcher = new WeakReference<>(sd); } } ServiceDispatcher(ServiceConnection conn, Context context, Handler activityThread, int flags) { mIServiceConnection = new InnerConnection(this); mConnection = conn; mContext = context; mActivityThread = activityThread; } IServiceConnection getIServiceConnection() { return mIServiceConnection; } } ``` ServiceDispatcher 类有一个类型为 InnerConnection 的成员变量 mIServiceConnection,它指向一个实现了 IServiceConnection 接口的 Binder 本地对象。 ServiceDispatcher 类还有另外三个成员变量 mConnection、mActivityThread 和 mContext,其中,成员变量 mContext 指向了一个 Activity 组件;成员变量 mActivityThread 和 mConnection 分别指向了与该 Activity 组件相关联的一个 Handler 对象和一个 ServiceConnection 对象。 回到 ContextImpl 类的成员函数 bindService 中,它将 ServiceConnection 对象 conn 封装成一个 InnerConnection 对象之后,最后就请求 AMS 将 Service 组件绑定到 Activity 组件中。 ```java // AMP public int bindService(IApplicationThread caller, IBinder token, Intent service, ...) { Parcel data = Parcel.obtain(); Parcel reply = Parcel.obtain(); data.writeInterfaceToken(IActivityManager.descriptor); data.writeStrongBinder(caller); service.writeToParcel(data, 0); mRemote.transact(BIND_SERVICE_TRANSACTION, data, reply, 0); int res = reply.readInt(); return res; } ``` 通过 AMP 内部的一个 Binder 代理对象 mRemote 向 AMS 发送一个类型为 BIND_SERVICE_TRANSACTION 的进程间通信请求。 以上都是在 Activity 组件中执行的,接下来就要到 AMS 中去处理该进程间通信请求了: ```java // AMS public int bindService(IApplicationThread caller, IBinder token, Intent service, ...) { final ProcessRecord callerApp = getRecordForAppLocked(caller); ActivityRecord activity = null; int aindex = mMainStack.indexOfTokenLocked(token); activity = (ActivityRecord)mMainStack.mHistory.get(aindex); ServiceLoopupResult = res = retrieveServiceLocked(service, resolvedType, Binder.getCallingPid(), Binder.getCallingUid()); ServiceRecord s = res.record; AppBindRecord b = s.retrieveAppBindingLocked(service, callerApp); ConnectionRecord c = new ConnectionRecord(b, activity, connection, ...); IBinder binder = connection.asBinder(); ArrayList clist = s.connection.get(binder); if(clist == null) { clist = new ArrayList<>(); s.connections.put(binder, clist); } clist.add(c); if((flags&Context.BIND_AUTO_CREATE) != 0) { if(!bringUpServiceLocked(s, service.getFlags(), false)) { return 0; } } return 1; } ``` ServiceRecord 类的成员函数 retrieveAppBindingLocked 的实现如下所示: ```java class ServiceRecord extends Binder { final HashMap bindings = new HashMap<>(); public AppBindRecord retrieveAppBindingLocked(Intent intent, ProcessRecord app) { Intent.FilterComparison filter = new Intent.FilterComparison(intent); IntentBindRecord i = bindings.get(filter); if(i==null) { i = new IntentBindRecord(this, filter); bindings.put(filter, i); } AppBindRecord a = i.apps.get(app); if(a!=null) { return a; } a = new AppBindRecord(this, i, app); i.apps.put(app, a); return a; } } ``` 当一个应用程序进程绑定了一个 Service 组件之后,用来描述这个应用程序进程的一个 ProcessRecord 对象就会被保存在用来描述这个被绑定的 Service 组件的一个 ServiceRecord 对象的成员变量 bingdigs 中。由于一个 Service 组件可能会被多个应用程序进程绑定,因此,用来描述这个 Service 组件的一个 ServiceRecord 对象就会使用一个 IntentBindRecord 对象来描述这些应用程序进程,并且以一个 FilterComparison 对象为关键字保存在该 ServiceRecord 对象的成员变量 bingdings 中。 ```java // AMS private final boolean bringUpServiceLocked(ServiceRecord r, int intentFlags, boolean whileRestarting) { final String appName = r.processName; ProcessRecord app = getProcessRecordLocked(appName, r.appInfo.uid); if (app != null && pp.thread != null) { realStartServiceLocked(r, app); return true; } } ``` 首先也是会先判断该 Service 对应的应用程序进程是否已经存在,如果存在,就调用 realStartServiceLocked 来启动 Service。 ```java // AMS private final void realStartServiceLocked(ServiceRecord r, ProcessRecord app) { r.app = app; app.services.add(r); app.thread.scheduleCreateService(r, r.serviceInfo); requestServiceBindingsLocked(r); } ``` 其实也是通过类型为 ApplicationThreadProxy 的 Binder 代理对象,它指向了 ProcessRecord 对象 app 所描述的应用程序进程中的一个 ApplicationThread 对象。让它来代理启动该 Service 组件。和之前一样,就是发了一个 SCHEDULE_CREATE_SERVICE_TRANSACTION 的进程间通信请求。 ServiceRecord 对象 r 所描述的 Service 组件启动完成之后,AMS 就需要将它连接到请求绑定它的一个 Activity 组件中,这是通过 AMS 的 requestServiceBindingsLocked 来实现的。 ```java // AMS private final void requestServiceBindingsLocked(ServiceRecord r) { Iterator bindings = r.bindings.values().iterator(); while (bindings.hasNext()) { IntentBindRecord i = bindings.next(); if (!requestServiceBindingLocked(r, i, false)) { break; } } } ``` 在 ServiceRecord 对象 r 的成员变量 bindings 中,保存了一系列 IntentBindRecord 对象,每一个 IntentBindRecord 对象都用来描述若干个需要将 ServiceRecord 对象 r 所描述的 Service 组件绑定到它们里面去的应用程序进程。 ```java // AMS private final boolean requestServiceBindingLocked(ServiceRecord r, IntentBindRecord i, boolean rebind) { if ((!i.requested || rebind) && i.apps.size() > 0) { r.app.thread.scheduleBindService(r, i.intent.getIntent(), rebind); if (!rebind) { i.requested = true; } return true; } } ``` 参数 rebind 用来描述是否要将 ServiceRecord 对象 r 所描述的 Service 组件重新绑定到 IntentBindRecord 对象 i 所描述的应用程序进程中。从前面的调用过程可以知道,参数 rebind 的值为 false,这意味着 IntentBindRecord 对象 i 所描述的应用程序进程是第一次请求绑定 ServiceRecord 对象 r 所描述的 Service 组件的。 接下来就是 AMS 会请求应用程序 Activity 组件返回 Service 组件内部的一个 Binder 本地对象。 ```java // ApplicationThreadProxy public final void scheduleBindService(IBinder token, Intent intent, boolean rebind) { Parcel data = Parcel.obtain(); data.writeInterfaceToken(IApplicationThread.descriptor); data.writeStrongBinder(token); intent.writeToParcel(data, 0); mRemote.transact(SCHEDULE_BIND_SERVICE_TRANSACTION, data, null, IBinder.FLAG_ONEWAY); data.recycle(); } ``` 通过 ApplicationThreadProxy 类内部的一个 Binder 代理对象 mRemote 向应用程序发送一个类型为 SCHEDULE_BIND_SERVICE_TRANSACION 的进程间通信请求。 接下来就会在应用程序进程处理一个 H.BIND_SERVICE 的消息: ```java // ActivityThread private final void handleBindService(BindServiceData data) { Service s = mService.get(data.token); IBinder binder = s.onBind(data.intent); ActivityManagerNative.getDefault().publishService(data.token, data.intent, binder); } ``` BindServiceData 对象 data 的成员变量 token 指向了一个 Binder 代理对象,它引用了 AMS 中的一个 ServiceRecord 对象,而这个 ServiceRecord 对象是用来描述应用程序的 Service 组件的。 首先通过 Service 的 onBind 函数获取一个 Binder 本地对象,然后通过 publishService 将它传给 AMS。 ```java // ActivityManagerNative public void publishService(IBinder token, Intent intent, IBinder service) { Parcel data = Parcel.obtain(); data.writeInterfaceToken(IActivityManager.descriptor); data.writeStrongBinder(token); mRemote.transact(PUBLISH_SERVICE_TRANSACTION, data, reply, 0); } ``` 在 AMS 中处理该请求: ```java // AMS public void publishService(IBinder token, Intent intent, IBinder service) { ServiceRecord r = (ServiceRecord)token; Intent.FilterComparison filter = new Intent.FilterComparison(intent); IntentBindRecord b = r.bindings.get(filter); IntentBindRecord b = r.bindings.get(filter); b.binder = service; b.requested = true; b.received = true; Iterator> it = r.connections.values().iterator(); while (it.hasNext()) { ArrayList clist = it.next(); for (int i=0; i mDispatcher; public void connected(ComponentName name, IBinder service) { LoadedApk.ServiceDispatcher sd = mDispatcher.get(); sd.connected(name, service); } } } ``` 有调到了 ServiceDispatcher 的 connected 方法: ```java // LoadedApk static final class ServiceDispatcher { private final Handler mActivityThread; public void connected(ComponentName name, IBinder service) { mActivityThread.post(new RunConnection(name, service, 0)); } static final class RunConnection implements Runnable { RunConnection(ComponentName name, IBinder service, int command) { mName = name; mService = service; mCommand = command; } public void run() { doConnected(mName, mService); } final ComponentName mName; final IBinder mService; final int mCommand; } } ``` 调用 ServiceDispatcher 类的成员函数 doConnected 将 RunConnection 类内部成员函数 mService 所描述的一个 Binder 本地对象传递给 Activity 组件。 ```java // ServiceDispatcher public void doConnected(ComponentName name, IBinder service) { mConnection.onServiceConnected(name, service); } ``` 至此,Activity 组件绑定 Service 组件的过程就分析完成了,Activity 组件获得了 Service 组件的访问接口之后,就可以调用相关服务了。