Merge "Better API for low disk space warnings."
This commit is contained in:
@@ -460,7 +460,8 @@ class ContextImpl extends Context {
|
|||||||
registerService(STORAGE_SERVICE, new ServiceFetcher() {
|
registerService(STORAGE_SERVICE, new ServiceFetcher() {
|
||||||
public Object createService(ContextImpl ctx) {
|
public Object createService(ContextImpl ctx) {
|
||||||
try {
|
try {
|
||||||
return new StorageManager(ctx.mMainThread.getHandler().getLooper());
|
return new StorageManager(
|
||||||
|
ctx.getContentResolver(), ctx.mMainThread.getHandler().getLooper());
|
||||||
} catch (RemoteException rex) {
|
} catch (RemoteException rex) {
|
||||||
Log.e(TAG, "Failed to create StorageManager", rex);
|
Log.e(TAG, "Failed to create StorageManager", rex);
|
||||||
return null;
|
return null;
|
||||||
|
|||||||
@@ -16,7 +16,9 @@
|
|||||||
|
|
||||||
package android.os.storage;
|
package android.os.storage;
|
||||||
|
|
||||||
import android.app.NotificationManager;
|
import static android.net.TrafficStats.MB_IN_BYTES;
|
||||||
|
|
||||||
|
import android.content.ContentResolver;
|
||||||
import android.content.Context;
|
import android.content.Context;
|
||||||
import android.os.Environment;
|
import android.os.Environment;
|
||||||
import android.os.Handler;
|
import android.os.Handler;
|
||||||
@@ -25,6 +27,7 @@ import android.os.Message;
|
|||||||
import android.os.Parcelable;
|
import android.os.Parcelable;
|
||||||
import android.os.RemoteException;
|
import android.os.RemoteException;
|
||||||
import android.os.ServiceManager;
|
import android.os.ServiceManager;
|
||||||
|
import android.provider.Settings;
|
||||||
import android.util.Log;
|
import android.util.Log;
|
||||||
import android.util.SparseArray;
|
import android.util.SparseArray;
|
||||||
|
|
||||||
@@ -54,20 +57,20 @@ import java.util.concurrent.atomic.AtomicInteger;
|
|||||||
* {@link android.content.Context#getSystemService(java.lang.String)} with an
|
* {@link android.content.Context#getSystemService(java.lang.String)} with an
|
||||||
* argument of {@link android.content.Context#STORAGE_SERVICE}.
|
* argument of {@link android.content.Context#STORAGE_SERVICE}.
|
||||||
*/
|
*/
|
||||||
|
public class StorageManager {
|
||||||
public class StorageManager
|
|
||||||
{
|
|
||||||
private static final String TAG = "StorageManager";
|
private static final String TAG = "StorageManager";
|
||||||
|
|
||||||
|
private final ContentResolver mResolver;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Our internal MountService binder reference
|
* Our internal MountService binder reference
|
||||||
*/
|
*/
|
||||||
final private IMountService mMountService;
|
private final IMountService mMountService;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* The looper target for callbacks
|
* The looper target for callbacks
|
||||||
*/
|
*/
|
||||||
Looper mTgtLooper;
|
private final Looper mTgtLooper;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Target listener for binder callbacks
|
* Target listener for binder callbacks
|
||||||
@@ -308,16 +311,16 @@ public class StorageManager
|
|||||||
*
|
*
|
||||||
* @hide
|
* @hide
|
||||||
*/
|
*/
|
||||||
public StorageManager(Looper tgtLooper) throws RemoteException {
|
public StorageManager(ContentResolver resolver, Looper tgtLooper) throws RemoteException {
|
||||||
|
mResolver = resolver;
|
||||||
|
mTgtLooper = tgtLooper;
|
||||||
mMountService = IMountService.Stub.asInterface(ServiceManager.getService("mount"));
|
mMountService = IMountService.Stub.asInterface(ServiceManager.getService("mount"));
|
||||||
if (mMountService == null) {
|
if (mMountService == null) {
|
||||||
Log.e(TAG, "Unable to connect to mount service! - is it running yet?");
|
Log.e(TAG, "Unable to connect to mount service! - is it running yet?");
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
mTgtLooper = tgtLooper;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Registers a {@link android.os.storage.StorageEventListener StorageEventListener}.
|
* Registers a {@link android.os.storage.StorageEventListener StorageEventListener}.
|
||||||
*
|
*
|
||||||
@@ -610,4 +613,36 @@ public class StorageManager
|
|||||||
Log.w(TAG, "No primary storage defined");
|
Log.w(TAG, "No primary storage defined");
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private static final int DEFAULT_THRESHOLD_PERCENTAGE = 10;
|
||||||
|
private static final long DEFAULT_THRESHOLD_MAX_BYTES = 500 * MB_IN_BYTES;
|
||||||
|
private static final long DEFAULT_FULL_THRESHOLD_BYTES = MB_IN_BYTES;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Return the number of available bytes at which the given path is
|
||||||
|
* considered running low on storage.
|
||||||
|
*
|
||||||
|
* @hide
|
||||||
|
*/
|
||||||
|
public long getStorageLowBytes(File path) {
|
||||||
|
final long lowPercent = Settings.Global.getInt(mResolver,
|
||||||
|
Settings.Global.SYS_STORAGE_THRESHOLD_PERCENTAGE, DEFAULT_THRESHOLD_PERCENTAGE);
|
||||||
|
final long lowBytes = (path.getTotalSpace() * lowPercent) / 100;
|
||||||
|
|
||||||
|
final long maxLowBytes = Settings.Global.getLong(mResolver,
|
||||||
|
Settings.Global.SYS_STORAGE_THRESHOLD_MAX_BYTES, DEFAULT_THRESHOLD_MAX_BYTES);
|
||||||
|
|
||||||
|
return Math.min(lowBytes, maxLowBytes);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Return the number of available bytes at which the given path is
|
||||||
|
* considered full.
|
||||||
|
*
|
||||||
|
* @hide
|
||||||
|
*/
|
||||||
|
public long getStorageFullBytes(File path) {
|
||||||
|
return Settings.Global.getLong(mResolver, Settings.Global.SYS_STORAGE_FULL_THRESHOLD_BYTES,
|
||||||
|
DEFAULT_FULL_THRESHOLD_BYTES);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -16,9 +16,6 @@
|
|||||||
|
|
||||||
package com.android.server;
|
package com.android.server;
|
||||||
|
|
||||||
import java.io.FileDescriptor;
|
|
||||||
import java.io.PrintWriter;
|
|
||||||
|
|
||||||
import android.app.Notification;
|
import android.app.Notification;
|
||||||
import android.app.NotificationManager;
|
import android.app.NotificationManager;
|
||||||
import android.app.PendingIntent;
|
import android.app.PendingIntent;
|
||||||
@@ -40,12 +37,17 @@ import android.os.StatFs;
|
|||||||
import android.os.SystemClock;
|
import android.os.SystemClock;
|
||||||
import android.os.SystemProperties;
|
import android.os.SystemProperties;
|
||||||
import android.os.UserHandle;
|
import android.os.UserHandle;
|
||||||
|
import android.os.storage.StorageManager;
|
||||||
import android.provider.Settings;
|
import android.provider.Settings;
|
||||||
import android.text.format.Formatter;
|
import android.text.format.Formatter;
|
||||||
import android.util.EventLog;
|
import android.util.EventLog;
|
||||||
import android.util.Slog;
|
import android.util.Slog;
|
||||||
import android.util.TimeUtils;
|
import android.util.TimeUtils;
|
||||||
|
|
||||||
|
import java.io.File;
|
||||||
|
import java.io.FileDescriptor;
|
||||||
|
import java.io.PrintWriter;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* This class implements a service to monitor the amount of disk
|
* This class implements a service to monitor the amount of disk
|
||||||
* storage space on the device. If the free storage on device is less
|
* storage space on the device. If the free storage on device is less
|
||||||
@@ -66,17 +68,18 @@ import android.util.TimeUtils;
|
|||||||
*/
|
*/
|
||||||
public class DeviceStorageMonitorService extends Binder {
|
public class DeviceStorageMonitorService extends Binder {
|
||||||
private static final String TAG = "DeviceStorageMonitorService";
|
private static final String TAG = "DeviceStorageMonitorService";
|
||||||
|
|
||||||
private static final boolean DEBUG = false;
|
private static final boolean DEBUG = false;
|
||||||
private static final boolean localLOGV = false;
|
private static final boolean localLOGV = false;
|
||||||
|
|
||||||
private static final int DEVICE_MEMORY_WHAT = 1;
|
private static final int DEVICE_MEMORY_WHAT = 1;
|
||||||
private static final int MONITOR_INTERVAL = 1; //in minutes
|
private static final int MONITOR_INTERVAL = 1; //in minutes
|
||||||
private static final int LOW_MEMORY_NOTIFICATION_ID = 1;
|
private static final int LOW_MEMORY_NOTIFICATION_ID = 1;
|
||||||
private static final int DEFAULT_THRESHOLD_PERCENTAGE = 10;
|
|
||||||
private static final int DEFAULT_THRESHOLD_MAX_BYTES = 500*1024*1024; // 500MB
|
|
||||||
private static final int DEFAULT_FREE_STORAGE_LOG_INTERVAL_IN_MINUTES = 12*60; //in minutes
|
private static final int DEFAULT_FREE_STORAGE_LOG_INTERVAL_IN_MINUTES = 12*60; //in minutes
|
||||||
private static final long DEFAULT_DISK_FREE_CHANGE_REPORTING_THRESHOLD = 2 * 1024 * 1024; // 2MB
|
private static final long DEFAULT_DISK_FREE_CHANGE_REPORTING_THRESHOLD = 2 * 1024 * 1024; // 2MB
|
||||||
private static final long DEFAULT_CHECK_INTERVAL = MONITOR_INTERVAL*60*1000;
|
private static final long DEFAULT_CHECK_INTERVAL = MONITOR_INTERVAL*60*1000;
|
||||||
private static final int DEFAULT_FULL_THRESHOLD_BYTES = 1024*1024; // 1MB
|
|
||||||
private long mFreeMem; // on /data
|
private long mFreeMem; // on /data
|
||||||
private long mFreeMemAfterLastCacheClear; // on /data
|
private long mFreeMemAfterLastCacheClear; // on /data
|
||||||
private long mLastReportedFreeMem;
|
private long mLastReportedFreeMem;
|
||||||
@@ -84,14 +87,16 @@ public class DeviceStorageMonitorService extends Binder {
|
|||||||
private boolean mLowMemFlag=false;
|
private boolean mLowMemFlag=false;
|
||||||
private boolean mMemFullFlag=false;
|
private boolean mMemFullFlag=false;
|
||||||
private Context mContext;
|
private Context mContext;
|
||||||
private ContentResolver mContentResolver;
|
private ContentResolver mResolver;
|
||||||
private long mTotalMemory; // on /data
|
private long mTotalMemory; // on /data
|
||||||
private StatFs mDataFileStats;
|
private StatFs mDataFileStats;
|
||||||
private StatFs mSystemFileStats;
|
private StatFs mSystemFileStats;
|
||||||
private StatFs mCacheFileStats;
|
private StatFs mCacheFileStats;
|
||||||
private static final String DATA_PATH = "/data";
|
|
||||||
private static final String SYSTEM_PATH = "/system";
|
private static final File DATA_PATH = Environment.getDataDirectory();
|
||||||
private static final String CACHE_PATH = "/cache";
|
private static final File SYSTEM_PATH = Environment.getRootDirectory();
|
||||||
|
private static final File CACHE_PATH = Environment.getDownloadCacheDirectory();
|
||||||
|
|
||||||
private long mThreadStartTime = -1;
|
private long mThreadStartTime = -1;
|
||||||
private boolean mClearSucceeded = false;
|
private boolean mClearSucceeded = false;
|
||||||
private boolean mClearingCache;
|
private boolean mClearingCache;
|
||||||
@@ -116,7 +121,7 @@ public class DeviceStorageMonitorService extends Binder {
|
|||||||
// more files than absolutely needed, to reduce the frequency that
|
// more files than absolutely needed, to reduce the frequency that
|
||||||
// flushing takes place.
|
// flushing takes place.
|
||||||
private long mMemCacheTrimToThreshold;
|
private long mMemCacheTrimToThreshold;
|
||||||
private int mMemFullThreshold;
|
private long mMemFullThreshold;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* This string is used for ServiceManager access to this class.
|
* This string is used for ServiceManager access to this class.
|
||||||
@@ -151,7 +156,7 @@ public class DeviceStorageMonitorService extends Binder {
|
|||||||
|
|
||||||
private final void restatDataDir() {
|
private final void restatDataDir() {
|
||||||
try {
|
try {
|
||||||
mDataFileStats.restat(DATA_PATH);
|
mDataFileStats.restat(DATA_PATH.getAbsolutePath());
|
||||||
mFreeMem = (long) mDataFileStats.getAvailableBlocks() *
|
mFreeMem = (long) mDataFileStats.getAvailableBlocks() *
|
||||||
mDataFileStats.getBlockSize();
|
mDataFileStats.getBlockSize();
|
||||||
} catch (IllegalArgumentException e) {
|
} catch (IllegalArgumentException e) {
|
||||||
@@ -163,7 +168,7 @@ public class DeviceStorageMonitorService extends Binder {
|
|||||||
mFreeMem = Long.parseLong(debugFreeMem);
|
mFreeMem = Long.parseLong(debugFreeMem);
|
||||||
}
|
}
|
||||||
// Read the log interval from secure settings
|
// Read the log interval from secure settings
|
||||||
long freeMemLogInterval = Settings.Global.getLong(mContentResolver,
|
long freeMemLogInterval = Settings.Global.getLong(mResolver,
|
||||||
Settings.Global.SYS_FREE_STORAGE_LOG_INTERVAL,
|
Settings.Global.SYS_FREE_STORAGE_LOG_INTERVAL,
|
||||||
DEFAULT_FREE_STORAGE_LOG_INTERVAL_IN_MINUTES)*60*1000;
|
DEFAULT_FREE_STORAGE_LOG_INTERVAL_IN_MINUTES)*60*1000;
|
||||||
//log the amount of free memory in event log
|
//log the amount of free memory in event log
|
||||||
@@ -173,14 +178,14 @@ public class DeviceStorageMonitorService extends Binder {
|
|||||||
mLastReportedFreeMemTime = currTime;
|
mLastReportedFreeMemTime = currTime;
|
||||||
long mFreeSystem = -1, mFreeCache = -1;
|
long mFreeSystem = -1, mFreeCache = -1;
|
||||||
try {
|
try {
|
||||||
mSystemFileStats.restat(SYSTEM_PATH);
|
mSystemFileStats.restat(SYSTEM_PATH.getAbsolutePath());
|
||||||
mFreeSystem = (long) mSystemFileStats.getAvailableBlocks() *
|
mFreeSystem = (long) mSystemFileStats.getAvailableBlocks() *
|
||||||
mSystemFileStats.getBlockSize();
|
mSystemFileStats.getBlockSize();
|
||||||
} catch (IllegalArgumentException e) {
|
} catch (IllegalArgumentException e) {
|
||||||
// ignore; report -1
|
// ignore; report -1
|
||||||
}
|
}
|
||||||
try {
|
try {
|
||||||
mCacheFileStats.restat(CACHE_PATH);
|
mCacheFileStats.restat(CACHE_PATH.getAbsolutePath());
|
||||||
mFreeCache = (long) mCacheFileStats.getAvailableBlocks() *
|
mFreeCache = (long) mCacheFileStats.getAvailableBlocks() *
|
||||||
mCacheFileStats.getBlockSize();
|
mCacheFileStats.getBlockSize();
|
||||||
} catch (IllegalArgumentException e) {
|
} catch (IllegalArgumentException e) {
|
||||||
@@ -190,7 +195,7 @@ public class DeviceStorageMonitorService extends Binder {
|
|||||||
mFreeMem, mFreeSystem, mFreeCache);
|
mFreeMem, mFreeSystem, mFreeCache);
|
||||||
}
|
}
|
||||||
// Read the reporting threshold from secure settings
|
// Read the reporting threshold from secure settings
|
||||||
long threshold = Settings.Global.getLong(mContentResolver,
|
long threshold = Settings.Global.getLong(mResolver,
|
||||||
Settings.Global.DISK_FREE_CHANGE_REPORTING_THRESHOLD,
|
Settings.Global.DISK_FREE_CHANGE_REPORTING_THRESHOLD,
|
||||||
DEFAULT_DISK_FREE_CHANGE_REPORTING_THRESHOLD);
|
DEFAULT_DISK_FREE_CHANGE_REPORTING_THRESHOLD);
|
||||||
// If mFree changed significantly log the new value
|
// If mFree changed significantly log the new value
|
||||||
@@ -303,40 +308,6 @@ public class DeviceStorageMonitorService extends Binder {
|
|||||||
delay);
|
delay);
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
|
||||||
* just query settings to retrieve the memory threshold.
|
|
||||||
* Preferred this over using a ContentObserver since Settings.Secure caches the value
|
|
||||||
* any way
|
|
||||||
*/
|
|
||||||
private long getMemThreshold() {
|
|
||||||
long value = Settings.Global.getInt(
|
|
||||||
mContentResolver,
|
|
||||||
Settings.Global.SYS_STORAGE_THRESHOLD_PERCENTAGE,
|
|
||||||
DEFAULT_THRESHOLD_PERCENTAGE);
|
|
||||||
if(localLOGV) Slog.v(TAG, "Threshold Percentage="+value);
|
|
||||||
value = (value*mTotalMemory)/100;
|
|
||||||
long maxValue = Settings.Global.getInt(
|
|
||||||
mContentResolver,
|
|
||||||
Settings.Global.SYS_STORAGE_THRESHOLD_MAX_BYTES,
|
|
||||||
DEFAULT_THRESHOLD_MAX_BYTES);
|
|
||||||
//evaluate threshold value
|
|
||||||
return value < maxValue ? value : maxValue;
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
* just query settings to retrieve the memory full threshold.
|
|
||||||
* Preferred this over using a ContentObserver since Settings.Secure caches the value
|
|
||||||
* any way
|
|
||||||
*/
|
|
||||||
private int getMemFullThreshold() {
|
|
||||||
int value = Settings.Global.getInt(
|
|
||||||
mContentResolver,
|
|
||||||
Settings.Global.SYS_STORAGE_FULL_THRESHOLD_BYTES,
|
|
||||||
DEFAULT_FULL_THRESHOLD_BYTES);
|
|
||||||
if(localLOGV) Slog.v(TAG, "Full Threshold Bytes="+value);
|
|
||||||
return value;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Constructor to run service. initializes the disk space threshold value
|
* Constructor to run service. initializes the disk space threshold value
|
||||||
* and posts an empty message to kickstart the process.
|
* and posts an empty message to kickstart the process.
|
||||||
@@ -344,11 +315,11 @@ public class DeviceStorageMonitorService extends Binder {
|
|||||||
public DeviceStorageMonitorService(Context context) {
|
public DeviceStorageMonitorService(Context context) {
|
||||||
mLastReportedFreeMemTime = 0;
|
mLastReportedFreeMemTime = 0;
|
||||||
mContext = context;
|
mContext = context;
|
||||||
mContentResolver = mContext.getContentResolver();
|
mResolver = mContext.getContentResolver();
|
||||||
//create StatFs object
|
//create StatFs object
|
||||||
mDataFileStats = new StatFs(DATA_PATH);
|
mDataFileStats = new StatFs(DATA_PATH.getAbsolutePath());
|
||||||
mSystemFileStats = new StatFs(SYSTEM_PATH);
|
mSystemFileStats = new StatFs(SYSTEM_PATH.getAbsolutePath());
|
||||||
mCacheFileStats = new StatFs(CACHE_PATH);
|
mCacheFileStats = new StatFs(CACHE_PATH.getAbsolutePath());
|
||||||
//initialize total storage on device
|
//initialize total storage on device
|
||||||
mTotalMemory = (long)mDataFileStats.getBlockCount() *
|
mTotalMemory = (long)mDataFileStats.getBlockCount() *
|
||||||
mDataFileStats.getBlockSize();
|
mDataFileStats.getBlockSize();
|
||||||
@@ -360,9 +331,12 @@ public class DeviceStorageMonitorService extends Binder {
|
|||||||
mStorageFullIntent.addFlags(Intent.FLAG_RECEIVER_REGISTERED_ONLY_BEFORE_BOOT);
|
mStorageFullIntent.addFlags(Intent.FLAG_RECEIVER_REGISTERED_ONLY_BEFORE_BOOT);
|
||||||
mStorageNotFullIntent = new Intent(Intent.ACTION_DEVICE_STORAGE_NOT_FULL);
|
mStorageNotFullIntent = new Intent(Intent.ACTION_DEVICE_STORAGE_NOT_FULL);
|
||||||
mStorageNotFullIntent.addFlags(Intent.FLAG_RECEIVER_REGISTERED_ONLY_BEFORE_BOOT);
|
mStorageNotFullIntent.addFlags(Intent.FLAG_RECEIVER_REGISTERED_ONLY_BEFORE_BOOT);
|
||||||
|
|
||||||
// cache storage thresholds
|
// cache storage thresholds
|
||||||
mMemLowThreshold = getMemThreshold();
|
final StorageManager sm = StorageManager.from(context);
|
||||||
mMemFullThreshold = getMemFullThreshold();
|
mMemLowThreshold = sm.getStorageLowBytes(DATA_PATH);
|
||||||
|
mMemFullThreshold = sm.getStorageFullBytes(DATA_PATH);
|
||||||
|
|
||||||
mMemCacheStartTrimThreshold = ((mMemLowThreshold*3)+mMemFullThreshold)/4;
|
mMemCacheStartTrimThreshold = ((mMemLowThreshold*3)+mMemFullThreshold)/4;
|
||||||
mMemCacheTrimToThreshold = mMemLowThreshold
|
mMemCacheTrimToThreshold = mMemLowThreshold
|
||||||
+ ((mMemLowThreshold-mMemCacheStartTrimThreshold)*2);
|
+ ((mMemLowThreshold-mMemCacheStartTrimThreshold)*2);
|
||||||
@@ -373,7 +347,6 @@ public class DeviceStorageMonitorService extends Binder {
|
|||||||
mCacheFileDeletedObserver.startWatching();
|
mCacheFileDeletedObserver.startWatching();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* This method sends a notification to NotificationManager to display
|
* This method sends a notification to NotificationManager to display
|
||||||
* an error dialog indicating low disk space and launch the Installer
|
* an error dialog indicating low disk space and launch the Installer
|
||||||
|
|||||||
Reference in New Issue
Block a user