本文以一个简单的例子详细介绍如何为Android 6.0 系统新增一个系统服务,以及如何使用自定义系统服务。
目标:1.创建一个自定义服务CCCService
2.APP 开发过程中可以使用 getSystemService("ccc") 获取 CCCManager 并且调用里面的函数。
step1 创建aidl文件
在源码frameworks/base/core/java/android/os/ 下面新增 一个 ICCCService.aidl
假设我们定义了5个函数,这些函数将会在SystemServer进程执行。
内容如下
// ICCCService.aidl package android.os; // Declare any non-default types here with import statements interface ICCCService { /** * Demonstrates some basic types that you can use as parameters * and return values in AIDL. */ void setVal(String key,String value); String getVal(String key); void appendLog(String log); void clearLog(); String readLog(); }
step2 创建Service文件 (CCCService)
在 frameworks/base/services/core/java/com/android/server/ 下面新增一个 CCCService.java 用来实现aidl文件定义的接口。
内容如下
package com.android.server; import java.io.InputStreamReader; import java.io.LineNumberReader; import java.lang.*; import java.util.HashMap; import android.os.RemoteException; import android.os.ICCCService; /** * * Created by zhuangqianliu on 2016/9/21. */ public class CCCService extends ICCCService.Stub { private static HashMap<String,String> map=new HashMap<>(); private static String inner_log=""; public CCCService() { } @Override public void setVal(String key, String value) throws RemoteException { map.put(key,value); } @Override public String getVal(String key) throws RemoteException { return map.get(key); } @Override public void appendLog(String log) throws RemoteException { inner_log+=log+"\n"; } @Override public void clearLog() throws RemoteException { inner_log=""; } @Override public String readLog() throws RemoteException { return inner_log; } }
step3 将自定义Service 加入到SystemServer 启动进程
先在 frameworks/base/core/java/android/content/Context.java 中添加一行
public static final String CCC_SERVICE="ccc";
修改 frameworks/base/services/java/com/android/server/SystemServer.java
在 startOtherServices() 函数 的try模块中增加以下代码
try { Slog.i(TAG, "CCC Service"); ServiceManager.addService(Context.CCC_SERVICE, new CCCService()); } catch (Throwable e) { Slog.e(TAG, "Failure starting CCC Service", e); }
最终效果如图
step4 创建Manager,即CCCManager
在frameworks/base/core/java/android/app/ 下创建CCCManager.java 文件 内容如下
package android.app; /** * Created by liam on 16/10/2. */ import android.annotation.SdkConstant; import android.annotation.SystemApi; import android.content.Context; import android.content.Intent; import android.os.Build; import android.os.Parcel; import android.os.Parcelable; import android.os.RemoteException; import android.os.ICCCService; import android.util.Log; public class CCCManager { ICCCService mService; public CCCManager(Context ctx,ICCCService service){ mService=service; } public void setVal(String key,String value){ try{ mService.setVal(key,value); }catch(Exception e){ Log.e("CCCManager",e.toString()); e.printStackTrace(); } } public String getVal(String key){ try{ return mService.getVal(key); }catch(Exception e){ Log.e("CCCManager",e.toString()); e.printStackTrace(); } return null; } public void appendLog(String log){ try{ mService.appendLog(log); }catch(Exception e){ Log.e("CCCManager",e.toString()); e.printStackTrace(); } } public void clearLog(){ try{ mService.clearLog(); }catch(Exception e){ Log.e("CCCManager",e.toString()); e.printStackTrace(); } } public String readLog(){ try{ return mService.readLog(); }catch(Exception e){ Log.e("CCCManager",e.toString()); e.printStackTrace(); } return null; } }
step5 注册到SystemService
修改frameworks/base/core/java/android/app/SystemServiceRegistry.java
在静态代码块中增加
registerService(Context.CCC_SERVICE, CCCManager.class, new CachedServiceFetcher<CCCManager>() { @Override public CCCManager createService(ContextImpl ctx) { IBinder b = ServiceManager.getService(Context.CCC_SERVICE); ICCCService service = ICCCService.Stub.asInterface(b); return new CCCManager(ctx, service); }});
step6 修改SePolicy的编译验证
修改 /external/sepolicy/service.te
在最后一行添加
type ccc_service, system_api_service, system_server_service, service_manager_type;
然后修改同目录下 /external/sepolicy/service_contexts 文件
中间插入一行
ccc u:object_r:ccc_service:s0
如图所示
step7 重新编译源码
别忘了先 make update-api。
Step8 测试
tip:
可以先创建一个java工程简单写一个CCCManager类,导出jar 在ide中使用provided 依赖。这样开发过程中就不会报错。
java临时工程如下使用 IDEA创建的java项目
Android studio 项目配置