本文以一個簡單的例子詳細介紹如何為Android 6.0 系統(tǒng)新增一個系統(tǒng)服務,以及如何使用自定義系統(tǒng)服務女轿。
目標:1.創(chuàng)建一個自定義服務CCCService
2.APP 開發(fā)過程中可以使用 getSystemService("ccc") 獲取 CCCManager 并且調(diào)用里面的函數(shù)。
** step1 創(chuàng)建aidl文件**
在源碼frameworks/base/core/java/android/os/ 下面新增 一個 ICCCService.aidl
假設我們定義了5個函數(shù)珍策,這些函數(shù)將會在SystemServer進程執(zhí)行蹭劈。
內(nèi)容如下
// 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 創(chuàng)建Service文件 (CCCService)
在 frameworks/base/services/core/java/com/android/server/ 下面新增一個 CCCService.java 用來實現(xiàn)aidl文件定義的接口肺孵。
內(nèi)容如下
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() 函數(shù) 的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 創(chuàng)建Manager凳怨,即CCCManager
在frameworks/base/core/java/android/app/ 下創(chuàng)建CCCManager.java 文件 內(nèi)容如下
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
在靜態(tài)代碼塊中增加
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:
可以先創(chuàng)建一個java工程簡單寫一個CCCManager類,導出jar 在ide中使用provided 依賴肤舞。這樣開發(fā)過程中就不會報錯紫新。
java臨時工程如下使用 IDEA創(chuàng)建的java項目
Android studio 項目配置