本章主要记录一下自己在工作中,遇到车机多屏幕操作时CarUserService中的逻辑实现流程,所以启动方面就简单分析一下:
- 和汽车相关的服务主要依赖系统服务 CarServiceHelperService 开机时在 SystemServer 中启动,如图:
判断是车,然后启动服务 - 随后通过类名加载类,并且通过反射构造服务类,并且调用
startService
启动如图:
通过包路径com.android.internal.car.CarServiceHelperService加载类
通过包路径com.android.internal.car.CarServiceHelperService加载类
通过反射调用构造函数 - 在
startService
方法中,将构造完成的对象添加到mServices
中,并且调用onStart
,如图:
startService(@NonNull final SystemService service) - 在 CarServiceHelperService 的
onStart
中会绑定一个包名为com.android.car
,action为android.car.ICar
的服务,这就是系统中和汽车相关的核心服务 CarService ,如图:
常量CAR_SERVICE_INTERFACE
绑定CarService - 绑定 CarService 的时候,在 CarService 的
onCreate
方法中,构造了一个具体实现类mICarImpl
,并且在onBind
方法中返回mICarImpl
,如图:
构造一个实现类
onBind返回实现类 - 在 ICarImpl 的构造方法中,新建了 CarUserService 和CarOccupantZoneService,CarUserService 是用来管理车载环境中多用户操作。而
CarOccupantZoneService 用来管理多用户和座位的对应关系,这里我们只分析和显示屏有关的,在车载环境中,可能有一个座位对应了一个显示屏提供给用户操作,初始化的代码如图:
初始化CarUserService和CarOccupantZoneService - 在 ICarImpl 构造方法的最后,将所有构造完成的服务对象添加到成员变量
mAllServices
中。
添加到mAllServices - 在 ICarImpl 中,
constructWithTrace
方法初始化了各个服务,它的逻辑是在调用对应类的构造方法之后,把它加入到 CarLocalServices 的成员变量sLocalServiceObjects
中:
callable调用call,把返回的对象加入到一个ArrayMap中
将class作为key,进行了判重 - 随后在 CarService 中调用了
init
方法,初始化车载的各个服务,每一个子服务都实现了 CarServiceBase 的接口,如图:
各个服务类在构造完成之后调用init初始化
ICarImpl内部是通过遍历mAllServices,对每一个子服务调用init
CarServiceBase -
CarService 绑定成功之后,又到了 CarServiceHelperService 成员变量
mCarServiceConnection
的onServiceConnected
方法中,并且执行handleCarServiceConnection
,如图
image.png - 在
handleCarServiceConnection
方法中,将mCarServiceBinder
指向iBinder
,至此 CarServiceHelperService 可以通过mCarServiceBinder
与 CarService 进行进程间通信;随后调用了sendSetSystemServerConnectionsCall
方法,如图:
mCarServiceBinder = iBinder - 在
sendSetSystemServerConnectionsCall
中,顾名思义,是将 CarServiceHelperService 的 Binder 发送给 CarService ,code
取值是FIRST_CALL_TRANSACTION = 0x00000001
到LAST_CALL_TRANSACTION = 0x00ffffff
之间,这里之直接传的FIRST_CALL_TRANSACTION
值为1,说明是aidl定义中为0的方法,如图:
transact(0,null,Binder.FLAG_ONEWAY) -
sendSetSystemServerConnectionsCall
方法中binder
的具体实现类是 ICarImpl ,通过查看 ICarImpl 源码,我们得知 ICarImpl 是 ICar.Stub 的子类,查看 ICar.aidl,得知,code = FIRST_CALL_TRANSACTION
时,实际调用的是setSystemServerConnections
方法,如图:
sendSetSystemServerConnectionsCall在aidl中的值为0 - 也可以在系统编译产物/out/target/common/obj/JAVA_LIBRARIES/android.car_intermediates/classes.jar中找到,解压classes.jar,在android/car/ICar$Stub.class中,反编译可以看到,在
onTransact
方法中,code值为1时,对应的方法是sendSetSystemServerConnectionsCall
,如图:
TRANSACTION_setSystemServerConnections = 1
case 1 : this.setSystemServerConnections(_arg0, _result);
- 查看实现类 ICarImpl 中的
sendSetSystemServerConnectionsCall
方法,将成员变量mICarServiceHelper
指向carServiceHelper
,并且将引用同步更新到各个功能模块中,至此 CarService 可以通过mICarServiceHelper
进行跨进程通信,如图:
mICarServiceHelper = carServiceHelper
下一章继续分析CarOccupantZoneService和CarUserService的init中,做了什么