1 /*
2  * Copyright (C) 2010 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 
17 package com.android.server;
18 
19 import android.content.Context;
20 import android.os.Binder;
21 import android.os.Environment;
22 import android.os.IBinder;
23 import android.os.IStoraged;
24 import android.os.RemoteException;
25 import android.os.ServiceManager;
26 import android.os.StatFs;
27 import android.os.SystemClock;
28 import android.os.storage.StorageManager;
29 import android.service.diskstats.DiskStatsAppSizesProto;
30 import android.service.diskstats.DiskStatsCachedValuesProto;
31 import android.service.diskstats.DiskStatsFreeSpaceProto;
32 import android.service.diskstats.DiskStatsServiceDumpProto;
33 import android.util.Log;
34 import android.util.Slog;
35 import android.util.proto.ProtoOutputStream;
36 
37 import com.android.internal.util.DumpUtils;
38 import com.android.server.storage.DiskStatsFileLogger;
39 import com.android.server.storage.DiskStatsLoggingService;
40 
41 import libcore.io.IoUtils;
42 
43 import org.json.JSONArray;
44 import org.json.JSONException;
45 import org.json.JSONObject;
46 
47 import java.io.File;
48 import java.io.FileDescriptor;
49 import java.io.FileOutputStream;
50 import java.io.IOException;
51 import java.io.PrintWriter;
52 
53 /**
54  * This service exists only as a "dumpsys" target which reports
55  * statistics about the status of the disk.
56  */
57 public class DiskStatsService extends Binder {
58     private static final String TAG = "DiskStatsService";
59     private static final String DISKSTATS_DUMP_FILE = "/data/system/diskstats_cache.json";
60 
61     private final Context mContext;
62 
DiskStatsService(Context context)63     public DiskStatsService(Context context) {
64         mContext = context;
65         DiskStatsLoggingService.schedule(context);
66     }
67 
68     @Override
dump(FileDescriptor fd, PrintWriter pw, String[] args)69     protected void dump(FileDescriptor fd, PrintWriter pw, String[] args) {
70         if (!DumpUtils.checkDumpAndUsageStatsPermission(mContext, TAG, pw)) return;
71 
72         // Run a quick-and-dirty performance test: write 512 bytes
73         byte[] junk = new byte[512];
74         for (int i = 0; i < junk.length; i++) junk[i] = (byte) i;  // Write nonzero bytes
75 
76         File tmp = new File(Environment.getDataDirectory(), "system/perftest.tmp");
77         FileOutputStream fos = null;
78         IOException error = null;
79 
80         long before = SystemClock.uptimeMillis();
81         try {
82             fos = new FileOutputStream(tmp);
83             fos.write(junk);
84         } catch (IOException e) {
85             error = e;
86         } finally {
87             try { if (fos != null) fos.close(); } catch (IOException e) {}
88         }
89 
90         long after = SystemClock.uptimeMillis();
91         if (tmp.exists()) tmp.delete();
92 
93         boolean protoFormat = hasOption(args, "--proto");
94         ProtoOutputStream proto = null;
95 
96         if (protoFormat) {
97             proto = new ProtoOutputStream(fd);
98             pw = null;
99             proto.write(DiskStatsServiceDumpProto.HAS_TEST_ERROR, error != null);
100             if (error != null) {
101                 proto.write(DiskStatsServiceDumpProto.ERROR_MESSAGE, error.toString());
102             } else {
103                 proto.write(DiskStatsServiceDumpProto.WRITE_512B_LATENCY_MILLIS, after - before);
104             }
105         } else {
106             if (error != null) {
107                 pw.print("Test-Error: ");
108                 pw.println(error.toString());
109             } else {
110                 pw.print("Latency: ");
111                 pw.print(after - before);
112                 pw.println("ms [512B Data Write]");
113             }
114         }
115 
116         if (protoFormat) {
117             reportDiskWriteSpeedProto(proto);
118         } else {
119             reportDiskWriteSpeed(pw);
120         }
121 
122         reportFreeSpace(Environment.getDataDirectory(), "Data", pw, proto,
123                 DiskStatsFreeSpaceProto.FOLDER_DATA);
124         reportFreeSpace(Environment.getDownloadCacheDirectory(), "Cache", pw, proto,
125                 DiskStatsFreeSpaceProto.FOLDER_CACHE);
126         reportFreeSpace(new File("/system"), "System", pw, proto,
127                 DiskStatsFreeSpaceProto.FOLDER_SYSTEM);
128 
129         boolean fileBased = StorageManager.isFileEncryptedNativeOnly();
130         boolean blockBased = fileBased ? false : StorageManager.isBlockEncrypted();
131         if (protoFormat) {
132             if (fileBased) {
133                 proto.write(DiskStatsServiceDumpProto.ENCRYPTION,
134                         DiskStatsServiceDumpProto.ENCRYPTION_FILE_BASED);
135             } else if (blockBased) {
136                 proto.write(DiskStatsServiceDumpProto.ENCRYPTION,
137                         DiskStatsServiceDumpProto.ENCRYPTION_FULL_DISK);
138             } else {
139                 proto.write(DiskStatsServiceDumpProto.ENCRYPTION,
140                         DiskStatsServiceDumpProto.ENCRYPTION_NONE);
141             }
142         } else if (fileBased) {
143             pw.println("File-based Encryption: true");
144         }
145 
146         if (protoFormat) {
147             reportCachedValuesProto(proto);
148         } else {
149             reportCachedValues(pw);
150         }
151 
152         if (protoFormat) {
153             proto.flush();
154         }
155         // TODO: Read /proc/yaffs and report interesting values;
156         // add configurable (through args) performance test parameters.
157     }
158 
reportFreeSpace(File path, String name, PrintWriter pw, ProtoOutputStream proto, int folderType)159     private void reportFreeSpace(File path, String name, PrintWriter pw,
160             ProtoOutputStream proto, int folderType) {
161         try {
162             StatFs statfs = new StatFs(path.getPath());
163             long bsize = statfs.getBlockSize();
164             long avail = statfs.getAvailableBlocks();
165             long total = statfs.getBlockCount();
166             if (bsize <= 0 || total <= 0) {
167                 throw new IllegalArgumentException(
168                         "Invalid stat: bsize=" + bsize + " avail=" + avail + " total=" + total);
169             }
170 
171             if (proto != null) {
172                 long freeSpaceToken = proto.start(DiskStatsServiceDumpProto.PARTITIONS_FREE_SPACE);
173                 proto.write(DiskStatsFreeSpaceProto.FOLDER, folderType);
174                 proto.write(DiskStatsFreeSpaceProto.AVAILABLE_SPACE_KB, avail * bsize / 1024);
175                 proto.write(DiskStatsFreeSpaceProto.TOTAL_SPACE_KB, total * bsize / 1024);
176                 proto.end(freeSpaceToken);
177             } else {
178                 pw.print(name);
179                 pw.print("-Free: ");
180                 pw.print(avail * bsize / 1024);
181                 pw.print("K / ");
182                 pw.print(total * bsize / 1024);
183                 pw.print("K total = ");
184                 pw.print(avail * 100 / total);
185                 pw.println("% free");
186             }
187         } catch (IllegalArgumentException e) {
188             if (proto != null) {
189                 // Empty proto
190             } else {
191                 pw.print(name);
192                 pw.print("-Error: ");
193                 pw.println(e.toString());
194             }
195             return;
196         }
197     }
198 
hasOption(String[] args, String arg)199     private boolean hasOption(String[] args, String arg) {
200         for (String opt : args) {
201             if (arg.equals(opt)) {
202                 return true;
203             }
204         }
205         return false;
206     }
207 
208     // If you change this method, make sure to modify the Proto version of this method as well.
reportCachedValues(PrintWriter pw)209     private void reportCachedValues(PrintWriter pw) {
210         try {
211             String jsonString = IoUtils.readFileAsString(DISKSTATS_DUMP_FILE);
212             JSONObject json = new JSONObject(jsonString);
213             pw.print("App Size: ");
214             pw.println(json.getLong(DiskStatsFileLogger.APP_SIZE_AGG_KEY));
215             pw.print("App Data Size: ");
216             pw.println(json.getLong(DiskStatsFileLogger.APP_DATA_SIZE_AGG_KEY));
217             pw.print("App Cache Size: ");
218             pw.println(json.getLong(DiskStatsFileLogger.APP_CACHE_AGG_KEY));
219             pw.print("Photos Size: ");
220             pw.println(json.getLong(DiskStatsFileLogger.PHOTOS_KEY));
221             pw.print("Videos Size: ");
222             pw.println(json.getLong(DiskStatsFileLogger.VIDEOS_KEY));
223             pw.print("Audio Size: ");
224             pw.println(json.getLong(DiskStatsFileLogger.AUDIO_KEY));
225             pw.print("Downloads Size: ");
226             pw.println(json.getLong(DiskStatsFileLogger.DOWNLOADS_KEY));
227             pw.print("System Size: ");
228             pw.println(json.getLong(DiskStatsFileLogger.SYSTEM_KEY));
229             pw.print("Other Size: ");
230             pw.println(json.getLong(DiskStatsFileLogger.MISC_KEY));
231             pw.print("Package Names: ");
232             pw.println(json.getJSONArray(DiskStatsFileLogger.PACKAGE_NAMES_KEY));
233             pw.print("App Sizes: ");
234             pw.println(json.getJSONArray(DiskStatsFileLogger.APP_SIZES_KEY));
235             pw.print("App Data Sizes: ");
236             pw.println(json.getJSONArray(DiskStatsFileLogger.APP_DATA_KEY));
237             pw.print("Cache Sizes: ");
238             pw.println(json.getJSONArray(DiskStatsFileLogger.APP_CACHES_KEY));
239         } catch (IOException | JSONException e) {
240             Log.w(TAG, "exception reading diskstats cache file", e);
241         }
242     }
243 
reportCachedValuesProto(ProtoOutputStream proto)244     private void reportCachedValuesProto(ProtoOutputStream proto) {
245         try {
246             String jsonString = IoUtils.readFileAsString(DISKSTATS_DUMP_FILE);
247             JSONObject json = new JSONObject(jsonString);
248             long cachedValuesToken = proto.start(DiskStatsServiceDumpProto.CACHED_FOLDER_SIZES);
249 
250             proto.write(DiskStatsCachedValuesProto.AGG_APPS_SIZE_KB,
251                     json.getLong(DiskStatsFileLogger.APP_SIZE_AGG_KEY));
252             proto.write(DiskStatsCachedValuesProto.AGG_APPS_DATA_SIZE_KB,
253                     json.getLong(DiskStatsFileLogger.APP_DATA_SIZE_AGG_KEY));
254             proto.write(DiskStatsCachedValuesProto.AGG_APPS_CACHE_SIZE_KB,
255                     json.getLong(DiskStatsFileLogger.APP_CACHE_AGG_KEY));
256             proto.write(DiskStatsCachedValuesProto.PHOTOS_SIZE_KB,
257                     json.getLong(DiskStatsFileLogger.PHOTOS_KEY));
258             proto.write(DiskStatsCachedValuesProto.VIDEOS_SIZE_KB,
259                     json.getLong(DiskStatsFileLogger.VIDEOS_KEY));
260             proto.write(DiskStatsCachedValuesProto.AUDIO_SIZE_KB,
261                     json.getLong(DiskStatsFileLogger.AUDIO_KEY));
262             proto.write(DiskStatsCachedValuesProto.DOWNLOADS_SIZE_KB,
263                     json.getLong(DiskStatsFileLogger.DOWNLOADS_KEY));
264             proto.write(DiskStatsCachedValuesProto.SYSTEM_SIZE_KB,
265                     json.getLong(DiskStatsFileLogger.SYSTEM_KEY));
266             proto.write(DiskStatsCachedValuesProto.OTHER_SIZE_KB,
267                     json.getLong(DiskStatsFileLogger.MISC_KEY));
268 
269             JSONArray packageNamesArray = json.getJSONArray(DiskStatsFileLogger.PACKAGE_NAMES_KEY);
270             JSONArray appSizesArray = json.getJSONArray(DiskStatsFileLogger.APP_SIZES_KEY);
271             JSONArray appDataSizesArray = json.getJSONArray(DiskStatsFileLogger.APP_DATA_KEY);
272             JSONArray cacheSizesArray = json.getJSONArray(DiskStatsFileLogger.APP_CACHES_KEY);
273             final int len = packageNamesArray.length();
274             if (len == appSizesArray.length()
275                     && len == appDataSizesArray.length()
276                     && len == cacheSizesArray.length()) {
277                 for (int i = 0; i < len; i++) {
278                     long packageToken = proto.start(DiskStatsCachedValuesProto.APP_SIZES);
279 
280                     proto.write(DiskStatsAppSizesProto.PACKAGE_NAME,
281                             packageNamesArray.getString(i));
282                     proto.write(DiskStatsAppSizesProto.APP_SIZE_KB, appSizesArray.getLong(i));
283                     proto.write(DiskStatsAppSizesProto.APP_DATA_SIZE_KB, appDataSizesArray.getLong(i));
284                     proto.write(DiskStatsAppSizesProto.CACHE_SIZE_KB, cacheSizesArray.getLong(i));
285 
286                     proto.end(packageToken);
287                 }
288             } else {
289                 Slog.wtf(TAG, "Sizes of packageNamesArray, appSizesArray, appDataSizesArray "
290                         + " and cacheSizesArray are not the same");
291             }
292 
293             proto.end(cachedValuesToken);
294         } catch (IOException | JSONException e) {
295             Log.w(TAG, "exception reading diskstats cache file", e);
296         }
297     }
298 
getRecentPerf()299     private int getRecentPerf() throws RemoteException, IllegalStateException {
300         IBinder binder = ServiceManager.getService("storaged");
301         if (binder == null) throw new IllegalStateException("storaged not found");
302         IStoraged storaged = IStoraged.Stub.asInterface(binder);
303         return storaged.getRecentPerf();
304     }
305 
306     // Keep reportDiskWriteSpeed and reportDiskWriteSpeedProto in sync
reportDiskWriteSpeed(PrintWriter pw)307     private void reportDiskWriteSpeed(PrintWriter pw) {
308         try {
309             long perf = getRecentPerf();
310             if (perf != 0) {
311                 pw.print("Recent Disk Write Speed (kB/s) = ");
312                 pw.println(perf);
313             } else {
314                 pw.println("Recent Disk Write Speed data unavailable");
315                 Log.w(TAG, "Recent Disk Write Speed data unavailable!");
316             }
317         } catch (RemoteException | IllegalStateException e) {
318             pw.println(e.toString());
319             Log.e(TAG, e.toString());
320         }
321     }
322 
reportDiskWriteSpeedProto(ProtoOutputStream proto)323     private void reportDiskWriteSpeedProto(ProtoOutputStream proto) {
324         try {
325             long perf = getRecentPerf();
326             if (perf != 0) {
327                 proto.write(DiskStatsServiceDumpProto.BENCHMARKED_WRITE_SPEED_KBPS, perf);
328             } else {
329                 Log.w(TAG, "Recent Disk Write Speed data unavailable!");
330             }
331         } catch (RemoteException | IllegalStateException e) {
332             Log.e(TAG, e.toString());
333         }
334     }
335 }
336