Android系統的啟動流程詳解

Android系統啟動流程.

1.當系統引導程序啟動Linux內核時, 內核會加載各種數據結構和驅動程序. 有了驅動之后, 開始啟動Android系統并加載用戶級別的第一個進程init(system/core/init/Init.c).

    int main(int argc, char **argv)
    {
        ...

        // 創(chuàng)建各種文件夾和掛載目錄.
        mkdir("/dev", 0755);

        ...

        // 初始化日志.
        log_init();

        // 解析配置文件.
        init_parse_config_file("/init.rc");

        ...

        return 0;
    }

2.加載Init.rc文件. 主要啟動了一個Zygote(孵化器)進程, 此進程是Android系統啟動關鍵服務的一個母進程.

    service zygote /system/bin/app_process -Xzygote /system/bin --zygote --start-system-server
        socket zygote stream 666
        onrestart write /sys/android_power/request_state wake
        onrestart write /sys/power/state on
        onrestart restart media
        onrestart restart netd

3.Zygote進程的初始化在App_main.cpp文件中開啟, 代碼片段如下:

int main(int argc, const char* const argv[])
{
    // 定義Android運行時環(huán)境.
    AppRuntime runtime;
    int i = runtime.addVmArguments(argc, argv);

    ...

    bool startSystemServer = (i < argc) ? 
            strcmp(argv[i], "--start-system-server") == 0 : false;
    setArgv0(argv0, "zygote");
    set_process_name("zygote");

    // 使用運行時環(huán)境啟動Zygote的初始化類.
    runtime.start("com.android.internal.os.ZygoteInit",
        startSystemServer);

    ...
}

4.現在從c或c++代碼進入到java代碼中, ZygoteInit.java初始化類, 代碼如下:

    public static void main(String argv[]) {
        // 加載系統運行依賴的class類.
        preloadClasses();

        ...

        if (argv[1].equals("true")) {
            // Zygote孵化器進程開始孵化系統核心服務.
            startSystemServer();
        } else if (!argv[1].equals("false")) {
            throw new RuntimeException(argv[0] + USAGE_STRING);
        }

        ...
    }

    private static boolean startSystemServer()
        throws MethodAndArgsCaller, RuntimeException {
        String args[] = {
            "--setuid=1000",
            "--setgid=1000",
            "--setgroups=1001,1002,1003,1004,1005,1006,1007,1008,1009,1010,3001,3002,3003",
            "--capabilities=130104352,130104352",
            "--runtime-init",
            "--nice-name=system_server",
            "com.android.server.SystemServer",
        };

        ...

        // 孵化器分叉開啟SystemServer類, 并且把上面定義的參數.
        // 傳遞給此類. 用于啟動系統關鍵服務.
        pid = Zygote.forkSystemServer(
                parsedArgs.uid, parsedArgs.gid,
                parsedArgs.gids, debugFlags, null,
                parsedArgs.permittedCapabilities,
                parsedArgs.effectiveCapabilities);

        ...
    }

5.Zygote進程分叉出SystemServer類, main函數如下:

    public static void main(String[] args) {
        ...

        // 加載本地的動態(tài)鏈接庫.
        System.loadLibrary("android_servers");

        // 調用動態(tài)鏈接庫中的c函數.
        init1(args);
    }

    // 這里init1的函數定義在frameworks\base\services\jni\com_android_server_SystemServer.cpp下的方法.
    native public static void init1(String[] args);

6.comandroidserver_SystemServer.cpp的代碼片段如下:

    static JNINativeMethod gMethods[] = {
        /* name, signature, funcPtr */
        // 把native方法init1, 映射到android_server_SystemServer_init1\. (這里是定義的函數指針)
        { "init1", "([Ljava/lang/String;)V", (void*) android_server_SystemServer_init1 },
    };

    static void android_server_SystemServer_init1(JNIEnv* env, jobject clazz)
    {
        // 轉調
        system_init();
    }

    // 此方法沒有方法體.
    extern "C" int system_init();

7.system_init方法的方法體, 在System_init.cpp類中. 代碼如下:

    extern "C" status_t system_init()
    {
        ...

        // 開啟一些硬件相關的服務.
        SensorService::instantiate();

        ...

        // 獲取Android運行時環(huán)境
        AndroidRuntime* runtime = AndroidRuntime::getRuntime();

        LOGI("System server: starting Android services.\n");
        // 調用SystemServer類中靜態(tài)方法init2\. 從native層轉到java層.
        runtime->callStatic("com/android/server/SystemServer", "init2");

        ...
    }

8.SystemServer下init2方法如下:

    public static final void init2() {
        Slog.i(TAG, "Entered the Android system server!");

        // 進入Android系統服務的初始化.
        Thread thr = new ServerThread();
        thr.setName("android.server.ServerThread");
        thr.start();
    }

