在Android 源码中添加系统服务

时间:2022-01-14 16:06:15

  

  Android系统本身提供了很多系统服务,如WindowManagerService,PowerManagerService等。下面描述一下添加一个系统服务的具体步骤。

 

1.定义自定义系统服务接口

  撰写一个aidl文件,定义服务的接口,将在编译过程中通过aidl工具生成对应的Java接口。

  一般系统服务的aidl文件都放在framework\base\core\java\android\os目录中。

  以IMyTool.aidl为例。在.aidl中定义自己需要加入的方法,编写规则和java接口差不多。

2.添加aidl到编译脚本

  将aidl文件名添加到frameworks\base\目录下的Android.mk编译脚本文件中。如:

1 LOCAL_SRC_FILES += \
2
3 core/java/android/accessibilityservice/IAccessibilityServiceConnection.aidl\
4
5 …\
6
7 core/java/android/os/IMyTool.aidl\
8
9

  IMyTool.aidl即加进去的aidl文件,加入后才能在make过程中编译到,否则将在后面的SystemServer添加系统服务时会报错提示找不到对应类。

3.实现添加的服务

  编写真正工作的服务类,继承IMyTool.Stub类(AIDL文件名.Stub,aidl生成的接口中的内部类,是一个Binder)。

  例如:

1 public class MyToolService extends IMyTool.Stub {
2
3   //实现IMyTool.aidl中定义的接口。
4
5 }

  服务类一般都放在framework\base\services\core\java\com\android\server目录中。

4.注册到SystemServer

  将自定义服务注册到SystemServer,使得开机过程中被添加。在framework\base\services\java\com\android\server目录中的SystemServer中启动服务代码处加入:

  在SystemServer.java中,分不同类型的服务启动。

 1     // Start services.
2 try {
3 startBootstrapServices();
4 startCoreServices();
5 startOtherServices();
6 } catch (Throwable ex) {
7 Slog.e("System", "******************************************");
8 Slog.e("System", "************ Failure starting system services", ex);
9 throw ex;
10 }

  根据自定义服务类型加入到相应函数中:

 1 try {
2
3   Slog.i(TAG, "MyToolService");
4
5   ServiceManager.addService(Context.MY_TOOL_SERVICE,new MyToolService(context));// MyToolService构造函数自己定义,一般都会用到Context
6
7 } catch(Throwable e) {
8
9   Slog.e(TAG, "Failure startingMyToolService", e);
10
11 }

  上面代码中Context.MY_TOOL_SERVICE是自己在Context类中定义的常量,也就是给服务定义的名字,使用常量方便获取服务,而不需要记住注册服务时用的名字,且想换名字时只需改一个常量的值。

5.更新api

  由于在工程中添加了自己定义的类及常量,系统的api没有更新,因此需要先在工程中make clean然后make update-api,执行完后会发现frameworks\base\api\current.xml文件中多出自己定义的一些东西。current.xml这个文件包含了所有系统所有能被应用层使用的类及其方法等。之后再使用make编出来的固件及jar包就能包含自定义的接口。

6.使用自定义的系统服务

  将编出来的jar包通过lib方式导入工程。jar包位置:out\target\common\obj\JAVA_LIBRARIES\framework_intermediates\classes.jar
调用以下代码获取自定义服务:

  IMyTool myTool = IMyTool.Stub.asInterface(ServiceManager.getService(MY_TOOL_SERVICE));

  MY_TOOL_SERVICE即在Context中定义的常量。获取到myTool后就可以调用在aidl文件中定义的接口了。