1. 编写 AIDL
在Binder 程序示例之 java 篇
咱们写了一个 Binder java 服务,不过需求咱们手动去履行 Java 程序。今天咱们就来看一下怎样增加一个在开机就自发动的 Java 体系服务。
在 frameworks/base/core/java
目录下创建如下的文件与文件夹:
com/yuandaima
└── IJavaHelloService.aidl
其间 IJavaHelloService.aidl 的内容如下:
package com.yuandaima;
interface IJavaHelloService
{
void sayhello();
int sayhello_to(String name);
}
AIDL 在咱们自定义的 com/yuandaima
包结构下。
2. 将 AIDL 加入 Android.bp
接着修正 frameworks/base/Android.bp
,在 framework-defaults 模块的 srcs 节点中增加 "core/java/android/rice14/com/yuandaima/IJavaHelloService.aidl"
:
java_defaults {
name: "framework-defaults",
installable: true,
srcs: [
//......
"core/java/com/yuandaima/IJavaHelloService.aidl"
]
//......
}
接着咱们能够履行:
source build/envsetup.sh
# 自定义 Product rice14-eng
lunch rice14-eng
make update-api
然后检查 Frameworks/base/api/current.txt
中检查是否存在对应 IJavaHelloService 接口。
3. 定义 Binder 服务端类
frameworks/base/services/core/java/com
目录下新建如下的目录与文件:
yuandaima
└── JavaHelloService.java
其间 JavaHelloService.java:
package com.yuandaima;
import android.util.Log;
public class JavaHelloService extends IJavaHelloService.Stub {
private static final String TAG = "JavaHelloService";
private int cnt1 = 0;
private int cnt2 = 0;
public void sayhello() throws android.os.RemoteException {
cnt1++;
Log.i(TAG, "sayhello : cnt = "+cnt1);
}
public int sayhello_to(java.lang.String name) throws android.os.RemoteException {
cnt2++;
Log.i(TAG, "sayhello_to "+name+" : cnt = "+cnt2);
return cnt2;
}
}
4. 开机发动服务
修正 frameworks/base/services/java/com/android/server/SystemServer.java
文件,在 startOtherServices
方法的最终增加以下代码:
//import 不要忘了
import com.yuandaima.JavaHelloService;
// add hello service
traceBeginAndSlog("JavaHelloService");
ServiceManager.addService("JavaHelloService", new JavaHelloService());
traceEnd();
5. selinux 配备
接着增加 selinux 配备:
一起修正 system/sepolicy/private 和 system/sepolicy/prebuilts/api/29.0/private 目录下的:
service.te:
type JavaHelloServiceType, system_server_service, service_manager_type;
service_contexts:
JavaHelloService u:object_r:JavaHelloServiceType:s0
platform_app.te
allow platform_app JavaHelloServiceType:service_manager find;
6. 增加 Java 接口白名单
咱们新增加了 com/yuandaima
Java 软件包,这些包需求在白名单中声明才干被上层 App 访问到:
在 build/make/core/tasks/check_boot_jars/package_whitelist.txt
中增加如下内容:
com\.yuandaima
com\.yuandaima\..*
7. 体系 App 使用体系服务
接着咱们就能够在咱们的体系 App 中使用咱们的自定义的体系服务了:
修正咱们在玩转 AOSP 之体系 App 源码增加中增加的体系 App Demo:
Android.bp:
android_app {
name: "FirstSystemApp",
srcs: ["src/**/*.java"],
resource_dirs: ["res"],
manifest: "AndroidManifest.xml",
platform_apis: true,
sdk_version: "",
certificate: "platform",
//去掉 product_specific
// product_specific: true,
static_libs: ["androidx.appcompat_appcompat",
"com.google.android.material_material",
"androidx-constraintlayout_constraintlayout",
"lib-lottie"],
}
接着修正自定义 Product 配备文件 device/jelly/rice14/rice14.mk
:
PRODUCT_PACKAGES += \
# ......
FirstSystemApp
PRODUCT_ARTIFACT_PATH_REQUIREMENT_WHITELIST += \
# ......
/system/app/FirstSystemApp/FirstSystemApp.apk
修正 device/jelly/rice14/FirstSystemApp/src/com/yuandaima/firstsystemapp/MainActivity.java
源码:
public class MainActivity extends AppCompatActivity {
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
try {
IJavaHelloService service = IJavaHelloService.Stub.asInterface(ServiceManager.getService("JavaHelloService"));
service.sayhello();
} catch (Exception e) {
e.printStackTrace();
}
}
}
接着咱们就能够从头编译体系,发动虚拟机了:
source build/envsetup.sh
# 自定义 Product rice14
lunch rice14-eng
make -j32
emulator
接着咱们就能够打开咱们的 FirstSystemApp 了,然后检查 log:
关于
我叫阿豪,2015 年本科毕业于国防科学技术大学指挥信息体系专业,毕业后从事信息化配备的研制作业,主要研讨方向是 Android Framework 与 Linux Kernel。
假如你对 Android Framework 感兴趣或者正在学习 Android Framework,能够重视我的微信公众号,我会继续共享我的学习经历,帮助正在学习的你少走一些弯路。学习过程中假如你有疑问或者你的经历想要共享给我们能够增加我的微信,我拉你进技术交流群。
假如你想体系学习 Anroid Framework 也能够重视我的抖音账号,在主页店肆即可购买付费课程。