1 /* 2 * Copyright (C) 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 17 package android.os.storage; 18 19 import static android.net.TrafficStats.MB_IN_BYTES; 20 21 import android.content.ContentResolver; 22 import android.content.Context; 23 import android.os.Environment; 24 import android.os.Handler; 25 import android.os.Looper; 26 import android.os.Message; 27 import android.os.Parcelable; 28 import android.os.RemoteException; 29 import android.os.ServiceManager; 30 import android.provider.Settings; 31 import android.util.Log; 32 import android.util.SparseArray; 33 34 import com.android.internal.util.Preconditions; 35 36 import java.io.File; 37 import java.io.IOException; 38 import java.lang.ref.WeakReference; 39 import java.util.ArrayList; 40 import java.util.List; 41 import java.util.concurrent.atomic.AtomicInteger; 42 43 /** 44 * StorageManager is the interface to the systems storage service. The storage 45 * manager handles storage-related items such as Opaque Binary Blobs (OBBs). 46 * <p> 47 * OBBs contain a filesystem that maybe be encrypted on disk and mounted 48 * on-demand from an application. OBBs are a good way of providing large amounts 49 * of binary assets without packaging them into APKs as they may be multiple 50 * gigabytes in size. However, due to their size, they're most likely stored in 51 * a shared storage pool accessible from all programs. The system does not 52 * guarantee the security of the OBB file itself: if any program modifies the 53 * OBB, there is no guarantee that a read from that OBB will produce the 54 * expected output. 55 * <p> 56 * Get an instance of this class by calling 57 * {@link android.content.Context#getSystemService(java.lang.String)} with an 58 * argument of {@link android.content.Context#STORAGE_SERVICE}. 59 */ 60 public class StorageManager { 61 private static final String TAG = "StorageManager"; 62 63 private final ContentResolver mResolver; 64 65 /* 66 * Our internal MountService binder reference 67 */ 68 private final IMountService mMountService; 69 70 /* 71 * The looper target for callbacks 72 */ 73 private final Looper mTgtLooper; 74 75 /* 76 * Target listener for binder callbacks 77 */ 78 private MountServiceBinderListener mBinderListener; 79 80 /* 81 * List of our listeners 82 */ 83 private List<ListenerDelegate> mListeners = new ArrayList<ListenerDelegate>(); 84 85 /* 86 * Next available nonce 87 */ 88 final private AtomicInteger mNextNonce = new AtomicInteger(0); 89 90 private class MountServiceBinderListener extends IMountServiceListener.Stub { 91 public void onUsbMassStorageConnectionChanged(boolean available) { 92 final int size = mListeners.size(); 93 for (int i = 0; i < size; i++) { 94 mListeners.get(i).sendShareAvailabilityChanged(available); 95 } 96 } 97 98 public void onStorageStateChanged(String path, String oldState, String newState) { 99 final int size = mListeners.size(); 100 for (int i = 0; i < size; i++) { 101 mListeners.get(i).sendStorageStateChanged(path, oldState, newState); 102 } 103 } 104 } 105 106 /** 107 * Binder listener for OBB action results. 108 */ 109 private final ObbActionListener mObbActionListener = new ObbActionListener(); 110 111 private class ObbActionListener extends IObbActionListener.Stub { 112 @SuppressWarnings("hiding") 113 private SparseArray<ObbListenerDelegate> mListeners = new SparseArray<ObbListenerDelegate>(); 114 115 @Override 116 public void onObbResult(String filename, int nonce, int status) { 117 final ObbListenerDelegate delegate; 118 synchronized (mListeners) { 119 delegate = mListeners.get(nonce); 120 if (delegate != null) { 121 mListeners.remove(nonce); 122 } 123 } 124 125 if (delegate != null) { 126 delegate.sendObbStateChanged(filename, status); 127 } 128 } 129 130 public int addListener(OnObbStateChangeListener listener) { 131 final ObbListenerDelegate delegate = new ObbListenerDelegate(listener); 132 133 synchronized (mListeners) { 134 mListeners.put(delegate.nonce, delegate); 135 } 136 137 return delegate.nonce; 138 } 139 } 140 141 private int getNextNonce() { 142 return mNextNonce.getAndIncrement(); 143 } 144 145 /** 146 * Private class containing sender and receiver code for StorageEvents. 147 */ 148 private class ObbListenerDelegate { 149 private final WeakReference<OnObbStateChangeListener> mObbEventListenerRef; 150 private final Handler mHandler; 151 152 private final int nonce; 153 154 ObbListenerDelegate(OnObbStateChangeListener listener) { 155 nonce = getNextNonce(); 156 mObbEventListenerRef = new WeakReference<OnObbStateChangeListener>(listener); 157 mHandler = new Handler(mTgtLooper) { 158 @Override 159 public void handleMessage(Message msg) { 160 final OnObbStateChangeListener changeListener = getListener(); 161 if (changeListener == null) { 162 return; 163 } 164 165 StorageEvent e = (StorageEvent) msg.obj; 166 167 if (msg.what == StorageEvent.EVENT_OBB_STATE_CHANGED) { 168 ObbStateChangedStorageEvent ev = (ObbStateChangedStorageEvent) e; 169 changeListener.onObbStateChange(ev.path, ev.state); 170 } else { 171 Log.e(TAG, "Unsupported event " + msg.what); 172 } 173 } 174 }; 175 } 176 177 OnObbStateChangeListener getListener() { 178 if (mObbEventListenerRef == null) { 179 return null; 180 } 181 return mObbEventListenerRef.get(); 182 } 183 184 void sendObbStateChanged(String path, int state) { 185 ObbStateChangedStorageEvent e = new ObbStateChangedStorageEvent(path, state); 186 mHandler.sendMessage(e.getMessage()); 187 } 188 } 189 190 /** 191 * Message sent during an OBB status change event. 192 */ 193 private class ObbStateChangedStorageEvent extends StorageEvent { 194 public final String path; 195 196 public final int state; 197 198 public ObbStateChangedStorageEvent(String path, int state) { 199 super(EVENT_OBB_STATE_CHANGED); 200 this.path = path; 201 this.state = state; 202 } 203 } 204 205 /** 206 * Private base class for messages sent between the callback thread 207 * and the target looper handler. 208 */ 209 private class StorageEvent { 210 static final int EVENT_UMS_CONNECTION_CHANGED = 1; 211 static final int EVENT_STORAGE_STATE_CHANGED = 2; 212 static final int EVENT_OBB_STATE_CHANGED = 3; 213 214 private Message mMessage; 215 216 public StorageEvent(int what) { 217 mMessage = Message.obtain(); 218 mMessage.what = what; 219 mMessage.obj = this; 220 } 221 222 public Message getMessage() { 223 return mMessage; 224 } 225 } 226 227 /** 228 * Message sent on a USB mass storage connection change. 229 */ 230 private class UmsConnectionChangedStorageEvent extends StorageEvent { 231 public boolean available; 232 233 public UmsConnectionChangedStorageEvent(boolean a) { 234 super(EVENT_UMS_CONNECTION_CHANGED); 235 available = a; 236 } 237 } 238 239 /** 240 * Message sent on volume state change. 241 */ 242 private class StorageStateChangedStorageEvent extends StorageEvent { 243 public String path; 244 public String oldState; 245 public String newState; 246 247 public StorageStateChangedStorageEvent(String p, String oldS, String newS) { 248 super(EVENT_STORAGE_STATE_CHANGED); 249 path = p; 250 oldState = oldS; 251 newState = newS; 252 } 253 } 254 255 /** 256 * Private class containing sender and receiver code for StorageEvents. 257 */ 258 private class ListenerDelegate { 259 final StorageEventListener mStorageEventListener; 260 private final Handler mHandler; 261 262 ListenerDelegate(StorageEventListener listener) { 263 mStorageEventListener = listener; 264 mHandler = new Handler(mTgtLooper) { 265 @Override 266 public void handleMessage(Message msg) { 267 StorageEvent e = (StorageEvent) msg.obj; 268 269 if (msg.what == StorageEvent.EVENT_UMS_CONNECTION_CHANGED) { 270 UmsConnectionChangedStorageEvent ev = (UmsConnectionChangedStorageEvent) e; 271 mStorageEventListener.onUsbMassStorageConnectionChanged(ev.available); 272 } else if (msg.what == StorageEvent.EVENT_STORAGE_STATE_CHANGED) { 273 StorageStateChangedStorageEvent ev = (StorageStateChangedStorageEvent) e; 274 mStorageEventListener.onStorageStateChanged(ev.path, ev.oldState, ev.newState); 275 } else { 276 Log.e(TAG, "Unsupported event " + msg.what); 277 } 278 } 279 }; 280 } 281 282 StorageEventListener getListener() { 283 return mStorageEventListener; 284 } 285 286 void sendShareAvailabilityChanged(boolean available) { 287 UmsConnectionChangedStorageEvent e = new UmsConnectionChangedStorageEvent(available); 288 mHandler.sendMessage(e.getMessage()); 289 } 290 291 void sendStorageStateChanged(String path, String oldState, String newState) { 292 StorageStateChangedStorageEvent e = new StorageStateChangedStorageEvent(path, oldState, newState); 293 mHandler.sendMessage(e.getMessage()); 294 } 295 } 296 297 /** {@hide} */ 298 public static StorageManager from(Context context) { 299 return (StorageManager) context.getSystemService(Context.STORAGE_SERVICE); 300 } 301 302 /** 303 * Constructs a StorageManager object through which an application can 304 * can communicate with the systems mount service. 305 * 306 * @param tgtLooper The {@link android.os.Looper} which events will be received on. 307 * 308 * <p>Applications can get instance of this class by calling 309 * {@link android.content.Context#getSystemService(java.lang.String)} with an argument 310 * of {@link android.content.Context#STORAGE_SERVICE}. 311 * 312 * @hide 313 */ 314 public StorageManager(ContentResolver resolver, Looper tgtLooper) throws RemoteException { 315 mResolver = resolver; 316 mTgtLooper = tgtLooper; 317 mMountService = IMountService.Stub.asInterface(ServiceManager.getService("mount")); 318 if (mMountService == null) { 319 Log.e(TAG, "Unable to connect to mount service! - is it running yet?"); 320 return; 321 } 322 } 323 324 /** 325 * Registers a {@link android.os.storage.StorageEventListener StorageEventListener}. 326 * 327 * @param listener A {@link android.os.storage.StorageEventListener StorageEventListener} object. 328 * 329 * @hide 330 */ 331 public void registerListener(StorageEventListener listener) { 332 if (listener == null) { 333 return; 334 } 335 336 synchronized (mListeners) { 337 if (mBinderListener == null ) { 338 try { 339 mBinderListener = new MountServiceBinderListener(); 340 mMountService.registerListener(mBinderListener); 341 } catch (RemoteException rex) { 342 Log.e(TAG, "Register mBinderListener failed"); 343 return; 344 } 345 } 346 mListeners.add(new ListenerDelegate(listener)); 347 } 348 } 349 350 /** 351 * Unregisters a {@link android.os.storage.StorageEventListener StorageEventListener}. 352 * 353 * @param listener A {@link android.os.storage.StorageEventListener StorageEventListener} object. 354 * 355 * @hide 356 */ 357 public void unregisterListener(StorageEventListener listener) { 358 if (listener == null) { 359 return; 360 } 361 362 synchronized (mListeners) { 363 final int size = mListeners.size(); 364 for (int i=0 ; i<size ; i++) { 365 ListenerDelegate l = mListeners.get(i); 366 if (l.getListener() == listener) { 367 mListeners.remove(i); 368 break; 369 } 370 } 371 if (mListeners.size() == 0 && mBinderListener != null) { 372 try { 373 mMountService.unregisterListener(mBinderListener); 374 } catch (RemoteException rex) { 375 Log.e(TAG, "Unregister mBinderListener failed"); 376 return; 377 } 378 } 379 } 380 } 381 382 /** 383 * Enables USB Mass Storage (UMS) on the device. 384 * 385 * @hide 386 */ 387 public void enableUsbMassStorage() { 388 try { 389 mMountService.setUsbMassStorageEnabled(true); 390 } catch (Exception ex) { 391 Log.e(TAG, "Failed to enable UMS", ex); 392 } 393 } 394 395 /** 396 * Disables USB Mass Storage (UMS) on the device. 397 * 398 * @hide 399 */ 400 public void disableUsbMassStorage() { 401 try { 402 mMountService.setUsbMassStorageEnabled(false); 403 } catch (Exception ex) { 404 Log.e(TAG, "Failed to disable UMS", ex); 405 } 406 } 407 408 /** 409 * Query if a USB Mass Storage (UMS) host is connected. 410 * @return true if UMS host is connected. 411 * 412 * @hide 413 */ 414 public boolean isUsbMassStorageConnected() { 415 try { 416 return mMountService.isUsbMassStorageConnected(); 417 } catch (Exception ex) { 418 Log.e(TAG, "Failed to get UMS connection state", ex); 419 } 420 return false; 421 } 422 423 /** 424 * Query if a USB Mass Storage (UMS) is enabled on the device. 425 * @return true if UMS host is enabled. 426 * 427 * @hide 428 */ 429 public boolean isUsbMassStorageEnabled() { 430 try { 431 return mMountService.isUsbMassStorageEnabled(); 432 } catch (RemoteException rex) { 433 Log.e(TAG, "Failed to get UMS enable state", rex); 434 } 435 return false; 436 } 437 438 /** 439 * Mount an Opaque Binary Blob (OBB) file. If a <code>key</code> is 440 * specified, it is supplied to the mounting process to be used in any 441 * encryption used in the OBB. 442 * <p> 443 * The OBB will remain mounted for as long as the StorageManager reference 444 * is held by the application. As soon as this reference is lost, the OBBs 445 * in use will be unmounted. The {@link OnObbStateChangeListener} registered 446 * with this call will receive the success or failure of this operation. 447 * <p> 448 * <em>Note:</em> you can only mount OBB files for which the OBB tag on the 449 * file matches a package ID that is owned by the calling program's UID. 450 * That is, shared UID applications can attempt to mount any other 451 * application's OBB that shares its UID. 452 * 453 * @param rawPath the path to the OBB file 454 * @param key secret used to encrypt the OBB; may be <code>null</code> if no 455 * encryption was used on the OBB. 456 * @param listener will receive the success or failure of the operation 457 * @return whether the mount call was successfully queued or not 458 */ 459 public boolean mountObb(String rawPath, String key, OnObbStateChangeListener listener) { 460 Preconditions.checkNotNull(rawPath, "rawPath cannot be null"); 461 Preconditions.checkNotNull(listener, "listener cannot be null"); 462 463 try { 464 final String canonicalPath = new File(rawPath).getCanonicalPath(); 465 final int nonce = mObbActionListener.addListener(listener); 466 mMountService.mountObb(rawPath, canonicalPath, key, mObbActionListener, nonce); 467 return true; 468 } catch (IOException e) { 469 throw new IllegalArgumentException("Failed to resolve path: " + rawPath, e); 470 } catch (RemoteException e) { 471 Log.e(TAG, "Failed to mount OBB", e); 472 } 473 474 return false; 475 } 476 477 /** 478 * Unmount an Opaque Binary Blob (OBB) file asynchronously. If the 479 * <code>force</code> flag is true, it will kill any application needed to 480 * unmount the given OBB (even the calling application). 481 * <p> 482 * The {@link OnObbStateChangeListener} registered with this call will 483 * receive the success or failure of this operation. 484 * <p> 485 * <em>Note:</em> you can only mount OBB files for which the OBB tag on the 486 * file matches a package ID that is owned by the calling program's UID. 487 * That is, shared UID applications can obtain access to any other 488 * application's OBB that shares its UID. 489 * <p> 490 * 491 * @param rawPath path to the OBB file 492 * @param force whether to kill any programs using this in order to unmount 493 * it 494 * @param listener will receive the success or failure of the operation 495 * @return whether the unmount call was successfully queued or not 496 */ 497 public boolean unmountObb(String rawPath, boolean force, OnObbStateChangeListener listener) { 498 Preconditions.checkNotNull(rawPath, "rawPath cannot be null"); 499 Preconditions.checkNotNull(listener, "listener cannot be null"); 500 501 try { 502 final int nonce = mObbActionListener.addListener(listener); 503 mMountService.unmountObb(rawPath, force, mObbActionListener, nonce); 504 return true; 505 } catch (RemoteException e) { 506 Log.e(TAG, "Failed to mount OBB", e); 507 } 508 509 return false; 510 } 511 512 /** 513 * Check whether an Opaque Binary Blob (OBB) is mounted or not. 514 * 515 * @param rawPath path to OBB image 516 * @return true if OBB is mounted; false if not mounted or on error 517 */ 518 public boolean isObbMounted(String rawPath) { 519 Preconditions.checkNotNull(rawPath, "rawPath cannot be null"); 520 521 try { 522 return mMountService.isObbMounted(rawPath); 523 } catch (RemoteException e) { 524 Log.e(TAG, "Failed to check if OBB is mounted", e); 525 } 526 527 return false; 528 } 529 530 /** 531 * Check the mounted path of an Opaque Binary Blob (OBB) file. This will 532 * give you the path to where you can obtain access to the internals of the 533 * OBB. 534 * 535 * @param rawPath path to OBB image 536 * @return absolute path to mounted OBB image data or <code>null</code> if 537 * not mounted or exception encountered trying to read status 538 */ 539 public String getMountedObbPath(String rawPath) { 540 Preconditions.checkNotNull(rawPath, "rawPath cannot be null"); 541 542 try { 543 return mMountService.getMountedObbPath(rawPath); 544 } catch (RemoteException e) { 545 Log.e(TAG, "Failed to find mounted path for OBB", e); 546 } 547 548 return null; 549 } 550 551 /** 552 * Gets the state of a volume via its mountpoint. 553 * @hide 554 */ 555 public String getVolumeState(String mountPoint) { 556 if (mMountService == null) return Environment.MEDIA_REMOVED; 557 try { 558 return mMountService.getVolumeState(mountPoint); 559 } catch (RemoteException e) { 560 Log.e(TAG, "Failed to get volume state", e); 561 return null; 562 } 563 } 564 565 /** 566 * Returns list of all mountable volumes. 567 * @hide 568 */ 569 public StorageVolume[] getVolumeList() { 570 if (mMountService == null) return new StorageVolume[0]; 571 try { 572 Parcelable[] list = mMountService.getVolumeList(); 573 if (list == null) return new StorageVolume[0]; 574 int length = list.length; 575 StorageVolume[] result = new StorageVolume[length]; 576 for (int i = 0; i < length; i++) { 577 result[i] = (StorageVolume)list[i]; 578 } 579 return result; 580 } catch (RemoteException e) { 581 Log.e(TAG, "Failed to get volume list", e); 582 return null; 583 } 584 } 585 586 /** 587 * Returns list of paths for all mountable volumes. 588 * @hide 589 */ 590 public String[] getVolumePaths() { 591 StorageVolume[] volumes = getVolumeList(); 592 if (volumes == null) return null; 593 int count = volumes.length; 594 String[] paths = new String[count]; 595 for (int i = 0; i < count; i++) { 596 paths[i] = volumes[i].getPath(); 597 } 598 return paths; 599 } 600 601 /** {@hide} */ 602 public StorageVolume getPrimaryVolume() { 603 return getPrimaryVolume(getVolumeList()); 604 } 605 606 /** {@hide} */ 607 public static StorageVolume getPrimaryVolume(StorageVolume[] volumes) { 608 for (StorageVolume volume : volumes) { 609 if (volume.isPrimary()) { 610 return volume; 611 } 612 } 613 Log.w(TAG, "No primary storage defined"); 614 return null; 615 } 616 617 private static final int DEFAULT_THRESHOLD_PERCENTAGE = 10; 618 private static final long DEFAULT_THRESHOLD_MAX_BYTES = 500 * MB_IN_BYTES; 619 private static final long DEFAULT_FULL_THRESHOLD_BYTES = MB_IN_BYTES; 620 621 /** 622 * Return the number of available bytes at which the given path is 623 * considered running low on storage. 624 * 625 * @hide 626 */ 627 public long getStorageLowBytes(File path) { 628 final long lowPercent = Settings.Global.getInt(mResolver, 629 Settings.Global.SYS_STORAGE_THRESHOLD_PERCENTAGE, DEFAULT_THRESHOLD_PERCENTAGE); 630 final long lowBytes = (path.getTotalSpace() * lowPercent) / 100; 631 632 final long maxLowBytes = Settings.Global.getLong(mResolver, 633 Settings.Global.SYS_STORAGE_THRESHOLD_MAX_BYTES, DEFAULT_THRESHOLD_MAX_BYTES); 634 635 return Math.min(lowBytes, maxLowBytes); 636 } 637 638 /** 639 * Return the number of available bytes at which the given path is 640 * considered full. 641 * 642 * @hide 643 */ 644 public long getStorageFullBytes(File path) { 645 return Settings.Global.getLong(mResolver, Settings.Global.SYS_STORAGE_FULL_THRESHOLD_BYTES, 646 DEFAULT_FULL_THRESHOLD_BYTES); 647 } 648 } 649