Android上层启动过程的几个关键点

在前面我简单的描述了Android的启动过程http://blog.csdn.net/codectq/article/details/7383231,但终究是不够完善。

        现在在工程中遇到了实际的问题,反过来阅读下代码,再重新审视一下在kernel启动之后的ANDROID启动过程。

        在frameworks/base/services/java/com/android/server文件夹下的SystemServer.cpp文件。在此文件中调用了init1()函数,文件中对次函数的描述为:这个方法是从Zygote中调用来初始化系统。这将引起本地的服务(Surfaceflinger,audioflinger等)开启。在这些完成之后将回调init2()来开始android服务。

       从上面的描述中我们可以得知两个方面:一是在android的服务中,我们分两种服务,一种是native服务,另一种是android服务。二是我们的本地服务是在init1()的过程中完成的,而android服务是在init2()的过程中完成的。

        从实际的真机的启动过程的打印信息我们也能够看出,在启动的过程中,我们首先会启动SurfaceFlinger,AudioFlinger然后开始启动dalikvm,从而进入对Zygote和System并且通过调用system_init()函数和sysproc来启动Android runtime 启动native service。通过回调init2(),Android  service\thread pool从而进入systemserver,运行systemserver线程,将注册到systemserver中的服务启动。

        如果我们想借助Android本身的架构来完成添加我们的服务的目的的情况下我们不妨就利用这些。

        在我们通过使用JNI机制使得上层的java应用能够调用native service。而具体的实现就是将.cpp文件放在JNI文件夹下,而将.java文件放在java文件夹下。具体步骤如下:

        在上文中,我们说过我们可以借助Android自身架构的东西来完成我们的JNI服务,而不必每个细节都自己完成。并且我们提到我们会调用AndroidRuntime所以我们可以将服务注册到AndroidRuntime.cpp文件中。下面是个经典的例子,此处做个摘抄地址为http://dongyulong.blog.51cto.com/1451604/545496(但是不知道是不是原作者)

frameworks/base/core/jni 路径下创建例子 android_mytest_hellojni.cpp 文件,这个文件就是在 JNI 层实现接口。文件内容如下:(可参考同一目录下的 android_debug_JNITest.cpp 文件编写)

#define LOG_TAG "HelloJNI"
#include "jni.h"
#include "nativehelper/JNIHelp.h"
#include "utils/Log.h"
#include "utils/misc.h"

namespace android {
static jstring android_mytest_hellojni_displayString(JNIEnv *env, jclass clazz)
{
 return env->NewStringUTF("Hello from JNI!");
} 

/*
 * JNI registration.
 */

static JNINativeMethod gMethods[] = {
    /* name, signature, funcPtr */
    { "displayString", "()Ljava/lang/String;",
            (void*) android_mytest_hellojni_displayString },
    
};
int register_android_mytest_hellojni(JNIEnv* env)
{ //此处的目录结构就是在Javaframework层的文件目录,且必须一致

return jniRegisterNativeMethods(env, "android/mytest/hellojni", 
        gMethods, NELEM(gMethods));
}
};

2. JNI 层:对编译的修改配置

  2.1修改/Android/android-1.6_r2/frameworks/base/core/jni目录下的Android.mk 文件,在LOCAL_SRC_FILES:= \ 下面加上
android_mytest_hellojni.cpp \ 
   2.2修改/Android/android-1.6_r2/frameworks/base/core/jni目录下的AndroidRuntime.cpp 文件在extern int 后面添加
extern int register_android_mytest_hellojni(JNIEnv* env); 
然后在static const RegJNIRec gRegJNI[] = {下面添加
REG_JNI(register_android_mytest_hellojni), 
这样,JNI层的修改就到此为止了。

frameworks/base/core/java/android/ 新建文件目录 mytest ,在该目录下新建文件 hellojni.java 声明接口。内容如下:(可以参考 android-1.6_r2/frameworks/base/core/java/android/debug 目录下的 JNITest.java 文件编写) 

package android.mytest;
public class hellojni{
    public hellojni(){}
//此处声明为public所以才可以被application调用

    public static native String displayString(); 
}

4 .下面我们要对我们做过更改的 libandroid_runtime.so 和 framework.jar 进行重新编译。

在源代码工程目录下输入 make libandroid_runtime 重新编译生成 libandroid_runtime.so

target thumb C++: libandroid_runtime <= frameworks/base/core/jni/android_mytest_hellojni.cpp

target thumb C++: libandroid_runtime <= frameworks/base/core/jni/AndroidRuntime.cpp

target SharedLib: libandroid_runtime (out/target/product/generic/obj/SHARED_LIBRARIES/libandroid_runtime_intermediates/LINKED/libandroid_runtime.so)

target Prelink: libandroid_runtime (out/target/product/generic/symbols/system/lib/libandroid_runtime.so)

target Strip: libandroid_runtime (out/target/product/generic/obj/lib/libandroid_runtime.so)

Install: out/target/product/generic/system/lib/libandroid_runtime.so

然后再输入 make framework 重新编译生成 framework.jar

Install: out/target/product/generic/system/framework/framework.jar

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值