blob: 8b7f321eb087fa5a343fc726b47076976aeab9e2 [file] [log] [blame]
Fyodor Kupolovca348512018-01-10 18:05:53 -08001/*
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 com.android.server;
18
Olivier Gaillardc17d2802018-11-19 17:08:17 +000019import static android.content.pm.PackageManager.MATCH_DIRECT_BOOT_AWARE;
20import static android.content.pm.PackageManager.MATCH_DIRECT_BOOT_UNAWARE;
21
Olivier Gaillard1d7f6152018-07-03 13:57:58 +010022import android.content.Context;
Fyodor Kupolovcf0fe2d2018-05-22 18:50:04 -070023import android.content.pm.PackageInfo;
24import android.content.pm.PackageManager;
Olivier Gaillardc17d2802018-11-19 17:08:17 +000025import android.content.pm.PackageManager.NameNotFoundException;
Olivier Gaillard1d7f6152018-07-03 13:57:58 +010026import android.database.ContentObserver;
27import android.net.Uri;
Fyodor Kupolovca348512018-01-10 18:05:53 -080028import android.os.Binder;
Olivier Gaillardc17d2802018-11-19 17:08:17 +000029import android.os.Process;
Fyodor Kupolovca348512018-01-10 18:05:53 -080030import android.os.SystemProperties;
Fyodor Kupolovcf0fe2d2018-05-22 18:50:04 -070031import android.os.UserHandle;
Olivier Gaillard1d7f6152018-07-03 13:57:58 +010032import android.provider.Settings;
Olivier Gaillard6f52d152018-07-25 12:13:12 +010033import android.util.ArrayMap;
Olivier Gaillardc17d2802018-11-19 17:08:17 +000034import android.util.ArraySet;
Olivier Gaillard1d7f6152018-07-03 13:57:58 +010035import android.util.KeyValueListParser;
Fyodor Kupolovca348512018-01-10 18:05:53 -080036import android.util.Slog;
37
Marcin Oczeretkoc4c45a82018-12-06 15:09:49 +000038import com.android.internal.os.AppIdToPackageMap;
Olivier Gaillard1d7f6152018-07-03 13:57:58 +010039import com.android.internal.os.BackgroundThread;
Fyodor Kupolovca348512018-01-10 18:05:53 -080040import com.android.internal.os.BinderCallsStats;
Olivier Gaillardc17d2802018-11-19 17:08:17 +000041import com.android.internal.os.BinderInternal;
Marcin Oczeretkoc80c81a2018-08-30 20:15:52 +010042import com.android.internal.os.CachedDeviceState;
Fyodor Kupolovca348512018-01-10 18:05:53 -080043
44import java.io.FileDescriptor;
45import java.io.PrintWriter;
Olivier Gaillard289ba402018-07-24 18:50:13 +010046import java.util.ArrayList;
Fyodor Kupolovcf0fe2d2018-05-22 18:50:04 -070047import java.util.List;
Fyodor Kupolovca348512018-01-10 18:05:53 -080048
49public class BinderCallsStatsService extends Binder {
50
51 private static final String TAG = "BinderCallsStatsService";
52
Fyodor Kupolov3f3af612018-04-18 17:26:43 -070053 private static final String PERSIST_SYS_BINDER_CALLS_DETAILED_TRACKING
54 = "persist.sys.binder_calls_detailed_tracking";
Fyodor Kupolovca348512018-01-10 18:05:53 -080055
Olivier Gaillardc17d2802018-11-19 17:08:17 +000056 /** Resolves the work source of an incoming binder transaction. */
Olivier Gaillard76c231d2018-12-05 12:52:08 +000057 static class AuthorizedWorkSourceProvider implements BinderInternal.WorkSourceProvider {
Olivier Gaillardc17d2802018-11-19 17:08:17 +000058 private ArraySet<Integer> mAppIdWhitelist;
59
Olivier Gaillard76c231d2018-12-05 12:52:08 +000060 AuthorizedWorkSourceProvider() {
Olivier Gaillardc17d2802018-11-19 17:08:17 +000061 mAppIdWhitelist = new ArraySet<>();
62 }
63
64 public int resolveWorkSourceUid() {
65 final int callingUid = getCallingUid();
66 final int appId = UserHandle.getAppId(callingUid);
67 if (mAppIdWhitelist.contains(appId)) {
68 final int workSource = getCallingWorkSourceUid();
69 final boolean isWorkSourceSet = workSource != Binder.UNSET_WORKSOURCE;
70 return isWorkSourceSet ? workSource : callingUid;
71 }
72 return callingUid;
73 }
74
75 public void systemReady(Context context) {
76 mAppIdWhitelist = createAppidWhitelist(context);
77 }
78
Marcin Oczeretkoc4c45a82018-12-06 15:09:49 +000079 public void dump(PrintWriter pw, AppIdToPackageMap packageMap) {
Olivier Gaillardc17d2802018-11-19 17:08:17 +000080 pw.println("AppIds of apps that can set the work source:");
81 final ArraySet<Integer> whitelist = mAppIdWhitelist;
82 for (Integer appId : whitelist) {
Marcin Oczeretkoc4c45a82018-12-06 15:09:49 +000083 pw.println("\t- " + packageMap.mapAppId(appId));
Olivier Gaillardc17d2802018-11-19 17:08:17 +000084 }
85 }
86
87 protected int getCallingUid() {
88 return Binder.getCallingUid();
89 }
90
91 protected int getCallingWorkSourceUid() {
92 return Binder.getCallingWorkSourceUid();
93 }
94
95 private ArraySet<Integer> createAppidWhitelist(Context context) {
96 // Use a local copy instead of mAppIdWhitelist to prevent concurrent read access.
97 final ArraySet<Integer> whitelist = new ArraySet<>();
98
99 // We trust our own process.
Olivier Gaillard182b4ed2018-12-07 11:18:52 +0000100 whitelist.add(UserHandle.getAppId(Process.myUid()));
Olivier Gaillardc17d2802018-11-19 17:08:17 +0000101 // We only need to initialize it once. UPDATE_DEVICE_STATS is a system permission.
102 final PackageManager pm = context.getPackageManager();
103 final String[] permissions = { android.Manifest.permission.UPDATE_DEVICE_STATS };
104 final int queryFlags = MATCH_DIRECT_BOOT_AWARE | MATCH_DIRECT_BOOT_UNAWARE;
105 final List<PackageInfo> packages =
106 pm.getPackagesHoldingPermissions(permissions, queryFlags);
107 final int packagesSize = packages.size();
108 for (int i = 0; i < packagesSize; i++) {
109 final PackageInfo pkgInfo = packages.get(i);
110 try {
111 final int uid = pm.getPackageUid(pkgInfo.packageName, queryFlags);
112 final int appId = UserHandle.getAppId(uid);
113 whitelist.add(appId);
114 } catch (NameNotFoundException e) {
115 Slog.e(TAG, "Cannot find uid for package name " + pkgInfo.packageName, e);
116 }
117 }
118 return whitelist;
119 }
120 }
121
Olivier Gaillard1d7f6152018-07-03 13:57:58 +0100122 /** Listens for flag changes. */
123 private static class SettingsObserver extends ContentObserver {
124 private static final String SETTINGS_ENABLED_KEY = "enabled";
125 private static final String SETTINGS_DETAILED_TRACKING_KEY = "detailed_tracking";
126 private static final String SETTINGS_UPLOAD_DATA_KEY = "upload_data";
127 private static final String SETTINGS_SAMPLING_INTERVAL_KEY = "sampling_interval";
Olivier Gaillard79490612018-11-30 16:22:23 +0000128 private static final String SETTINGS_MAX_CALL_STATS_KEY = "max_call_stats_count";
Fyodor Kupolovca348512018-01-10 18:05:53 -0800129
Olivier Gaillard289ba402018-07-24 18:50:13 +0100130 private boolean mEnabled;
Olivier Gaillard1d7f6152018-07-03 13:57:58 +0100131 private final Uri mUri = Settings.Global.getUriFor(Settings.Global.BINDER_CALLS_STATS);
132 private final Context mContext;
133 private final KeyValueListParser mParser = new KeyValueListParser(',');
Olivier Gaillard289ba402018-07-24 18:50:13 +0100134 private final BinderCallsStats mBinderCallsStats;
Olivier Gaillard76c231d2018-12-05 12:52:08 +0000135 private final AuthorizedWorkSourceProvider mWorkSourceProvider;
Olivier Gaillard1d7f6152018-07-03 13:57:58 +0100136
Olivier Gaillardc17d2802018-11-19 17:08:17 +0000137 SettingsObserver(Context context, BinderCallsStats binderCallsStats,
Olivier Gaillard76c231d2018-12-05 12:52:08 +0000138 AuthorizedWorkSourceProvider workSourceProvider) {
Olivier Gaillard1d7f6152018-07-03 13:57:58 +0100139 super(BackgroundThread.getHandler());
140 mContext = context;
141 context.getContentResolver().registerContentObserver(mUri, false, this,
142 UserHandle.USER_SYSTEM);
Olivier Gaillard289ba402018-07-24 18:50:13 +0100143 mBinderCallsStats = binderCallsStats;
Olivier Gaillard76c231d2018-12-05 12:52:08 +0000144 mWorkSourceProvider = workSourceProvider;
Olivier Gaillard1d7f6152018-07-03 13:57:58 +0100145 // Always kick once to ensure that we match current state
146 onChange();
Fyodor Kupolovca348512018-01-10 18:05:53 -0800147 }
Olivier Gaillard1d7f6152018-07-03 13:57:58 +0100148
149 @Override
150 public void onChange(boolean selfChange, Uri uri, int userId) {
151 if (mUri.equals(uri)) {
152 onChange();
153 }
154 }
155
156 public void onChange() {
157 // Do not overwrite the default set manually.
158 if (!SystemProperties.get(PERSIST_SYS_BINDER_CALLS_DETAILED_TRACKING).isEmpty()) {
159 return;
160 }
161
Olivier Gaillard1d7f6152018-07-03 13:57:58 +0100162 try {
163 mParser.setString(Settings.Global.getString(mContext.getContentResolver(),
164 Settings.Global.BINDER_CALLS_STATS));
165 } catch (IllegalArgumentException e) {
166 Slog.e(TAG, "Bad binder call stats settings", e);
167 }
Olivier Gaillard289ba402018-07-24 18:50:13 +0100168 mBinderCallsStats.setDetailedTracking(mParser.getBoolean(
Olivier Gaillard1d7f6152018-07-03 13:57:58 +0100169 SETTINGS_DETAILED_TRACKING_KEY, BinderCallsStats.DETAILED_TRACKING_DEFAULT));
Olivier Gaillard289ba402018-07-24 18:50:13 +0100170 mBinderCallsStats.setSamplingInterval(mParser.getInt(
Olivier Gaillard1d7f6152018-07-03 13:57:58 +0100171 SETTINGS_SAMPLING_INTERVAL_KEY,
172 BinderCallsStats.PERIODIC_SAMPLING_INTERVAL_DEFAULT));
Olivier Gaillard2c9d47a2018-12-03 21:49:39 +0000173 mBinderCallsStats.setMaxBinderCallStats(mParser.getInt(
Olivier Gaillard79490612018-11-30 16:22:23 +0000174 SETTINGS_MAX_CALL_STATS_KEY,
175 BinderCallsStats.MAX_BINDER_CALL_STATS_COUNT_DEFAULT));
Olivier Gaillard289ba402018-07-24 18:50:13 +0100176
177
178 final boolean enabled =
179 mParser.getBoolean(SETTINGS_ENABLED_KEY, BinderCallsStats.ENABLED_DEFAULT);
180 if (mEnabled != enabled) {
Olivier Gaillarddef1b902018-10-17 17:10:41 +0100181 if (enabled) {
Olivier Gaillard76c231d2018-12-05 12:52:08 +0000182 Binder.setObserver(mBinderCallsStats);
Olivier Gaillarddef1b902018-10-17 17:10:41 +0100183 Binder.setProxyTransactListener(
184 new Binder.PropagateWorkSourceTransactListener());
Olivier Gaillard76c231d2018-12-05 12:52:08 +0000185 Binder.setWorkSourceProvider(mWorkSourceProvider);
Olivier Gaillarddef1b902018-10-17 17:10:41 +0100186 } else {
187 Binder.setObserver(null);
188 Binder.setProxyTransactListener(null);
Olivier Gaillard76c231d2018-12-05 12:52:08 +0000189 Binder.setWorkSourceProvider(Binder::getCallingUid);
Olivier Gaillarddef1b902018-10-17 17:10:41 +0100190 }
Olivier Gaillard289ba402018-07-24 18:50:13 +0100191 mEnabled = enabled;
192 mBinderCallsStats.reset();
Marcin Oczeretko772e8f22018-11-21 13:00:32 +0000193 mBinderCallsStats.setAddDebugEntries(enabled);
Olivier Gaillard289ba402018-07-24 18:50:13 +0100194 }
195 }
196 }
197
198 /**
199 * @hide Only for use within the system server.
200 */
201 public static class Internal {
202 private final BinderCallsStats mBinderCallsStats;
203
204 Internal(BinderCallsStats binderCallsStats) {
205 this.mBinderCallsStats = binderCallsStats;
206 }
207
Olivier Gaillardd25f7a82018-09-12 14:28:48 +0100208 /** @see BinderCallsStats#reset */
209 public void reset() {
210 mBinderCallsStats.reset();
211 }
212
213 /**
214 * @see BinderCallsStats#getExportedCallStats.
215 *
216 * Note that binder calls stats will be reset by statsd every time
217 * the data is exported.
218 */
Olivier Gaillard289ba402018-07-24 18:50:13 +0100219 public ArrayList<BinderCallsStats.ExportedCallStat> getExportedCallStats() {
220 return mBinderCallsStats.getExportedCallStats();
Olivier Gaillard1d7f6152018-07-03 13:57:58 +0100221 }
Olivier Gaillard6f52d152018-07-25 12:13:12 +0100222
Olivier Gaillardd25f7a82018-09-12 14:28:48 +0100223 /** @see BinderCallsStats#getExportedExceptionStats */
Olivier Gaillard6f52d152018-07-25 12:13:12 +0100224 public ArrayMap<String, Integer> getExportedExceptionStats() {
225 return mBinderCallsStats.getExportedExceptionStats();
226 }
Olivier Gaillard1d7f6152018-07-03 13:57:58 +0100227 }
228
229 public static class LifeCycle extends SystemService {
230 private BinderCallsStatsService mService;
Olivier Gaillard86714d12018-08-01 15:05:36 +0100231 private BinderCallsStats mBinderCallsStats;
Olivier Gaillard76c231d2018-12-05 12:52:08 +0000232 private AuthorizedWorkSourceProvider mWorkSourceProvider;
Olivier Gaillard1d7f6152018-07-03 13:57:58 +0100233
234 public LifeCycle(Context context) {
235 super(context);
236 }
237
238 @Override
239 public void onStart() {
Olivier Gaillard86714d12018-08-01 15:05:36 +0100240 mBinderCallsStats = new BinderCallsStats(new BinderCallsStats.Injector());
Olivier Gaillard76c231d2018-12-05 12:52:08 +0000241 mWorkSourceProvider = new AuthorizedWorkSourceProvider();
Olivier Gaillardc17d2802018-11-19 17:08:17 +0000242 mService = new BinderCallsStatsService(
Olivier Gaillard76c231d2018-12-05 12:52:08 +0000243 mBinderCallsStats, mWorkSourceProvider);
Olivier Gaillard86714d12018-08-01 15:05:36 +0100244 publishLocalService(Internal.class, new Internal(mBinderCallsStats));
Olivier Gaillard1d7f6152018-07-03 13:57:58 +0100245 publishBinderService("binder_calls_stats", mService);
246 boolean detailedTrackingEnabled = SystemProperties.getBoolean(
247 PERSIST_SYS_BINDER_CALLS_DETAILED_TRACKING, false);
248
249 if (detailedTrackingEnabled) {
250 Slog.i(TAG, "Enabled CPU usage tracking for binder calls. Controlled by "
251 + PERSIST_SYS_BINDER_CALLS_DETAILED_TRACKING
252 + " or via dumpsys binder_calls_stats --enable-detailed-tracking");
Olivier Gaillard86714d12018-08-01 15:05:36 +0100253 mBinderCallsStats.setDetailedTracking(true);
Olivier Gaillard1d7f6152018-07-03 13:57:58 +0100254 }
255 }
256
257 @Override
258 public void onBootPhase(int phase) {
259 if (SystemService.PHASE_SYSTEM_SERVICES_READY == phase) {
Marcin Oczeretkoc80c81a2018-08-30 20:15:52 +0100260 CachedDeviceState.Readonly deviceState = getLocalService(
261 CachedDeviceState.Readonly.class);
Marcin Oczeretkoc80c81a2018-08-30 20:15:52 +0100262 mBinderCallsStats.setDeviceState(deviceState);
Olivier Gaillardc17d2802018-11-19 17:08:17 +0000263 // It needs to be called before mService.systemReady to make sure the observer is
264 // initialized before installing it.
265 mWorkSourceProvider.systemReady(getContext());
266 mService.systemReady(getContext());
Olivier Gaillard1d7f6152018-07-03 13:57:58 +0100267 }
268 }
269 }
270
271 private SettingsObserver mSettingsObserver;
Olivier Gaillard289ba402018-07-24 18:50:13 +0100272 private final BinderCallsStats mBinderCallsStats;
Olivier Gaillard76c231d2018-12-05 12:52:08 +0000273 private final AuthorizedWorkSourceProvider mWorkSourceProvider;
Olivier Gaillard1d7f6152018-07-03 13:57:58 +0100274
Olivier Gaillard76c231d2018-12-05 12:52:08 +0000275 BinderCallsStatsService(BinderCallsStats binderCallsStats,
276 AuthorizedWorkSourceProvider workSourceProvider) {
Olivier Gaillard289ba402018-07-24 18:50:13 +0100277 mBinderCallsStats = binderCallsStats;
Olivier Gaillardc17d2802018-11-19 17:08:17 +0000278 mWorkSourceProvider = workSourceProvider;
Fyodor Kupolovca348512018-01-10 18:05:53 -0800279 }
280
Olivier Gaillard289ba402018-07-24 18:50:13 +0100281 public void systemReady(Context context) {
Olivier Gaillard76c231d2018-12-05 12:52:08 +0000282 mSettingsObserver = new SettingsObserver(context, mBinderCallsStats, mWorkSourceProvider);
Olivier Gaillard289ba402018-07-24 18:50:13 +0100283 }
284
285 public void reset() {
Fyodor Kupolov3f3af612018-04-18 17:26:43 -0700286 Slog.i(TAG, "Resetting stats");
Olivier Gaillard289ba402018-07-24 18:50:13 +0100287 mBinderCallsStats.reset();
Fyodor Kupolov3f3af612018-04-18 17:26:43 -0700288 }
289
Fyodor Kupolovca348512018-01-10 18:05:53 -0800290 @Override
291 protected void dump(FileDescriptor fd, PrintWriter pw, String[] args) {
Fyodor Kupolovcf0fe2d2018-05-22 18:50:04 -0700292 boolean verbose = false;
Fyodor Kupolov3f3af612018-04-18 17:26:43 -0700293 if (args != null) {
294 for (final String arg : args) {
Fyodor Kupolov32a71122018-04-25 10:41:01 -0700295 if ("-a".equals(arg)) {
Fyodor Kupolovcf0fe2d2018-05-22 18:50:04 -0700296 verbose = true;
Fyodor Kupolov32a71122018-04-25 10:41:01 -0700297 } else if ("--reset".equals(arg)) {
Fyodor Kupolov3f3af612018-04-18 17:26:43 -0700298 reset();
299 pw.println("binder_calls_stats reset.");
300 return;
Olivier Gaillard431cb4c2018-09-13 17:06:12 +0100301 } else if ("--enable".equals(arg)) {
Olivier Gaillard76c231d2018-12-05 12:52:08 +0000302 Binder.setObserver(mBinderCallsStats);
Olivier Gaillard431cb4c2018-09-13 17:06:12 +0100303 return;
304 } else if ("--disable".equals(arg)) {
305 Binder.setObserver(null);
306 return;
307 } else if ("--no-sampling".equals(arg)) {
308 mBinderCallsStats.setSamplingInterval(1);
309 return;
Fyodor Kupolov3f3af612018-04-18 17:26:43 -0700310 } else if ("--enable-detailed-tracking".equals(arg)) {
311 SystemProperties.set(PERSIST_SYS_BINDER_CALLS_DETAILED_TRACKING, "1");
Olivier Gaillard289ba402018-07-24 18:50:13 +0100312 mBinderCallsStats.setDetailedTracking(true);
Fyodor Kupolov3f3af612018-04-18 17:26:43 -0700313 pw.println("Detailed tracking enabled");
314 return;
315 } else if ("--disable-detailed-tracking".equals(arg)) {
316 SystemProperties.set(PERSIST_SYS_BINDER_CALLS_DETAILED_TRACKING, "");
Olivier Gaillard289ba402018-07-24 18:50:13 +0100317 mBinderCallsStats.setDetailedTracking(false);
Fyodor Kupolov3f3af612018-04-18 17:26:43 -0700318 pw.println("Detailed tracking disabled");
319 return;
Olivier Gaillardc17d2802018-11-19 17:08:17 +0000320 } else if ("--dump-worksource-provider".equals(arg)) {
Marcin Oczeretkoc4c45a82018-12-06 15:09:49 +0000321 mWorkSourceProvider.dump(pw, AppIdToPackageMap.getSnapshot());
Olivier Gaillardc17d2802018-11-19 17:08:17 +0000322 return;
Fyodor Kupolov3f3af612018-04-18 17:26:43 -0700323 } else if ("-h".equals(arg)) {
324 pw.println("binder_calls_stats commands:");
325 pw.println(" --reset: Reset stats");
Olivier Gaillard431cb4c2018-09-13 17:06:12 +0100326 pw.println(" --enable: Enable tracking binder calls");
327 pw.println(" --disable: Disables tracking binder calls");
328 pw.println(" --no-sampling: Tracks all calls");
Fyodor Kupolov3f3af612018-04-18 17:26:43 -0700329 pw.println(" --enable-detailed-tracking: Enables detailed tracking");
330 pw.println(" --disable-detailed-tracking: Disables detailed tracking");
331 return;
332 } else {
333 pw.println("Unknown option: " + arg);
Fyodor Kupolov3f3af612018-04-18 17:26:43 -0700334 }
335 }
336 }
Marcin Oczeretkoc4c45a82018-12-06 15:09:49 +0000337 mBinderCallsStats.dump(pw, AppIdToPackageMap.getSnapshot(), verbose);
Fyodor Kupolovcf0fe2d2018-05-22 18:50:04 -0700338 }
Fyodor Kupolovca348512018-01-10 18:05:53 -0800339}