zygote分析--AndroidRuntime::start()

AndroidRuntime.cpp的路径为/frameworks/base/core/jni/AndroidRuntime.cpp

在上一篇文章中分析了app_main.cpp,得到app_main.cppmain()方法最终调用了AndroidRuntimestart()方法(runtime为apptime类型的对象,apptime继承了androidruntime的start方法)。
runtime.start("com.android.internal.os.ZygoteInit",args)

  • runtime.addOption("-Xzygote"),runtime.setArgv0("zygote")
  • args中存储的是关于虚拟机的参数,主要有start-system-serverabi-listsocket-name=zygote

在android源码的注释中,写到start方法的主要作用分为两点

  1. 启动虚拟机
  2. 调用className类的static void main(String args[])方法

start方法源代码解析

    ALOGD(">>>>>>START %s uid %d<<<<<<<<<<<<<\n",className!=NULL?className:"(unknown)",getuid());
    static const String8 startSystemServer("start-system-server");

代码首先调用了ALOGD方法,用来记录日志内容(ALOGD记录的日志在编译的时候存在,但是在运行的时候会被剔除)。


static const String8 startSystemServer("start-system-server")
接着是一个for循环

for(size_t i=0;i<options.size();++i)
{ 
    if(options[i] == startSystemServer)
    {
        const int LOG_BOOT_PROGRESS_START = 3000;      
        LOG_EVENT_LONG(LOG_BOOT_PROGRESS_START,ns2ms(systemTime(SYSTEM_TIME_MONOTONIC)));
    }
}

typedef long unsigned int size_t
systemTime(SYSTEM_TIME_MONOTONIC)获取系统当前时间

循环对传入的参数args进行匹配,如果传入的参数有startSystemServer,则进行一个日志记录。记录当前时间之后一段时间内的日志。

const char * rootDir = getenv(ANDROID_ROOT);
if(rootDir == NULL)
{
    rootDir = "/system";
    if(!hasDir("/system"))
    {
        LOG_FATAL("No root directory specified,and / android dose not exist.");
        return ;
    }
    setenv("ANDROID_ROOT",rootDir,1);
}

在上边这段代码中,主要是用来完成环境变量ANDROID_ROOT的设置,调用了getenv()方法和setenv()方法。通过这段代码,环境变量ANDROID_ROOT的值被设置成了/system
接着定义了一些变量。

JniInvocation jni_invocation;
JNIEnv *env;

接着的代码是

jni_invocation.Init(NULL);

通过jni_invocation.Init(NULL)完成jni接口的初始化。
接着是创建虚拟机的代码

if(startVm(&mJavaVM,&env)!=0)
{
    return ;
}

调用startVm启动虚拟机,在startVm方法中,定义了虚拟机的一系列参数。通过property_get()方法来进行参数的设置。
接着是注册JNI的代码

if(startReg(env)<0)
{
    ALOGE("Unable to register all android natives\n");
    return ;
}

调用startReg方法 注册android方法
接着定义了三个变量,定义参数的目的是将AndroidRuntime::start的两个参数传入到ZygoteInit的main方法。

两个参数是ZygoteInit类名和args

jclass stringClass;
jobjectArray strArray;
jstring classNameStr;

接着对三个变量分别进行了赋值

    stringClass = env->FindClass("java/lang/String");
    assert(stringClass != NULL);
    strArray = env->NewObjectArray(options.size() + 1, stringClass, NULL);
    assert(strArray != NULL);
    classNameStr = env->NewStringUTF(className);
    assert(classNameStr != NULL);
    env->SetObjectArrayElement(strArray, 0, classNameStr);

    for (size_t i = 0; i < options.size(); ++i) {
        jstring optionsStr = env->NewStringUTF(options.itemAt(i).string());
        assert(optionsStr != NULL);
        env->SetObjectArrayElement(strArray, i + 1, optionsStr);
    }

className的值是com.android.internal.os.ZygoteInit
args中存储的是关于虚拟机的参数,主要有start-system-serverabi-listsocket-name=zygote

上边的操作将ZygoteInit和args的值都存入到了env中。
接着的代码是

char* slashClassName=toSlashClassName(className);
jchar startClass = env->FindClass(slashClassName);

if(startClass == NULL)
{
    ALOGE("JavaVM unable to locate class '%s'\n",slashClassName);
}
else
{
    jmethodID startMeth = env->GetStaticMethodID(startClass,"main","(Ljava/lang/String;)V");
    if(startMeth==NULL)
    {
        ALOGE("JavaVM unable to find main() in '%s'\n",className);
    }
    else 
    {
        env->CallStaticVoidMethod(startClass,startMeth,strArray);
    }
}

这一段代码主要作用是调用CallStaticVoidMethod方法,启动ZygoteInit.main()。
在进入函数时,从env中获取startClass名字,然后调用GetStaticMethodID方法,获取startClass的main方法。最后调用CallStaticVoidMethod方法,启动ZygoteInit.main()。
至此,从app_main.cpp-->AndroidRuntime.cpp通过CallStaticVoidMethod方法进入到了java世界,调用的第一个java方法是ZygoteInit类的mian方法

总结

start主要完成的操作有

  • jni_invocation.Init()方法初始化jni接口
  • startVm()方法创建虚拟机
  • startReg()方法注册Android方法
  • CallStaticVoidMethod()方法打开ZygoteInit类的main方法。完成从c/c++到java代码。
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 204,189评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,577评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,857评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,703评论 1 276
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,705评论 5 366
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,620评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,995评论 3 396
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,656评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,898评论 1 298
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,639评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,720评论 1 330
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,395评论 4 319
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,982评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,953评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,195评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 44,907评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,472评论 2 342

推荐阅读更多精彩内容