diff options
author | San Mehat <san@google.com> | 2010-01-12 07:57:42 -0800 |
---|---|---|
committer | San Mehat <san@google.com> | 2010-01-12 07:57:42 -0800 |
commit | 7ebf017658070323ed1c2bbd80c46c7cd2390d87 (patch) | |
tree | 3e5d6d274f3ef4f56e79883e21a04078ef941871 /services | |
parent | 595f48b2a4fa2751ad638b60d2476e77738b21da (diff) | |
download | frameworks_base-7ebf017658070323ed1c2bbd80c46c7cd2390d87.zip frameworks_base-7ebf017658070323ed1c2bbd80c46c7cd2390d87.tar.gz frameworks_base-7ebf017658070323ed1c2bbd80c46c7cd2390d87.tar.bz2 |
framework: storage: Ensure that filesystems are unmounted before shutdown/reboot
Signed-off-by: San Mehat <san@google.com>
Diffstat (limited to 'services')
-rw-r--r-- | services/java/com/android/server/MountService.java | 75 |
1 files changed, 74 insertions, 1 deletions
diff --git a/services/java/com/android/server/MountService.java b/services/java/com/android/server/MountService.java index 204389e..ddf7c56 100644 --- a/services/java/com/android/server/MountService.java +++ b/services/java/com/android/server/MountService.java @@ -108,13 +108,86 @@ class MountService extends IMountService.Stub { BroadcastReceiver mBroadcastReceiver = new BroadcastReceiver() { public void onReceive(Context context, Intent intent) { - if (intent.getAction().equals(Intent.ACTION_BOOT_COMPLETED)) { + String action = intent.getAction(); + + if (action.equals(Intent.ACTION_BOOT_COMPLETED)) { Thread thread = new Thread(mListener, MountListener.class.getName()); thread.start(); } } }; + public void shutdown() { + if (mContext.checkCallingOrSelfPermission( + android.Manifest.permission.SHUTDOWN) + != PackageManager.PERMISSION_GRANTED) { + throw new SecurityException("Requires SHUTDOWN permission"); + } + + Log.d(TAG, "Shutting down"); + String state = Environment.getExternalStorageState(); + + if (state.equals(Environment.MEDIA_SHARED)) { + /* + * If the media is currently shared, unshare it. + * XXX: This is still dangerous!. We should not + * be rebooting at *all* if UMS is enabled, since + * the UMS host could have dirty FAT cache entries + * yet to flush. + */ + try { + setMassStorageEnabled(false); + } catch (Exception e) { + Log.e(TAG, "ums disable failed", e); + } + } else if (state.equals(Environment.MEDIA_CHECKING)) { + /* + * If the media is being checked, then we need to wait for + * it to complete before being able to proceed. + */ + // XXX: @hackbod - Should we disable the ANR timer here? + int retries = 30; + while (state.equals(Environment.MEDIA_CHECKING) && (retries-- >=0)) { + try { + Thread.sleep(1000); + } catch (InterruptedException iex) { + Log.e(TAG, "Interrupted while waiting for media", iex); + break; + } + state = Environment.getExternalStorageState(); + } + if (retries == 0) { + Log.e(TAG, "Timed out waiting for media to check"); + } + } + + if (state.equals(Environment.MEDIA_MOUNTED)) { + /* + * If the media is mounted, then gracefully unmount it. + */ + try { + String m = Environment.getExternalStorageDirectory().toString(); + unmountMedia(m); + + int retries = 12; + while (!state.equals(Environment.MEDIA_UNMOUNTED) && (retries-- >=0)) { + try { + Thread.sleep(1000); + } catch (InterruptedException iex) { + Log.e(TAG, "Interrupted while waiting for media", iex); + break; + } + state = Environment.getExternalStorageState(); + } + if (retries == 0) { + Log.e(TAG, "Timed out waiting for media to unmount"); + } + } catch (Exception e) { + Log.e(TAG, "external storage unmount failed", e); + } + } + } + /** * @return true if USB mass storage support is enabled. */ |