blob: 6daa5b4dc6d8bf7bd7e860b8897b6901ead444bb [file] [log] [blame]
Rahul Ravikumar05336002019-10-14 15:04:32 -07001/*
2 * Copyright (C) 2018 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
17package android.os;
18
19import android.content.pm.ApplicationInfo;
20import android.util.Log;
21
22import com.android.internal.annotations.GuardedBy;
23
24/**
25 * AppZygote is responsible for interfacing with an application-specific zygote.
26 *
27 * Application zygotes can pre-load app-specific code and data, and this interface can
28 * be used to spawn isolated services from such an application zygote.
29 *
30 * Note that we'll have only one instance of this per application / uid combination.
31 *
32 * @hide
33 */
34public class AppZygote {
35 private static final String LOG_TAG = "AppZygote";
36
37 // UID of the Zygote itself
38 private final int mZygoteUid;
39
40 // First UID/GID of the range the AppZygote can setuid()/setgid() to
41 private final int mZygoteUidGidMin;
42
43 // Last UID/GID of the range the AppZygote can setuid()/setgid() to
44 private final int mZygoteUidGidMax;
45
46 private final Object mLock = new Object();
47
48 /**
49 * Instance that maintains the socket connection to the zygote. This is {@code null} if the
50 * zygote is not running or is not connected.
51 */
52 @GuardedBy("mLock")
53 private ChildZygoteProcess mZygote;
54
55 private final ApplicationInfo mAppInfo;
56
57 public AppZygote(ApplicationInfo appInfo, int zygoteUid, int uidGidMin, int uidGidMax) {
58 mAppInfo = appInfo;
59 mZygoteUid = zygoteUid;
60 mZygoteUidGidMin = uidGidMin;
61 mZygoteUidGidMax = uidGidMax;
62 }
63
64 /**
65 * Returns the zygote process associated with this app zygote.
66 * Creates the process if it's not already running.
67 */
68 public ChildZygoteProcess getProcess() {
69 synchronized (mLock) {
70 if (mZygote != null) return mZygote;
71
72 connectToZygoteIfNeededLocked();
73 return mZygote;
74 }
75 }
76
77 /**
78 * Stops the Zygote and kills the zygote process.
79 */
80 public void stopZygote() {
81 synchronized (mLock) {
82 stopZygoteLocked();
83 }
84 }
85
86 public ApplicationInfo getAppInfo() {
87 return mAppInfo;
88 }
89
90 @GuardedBy("mLock")
91 private void stopZygoteLocked() {
92 if (mZygote != null) {
93 // Close the connection and kill the zygote process. This will not cause
94 // child processes to be killed by itself.
95 mZygote.close();
96 Process.killProcess(mZygote.getPid());
97 mZygote = null;
98 }
99 }
100
101 @GuardedBy("mLock")
102 private void connectToZygoteIfNeededLocked() {
103 String abi = mAppInfo.primaryCpuAbi != null ? mAppInfo.primaryCpuAbi :
104 Build.SUPPORTED_ABIS[0];
105 try {
106 mZygote = Process.ZYGOTE_PROCESS.startChildZygote(
107 "com.android.internal.os.AppZygoteInit",
108 mAppInfo.processName + "_zygote",
109 mZygoteUid,
110 mZygoteUid,
111 null, // gids
112 0, // runtimeFlags
113 "app_zygote", // seInfo
114 abi, // abi
115 abi, // acceptedAbiList
116 null, // instructionSet
117 mZygoteUidGidMin,
118 mZygoteUidGidMax);
119
120 ZygoteProcess.waitForConnectionToZygote(mZygote.getPrimarySocketAddress());
121 // preload application code in the zygote
122 Log.i(LOG_TAG, "Starting application preload.");
123 mZygote.preloadApp(mAppInfo, abi);
124 Log.i(LOG_TAG, "Application preload done.");
125 } catch (Exception e) {
126 Log.e(LOG_TAG, "Error connecting to app zygote", e);
127 stopZygoteLocked();
128 }
129 }
130}