blob: 26735576cdff9c76401491c5f1f93380bc76d84d [file] [log] [blame]
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08001/*
2 * Copyright (C) 2007-2008 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
Adam Lesinski182f73f2013-12-05 16:48:06 -080017package com.android.server.storage;
18
19import com.android.server.EventLogTags;
20import com.android.server.SystemService;
Fyodor Kupoloveeea67b2015-02-23 17:14:45 -080021import com.android.server.pm.InstructionSets;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080022import android.app.Notification;
23import android.app.NotificationManager;
24import android.app.PendingIntent;
25import android.content.ContentResolver;
26import android.content.Context;
27import android.content.Intent;
28import android.content.pm.IPackageDataObserver;
29import android.content.pm.IPackageManager;
Dianne Hackborn197a0c82012-07-12 14:46:04 -070030import android.content.pm.PackageManager;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080031import android.os.Binder;
Dianne Hackbornf882efa2012-04-11 16:04:12 -070032import android.os.Environment;
Jeff Sharkey4b496572012-04-19 14:17:03 -070033import android.os.FileObserver;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080034import android.os.Handler;
Adam Lesinski182f73f2013-12-05 16:48:06 -080035import android.os.IBinder;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080036import android.os.Message;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080037import android.os.RemoteException;
38import android.os.ServiceManager;
39import android.os.StatFs;
40import android.os.SystemClock;
41import android.os.SystemProperties;
Dianne Hackborn5ac72a22012-08-29 18:32:08 -070042import android.os.UserHandle;
Jeff Sharkeybe722152013-02-15 16:56:38 -080043import android.os.storage.StorageManager;
Doug Zongker43866e02010-01-07 12:09:54 -080044import android.provider.Settings;
Dianne Hackborn197a0c82012-07-12 14:46:04 -070045import android.text.format.Formatter;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080046import android.util.EventLog;
Joe Onorato8a9b2202010-02-26 18:56:32 -080047import android.util.Slog;
Dianne Hackborn197a0c82012-07-12 14:46:04 -070048import android.util.TimeUtils;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080049
Jeff Sharkeybe722152013-02-15 16:56:38 -080050import java.io.File;
51import java.io.FileDescriptor;
52import java.io.PrintWriter;
53
Brian Carlstroma39871e2014-09-29 13:44:04 -070054import dalvik.system.VMRuntime;
55
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080056/**
Doug Zongker43866e02010-01-07 12:09:54 -080057 * This class implements a service to monitor the amount of disk
58 * storage space on the device. If the free storage on device is less
59 * than a tunable threshold value (a secure settings parameter;
60 * default 10%) a low memory notification is displayed to alert the
61 * user. If the user clicks on the low memory notification the
62 * Application Manager application gets launched to let the user free
63 * storage space.
64 *
65 * Event log events: A low memory event with the free storage on
66 * device in bytes is logged to the event log when the device goes low
67 * on storage space. The amount of free storage on the device is
68 * periodically logged to the event log. The log interval is a secure
69 * settings parameter with a default value of 12 hours. When the free
70 * storage differential goes below a threshold (again a secure
71 * settings parameter with a default value of 2MB), the free memory is
72 * logged to the event log.
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080073 */
Adam Lesinski182f73f2013-12-05 16:48:06 -080074public class DeviceStorageMonitorService extends SystemService {
75 static final String TAG = "DeviceStorageMonitorService";
Jeff Sharkeybe722152013-02-15 16:56:38 -080076
Adam Lesinski182f73f2013-12-05 16:48:06 -080077 static final boolean DEBUG = false;
78 static final boolean localLOGV = false;
Jeff Sharkeybe722152013-02-15 16:56:38 -080079
Adam Lesinski182f73f2013-12-05 16:48:06 -080080 static final int DEVICE_MEMORY_WHAT = 1;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080081 private static final int MONITOR_INTERVAL = 1; //in minutes
82 private static final int LOW_MEMORY_NOTIFICATION_ID = 1;
Jeff Sharkeybe722152013-02-15 16:56:38 -080083
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080084 private static final int DEFAULT_FREE_STORAGE_LOG_INTERVAL_IN_MINUTES = 12*60; //in minutes
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080085 private static final long DEFAULT_DISK_FREE_CHANGE_REPORTING_THRESHOLD = 2 * 1024 * 1024; // 2MB
86 private static final long DEFAULT_CHECK_INTERVAL = MONITOR_INTERVAL*60*1000;
Jeff Sharkeybe722152013-02-15 16:56:38 -080087
Doug Zongker3161795b2009-10-07 15:14:03 -070088 private long mFreeMem; // on /data
Dianne Hackborn197a0c82012-07-12 14:46:04 -070089 private long mFreeMemAfterLastCacheClear; // on /data
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080090 private long mLastReportedFreeMem;
91 private long mLastReportedFreeMemTime;
Adam Lesinski182f73f2013-12-05 16:48:06 -080092 boolean mLowMemFlag=false;
Jake Hambybb371632010-08-23 18:16:48 -070093 private boolean mMemFullFlag=false;
Brian Carlstroma39871e2014-09-29 13:44:04 -070094 private final boolean mIsBootImageOnDisk;
Jeff Brownb880d882014-02-10 19:47:07 -080095 private final ContentResolver mResolver;
96 private final long mTotalMemory; // on /data
97 private final StatFs mDataFileStats;
98 private final StatFs mSystemFileStats;
99 private final StatFs mCacheFileStats;
Jeff Sharkeybe722152013-02-15 16:56:38 -0800100
101 private static final File DATA_PATH = Environment.getDataDirectory();
102 private static final File SYSTEM_PATH = Environment.getRootDirectory();
103 private static final File CACHE_PATH = Environment.getDownloadCacheDirectory();
104
Doug Zongker3161795b2009-10-07 15:14:03 -0700105 private long mThreadStartTime = -1;
Adam Lesinski182f73f2013-12-05 16:48:06 -0800106 boolean mClearSucceeded = false;
107 boolean mClearingCache;
Jeff Brownb880d882014-02-10 19:47:07 -0800108 private final Intent mStorageLowIntent;
109 private final Intent mStorageOkIntent;
110 private final Intent mStorageFullIntent;
111 private final Intent mStorageNotFullIntent;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800112 private CachePackageDataObserver mClearCacheObserver;
Adam Lesinski182f73f2013-12-05 16:48:06 -0800113 private CacheFileDeletedObserver mCacheFileDeletedObserver;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800114 private static final int _TRUE = 1;
115 private static final int _FALSE = 0;
Dianne Hackborn197a0c82012-07-12 14:46:04 -0700116 // This is the raw threshold that has been set at which we consider
117 // storage to be low.
Adam Lesinski182f73f2013-12-05 16:48:06 -0800118 long mMemLowThreshold;
Dianne Hackborn197a0c82012-07-12 14:46:04 -0700119 // This is the threshold at which we start trying to flush caches
120 // to get below the low threshold limit. It is less than the low
121 // threshold; we will allow storage to get a bit beyond the limit
122 // before flushing and checking if we are actually low.
123 private long mMemCacheStartTrimThreshold;
124 // This is the threshold that we try to get to when deleting cache
125 // files. This is greater than the low threshold so that we will flush
126 // more files than absolutely needed, to reduce the frequency that
127 // flushing takes place.
128 private long mMemCacheTrimToThreshold;
Jeff Sharkeybe722152013-02-15 16:56:38 -0800129 private long mMemFullThreshold;
Doug Zongker3161795b2009-10-07 15:14:03 -0700130
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800131 /**
132 * This string is used for ServiceManager access to this class.
133 */
Adam Lesinski182f73f2013-12-05 16:48:06 -0800134 static final String SERVICE = "devicestoragemonitor";
Doug Zongker3161795b2009-10-07 15:14:03 -0700135
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800136 /**
Doug Zongker3161795b2009-10-07 15:14:03 -0700137 * Handler that checks the amount of disk space on the device and sends a
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800138 * notification if the device runs low on disk space
139 */
Jeff Brownb880d882014-02-10 19:47:07 -0800140 private final Handler mHandler = new Handler() {
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800141 @Override
142 public void handleMessage(Message msg) {
Jake Hambybb371632010-08-23 18:16:48 -0700143 //don't handle an invalid message
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800144 if (msg.what != DEVICE_MEMORY_WHAT) {
Joe Onorato8a9b2202010-02-26 18:56:32 -0800145 Slog.e(TAG, "Will not process invalid message");
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800146 return;
147 }
148 checkMemory(msg.arg1 == _TRUE);
149 }
150 };
Doug Zongker3161795b2009-10-07 15:14:03 -0700151
Adam Lesinski182f73f2013-12-05 16:48:06 -0800152 private class CachePackageDataObserver extends IPackageDataObserver.Stub {
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800153 public void onRemoveCompleted(String packageName, boolean succeeded) {
154 mClearSucceeded = succeeded;
155 mClearingCache = false;
Joe Onorato8a9b2202010-02-26 18:56:32 -0800156 if(localLOGV) Slog.i(TAG, " Clear succeeded:"+mClearSucceeded
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800157 +", mClearingCache:"+mClearingCache+" Forcing memory check");
158 postCheckMemoryMsg(false, 0);
Doug Zongker3161795b2009-10-07 15:14:03 -0700159 }
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800160 }
Doug Zongker3161795b2009-10-07 15:14:03 -0700161
Adam Lesinski182f73f2013-12-05 16:48:06 -0800162 private void restatDataDir() {
Doug Zongker3161795b2009-10-07 15:14:03 -0700163 try {
Jeff Sharkeybe722152013-02-15 16:56:38 -0800164 mDataFileStats.restat(DATA_PATH.getAbsolutePath());
Doug Zongker3161795b2009-10-07 15:14:03 -0700165 mFreeMem = (long) mDataFileStats.getAvailableBlocks() *
166 mDataFileStats.getBlockSize();
167 } catch (IllegalArgumentException e) {
168 // use the old value of mFreeMem
169 }
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800170 // Allow freemem to be overridden by debug.freemem for testing
171 String debugFreeMem = SystemProperties.get("debug.freemem");
172 if (!"".equals(debugFreeMem)) {
173 mFreeMem = Long.parseLong(debugFreeMem);
174 }
Doug Zongker43866e02010-01-07 12:09:54 -0800175 // Read the log interval from secure settings
Jeff Sharkeybe722152013-02-15 16:56:38 -0800176 long freeMemLogInterval = Settings.Global.getLong(mResolver,
Jeff Sharkey625239a2012-09-26 22:03:49 -0700177 Settings.Global.SYS_FREE_STORAGE_LOG_INTERVAL,
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800178 DEFAULT_FREE_STORAGE_LOG_INTERVAL_IN_MINUTES)*60*1000;
179 //log the amount of free memory in event log
180 long currTime = SystemClock.elapsedRealtime();
Doug Zongker3161795b2009-10-07 15:14:03 -0700181 if((mLastReportedFreeMemTime == 0) ||
182 (currTime-mLastReportedFreeMemTime) >= freeMemLogInterval) {
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800183 mLastReportedFreeMemTime = currTime;
Doug Zongker3161795b2009-10-07 15:14:03 -0700184 long mFreeSystem = -1, mFreeCache = -1;
185 try {
Jeff Sharkeybe722152013-02-15 16:56:38 -0800186 mSystemFileStats.restat(SYSTEM_PATH.getAbsolutePath());
Doug Zongker3161795b2009-10-07 15:14:03 -0700187 mFreeSystem = (long) mSystemFileStats.getAvailableBlocks() *
188 mSystemFileStats.getBlockSize();
189 } catch (IllegalArgumentException e) {
190 // ignore; report -1
191 }
192 try {
Jeff Sharkeybe722152013-02-15 16:56:38 -0800193 mCacheFileStats.restat(CACHE_PATH.getAbsolutePath());
Doug Zongker3161795b2009-10-07 15:14:03 -0700194 mFreeCache = (long) mCacheFileStats.getAvailableBlocks() *
195 mCacheFileStats.getBlockSize();
196 } catch (IllegalArgumentException e) {
197 // ignore; report -1
198 }
Doug Zongkerab5c49c2009-12-04 10:31:43 -0800199 EventLog.writeEvent(EventLogTags.FREE_STORAGE_LEFT,
Doug Zongker3161795b2009-10-07 15:14:03 -0700200 mFreeMem, mFreeSystem, mFreeCache);
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800201 }
Doug Zongker43866e02010-01-07 12:09:54 -0800202 // Read the reporting threshold from secure settings
Jeff Sharkeybe722152013-02-15 16:56:38 -0800203 long threshold = Settings.Global.getLong(mResolver,
Jeff Sharkey625239a2012-09-26 22:03:49 -0700204 Settings.Global.DISK_FREE_CHANGE_REPORTING_THRESHOLD,
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800205 DEFAULT_DISK_FREE_CHANGE_REPORTING_THRESHOLD);
206 // If mFree changed significantly log the new value
207 long delta = mFreeMem - mLastReportedFreeMem;
208 if (delta > threshold || delta < -threshold) {
209 mLastReportedFreeMem = mFreeMem;
Doug Zongkerab5c49c2009-12-04 10:31:43 -0800210 EventLog.writeEvent(EventLogTags.FREE_STORAGE_CHANGED, mFreeMem);
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800211 }
212 }
Doug Zongker3161795b2009-10-07 15:14:03 -0700213
Adam Lesinski182f73f2013-12-05 16:48:06 -0800214 private void clearCache() {
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800215 if (mClearCacheObserver == null) {
216 // Lazy instantiation
217 mClearCacheObserver = new CachePackageDataObserver();
218 }
219 mClearingCache = true;
220 try {
Joe Onorato8a9b2202010-02-26 18:56:32 -0800221 if (localLOGV) Slog.i(TAG, "Clearing cache");
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800222 IPackageManager.Stub.asInterface(ServiceManager.getService("package")).
Dianne Hackborn197a0c82012-07-12 14:46:04 -0700223 freeStorageAndNotify(mMemCacheTrimToThreshold, mClearCacheObserver);
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800224 } catch (RemoteException e) {
Joe Onorato8a9b2202010-02-26 18:56:32 -0800225 Slog.w(TAG, "Failed to get handle for PackageManger Exception: "+e);
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800226 mClearingCache = false;
227 mClearSucceeded = false;
228 }
229 }
Doug Zongker3161795b2009-10-07 15:14:03 -0700230
Adam Lesinski182f73f2013-12-05 16:48:06 -0800231 void checkMemory(boolean checkCache) {
Doug Zongker3161795b2009-10-07 15:14:03 -0700232 //if the thread that was started to clear cache is still running do nothing till its
233 //finished clearing cache. Ideally this flag could be modified by clearCache
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800234 // and should be accessed via a lock but even if it does this test will fail now and
235 //hopefully the next time this flag will be set to the correct value.
236 if(mClearingCache) {
Joe Onorato8a9b2202010-02-26 18:56:32 -0800237 if(localLOGV) Slog.i(TAG, "Thread already running just skip");
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800238 //make sure the thread is not hung for too long
239 long diffTime = System.currentTimeMillis() - mThreadStartTime;
240 if(diffTime > (10*60*1000)) {
Joe Onorato8a9b2202010-02-26 18:56:32 -0800241 Slog.w(TAG, "Thread that clears cache file seems to run for ever");
Doug Zongker3161795b2009-10-07 15:14:03 -0700242 }
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800243 } else {
244 restatDataDir();
Joe Onorato8a9b2202010-02-26 18:56:32 -0800245 if (localLOGV) Slog.v(TAG, "freeMemory="+mFreeMem);
Doug Zongker3161795b2009-10-07 15:14:03 -0700246
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800247 //post intent to NotificationManager to display icon if necessary
Jake Hambybb371632010-08-23 18:16:48 -0700248 if (mFreeMem < mMemLowThreshold) {
Dianne Hackborn197a0c82012-07-12 14:46:04 -0700249 if (checkCache) {
250 // We are allowed to clear cache files at this point to
251 // try to get down below the limit, because this is not
252 // the initial call after a cache clear has been attempted.
253 // In this case we will try a cache clear if our free
254 // space has gone below the cache clear limit.
255 if (mFreeMem < mMemCacheStartTrimThreshold) {
256 // We only clear the cache if the free storage has changed
257 // a significant amount since the last time.
258 if ((mFreeMemAfterLastCacheClear-mFreeMem)
259 >= ((mMemLowThreshold-mMemCacheStartTrimThreshold)/4)) {
260 // See if clearing cache helps
261 // Note that clearing cache is asynchronous and so we do a
262 // memory check again once the cache has been cleared.
263 mThreadStartTime = System.currentTimeMillis();
264 mClearSucceeded = false;
265 clearCache();
266 }
267 }
268 } else {
269 // This is a call from after clearing the cache. Note
270 // the amount of free storage at this point.
271 mFreeMemAfterLastCacheClear = mFreeMem;
272 if (!mLowMemFlag) {
273 // We tried to clear the cache, but that didn't get us
274 // below the low storage limit. Tell the user.
Joe Onorato8a9b2202010-02-26 18:56:32 -0800275 Slog.i(TAG, "Running low on memory. Sending notification");
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800276 sendNotification();
277 mLowMemFlag = true;
Dianne Hackborn197a0c82012-07-12 14:46:04 -0700278 } else {
279 if (localLOGV) Slog.v(TAG, "Running low on memory " +
280 "notification already sent. do nothing");
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800281 }
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800282 }
283 } else {
Dianne Hackborn197a0c82012-07-12 14:46:04 -0700284 mFreeMemAfterLastCacheClear = mFreeMem;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800285 if (mLowMemFlag) {
Joe Onorato8a9b2202010-02-26 18:56:32 -0800286 Slog.i(TAG, "Memory available. Cancelling notification");
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800287 cancelNotification();
288 mLowMemFlag = false;
289 }
290 }
Brian Carlstroma39871e2014-09-29 13:44:04 -0700291 if (!mLowMemFlag && !mIsBootImageOnDisk) {
292 Slog.i(TAG, "No boot image on disk due to lack of space. Sending notification");
293 sendNotification();
294 }
Jake Hambybb371632010-08-23 18:16:48 -0700295 if (mFreeMem < mMemFullThreshold) {
296 if (!mMemFullFlag) {
297 sendFullNotification();
298 mMemFullFlag = true;
299 }
300 } else {
301 if (mMemFullFlag) {
302 cancelFullNotification();
303 mMemFullFlag = false;
304 }
305 }
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800306 }
Joe Onorato8a9b2202010-02-26 18:56:32 -0800307 if(localLOGV) Slog.i(TAG, "Posting Message again");
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800308 //keep posting messages to itself periodically
309 postCheckMemoryMsg(true, DEFAULT_CHECK_INTERVAL);
310 }
Doug Zongker3161795b2009-10-07 15:14:03 -0700311
Adam Lesinski182f73f2013-12-05 16:48:06 -0800312 void postCheckMemoryMsg(boolean clearCache, long delay) {
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800313 // Remove queued messages
314 mHandler.removeMessages(DEVICE_MEMORY_WHAT);
315 mHandler.sendMessageDelayed(mHandler.obtainMessage(DEVICE_MEMORY_WHAT,
316 clearCache ?_TRUE : _FALSE, 0),
317 delay);
318 }
Doug Zongker3161795b2009-10-07 15:14:03 -0700319
Jeff Brownb880d882014-02-10 19:47:07 -0800320 public DeviceStorageMonitorService(Context context) {
321 super(context);
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800322 mLastReportedFreeMemTime = 0;
Adam Lesinski182f73f2013-12-05 16:48:06 -0800323 mResolver = context.getContentResolver();
Brian Carlstroma39871e2014-09-29 13:44:04 -0700324 mIsBootImageOnDisk = isBootImageOnDisk();
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800325 //create StatFs object
Jeff Sharkeybe722152013-02-15 16:56:38 -0800326 mDataFileStats = new StatFs(DATA_PATH.getAbsolutePath());
327 mSystemFileStats = new StatFs(SYSTEM_PATH.getAbsolutePath());
328 mCacheFileStats = new StatFs(CACHE_PATH.getAbsolutePath());
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800329 //initialize total storage on device
Dianne Hackborn197a0c82012-07-12 14:46:04 -0700330 mTotalMemory = (long)mDataFileStats.getBlockCount() *
331 mDataFileStats.getBlockSize();
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800332 mStorageLowIntent = new Intent(Intent.ACTION_DEVICE_STORAGE_LOW);
Jeff Hamilton4b330922010-06-04 15:16:06 -0500333 mStorageLowIntent.addFlags(Intent.FLAG_RECEIVER_REGISTERED_ONLY_BEFORE_BOOT);
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800334 mStorageOkIntent = new Intent(Intent.ACTION_DEVICE_STORAGE_OK);
Jeff Hamilton4b330922010-06-04 15:16:06 -0500335 mStorageOkIntent.addFlags(Intent.FLAG_RECEIVER_REGISTERED_ONLY_BEFORE_BOOT);
Jake Hambybb371632010-08-23 18:16:48 -0700336 mStorageFullIntent = new Intent(Intent.ACTION_DEVICE_STORAGE_FULL);
337 mStorageFullIntent.addFlags(Intent.FLAG_RECEIVER_REGISTERED_ONLY_BEFORE_BOOT);
338 mStorageNotFullIntent = new Intent(Intent.ACTION_DEVICE_STORAGE_NOT_FULL);
339 mStorageNotFullIntent.addFlags(Intent.FLAG_RECEIVER_REGISTERED_ONLY_BEFORE_BOOT);
Adam Lesinski182f73f2013-12-05 16:48:06 -0800340 }
Jeff Sharkeybe722152013-02-15 16:56:38 -0800341
Brian Carlstroma39871e2014-09-29 13:44:04 -0700342 private static boolean isBootImageOnDisk() {
Fyodor Kupoloveeea67b2015-02-23 17:14:45 -0800343 for (String instructionSet : InstructionSets.getAllDexCodeInstructionSets()) {
Brian Carlstroma39871e2014-09-29 13:44:04 -0700344 if (!VMRuntime.isBootClassPathOnDisk(instructionSet)) {
345 return false;
346 }
347 }
348 return true;
349 }
350
Jeff Brownb880d882014-02-10 19:47:07 -0800351 /**
352 * Initializes the disk space threshold value and posts an empty message to
353 * kickstart the process.
354 */
Adam Lesinski182f73f2013-12-05 16:48:06 -0800355 @Override
356 public void onStart() {
Jake Hambybb371632010-08-23 18:16:48 -0700357 // cache storage thresholds
Adam Lesinski182f73f2013-12-05 16:48:06 -0800358 final StorageManager sm = StorageManager.from(getContext());
Jeff Sharkeybe722152013-02-15 16:56:38 -0800359 mMemLowThreshold = sm.getStorageLowBytes(DATA_PATH);
360 mMemFullThreshold = sm.getStorageFullBytes(DATA_PATH);
361
Dianne Hackborn197a0c82012-07-12 14:46:04 -0700362 mMemCacheStartTrimThreshold = ((mMemLowThreshold*3)+mMemFullThreshold)/4;
363 mMemCacheTrimToThreshold = mMemLowThreshold
364 + ((mMemLowThreshold-mMemCacheStartTrimThreshold)*2);
365 mFreeMemAfterLastCacheClear = mTotalMemory;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800366 checkMemory(true);
Jeff Sharkey4b496572012-04-19 14:17:03 -0700367
368 mCacheFileDeletedObserver = new CacheFileDeletedObserver();
369 mCacheFileDeletedObserver.startWatching();
Adam Lesinski182f73f2013-12-05 16:48:06 -0800370
371 publishBinderService(SERVICE, mRemoteService);
372 publishLocalService(DeviceStorageMonitorInternal.class, mLocalService);
373 }
374
375 private final DeviceStorageMonitorInternal mLocalService = new DeviceStorageMonitorInternal() {
376 @Override
377 public void checkMemory() {
378 // force an early check
379 postCheckMemoryMsg(true, 0);
380 }
381
382 @Override
383 public boolean isMemoryLow() {
Brian Carlstroma39871e2014-09-29 13:44:04 -0700384 return mLowMemFlag || !mIsBootImageOnDisk;
Adam Lesinski182f73f2013-12-05 16:48:06 -0800385 }
386
387 @Override
388 public long getMemoryLowThreshold() {
389 return mMemLowThreshold;
390 }
391 };
392
393 private final IBinder mRemoteService = new Binder() {
394 @Override
395 protected void dump(FileDescriptor fd, PrintWriter pw, String[] args) {
396 if (getContext().checkCallingOrSelfPermission(android.Manifest.permission.DUMP)
397 != PackageManager.PERMISSION_GRANTED) {
398
399 pw.println("Permission Denial: can't dump " + SERVICE + " from from pid="
400 + Binder.getCallingPid()
401 + ", uid=" + Binder.getCallingUid());
402 return;
403 }
404
405 dumpImpl(pw);
406 }
407 };
408
409 void dumpImpl(PrintWriter pw) {
410 final Context context = getContext();
411
412 pw.println("Current DeviceStorageMonitor state:");
413
414 pw.print(" mFreeMem="); pw.print(Formatter.formatFileSize(context, mFreeMem));
415 pw.print(" mTotalMemory=");
416 pw.println(Formatter.formatFileSize(context, mTotalMemory));
417
418 pw.print(" mFreeMemAfterLastCacheClear=");
419 pw.println(Formatter.formatFileSize(context, mFreeMemAfterLastCacheClear));
420
421 pw.print(" mLastReportedFreeMem=");
422 pw.print(Formatter.formatFileSize(context, mLastReportedFreeMem));
423 pw.print(" mLastReportedFreeMemTime=");
424 TimeUtils.formatDuration(mLastReportedFreeMemTime, SystemClock.elapsedRealtime(), pw);
425 pw.println();
426
427 pw.print(" mLowMemFlag="); pw.print(mLowMemFlag);
428 pw.print(" mMemFullFlag="); pw.println(mMemFullFlag);
Brian Carlstroma39871e2014-09-29 13:44:04 -0700429 pw.print(" mIsBootImageOnDisk="); pw.print(mIsBootImageOnDisk);
Adam Lesinski182f73f2013-12-05 16:48:06 -0800430
431 pw.print(" mClearSucceeded="); pw.print(mClearSucceeded);
432 pw.print(" mClearingCache="); pw.println(mClearingCache);
433
434 pw.print(" mMemLowThreshold=");
435 pw.print(Formatter.formatFileSize(context, mMemLowThreshold));
436 pw.print(" mMemFullThreshold=");
437 pw.println(Formatter.formatFileSize(context, mMemFullThreshold));
438
439 pw.print(" mMemCacheStartTrimThreshold=");
440 pw.print(Formatter.formatFileSize(context, mMemCacheStartTrimThreshold));
441 pw.print(" mMemCacheTrimToThreshold=");
442 pw.println(Formatter.formatFileSize(context, mMemCacheTrimToThreshold));
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800443 }
Doug Zongker3161795b2009-10-07 15:14:03 -0700444
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800445 /**
446 * This method sends a notification to NotificationManager to display
447 * an error dialog indicating low disk space and launch the Installer
448 * application
449 */
Adam Lesinski182f73f2013-12-05 16:48:06 -0800450 private void sendNotification() {
451 final Context context = getContext();
Joe Onorato8a9b2202010-02-26 18:56:32 -0800452 if(localLOGV) Slog.i(TAG, "Sending low memory notification");
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800453 //log the event to event log with the amount of free storage(in bytes) left on the device
Doug Zongkerab5c49c2009-12-04 10:31:43 -0800454 EventLog.writeEvent(EventLogTags.LOW_STORAGE, mFreeMem);
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800455 // Pack up the values and broadcast them to everyone
Dianne Hackbornf882efa2012-04-11 16:04:12 -0700456 Intent lowMemIntent = new Intent(Environment.isExternalStorageEmulated()
457 ? Settings.ACTION_INTERNAL_STORAGE_SETTINGS
458 : Intent.ACTION_MANAGE_PACKAGE_STORAGE);
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800459 lowMemIntent.putExtra("memory", mFreeMem);
460 lowMemIntent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
Doug Zongker3161795b2009-10-07 15:14:03 -0700461 NotificationManager mNotificationMgr =
Adam Lesinski182f73f2013-12-05 16:48:06 -0800462 (NotificationManager)context.getSystemService(
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800463 Context.NOTIFICATION_SERVICE);
Adam Lesinski182f73f2013-12-05 16:48:06 -0800464 CharSequence title = context.getText(
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800465 com.android.internal.R.string.low_internal_storage_view_title);
Brian Carlstroma39871e2014-09-29 13:44:04 -0700466 CharSequence details = context.getText(mIsBootImageOnDisk
467 ? com.android.internal.R.string.low_internal_storage_view_text
468 : com.android.internal.R.string.low_internal_storage_view_text_no_boot);
Adam Lesinski182f73f2013-12-05 16:48:06 -0800469 PendingIntent intent = PendingIntent.getActivityAsUser(context, 0, lowMemIntent, 0,
Dianne Hackborn50cdf7c32012-09-23 17:08:57 -0700470 null, UserHandle.CURRENT);
Brian Carlstroma39871e2014-09-29 13:44:04 -0700471 Notification notification = new Notification.Builder(context)
472 .setSmallIcon(com.android.internal.R.drawable.stat_notify_disk_full)
473 .setTicker(title)
Alan Viverette4a357cd2015-03-18 18:37:18 -0700474 .setColor(context.getColor(
Brian Carlstroma39871e2014-09-29 13:44:04 -0700475 com.android.internal.R.color.system_notification_accent_color))
476 .setContentTitle(title)
477 .setContentText(details)
478 .setContentIntent(intent)
479 .setStyle(new Notification.BigTextStyle()
480 .bigText(details))
481 .setVisibility(Notification.VISIBILITY_PUBLIC)
482 .setCategory(Notification.CATEGORY_SYSTEM)
483 .build();
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800484 notification.flags |= Notification.FLAG_NO_CLEAR;
Dianne Hackborn50cdf7c32012-09-23 17:08:57 -0700485 mNotificationMgr.notifyAsUser(null, LOW_MEMORY_NOTIFICATION_ID, notification,
486 UserHandle.ALL);
Adam Lesinski182f73f2013-12-05 16:48:06 -0800487 context.sendStickyBroadcastAsUser(mStorageLowIntent, UserHandle.ALL);
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800488 }
489
490 /**
491 * Cancels low storage notification and sends OK intent.
492 */
Adam Lesinski182f73f2013-12-05 16:48:06 -0800493 private void cancelNotification() {
494 final Context context = getContext();
Joe Onorato8a9b2202010-02-26 18:56:32 -0800495 if(localLOGV) Slog.i(TAG, "Canceling low memory notification");
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800496 NotificationManager mNotificationMgr =
Adam Lesinski182f73f2013-12-05 16:48:06 -0800497 (NotificationManager)context.getSystemService(
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800498 Context.NOTIFICATION_SERVICE);
499 //cancel notification since memory has been freed
Dianne Hackborn50cdf7c32012-09-23 17:08:57 -0700500 mNotificationMgr.cancelAsUser(null, LOW_MEMORY_NOTIFICATION_ID, UserHandle.ALL);
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800501
Adam Lesinski182f73f2013-12-05 16:48:06 -0800502 context.removeStickyBroadcastAsUser(mStorageLowIntent, UserHandle.ALL);
503 context.sendBroadcastAsUser(mStorageOkIntent, UserHandle.ALL);
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800504 }
Doug Zongker3161795b2009-10-07 15:14:03 -0700505
Jake Hambybb371632010-08-23 18:16:48 -0700506 /**
507 * Send a notification when storage is full.
508 */
Adam Lesinski182f73f2013-12-05 16:48:06 -0800509 private void sendFullNotification() {
Jake Hambybb371632010-08-23 18:16:48 -0700510 if(localLOGV) Slog.i(TAG, "Sending memory full notification");
Adam Lesinski182f73f2013-12-05 16:48:06 -0800511 getContext().sendStickyBroadcastAsUser(mStorageFullIntent, UserHandle.ALL);
Jake Hambybb371632010-08-23 18:16:48 -0700512 }
513
514 /**
515 * Cancels memory full notification and sends "not full" intent.
516 */
Adam Lesinski182f73f2013-12-05 16:48:06 -0800517 private void cancelFullNotification() {
Jake Hambybb371632010-08-23 18:16:48 -0700518 if(localLOGV) Slog.i(TAG, "Canceling memory full notification");
Adam Lesinski182f73f2013-12-05 16:48:06 -0800519 getContext().removeStickyBroadcastAsUser(mStorageFullIntent, UserHandle.ALL);
520 getContext().sendBroadcastAsUser(mStorageNotFullIntent, UserHandle.ALL);
Jake Hambybb371632010-08-23 18:16:48 -0700521 }
522
Adam Lesinski182f73f2013-12-05 16:48:06 -0800523 private static class CacheFileDeletedObserver extends FileObserver {
Jeff Sharkey4b496572012-04-19 14:17:03 -0700524 public CacheFileDeletedObserver() {
525 super(Environment.getDownloadCacheDirectory().getAbsolutePath(), FileObserver.DELETE);
526 }
527
528 @Override
529 public void onEvent(int event, String path) {
530 EventLogTags.writeCacheFileDeleted(path);
531 }
532 }
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800533}