9.ServerThread中的run方法如下:

    @Override
    public void run() {
        ...

        // 初始化系統的服務, 并且把服務添加ServiceManager中, 便于以后系統進行統一管理.
        ServiceManager.addService("entropy", new EntropyService());

        ...

        // 調用了ActivityManagerService的systemReady的方法.
        ((ActivityManagerService)ActivityManagerNative.getDefault())
                .systemReady(new Runnable() {
            public void run() {
                ...
            }
        });

        ...
    }

10.ActivityManagerService下的systemReady方法如下:

public void systemReady(final Runnable goingCallback) {
    ...

    // 調用了ActivityStack中的resumeTopActivityLocked去啟動Activity
    mMainStack.resumeTopActivityLocked(null);
}

11.ActivityStack中的resumeTopActivityLocked方法如下:

    final boolean resumeTopActivityLocked(ActivityRecord prev) {
            // 找到第一個當前沒有關閉的Activity, 系統剛剛系統沒有任何Activity執(zhí)行, 所以next為null
            ActivityRecord next = topRunningActivityLocked(null);

            // Remember how we'll process this pause/resume situation, and ensure
            // that the state is reset however we wind up proceeding.
            final boolean userLeaving = mUserLeaving;
            mUserLeaving = false;

            if (next == null) {
                // There are no more activities!  Let's just start up the
                // Launcher...
                if (mMainStack) {
                    // 開啟Launcher應用的第一個Activity界面.
                    return mService.startHomeActivityLocked();
                }
            }
    }

12.home界面顯示, 這時Android系統啟動完畢. 進入到待機畫面.

?著作權歸作者所有,轉載或內容合作請聯系作者
  • 序言:七十年代末播赁,一起剝皮案震驚了整個濱河市颂郎,隨后出現的幾起案子,更是在濱河造成了極大的恐慌容为,老刑警劉巖乓序,帶你破解...
    沈念sama閱讀 217,542評論 6 504
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現場離奇詭異坎背,居然都是意外死亡替劈,警方通過查閱死者的電腦和手機,發(fā)現死者居然都...
    沈念sama閱讀 92,822評論 3 394
  • 文/潘曉璐 我一進店門得滤,熙熙樓的掌柜王于貴愁眉苦臉地迎上來陨献,“玉大人,你說我怎么就攤上這事懂更≌R担” “怎么了?”我有些...
    開封第一講書人閱讀 163,912評論 0 354
  • 文/不壞的土叔 我叫張陵沮协,是天一觀的道長龄捡。 經常有香客問我,道長慷暂,這世上最難降的妖魔是什么聘殖? 我笑而不...
    開封第一講書人閱讀 58,449評論 1 293
  • 正文 為了忘掉前任,我火速辦了婚禮行瑞,結果婚禮上奸腺,老公的妹妹穿的比我還像新娘。我一直安慰自己蘑辑,他們只是感情好洋机,可當我...
    茶點故事閱讀 67,500評論 6 392
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著洋魂,像睡著了一般绷旗。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上副砍,一...
    開封第一講書人閱讀 51,370評論 1 302
  • 那天衔肢,我揣著相機與錄音,去河邊找鬼豁翎。 笑死角骤,一個胖子當著我的面吹牛,可吹牛的內容都是我干的。 我是一名探鬼主播邦尊,決...
    沈念sama閱讀 40,193評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼背桐,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了蝉揍?” 一聲冷哼從身側響起链峭,我...
    開封第一講書人閱讀 39,074評論 0 276
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎又沾,沒想到半個月后弊仪,有當地人在樹林里發(fā)現了一具尸體,經...
    沈念sama閱讀 45,505評論 1 314
  • 正文 獨居荒郊野嶺守林人離奇死亡杖刷,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 37,722評論 3 335
  • 正文 我和宋清朗相戀三年励饵,在試婚紗的時候發(fā)現自己被綠了。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片滑燃。...
    茶點故事閱讀 39,841評論 1 348
  • 序言:一個原本活蹦亂跳的男人離奇死亡役听,死狀恐怖,靈堂內的尸體忽然破棺而出不瓶,到底是詐尸還是另有隱情禾嫉,我是刑警寧澤,帶...
    沈念sama閱讀 35,569評論 5 345
  • 正文 年R本政府宣布蚊丐,位于F島的核電站,受9級特大地震影響艳吠,放射性物質發(fā)生泄漏麦备。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 41,168評論 3 328
  • 文/蒙蒙 一昭娩、第九天 我趴在偏房一處隱蔽的房頂上張望凛篙。 院中可真熱鬧,春花似錦栏渺、人聲如沸呛梆。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,783評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽填物。三九已至,卻和暖如春霎终,著一層夾襖步出監(jiān)牢的瞬間滞磺,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 32,918評論 1 269
  • 我被黑心中介騙來泰國打工莱褒, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留击困,地道東北人。 一個月前我還...
    沈念sama閱讀 47,962評論 2 370
  • 正文 我出身青樓广凸,卻偏偏與公主長得像阅茶,于是被迫代替她去往敵國和親蛛枚。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 44,781評論 2 354

推薦閱讀更多精彩內容