前言
本系列文章的主题是从0开始写android。从0是指从linux操作系统开始, 记录笔者在linux 系统的基础复现生长出android 的过程。生长是个以时间为轴,事物发展变化的过程。瑞士计算机科学家尼古拉斯•威茨(Niklaus Wirth)曾提出程序=数据结构+算法,同样这系列文章将着重分析 数据结构和算法。在这系列文章里将看不到binder等复杂的跨进程调用,而是用极简的风格描述程序。
这里的开发环境为 ,操作系统:ubuntu20.04 , java -version 为openjdk version "1.8.0_392"。
1、从 init.rc 到 jvm
linux 从init.rc 启动jvm 的第一个程序为 app_process,我们这里尝试在ubuntu 上重写app_process的主要流程。
#include <iostream>
#include<jni.h>
JNIEnv* create_vm(JavaVM **jvm)
{
printf("%s\n",__func__);
JNIEnv* env;
JavaVMInitArgs args;
JavaVMOption options;
args.version = JNI_VERSION_1_8;
args.nOptions = 1;
options.optionString ="-Djava.class.path=./";
args.options = &options;
args.ignoreUnrecognized = 0;
int rv;
rv = JNI_CreateJavaVM(jvm,(void**)&env, &args);
if (rv < 0 || !env)
printf("Unable to Launch JVM%d\n",rv);
else
printf("Launched JVM! :)\n");
return env;
}
void invoke_class(JNIEnv* env)
{
printf("%s\n",__func__);
jclass zygote_init_class;
jmethodID main_method;
zygote_init_class =(env)->FindClass( "com/android/ZygoteInit");
main_method =(env)->GetStaticMethodID( zygote_init_class, "main","([Ljava/lang/String;)V");
(env)->CallStaticVoidMethod(zygote_init_class, main_method, NULL);
}
int main() {
JavaVM *jvm;
JNIEnv *env;
env = create_vm(&jvm);
if(env == NULL)
return 1;
invoke_class(env);
return 0;
}
在ubuntu 上我们可以选用自己喜欢的C/C++开发工具,我这里选用的是clion。上面的代码中,我们主要是调用了JVM 的
JNI_CreateJavaVM
函数 ,创建出jvm虚拟机,然后去反射 ZygoteInit.java 中的ZygoteInit 类。
对应的编译脚本如下。
cmake_minimum_required(VERSION 3.15)
project(HDroid)
set(CMAKE_CXX_STANDARD 14)
set(SRC_LIST main.cpp)
# include_directories 将指定目录添加到编译器的头文件搜索路径之下,指定的目录被解释成当前源码路径的相对路径。
include_directories(/usr/lib/jvm/java-8-openjdk-amd64/include /usr/lib/jvm/java-8-openjdk-amd64/include/linux)
# 把该路径添加到第三方库搜索路径中
link_directories(/usr/lib/jvm/java-8-openjdk-amd64/jre/lib/amd64/server)
2. 从jvm中加载ZygoteInit
写一个最简单的ZygoteInit类.
public class ZygoteInit{
public static void main(String[] args){
Log.d("Hello, ZygoteInit");
}
}
放在当前目录的 com/android/ 路径下。
用clion的环境编译 ,如果没有问题会输出
Hello, ZygoteInit. 至此我们完成了第一步 ,从init.rc到启动JVM, 并加载了第一个java类,ZygoteInit.
总结
本文详述了从linux 启动程序 init.rc到 启动第一个java程序的过程。完成了从linux环境到jvm 环境的切换。
对应的代码放在: