blob: 41293981102fc1533e2fdb3b25f483639d5650b3 [file] [log] [blame]
Justin Klaassen98fe7812018-01-03 13:39:41 -05001/**
2 * Copyright 2017 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.content.pm.dex;
18
Justin Klaassen4d01eea2018-04-03 23:21:57 -040019import android.annotation.CallbackExecutor;
20import android.annotation.IntDef;
Justin Klaassen98fe7812018-01-03 13:39:41 -050021import android.annotation.NonNull;
Justin Klaassen4d01eea2018-04-03 23:21:57 -040022import android.annotation.Nullable;
Justin Klaassen98fe7812018-01-03 13:39:41 -050023import android.annotation.RequiresPermission;
24import android.annotation.SystemApi;
Justin Klaassen4d01eea2018-04-03 23:21:57 -040025import android.os.Environment;
Justin Klaassen98fe7812018-01-03 13:39:41 -050026import android.os.ParcelFileDescriptor;
27import android.os.RemoteException;
28import android.util.Slog;
29
Justin Klaassen4d01eea2018-04-03 23:21:57 -040030import java.io.File;
31import java.lang.annotation.Retention;
32import java.lang.annotation.RetentionPolicy;
33import java.util.concurrent.Executor;
34
Justin Klaassen98fe7812018-01-03 13:39:41 -050035/**
36 * Class for retrieving various kinds of information related to the runtime artifacts of
37 * packages that are currently installed on the device.
38 *
39 * @hide
40 */
41@SystemApi
42public class ArtManager {
43 private static final String TAG = "ArtManager";
44
45 /** The snapshot failed because the package was not found. */
46 public static final int SNAPSHOT_FAILED_PACKAGE_NOT_FOUND = 0;
47 /** The snapshot failed because the package code path does not exist. */
48 public static final int SNAPSHOT_FAILED_CODE_PATH_NOT_FOUND = 1;
49 /** The snapshot failed because of an internal error (e.g. error during opening profiles). */
50 public static final int SNAPSHOT_FAILED_INTERNAL_ERROR = 2;
51
Justin Klaassen4d01eea2018-04-03 23:21:57 -040052 /** Constant used for applications profiles. */
53 public static final int PROFILE_APPS = 0;
54 /** Constant used for the boot image profile. */
55 public static final int PROFILE_BOOT_IMAGE = 1;
56
57 /** @hide */
58 @IntDef(flag = true, prefix = { "PROFILE_" }, value = {
59 PROFILE_APPS,
60 PROFILE_BOOT_IMAGE,
61 })
62 @Retention(RetentionPolicy.SOURCE)
63 public @interface ProfileType {}
64
65
Justin Klaassen98fe7812018-01-03 13:39:41 -050066 private IArtManager mArtManager;
67
68 /**
69 * @hide
70 */
71 public ArtManager(@NonNull IArtManager manager) {
72 mArtManager = manager;
73 }
74
75 /**
Justin Klaassen4d01eea2018-04-03 23:21:57 -040076 * Snapshots a runtime profile according to the {@code profileType} parameter.
Justin Klaassen98fe7812018-01-03 13:39:41 -050077 *
Justin Klaassen4d01eea2018-04-03 23:21:57 -040078 * If {@code profileType} is {@link ArtManager#PROFILE_APPS} the method will snapshot
79 * the profile for for an apk belonging to the package {@code packageName}.
80 * The apk is identified by {@code codePath}.
Justin Klaassen98fe7812018-01-03 13:39:41 -050081 *
Justin Klaassen4d01eea2018-04-03 23:21:57 -040082 * If {@code profileType} is {@code ArtManager.PROFILE_BOOT_IMAGE} the method will snapshot
83 * the profile for the boot image. In this case {@code codePath can be null}. The parameters
84 * {@code packageName} and {@code codePath} are ignored.
85 *u
86 * The calling process must have {@code android.permission.READ_RUNTIME_PROFILE} permission.
87 *
88 * The result will be posted on the {@code executor} using the given {@code callback}.
89 * The profile will be available as a read-only {@link android.os.ParcelFileDescriptor}.
90 *
91 * This method will throw {@link IllegalStateException} if
92 * {@link ArtManager#isRuntimeProfilingEnabled(int)} does not return true for the given
93 * {@code profileType}.
94 *
95 * @param profileType the type of profile that should be snapshot (boot image or app)
96 * @param packageName the target package name or null if the target is the boot image
97 * @param codePath the code path for which the profile should be retrieved or null if
98 * the target is the boot image
Justin Klaassen98fe7812018-01-03 13:39:41 -050099 * @param callback the callback which should be used for the result
Justin Klaassen4d01eea2018-04-03 23:21:57 -0400100 * @param executor the executor which should be used to post the result
Justin Klaassen98fe7812018-01-03 13:39:41 -0500101 */
102 @RequiresPermission(android.Manifest.permission.READ_RUNTIME_PROFILES)
Justin Klaassen4d01eea2018-04-03 23:21:57 -0400103 public void snapshotRuntimeProfile(@ProfileType int profileType, @Nullable String packageName,
104 @Nullable String codePath, @NonNull @CallbackExecutor Executor executor,
105 @NonNull SnapshotRuntimeProfileCallback callback) {
Justin Klaassen98fe7812018-01-03 13:39:41 -0500106 Slog.d(TAG, "Requesting profile snapshot for " + packageName + ":" + codePath);
107
108 SnapshotRuntimeProfileCallbackDelegate delegate =
Justin Klaassen4d01eea2018-04-03 23:21:57 -0400109 new SnapshotRuntimeProfileCallbackDelegate(callback, executor);
Justin Klaassen98fe7812018-01-03 13:39:41 -0500110 try {
Justin Klaassen4d01eea2018-04-03 23:21:57 -0400111 mArtManager.snapshotRuntimeProfile(profileType, packageName, codePath, delegate);
Justin Klaassen98fe7812018-01-03 13:39:41 -0500112 } catch (RemoteException e) {
113 e.rethrowAsRuntimeException();
114 }
115 }
116
117 /**
Justin Klaassen4d01eea2018-04-03 23:21:57 -0400118 * Returns true if runtime profiles are enabled for the given type, false otherwise.
Justin Klaassen98fe7812018-01-03 13:39:41 -0500119 *
120 * The calling process must have {@code android.permission.READ_RUNTIME_PROFILE} permission.
Justin Klaassen4d01eea2018-04-03 23:21:57 -0400121 *
122 * @param profileType can be either {@link ArtManager#PROFILE_APPS}
123 * or {@link ArtManager#PROFILE_BOOT_IMAGE}
Justin Klaassen98fe7812018-01-03 13:39:41 -0500124 */
125 @RequiresPermission(android.Manifest.permission.READ_RUNTIME_PROFILES)
Justin Klaassen4d01eea2018-04-03 23:21:57 -0400126 public boolean isRuntimeProfilingEnabled(@ProfileType int profileType) {
Justin Klaassen98fe7812018-01-03 13:39:41 -0500127 try {
Justin Klaassen4d01eea2018-04-03 23:21:57 -0400128 return mArtManager.isRuntimeProfilingEnabled(profileType);
Justin Klaassen98fe7812018-01-03 13:39:41 -0500129 } catch (RemoteException e) {
130 e.rethrowAsRuntimeException();
131 }
132 return false;
133 }
134
135 /**
136 * Callback used for retrieving runtime profiles.
137 */
138 public abstract static class SnapshotRuntimeProfileCallback {
139 /**
140 * Called when the profile snapshot finished with success.
141 *
142 * @param profileReadFd the file descriptor that can be used to read the profile. Note that
143 * the file might be empty (which is valid profile).
144 */
145 public abstract void onSuccess(ParcelFileDescriptor profileReadFd);
146
147 /**
148 * Called when the profile snapshot finished with an error.
149 *
150 * @param errCode the error code {@see SNAPSHOT_FAILED_PACKAGE_NOT_FOUND,
151 * SNAPSHOT_FAILED_CODE_PATH_NOT_FOUND, SNAPSHOT_FAILED_INTERNAL_ERROR}.
152 */
153 public abstract void onError(int errCode);
154 }
155
156 private static class SnapshotRuntimeProfileCallbackDelegate
Justin Klaassen4d01eea2018-04-03 23:21:57 -0400157 extends android.content.pm.dex.ISnapshotRuntimeProfileCallback.Stub {
Justin Klaassen98fe7812018-01-03 13:39:41 -0500158 private final ArtManager.SnapshotRuntimeProfileCallback mCallback;
Justin Klaassen4d01eea2018-04-03 23:21:57 -0400159 private final Executor mExecutor;
Justin Klaassen98fe7812018-01-03 13:39:41 -0500160
161 private SnapshotRuntimeProfileCallbackDelegate(
Justin Klaassen4d01eea2018-04-03 23:21:57 -0400162 ArtManager.SnapshotRuntimeProfileCallback callback, Executor executor) {
Justin Klaassen98fe7812018-01-03 13:39:41 -0500163 mCallback = callback;
Justin Klaassen4d01eea2018-04-03 23:21:57 -0400164 mExecutor = executor;
Justin Klaassen98fe7812018-01-03 13:39:41 -0500165 }
166
167 @Override
Justin Klaassen4d01eea2018-04-03 23:21:57 -0400168 public void onSuccess(final ParcelFileDescriptor profileReadFd) {
169 mExecutor.execute(() -> mCallback.onSuccess(profileReadFd));
Justin Klaassen98fe7812018-01-03 13:39:41 -0500170 }
171
172 @Override
173 public void onError(int errCode) {
Justin Klaassen4d01eea2018-04-03 23:21:57 -0400174 mExecutor.execute(() -> mCallback.onError(errCode));
Justin Klaassen98fe7812018-01-03 13:39:41 -0500175 }
176 }
Jeff Davidsona192cc22018-02-08 15:30:06 -0800177
178 /**
179 * Return the profile name for the given split. If {@code splitName} is null the
180 * method returns the profile name for the base apk.
181 *
182 * @hide
183 */
184 public static String getProfileName(String splitName) {
185 return splitName == null ? "primary.prof" : splitName + ".split.prof";
186 }
Justin Klaassen4d01eea2018-04-03 23:21:57 -0400187
188 /**
189 * Return the path to the current profile corresponding to given package and split.
190 *
191 * @hide
192 */
193 public static String getCurrentProfilePath(String packageName, int userId, String splitName) {
194 File profileDir = Environment.getDataProfilesDePackageDirectory(userId, packageName);
195 return new File(profileDir, getProfileName(splitName)).getAbsolutePath();
196 }
197
198 /**
199 * Return the snapshot profile file for the given package and profile name.
200 *
201 * KEEP in sync with installd dexopt.cpp.
202 * TODO(calin): inject the snapshot profile name from PM to avoid the dependency.
203 *
204 * @hide
205 */
206 public static File getProfileSnapshotFileForName(String packageName, String profileName) {
207 File profileDir = Environment.getDataRefProfilesDePackageDirectory(packageName);
208 return new File(profileDir, profileName + ".snapshot");
209 }
Justin Klaassen98fe7812018-01-03 13:39:41 -0500210}