blob: f2ce444a8b90229440bd892b537c3b95a00706a9 [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
Felipe Lemea94024a2019-11-14 16:16:16 -080022import android.app.ActivityThread;
Olivier Gaillard1d7f6152018-07-03 13:57:58 +010023import android.content.Context;
Fyodor Kupolovcf0fe2d2018-05-22 18:50:04 -070024import android.content.pm.PackageInfo;
25import android.content.pm.PackageManager;
Olivier Gaillardc17d2802018-11-19 17:08:17 +000026import android.content.pm.PackageManager.NameNotFoundException;
Olivier Gaillard1d7f6152018-07-03 13:57:58 +010027import android.database.ContentObserver;
28import android.net.Uri;
Fyodor Kupolovca348512018-01-10 18:05:53 -080029import android.os.Binder;
Olivier Gaillardc17d2802018-11-19 17:08:17 +000030import android.os.Process;
Fyodor Kupolovca348512018-01-10 18:05:53 -080031import android.os.SystemProperties;
Fyodor Kupolovcf0fe2d2018-05-22 18:50:04 -070032import android.os.UserHandle;
Olivier Gaillard1d7f6152018-07-03 13:57:58 +010033import android.provider.Settings;
Olivier Gaillard6f52d152018-07-25 12:13:12 +010034import android.util.ArrayMap;
Olivier Gaillardc17d2802018-11-19 17:08:17 +000035import android.util.ArraySet;
Olivier Gaillard1d7f6152018-07-03 13:57:58 +010036import android.util.KeyValueListParser;
Fyodor Kupolovca348512018-01-10 18:05:53 -080037import android.util.Slog;
38
Marcin Oczeretkoc4c45a82018-12-06 15:09:49 +000039import com.android.internal.os.AppIdToPackageMap;
Olivier Gaillard1d7f6152018-07-03 13:57:58 +010040import com.android.internal.os.BackgroundThread;
Fyodor Kupolovca348512018-01-10 18:05:53 -080041import com.android.internal.os.BinderCallsStats;
Olivier Gaillardc17d2802018-11-19 17:08:17 +000042import com.android.internal.os.BinderInternal;
Marcin Oczeretkoc80c81a2018-08-30 20:15:52 +010043import com.android.internal.os.CachedDeviceState;
Felipe Lemea94024a2019-11-14 16:16:16 -080044import com.android.internal.util.DumpUtils;
Fyodor Kupolovca348512018-01-10 18:05:53 -080045
46import java.io.FileDescriptor;
47import java.io.PrintWriter;
Olivier Gaillard289ba402018-07-24 18:50:13 +010048import java.util.ArrayList;
Fyodor Kupolovcf0fe2d2018-05-22 18:50:04 -070049import java.util.List;
Fyodor Kupolovca348512018-01-10 18:05:53 -080050
51public class BinderCallsStatsService extends Binder {
52
53 private static final String TAG = "BinderCallsStatsService";
Felipe Lemea94024a2019-11-14 16:16:16 -080054 private static final String SERVICE_NAME = "binder_calls_stats";
Fyodor Kupolovca348512018-01-10 18:05:53 -080055
Fyodor Kupolov3f3af612018-04-18 17:26:43 -070056 private static final String PERSIST_SYS_BINDER_CALLS_DETAILED_TRACKING
57 = "persist.sys.binder_calls_detailed_tracking";
Fyodor Kupolovca348512018-01-10 18:05:53 -080058
Olivier Gaillardc17d2802018-11-19 17:08:17 +000059 /** Resolves the work source of an incoming binder transaction. */
Olivier Gaillard76c231d2018-12-05 12:52:08 +000060 static class AuthorizedWorkSourceProvider implements BinderInternal.WorkSourceProvider {
Olivier Gaillardc17d2802018-11-19 17:08:17 +000061 private ArraySet<Integer> mAppIdWhitelist;
62
Olivier Gaillard76c231d2018-12-05 12:52:08 +000063 AuthorizedWorkSourceProvider() {
Olivier Gaillardc17d2802018-11-19 17:08:17 +000064 mAppIdWhitelist = new ArraySet<>();
65 }
66
Olivier Gaillardbab444a2019-01-30 17:11:40 +000067 public int resolveWorkSourceUid(int untrustedWorkSourceUid) {
Olivier Gaillardc17d2802018-11-19 17:08:17 +000068 final int callingUid = getCallingUid();
69 final int appId = UserHandle.getAppId(callingUid);
70 if (mAppIdWhitelist.contains(appId)) {
Olivier Gaillardbab444a2019-01-30 17:11:40 +000071 final int workSource = untrustedWorkSourceUid;
Olivier Gaillardc17d2802018-11-19 17:08:17 +000072 final boolean isWorkSourceSet = workSource != Binder.UNSET_WORKSOURCE;
73 return isWorkSourceSet ? workSource : callingUid;
74 }
75 return callingUid;
76 }
77
78 public void systemReady(Context context) {
79 mAppIdWhitelist = createAppidWhitelist(context);
80 }
81
Marcin Oczeretkoc4c45a82018-12-06 15:09:49 +000082 public void dump(PrintWriter pw, AppIdToPackageMap packageMap) {
Olivier Gaillardc17d2802018-11-19 17:08:17 +000083 pw.println("AppIds of apps that can set the work source:");
84 final ArraySet<Integer> whitelist = mAppIdWhitelist;
85 for (Integer appId : whitelist) {
Marcin Oczeretkoc4c45a82018-12-06 15:09:49 +000086 pw.println("\t- " + packageMap.mapAppId(appId));
Olivier Gaillardc17d2802018-11-19 17:08:17 +000087 }
88 }
89
90 protected int getCallingUid() {
91 return Binder.getCallingUid();
92 }
93
Olivier Gaillardc17d2802018-11-19 17:08:17 +000094 private ArraySet<Integer> createAppidWhitelist(Context context) {
95 // Use a local copy instead of mAppIdWhitelist to prevent concurrent read access.
96 final ArraySet<Integer> whitelist = new ArraySet<>();
97
98 // We trust our own process.
Olivier Gaillard182b4ed2018-12-07 11:18:52 +000099 whitelist.add(UserHandle.getAppId(Process.myUid()));
Olivier Gaillardc17d2802018-11-19 17:08:17 +0000100 // We only need to initialize it once. UPDATE_DEVICE_STATS is a system permission.
101 final PackageManager pm = context.getPackageManager();
102 final String[] permissions = { android.Manifest.permission.UPDATE_DEVICE_STATS };
103 final int queryFlags = MATCH_DIRECT_BOOT_AWARE | MATCH_DIRECT_BOOT_UNAWARE;
104 final List<PackageInfo> packages =
105 pm.getPackagesHoldingPermissions(permissions, queryFlags);
106 final int packagesSize = packages.size();
107 for (int i = 0; i < packagesSize; i++) {
108 final PackageInfo pkgInfo = packages.get(i);
109 try {
110 final int uid = pm.getPackageUid(pkgInfo.packageName, queryFlags);
111 final int appId = UserHandle.getAppId(uid);
112 whitelist.add(appId);
113 } catch (NameNotFoundException e) {
114 Slog.e(TAG, "Cannot find uid for package name " + pkgInfo.packageName, e);
115 }
116 }
117 return whitelist;
118 }
119 }
120
Olivier Gaillard1d7f6152018-07-03 13:57:58 +0100121 /** Listens for flag changes. */
122 private static class SettingsObserver extends ContentObserver {
123 private static final String SETTINGS_ENABLED_KEY = "enabled";
124 private static final String SETTINGS_DETAILED_TRACKING_KEY = "detailed_tracking";
125 private static final String SETTINGS_UPLOAD_DATA_KEY = "upload_data";
126 private static final String SETTINGS_SAMPLING_INTERVAL_KEY = "sampling_interval";
Olivier Gaillard36b80ca2019-02-11 11:41:39 +0000127 private static final String SETTINGS_TRACK_SCREEN_INTERACTIVE_KEY = "track_screen_state";
128 private static final String SETTINGS_TRACK_DIRECT_CALLING_UID_KEY = "track_calling_uid";
Olivier Gaillard79490612018-11-30 16:22:23 +0000129 private static final String SETTINGS_MAX_CALL_STATS_KEY = "max_call_stats_count";
Fyodor Kupolovca348512018-01-10 18:05:53 -0800130
Olivier Gaillard289ba402018-07-24 18:50:13 +0100131 private boolean mEnabled;
Olivier Gaillard1d7f6152018-07-03 13:57:58 +0100132 private final Uri mUri = Settings.Global.getUriFor(Settings.Global.BINDER_CALLS_STATS);
133 private final Context mContext;
134 private final KeyValueListParser mParser = new KeyValueListParser(',');
Olivier Gaillard289ba402018-07-24 18:50:13 +0100135 private final BinderCallsStats mBinderCallsStats;
Olivier Gaillard76c231d2018-12-05 12:52:08 +0000136 private final AuthorizedWorkSourceProvider mWorkSourceProvider;
Olivier Gaillard1d7f6152018-07-03 13:57:58 +0100137
Olivier Gaillardc17d2802018-11-19 17:08:17 +0000138 SettingsObserver(Context context, BinderCallsStats binderCallsStats,
Olivier Gaillard76c231d2018-12-05 12:52:08 +0000139 AuthorizedWorkSourceProvider workSourceProvider) {
Olivier Gaillard1d7f6152018-07-03 13:57:58 +0100140 super(BackgroundThread.getHandler());
141 mContext = context;
142 context.getContentResolver().registerContentObserver(mUri, false, this,
143 UserHandle.USER_SYSTEM);
Olivier Gaillard289ba402018-07-24 18:50:13 +0100144 mBinderCallsStats = binderCallsStats;
Olivier Gaillard76c231d2018-12-05 12:52:08 +0000145 mWorkSourceProvider = workSourceProvider;
Olivier Gaillard1d7f6152018-07-03 13:57:58 +0100146 // Always kick once to ensure that we match current state
147 onChange();
Fyodor Kupolovca348512018-01-10 18:05:53 -0800148 }
Olivier Gaillard1d7f6152018-07-03 13:57:58 +0100149
150 @Override
151 public void onChange(boolean selfChange, Uri uri, int userId) {
152 if (mUri.equals(uri)) {
153 onChange();
154 }
155 }
156
157 public void onChange() {
158 // Do not overwrite the default set manually.
159 if (!SystemProperties.get(PERSIST_SYS_BINDER_CALLS_DETAILED_TRACKING).isEmpty()) {
160 return;
161 }
162
Olivier Gaillard1d7f6152018-07-03 13:57:58 +0100163 try {
164 mParser.setString(Settings.Global.getString(mContext.getContentResolver(),
165 Settings.Global.BINDER_CALLS_STATS));
166 } catch (IllegalArgumentException e) {
167 Slog.e(TAG, "Bad binder call stats settings", e);
168 }
Olivier Gaillard289ba402018-07-24 18:50:13 +0100169 mBinderCallsStats.setDetailedTracking(mParser.getBoolean(
Olivier Gaillard1d7f6152018-07-03 13:57:58 +0100170 SETTINGS_DETAILED_TRACKING_KEY, BinderCallsStats.DETAILED_TRACKING_DEFAULT));
Olivier Gaillard289ba402018-07-24 18:50:13 +0100171 mBinderCallsStats.setSamplingInterval(mParser.getInt(
Olivier Gaillard1d7f6152018-07-03 13:57:58 +0100172 SETTINGS_SAMPLING_INTERVAL_KEY,
173 BinderCallsStats.PERIODIC_SAMPLING_INTERVAL_DEFAULT));
Olivier Gaillard2c9d47a2018-12-03 21:49:39 +0000174 mBinderCallsStats.setMaxBinderCallStats(mParser.getInt(
Olivier Gaillard79490612018-11-30 16:22:23 +0000175 SETTINGS_MAX_CALL_STATS_KEY,
176 BinderCallsStats.MAX_BINDER_CALL_STATS_COUNT_DEFAULT));
Olivier Gaillard36b80ca2019-02-11 11:41:39 +0000177 mBinderCallsStats.setTrackScreenInteractive(
178 mParser.getBoolean(SETTINGS_TRACK_SCREEN_INTERACTIVE_KEY,
179 BinderCallsStats.DEFAULT_TRACK_SCREEN_INTERACTIVE));
180 mBinderCallsStats.setTrackDirectCallerUid(
181 mParser.getBoolean(SETTINGS_TRACK_DIRECT_CALLING_UID_KEY,
182 BinderCallsStats.DEFAULT_TRACK_DIRECT_CALLING_UID));
Olivier Gaillard289ba402018-07-24 18:50:13 +0100183
184
185 final boolean enabled =
186 mParser.getBoolean(SETTINGS_ENABLED_KEY, BinderCallsStats.ENABLED_DEFAULT);
187 if (mEnabled != enabled) {
Olivier Gaillarddef1b902018-10-17 17:10:41 +0100188 if (enabled) {
Olivier Gaillard76c231d2018-12-05 12:52:08 +0000189 Binder.setObserver(mBinderCallsStats);
Olivier Gaillarddef1b902018-10-17 17:10:41 +0100190 Binder.setProxyTransactListener(
191 new Binder.PropagateWorkSourceTransactListener());
Olivier Gaillard76c231d2018-12-05 12:52:08 +0000192 Binder.setWorkSourceProvider(mWorkSourceProvider);
Olivier Gaillarddef1b902018-10-17 17:10:41 +0100193 } else {
194 Binder.setObserver(null);
195 Binder.setProxyTransactListener(null);
Olivier Gaillardbab444a2019-01-30 17:11:40 +0000196 Binder.setWorkSourceProvider((x) -> Binder.getCallingUid());
Olivier Gaillarddef1b902018-10-17 17:10:41 +0100197 }
Olivier Gaillard289ba402018-07-24 18:50:13 +0100198 mEnabled = enabled;
199 mBinderCallsStats.reset();
Marcin Oczeretko772e8f22018-11-21 13:00:32 +0000200 mBinderCallsStats.setAddDebugEntries(enabled);
Olivier Gaillard289ba402018-07-24 18:50:13 +0100201 }
202 }
203 }
204
205 /**
206 * @hide Only for use within the system server.
207 */
208 public static class Internal {
209 private final BinderCallsStats mBinderCallsStats;
210
211 Internal(BinderCallsStats binderCallsStats) {
212 this.mBinderCallsStats = binderCallsStats;
213 }
214
Olivier Gaillardd25f7a82018-09-12 14:28:48 +0100215 /** @see BinderCallsStats#reset */
216 public void reset() {
217 mBinderCallsStats.reset();
218 }
219
220 /**
221 * @see BinderCallsStats#getExportedCallStats.
222 *
223 * Note that binder calls stats will be reset by statsd every time
224 * the data is exported.
225 */
Olivier Gaillard289ba402018-07-24 18:50:13 +0100226 public ArrayList<BinderCallsStats.ExportedCallStat> getExportedCallStats() {
227 return mBinderCallsStats.getExportedCallStats();
Olivier Gaillard1d7f6152018-07-03 13:57:58 +0100228 }
Olivier Gaillard6f52d152018-07-25 12:13:12 +0100229
Olivier Gaillardd25f7a82018-09-12 14:28:48 +0100230 /** @see BinderCallsStats#getExportedExceptionStats */
Olivier Gaillard6f52d152018-07-25 12:13:12 +0100231 public ArrayMap<String, Integer> getExportedExceptionStats() {
232 return mBinderCallsStats.getExportedExceptionStats();
233 }
Olivier Gaillard1d7f6152018-07-03 13:57:58 +0100234 }
235
236 public static class LifeCycle extends SystemService {
237 private BinderCallsStatsService mService;
Olivier Gaillard86714d12018-08-01 15:05:36 +0100238 private BinderCallsStats mBinderCallsStats;
Olivier Gaillard76c231d2018-12-05 12:52:08 +0000239 private AuthorizedWorkSourceProvider mWorkSourceProvider;
Olivier Gaillard1d7f6152018-07-03 13:57:58 +0100240
241 public LifeCycle(Context context) {
242 super(context);
243 }
244
245 @Override
246 public void onStart() {
Olivier Gaillard86714d12018-08-01 15:05:36 +0100247 mBinderCallsStats = new BinderCallsStats(new BinderCallsStats.Injector());
Olivier Gaillard76c231d2018-12-05 12:52:08 +0000248 mWorkSourceProvider = new AuthorizedWorkSourceProvider();
Olivier Gaillardc17d2802018-11-19 17:08:17 +0000249 mService = new BinderCallsStatsService(
Olivier Gaillard76c231d2018-12-05 12:52:08 +0000250 mBinderCallsStats, mWorkSourceProvider);
Olivier Gaillard86714d12018-08-01 15:05:36 +0100251 publishLocalService(Internal.class, new Internal(mBinderCallsStats));
Felipe Lemea94024a2019-11-14 16:16:16 -0800252 publishBinderService(SERVICE_NAME, mService);
Olivier Gaillard1d7f6152018-07-03 13:57:58 +0100253 boolean detailedTrackingEnabled = SystemProperties.getBoolean(
254 PERSIST_SYS_BINDER_CALLS_DETAILED_TRACKING, false);
255
256 if (detailedTrackingEnabled) {
257 Slog.i(TAG, "Enabled CPU usage tracking for binder calls. Controlled by "
258 + PERSIST_SYS_BINDER_CALLS_DETAILED_TRACKING
259 + " or via dumpsys binder_calls_stats --enable-detailed-tracking");
Olivier Gaillard86714d12018-08-01 15:05:36 +0100260 mBinderCallsStats.setDetailedTracking(true);
Olivier Gaillard1d7f6152018-07-03 13:57:58 +0100261 }
262 }
263
264 @Override
265 public void onBootPhase(int phase) {
266 if (SystemService.PHASE_SYSTEM_SERVICES_READY == phase) {
Marcin Oczeretkoc80c81a2018-08-30 20:15:52 +0100267 CachedDeviceState.Readonly deviceState = getLocalService(
268 CachedDeviceState.Readonly.class);
Marcin Oczeretkoc80c81a2018-08-30 20:15:52 +0100269 mBinderCallsStats.setDeviceState(deviceState);
Olivier Gaillardc17d2802018-11-19 17:08:17 +0000270 // It needs to be called before mService.systemReady to make sure the observer is
271 // initialized before installing it.
272 mWorkSourceProvider.systemReady(getContext());
273 mService.systemReady(getContext());
Olivier Gaillard1d7f6152018-07-03 13:57:58 +0100274 }
275 }
276 }
277
278 private SettingsObserver mSettingsObserver;
Olivier Gaillard289ba402018-07-24 18:50:13 +0100279 private final BinderCallsStats mBinderCallsStats;
Olivier Gaillard76c231d2018-12-05 12:52:08 +0000280 private final AuthorizedWorkSourceProvider mWorkSourceProvider;
Olivier Gaillard1d7f6152018-07-03 13:57:58 +0100281
Olivier Gaillard76c231d2018-12-05 12:52:08 +0000282 BinderCallsStatsService(BinderCallsStats binderCallsStats,
283 AuthorizedWorkSourceProvider workSourceProvider) {
Olivier Gaillard289ba402018-07-24 18:50:13 +0100284 mBinderCallsStats = binderCallsStats;
Olivier Gaillardc17d2802018-11-19 17:08:17 +0000285 mWorkSourceProvider = workSourceProvider;
Fyodor Kupolovca348512018-01-10 18:05:53 -0800286 }
287
Olivier Gaillard289ba402018-07-24 18:50:13 +0100288 public void systemReady(Context context) {
Olivier Gaillard76c231d2018-12-05 12:52:08 +0000289 mSettingsObserver = new SettingsObserver(context, mBinderCallsStats, mWorkSourceProvider);
Olivier Gaillard289ba402018-07-24 18:50:13 +0100290 }
291
292 public void reset() {
Fyodor Kupolov3f3af612018-04-18 17:26:43 -0700293 Slog.i(TAG, "Resetting stats");
Olivier Gaillard289ba402018-07-24 18:50:13 +0100294 mBinderCallsStats.reset();
Fyodor Kupolov3f3af612018-04-18 17:26:43 -0700295 }
296
Fyodor Kupolovca348512018-01-10 18:05:53 -0800297 @Override
298 protected void dump(FileDescriptor fd, PrintWriter pw, String[] args) {
Felipe Lemea94024a2019-11-14 16:16:16 -0800299 if (!DumpUtils.checkDumpAndUsageStatsPermission(ActivityThread.currentApplication(),
300 SERVICE_NAME, pw)) {
301 return;
302 }
303
Fyodor Kupolovcf0fe2d2018-05-22 18:50:04 -0700304 boolean verbose = false;
Fyodor Kupolov3f3af612018-04-18 17:26:43 -0700305 if (args != null) {
306 for (final String arg : args) {
Fyodor Kupolov32a71122018-04-25 10:41:01 -0700307 if ("-a".equals(arg)) {
Fyodor Kupolovcf0fe2d2018-05-22 18:50:04 -0700308 verbose = true;
Fyodor Kupolov32a71122018-04-25 10:41:01 -0700309 } else if ("--reset".equals(arg)) {
Fyodor Kupolov3f3af612018-04-18 17:26:43 -0700310 reset();
311 pw.println("binder_calls_stats reset.");
312 return;
Olivier Gaillard431cb4c2018-09-13 17:06:12 +0100313 } else if ("--enable".equals(arg)) {
Olivier Gaillard76c231d2018-12-05 12:52:08 +0000314 Binder.setObserver(mBinderCallsStats);
Olivier Gaillard431cb4c2018-09-13 17:06:12 +0100315 return;
316 } else if ("--disable".equals(arg)) {
317 Binder.setObserver(null);
318 return;
319 } else if ("--no-sampling".equals(arg)) {
320 mBinderCallsStats.setSamplingInterval(1);
321 return;
Fyodor Kupolov3f3af612018-04-18 17:26:43 -0700322 } else if ("--enable-detailed-tracking".equals(arg)) {
323 SystemProperties.set(PERSIST_SYS_BINDER_CALLS_DETAILED_TRACKING, "1");
Olivier Gaillard289ba402018-07-24 18:50:13 +0100324 mBinderCallsStats.setDetailedTracking(true);
Fyodor Kupolov3f3af612018-04-18 17:26:43 -0700325 pw.println("Detailed tracking enabled");
326 return;
327 } else if ("--disable-detailed-tracking".equals(arg)) {
328 SystemProperties.set(PERSIST_SYS_BINDER_CALLS_DETAILED_TRACKING, "");
Olivier Gaillard289ba402018-07-24 18:50:13 +0100329 mBinderCallsStats.setDetailedTracking(false);
Fyodor Kupolov3f3af612018-04-18 17:26:43 -0700330 pw.println("Detailed tracking disabled");
331 return;
Olivier Gaillardc17d2802018-11-19 17:08:17 +0000332 } else if ("--dump-worksource-provider".equals(arg)) {
Marcin Oczeretkoc4c45a82018-12-06 15:09:49 +0000333 mWorkSourceProvider.dump(pw, AppIdToPackageMap.getSnapshot());
Olivier Gaillardc17d2802018-11-19 17:08:17 +0000334 return;
Fyodor Kupolov3f3af612018-04-18 17:26:43 -0700335 } else if ("-h".equals(arg)) {
336 pw.println("binder_calls_stats commands:");
337 pw.println(" --reset: Reset stats");
Olivier Gaillard431cb4c2018-09-13 17:06:12 +0100338 pw.println(" --enable: Enable tracking binder calls");
339 pw.println(" --disable: Disables tracking binder calls");
340 pw.println(" --no-sampling: Tracks all calls");
Fyodor Kupolov3f3af612018-04-18 17:26:43 -0700341 pw.println(" --enable-detailed-tracking: Enables detailed tracking");
342 pw.println(" --disable-detailed-tracking: Disables detailed tracking");
343 return;
344 } else {
345 pw.println("Unknown option: " + arg);
Fyodor Kupolov3f3af612018-04-18 17:26:43 -0700346 }
347 }
348 }
Marcin Oczeretkoc4c45a82018-12-06 15:09:49 +0000349 mBinderCallsStats.dump(pw, AppIdToPackageMap.getSnapshot(), verbose);
Fyodor Kupolovcf0fe2d2018-05-22 18:50:04 -0700350 }
Fyodor Kupolovca348512018-01-10 18:05:53 -0800351}