1 /* 2 * Copyright (C) 2014 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; 18 19 import static java.util.Objects.requireNonNull; 20 21 import android.annotation.NonNull; 22 import android.annotation.Nullable; 23 import android.compat.annotation.UnsupportedAppUsage; 24 import android.util.ArrayMap; 25 import android.util.Log; 26 import android.util.MathUtils; 27 import android.util.Slog; 28 import android.util.SparseArray; 29 30 import com.android.internal.annotations.GuardedBy; 31 import com.android.internal.annotations.VisibleForTesting; 32 import com.android.internal.util.IndentingPrintWriter; 33 import com.android.internal.util.Preconditions; 34 35 import java.io.Serializable; 36 import java.lang.ref.WeakReference; 37 import java.util.ArrayList; 38 import java.util.Set; 39 import java.util.function.BiFunction; 40 41 /** 42 * A mapping from String keys to values of various types. In most cases, you 43 * should work directly with either the {@link Bundle} or 44 * {@link PersistableBundle} subclass. 45 */ 46 public class BaseBundle { 47 /** @hide */ 48 protected static final String TAG = "Bundle"; 49 static final boolean DEBUG = false; 50 51 /** 52 * Keep them in sync with frameworks/native/libs/binder/PersistableBundle.cpp. 53 * 54 * @hide 55 */ 56 @VisibleForTesting 57 static final int BUNDLE_MAGIC = 0x4C444E42; // 'B' 'N' 'D' 'L' 58 private static final int BUNDLE_MAGIC_NATIVE = 0x4C444E44; // 'B' 'N' 'D' 'N' 59 60 /** 61 * Flag indicating that this Bundle is okay to "defuse", see {@link #setShouldDefuse(boolean)} 62 * for more details. 63 * <p> 64 * This should <em>only</em> be set when the Bundle reaches its final destination, otherwise a 65 * system process may clobber contents that were destined for an app that could have unparceled 66 * them. 67 */ 68 static final int FLAG_DEFUSABLE = 1 << 0; 69 70 private static final boolean LOG_DEFUSABLE = false; 71 72 private static volatile boolean sShouldDefuse = false; 73 74 /** 75 * Set global variable indicating that any Bundles parsed in this process should be "defused". 76 * That is, any {@link BadParcelableException} encountered will be suppressed and logged. Also: 77 * <ul> 78 * <li>If it was the deserialization of a custom item (eg. {@link Parcelable}) that caused the 79 * exception, {@code null} will be returned but the item will be held in the map in its 80 * serialized form (lazy value). 81 * <li>If the exception happened during partial deserialization, that is, during the read of 82 * the map and its basic types (while skipping custom types), the map will be left empty. 83 * </ul> 84 * 85 * @hide 86 */ setShouldDefuse(boolean shouldDefuse)87 public static void setShouldDefuse(boolean shouldDefuse) { 88 sShouldDefuse = shouldDefuse; 89 } 90 91 // A parcel cannot be obtained during compile-time initialization. Put the 92 // empty parcel into an inner class that can be initialized separately. This 93 // allows to initialize BaseBundle, and classes depending on it. 94 /** {@hide} */ 95 static final class NoImagePreloadHolder { 96 public static final Parcel EMPTY_PARCEL = Parcel.obtain(); 97 } 98 99 // Invariant - exactly one of mMap / mParcelledData will be null 100 // (except inside a call to unparcel) 101 102 @UnsupportedAppUsage 103 ArrayMap<String, Object> mMap = null; 104 105 /* 106 * If mParcelledData is non-null, then mMap will be null and the 107 * data are stored as a Parcel containing a Bundle. When the data 108 * are unparcelled, mParcelledData will be set to null. 109 */ 110 @UnsupportedAppUsage 111 volatile Parcel mParcelledData = null; 112 113 /** 114 * Whether {@link #mParcelledData} was generated by native code or not. 115 */ 116 private boolean mParcelledByNative; 117 118 /** 119 * Flag indicating if mParcelledData is only referenced in this bundle. 120 * mParcelledData could be referenced elsewhere if mMap contains lazy values, 121 * and bundle data is copied to another bundle using putAll or the copy constructors. 122 */ 123 boolean mOwnsLazyValues = true; 124 125 /** Tracks how many lazy values are referenced in mMap */ 126 private int mLazyValues = 0; 127 128 /** 129 * As mParcelledData is set to null when it is unparcelled, we keep a weak reference to 130 * it to aid in recycling it. Do not use this reference otherwise. 131 * Is non-null iff mMap contains lazy values. 132 */ 133 private WeakReference<Parcel> mWeakParcelledData = null; 134 135 /** 136 * The ClassLoader used when unparcelling data from mParcelledData. 137 */ 138 private ClassLoader mClassLoader; 139 140 /** {@hide} */ 141 @VisibleForTesting 142 public int mFlags; 143 144 /** 145 * Constructs a new, empty Bundle that uses a specific ClassLoader for 146 * instantiating Parcelable and Serializable objects. 147 * 148 * @param loader An explicit ClassLoader to use when instantiating objects 149 * inside of the Bundle. 150 * @param capacity Initial size of the ArrayMap. 151 */ BaseBundle(@ullable ClassLoader loader, int capacity)152 BaseBundle(@Nullable ClassLoader loader, int capacity) { 153 mMap = capacity > 0 ? 154 new ArrayMap<String, Object>(capacity) : new ArrayMap<String, Object>(); 155 mClassLoader = loader == null ? getClass().getClassLoader() : loader; 156 } 157 158 /** 159 * Constructs a new, empty Bundle. 160 */ BaseBundle()161 BaseBundle() { 162 this((ClassLoader) null, 0); 163 } 164 165 /** 166 * Constructs a Bundle whose data is stored as a Parcel. The data 167 * will be unparcelled on first contact, using the assigned ClassLoader. 168 * 169 * @param parcelledData a Parcel containing a Bundle 170 */ BaseBundle(Parcel parcelledData)171 BaseBundle(Parcel parcelledData) { 172 readFromParcelInner(parcelledData); 173 } 174 BaseBundle(Parcel parcelledData, int length)175 BaseBundle(Parcel parcelledData, int length) { 176 readFromParcelInner(parcelledData, length); 177 } 178 179 /** 180 * Constructs a new, empty Bundle that uses a specific ClassLoader for 181 * instantiating Parcelable and Serializable objects. 182 * 183 * @param loader An explicit ClassLoader to use when instantiating objects 184 * inside of the Bundle. 185 */ BaseBundle(ClassLoader loader)186 BaseBundle(ClassLoader loader) { 187 this(loader, 0); 188 } 189 190 /** 191 * Constructs a new, empty Bundle sized to hold the given number of 192 * elements. The Bundle will grow as needed. 193 * 194 * @param capacity the initial capacity of the Bundle 195 */ BaseBundle(int capacity)196 BaseBundle(int capacity) { 197 this((ClassLoader) null, capacity); 198 } 199 200 /** 201 * Constructs a Bundle containing a copy of the mappings from the given 202 * Bundle. 203 * 204 * @param b a Bundle to be copied. 205 */ BaseBundle(BaseBundle b)206 BaseBundle(BaseBundle b) { 207 this(b, /* deep */ false); 208 } 209 210 /** 211 * Constructs a {@link BaseBundle} containing a copy of {@code from}. 212 * 213 * @param from The bundle to be copied. 214 * @param deep Whether is a deep or shallow copy. 215 * 216 * @hide 217 */ BaseBundle(BaseBundle from, boolean deep)218 BaseBundle(BaseBundle from, boolean deep) { 219 synchronized (from) { 220 mClassLoader = from.mClassLoader; 221 222 if (from.mMap != null) { 223 mOwnsLazyValues = false; 224 from.mOwnsLazyValues = false; 225 226 if (!deep) { 227 mMap = new ArrayMap<>(from.mMap); 228 } else { 229 final ArrayMap<String, Object> fromMap = from.mMap; 230 final int n = fromMap.size(); 231 mMap = new ArrayMap<>(n); 232 for (int i = 0; i < n; i++) { 233 mMap.append(fromMap.keyAt(i), deepCopyValue(fromMap.valueAt(i))); 234 } 235 } 236 } else { 237 mMap = null; 238 } 239 240 final Parcel parcelledData; 241 if (from.mParcelledData != null) { 242 if (from.isEmptyParcel()) { 243 parcelledData = NoImagePreloadHolder.EMPTY_PARCEL; 244 mParcelledByNative = false; 245 } else { 246 parcelledData = Parcel.obtain(); 247 parcelledData.appendFrom(from.mParcelledData, 0, 248 from.mParcelledData.dataSize()); 249 parcelledData.setDataPosition(0); 250 mParcelledByNative = from.mParcelledByNative; 251 } 252 } else { 253 parcelledData = null; 254 mParcelledByNative = false; 255 } 256 257 // Keep as last statement to ensure visibility of other fields 258 mParcelledData = parcelledData; 259 } 260 } 261 262 /** 263 * TODO: optimize this later (getting just the value part of a Bundle 264 * with a single pair) once Bundle.forPair() above is implemented 265 * with a special single-value Map implementation/serialization. 266 * 267 * Note: value in single-pair Bundle may be null. 268 * 269 * @hide 270 */ getPairValue()271 public String getPairValue() { 272 unparcel(); 273 int size = mMap.size(); 274 if (size > 1) { 275 Log.w(TAG, "getPairValue() used on Bundle with multiple pairs."); 276 } 277 if (size == 0) { 278 return null; 279 } 280 try { 281 return getValueAt(0, String.class); 282 } catch (ClassCastException | BadTypeParcelableException e) { 283 typeWarning("getPairValue()", "String", e); 284 return null; 285 } 286 } 287 288 /** 289 * Changes the ClassLoader this Bundle uses when instantiating objects. 290 * 291 * @param loader An explicit ClassLoader to use when instantiating objects 292 * inside of the Bundle. 293 */ setClassLoader(ClassLoader loader)294 void setClassLoader(ClassLoader loader) { 295 mClassLoader = loader; 296 } 297 298 /** 299 * Return the ClassLoader currently associated with this Bundle. 300 */ getClassLoader()301 ClassLoader getClassLoader() { 302 return mClassLoader; 303 } 304 305 /** 306 * If the underlying data are stored as a Parcel, unparcel them 307 * using the currently assigned class loader. 308 */ 309 @UnsupportedAppUsage unparcel()310 final void unparcel() { 311 unparcel(/* itemwise */ false); 312 } 313 314 /** Deserializes the underlying data and each item if {@code itemwise} is true. */ unparcel(boolean itemwise)315 final void unparcel(boolean itemwise) { 316 synchronized (this) { 317 final Parcel source = mParcelledData; 318 if (source != null) { 319 Preconditions.checkState(mOwnsLazyValues); 320 initializeFromParcelLocked(source, /*ownsParcel*/ true, mParcelledByNative); 321 } else { 322 if (DEBUG) { 323 Log.d(TAG, "unparcel " 324 + Integer.toHexString(System.identityHashCode(this)) 325 + ": no parcelled data"); 326 } 327 } 328 if (itemwise) { 329 if (LOG_DEFUSABLE && sShouldDefuse && (mFlags & FLAG_DEFUSABLE) == 0) { 330 Slog.wtf(TAG, 331 "Attempting to unparcel all items in a Bundle while in transit; this " 332 + "may remove elements intended for the final desitination.", 333 new Throwable()); 334 } 335 for (int i = 0, n = mMap.size(); i < n; i++) { 336 // Triggers deserialization of i-th item, if needed 337 getValueAt(i, /* clazz */ null); 338 } 339 } 340 } 341 } 342 343 /** 344 * Returns the value for key {@code key}. 345 * 346 * This call should always be made after {@link #unparcel()} or inside a lock after making sure 347 * {@code mMap} is not null. 348 * 349 * @deprecated Use {@link #getValue(String, Class, Class[])}. This method should only be used in 350 * other deprecated APIs. 351 * 352 * @hide 353 */ 354 @Deprecated 355 @Nullable getValue(String key)356 final Object getValue(String key) { 357 return getValue(key, /* clazz */ null); 358 } 359 360 /** Same as {@link #getValue(String, Class, Class[])} with no item types. */ 361 @Nullable getValue(String key, @Nullable Class<T> clazz)362 final <T> T getValue(String key, @Nullable Class<T> clazz) { 363 // Avoids allocating Class[0] array 364 return getValue(key, clazz, (Class<?>[]) null); 365 } 366 367 /** 368 * Returns the value for key {@code key} for expected return type {@code clazz} (or pass {@code 369 * null} for no type check). 370 * 371 * For {@code itemTypes}, see {@link Parcel#readValue(int, ClassLoader, Class, Class[])}. 372 * 373 * This call should always be made after {@link #unparcel()} or inside a lock after making sure 374 * {@code mMap} is not null. 375 * 376 * @hide 377 */ 378 @Nullable getValue(String key, @Nullable Class<T> clazz, @Nullable Class<?>... itemTypes)379 final <T> T getValue(String key, @Nullable Class<T> clazz, @Nullable Class<?>... itemTypes) { 380 int i = mMap.indexOfKey(key); 381 return (i >= 0) ? getValueAt(i, clazz, itemTypes) : null; 382 } 383 384 /** 385 * Returns the value for a certain position in the array map for expected return type {@code 386 * clazz} (or pass {@code null} for no type check). 387 * 388 * For {@code itemTypes}, see {@link Parcel#readValue(int, ClassLoader, Class, Class[])}. 389 * 390 * This call should always be made after {@link #unparcel()} or inside a lock after making sure 391 * {@code mMap} is not null. 392 * 393 * @hide 394 */ 395 @SuppressWarnings("unchecked") 396 @Nullable getValueAt(int i, @Nullable Class<T> clazz, @Nullable Class<?>... itemTypes)397 final <T> T getValueAt(int i, @Nullable Class<T> clazz, @Nullable Class<?>... itemTypes) { 398 Object object = mMap.valueAt(i); 399 if (object instanceof BiFunction<?, ?, ?>) { 400 synchronized (this) { 401 object = unwrapLazyValueFromMapLocked(i, clazz, itemTypes); 402 } 403 } 404 return (clazz != null) ? clazz.cast(object) : (T) object; 405 } 406 407 @SuppressWarnings("unchecked") 408 @Nullable 409 @GuardedBy("this") unwrapLazyValueFromMapLocked(int i, @Nullable Class<?> clazz, @Nullable Class<?>... itemTypes)410 private Object unwrapLazyValueFromMapLocked(int i, @Nullable Class<?> clazz, 411 @Nullable Class<?>... itemTypes) { 412 Object object = mMap.valueAt(i); 413 if (object instanceof BiFunction<?, ?, ?>) { 414 try { 415 object = ((BiFunction<Class<?>, Class<?>[], ?>) object).apply(clazz, itemTypes); 416 } catch (BadParcelableException e) { 417 if (sShouldDefuse) { 418 Log.w(TAG, "Failed to parse item " + mMap.keyAt(i) + ", returning null.", e); 419 return null; 420 } else { 421 throw e; 422 } 423 } 424 mMap.setValueAt(i, object); 425 mLazyValues--; 426 if (mOwnsLazyValues) { 427 Preconditions.checkState(mLazyValues >= 0, 428 "Lazy values ref count below 0"); 429 // No more lazy values in mMap, so we can recycle the parcel early rather than 430 // waiting for the next GC run 431 if (mLazyValues == 0) { 432 Preconditions.checkState(mWeakParcelledData.get() != null, 433 "Parcel recycled earlier than expected"); 434 recycleParcel(mWeakParcelledData.get()); 435 mWeakParcelledData = null; 436 } 437 } 438 } 439 return object; 440 } 441 initializeFromParcelLocked(@onNull Parcel parcelledData, boolean ownsParcel, boolean parcelledByNative)442 private void initializeFromParcelLocked(@NonNull Parcel parcelledData, boolean ownsParcel, 443 boolean parcelledByNative) { 444 if (isEmptyParcel(parcelledData)) { 445 if (DEBUG) { 446 Log.d(TAG, "unparcel " 447 + Integer.toHexString(System.identityHashCode(this)) + ": empty"); 448 } 449 if (mMap == null) { 450 mMap = new ArrayMap<>(1); 451 } else { 452 mMap.erase(); 453 } 454 mParcelledByNative = false; 455 mParcelledData = null; 456 return; 457 } 458 459 final int count = parcelledData.readInt(); 460 if (DEBUG) { 461 Log.d(TAG, "unparcel " + Integer.toHexString(System.identityHashCode(this)) 462 + ": reading " + count + " maps"); 463 } 464 if (count < 0) { 465 return; 466 } 467 ArrayMap<String, Object> map = mMap; 468 if (map == null) { 469 map = new ArrayMap<>(count); 470 } else { 471 map.erase(); 472 map.ensureCapacity(count); 473 } 474 int numLazyValues = 0; 475 try { 476 numLazyValues = parcelledData.readArrayMap(map, count, !parcelledByNative, 477 /* lazy */ ownsParcel, mClassLoader); 478 } catch (BadParcelableException e) { 479 if (sShouldDefuse) { 480 Log.w(TAG, "Failed to parse Bundle, but defusing quietly", e); 481 map.erase(); 482 } else { 483 throw e; 484 } 485 } finally { 486 mWeakParcelledData = null; 487 if (ownsParcel) { 488 if (numLazyValues == 0) { 489 recycleParcel(parcelledData); 490 } else { 491 mWeakParcelledData = new WeakReference<>(parcelledData); 492 } 493 } 494 495 mLazyValues = numLazyValues; 496 mParcelledByNative = false; 497 mMap = map; 498 // Set field last as it is volatile 499 mParcelledData = null; 500 } 501 if (DEBUG) { 502 Log.d(TAG, "unparcel " + Integer.toHexString(System.identityHashCode(this)) 503 + " final map: " + mMap); 504 } 505 } 506 507 /** 508 * @hide 509 */ 510 @UnsupportedAppUsage(maxTargetSdk = Build.VERSION_CODES.R, trackingBug = 170729553) isParcelled()511 public boolean isParcelled() { 512 return mParcelledData != null; 513 } 514 515 /** 516 * @hide 517 */ isEmptyParcel()518 public boolean isEmptyParcel() { 519 return isEmptyParcel(mParcelledData); 520 } 521 522 /** 523 * @hide 524 */ isEmptyParcel(Parcel p)525 private static boolean isEmptyParcel(Parcel p) { 526 return p == NoImagePreloadHolder.EMPTY_PARCEL; 527 } 528 recycleParcel(Parcel p)529 private static void recycleParcel(Parcel p) { 530 if (p != null && !isEmptyParcel(p)) { 531 p.recycle(); 532 } 533 } 534 535 /** 536 * Returns the backing map of this bundle after deserializing every item. 537 * 538 * <p><b>Warning:</b> This method will deserialize every item on the bundle, including custom 539 * types such as {@link Parcelable} and {@link Serializable}, so only use this when you trust 540 * the source. Specifically don't use this method on app-provided bundles. 541 * 542 * @hide 543 */ getItemwiseMap()544 ArrayMap<String, Object> getItemwiseMap() { 545 unparcel(/* itemwise */ true); 546 return mMap; 547 } 548 549 /** 550 * Returns the number of mappings contained in this Bundle. 551 * 552 * @return the number of mappings as an int. 553 */ size()554 public int size() { 555 unparcel(); 556 return mMap.size(); 557 } 558 559 /** 560 * Returns true if the mapping of this Bundle is empty, false otherwise. 561 */ isEmpty()562 public boolean isEmpty() { 563 unparcel(); 564 return mMap.isEmpty(); 565 } 566 567 /** 568 * This method returns true when the parcel is 'definitely' empty. 569 * That is, it may return false for an empty parcel. But will never return true for a non-empty 570 * one. 571 * 572 * @hide this should probably be the implementation of isEmpty(). To do that we 573 * need to ensure we always use the special empty parcel form when the bundle is 574 * empty. (This may already be the case, but to be safe we'll do this later when 575 * we aren't trying to stabilize.) 576 */ isDefinitelyEmpty()577 public boolean isDefinitelyEmpty() { 578 if (isParcelled()) { 579 return isEmptyParcel(); 580 } else { 581 return isEmpty(); 582 } 583 } 584 585 /** 586 * Does a loose equality check between two given {@link BaseBundle} objects. 587 * Returns {@code true} if both are {@code null}, or if both are equal as per 588 * {@link #kindofEquals(BaseBundle)} 589 * 590 * @param a A {@link BaseBundle} object 591 * @param b Another {@link BaseBundle} to compare with a 592 * @return {@code true} if both are the same, {@code false} otherwise 593 * 594 * @see #kindofEquals(BaseBundle) 595 * 596 * @hide 597 */ kindofEquals(@ullable BaseBundle a, @Nullable BaseBundle b)598 public static boolean kindofEquals(@Nullable BaseBundle a, @Nullable BaseBundle b) { 599 return (a == b) || (a != null && a.kindofEquals(b)); 600 } 601 602 /** 603 * Performs a loose equality check, which means there can be false negatives but if the method 604 * returns true than both objects are guaranteed to be equal. 605 * 606 * The point is that this method is a light-weight check in performance terms. 607 * 608 * @hide 609 */ kindofEquals(BaseBundle other)610 public boolean kindofEquals(BaseBundle other) { 611 if (other == null) { 612 return false; 613 } 614 if (isDefinitelyEmpty() && other.isDefinitelyEmpty()) { 615 return true; 616 } 617 if (isParcelled() != other.isParcelled()) { 618 // Big kind-of here! 619 return false; 620 } else if (isParcelled()) { 621 return mParcelledData.compareData(other.mParcelledData) == 0; 622 } else { 623 // Following semantic above of failing in case we get a serialized value vs a 624 // deserialized one, we'll compare the map. If a certain element hasn't been 625 // deserialized yet, it's a function object (or more specifically a LazyValue, but let's 626 // pretend we don't know that here :P), we'll use that element's equality comparison as 627 // map naturally does. That will takes care of comparing the payload if needed (see 628 // Parcel.readLazyValue() for details). 629 return mMap.equals(other.mMap); 630 } 631 } 632 633 /** 634 * Removes all elements from the mapping of this Bundle. 635 * Recycles the underlying parcel if it is still present. 636 */ clear()637 public void clear() { 638 unparcel(); 639 if (mOwnsLazyValues && mWeakParcelledData != null) { 640 recycleParcel(mWeakParcelledData.get()); 641 } 642 643 mWeakParcelledData = null; 644 mLazyValues = 0; 645 mOwnsLazyValues = true; 646 mMap.clear(); 647 } 648 deepCopyValue(Object value)649 private Object deepCopyValue(Object value) { 650 if (value == null) { 651 return null; 652 } 653 if (value instanceof Bundle) { 654 return ((Bundle)value).deepCopy(); 655 } else if (value instanceof PersistableBundle) { 656 return ((PersistableBundle)value).deepCopy(); 657 } else if (value instanceof ArrayList) { 658 return deepcopyArrayList((ArrayList) value); 659 } else if (value.getClass().isArray()) { 660 if (value instanceof int[]) { 661 return ((int[])value).clone(); 662 } else if (value instanceof long[]) { 663 return ((long[])value).clone(); 664 } else if (value instanceof float[]) { 665 return ((float[])value).clone(); 666 } else if (value instanceof double[]) { 667 return ((double[])value).clone(); 668 } else if (value instanceof Object[]) { 669 return ((Object[])value).clone(); 670 } else if (value instanceof byte[]) { 671 return ((byte[])value).clone(); 672 } else if (value instanceof short[]) { 673 return ((short[])value).clone(); 674 } else if (value instanceof char[]) { 675 return ((char[]) value).clone(); 676 } 677 } 678 return value; 679 } 680 deepcopyArrayList(ArrayList from)681 private ArrayList deepcopyArrayList(ArrayList from) { 682 final int N = from.size(); 683 ArrayList out = new ArrayList(N); 684 for (int i=0; i<N; i++) { 685 out.add(deepCopyValue(from.get(i))); 686 } 687 return out; 688 } 689 690 /** 691 * Returns true if the given key is contained in the mapping 692 * of this Bundle. 693 * 694 * @param key a String key 695 * @return true if the key is part of the mapping, false otherwise 696 */ containsKey(String key)697 public boolean containsKey(String key) { 698 unparcel(); 699 return mMap.containsKey(key); 700 } 701 702 /** 703 * Returns the entry with the given key as an object. 704 * 705 * @param key a String key 706 * @return an Object, or null 707 * 708 * @deprecated Use the type-safe specific APIs depending on the type of the item to be 709 * retrieved, eg. {@link #getString(String)}. 710 */ 711 @Deprecated 712 @Nullable get(String key)713 public Object get(String key) { 714 unparcel(); 715 return getValue(key); 716 } 717 718 /** 719 * Returns the object of type {@code clazz} for the given {@code key}, or {@code null} if: 720 * <ul> 721 * <li>No mapping of the desired type exists for the given key. 722 * <li>A {@code null} value is explicitly associated with the key. 723 * <li>The object is not of type {@code clazz}. 724 * </ul> 725 * 726 * <p>Use the more specific APIs where possible, especially in the case of containers such as 727 * lists, since those APIs allow you to specify the type of the items. 728 * 729 * @param key String key 730 * @param clazz The type of the object expected 731 * @return an Object, or null 732 */ 733 @Nullable get(@ullable String key, @NonNull Class<T> clazz)734 <T> T get(@Nullable String key, @NonNull Class<T> clazz) { 735 unparcel(); 736 try { 737 return getValue(key, requireNonNull(clazz)); 738 } catch (ClassCastException | BadTypeParcelableException e) { 739 typeWarning(key, clazz.getCanonicalName(), e); 740 return null; 741 } 742 } 743 744 /** 745 * Removes any entry with the given key from the mapping of this Bundle. 746 * 747 * @param key a String key 748 */ remove(String key)749 public void remove(String key) { 750 unparcel(); 751 mMap.remove(key); 752 } 753 754 /** 755 * Inserts all mappings from the given PersistableBundle into this BaseBundle. 756 * 757 * @param bundle a PersistableBundle 758 */ putAll(PersistableBundle bundle)759 public void putAll(PersistableBundle bundle) { 760 unparcel(); 761 bundle.unparcel(); 762 mMap.putAll(bundle.mMap); 763 } 764 765 /** 766 * Inserts all mappings from the given Map into this BaseBundle. 767 * 768 * @param map a Map 769 */ putAll(ArrayMap map)770 void putAll(ArrayMap map) { 771 unparcel(); 772 mMap.putAll(map); 773 } 774 775 /** 776 * Returns a Set containing the Strings used as keys in this Bundle. 777 * 778 * @return a Set of String keys 779 */ keySet()780 public Set<String> keySet() { 781 unparcel(); 782 return mMap.keySet(); 783 } 784 785 /** {@hide} */ putObject(@ullable String key, @Nullable Object value)786 public void putObject(@Nullable String key, @Nullable Object value) { 787 if (value == null) { 788 putString(key, null); 789 } else if (value instanceof Boolean) { 790 putBoolean(key, (Boolean) value); 791 } else if (value instanceof Integer) { 792 putInt(key, (Integer) value); 793 } else if (value instanceof Long) { 794 putLong(key, (Long) value); 795 } else if (value instanceof Double) { 796 putDouble(key, (Double) value); 797 } else if (value instanceof String) { 798 putString(key, (String) value); 799 } else if (value instanceof boolean[]) { 800 putBooleanArray(key, (boolean[]) value); 801 } else if (value instanceof int[]) { 802 putIntArray(key, (int[]) value); 803 } else if (value instanceof long[]) { 804 putLongArray(key, (long[]) value); 805 } else if (value instanceof double[]) { 806 putDoubleArray(key, (double[]) value); 807 } else if (value instanceof String[]) { 808 putStringArray(key, (String[]) value); 809 } else { 810 throw new IllegalArgumentException("Unsupported type " + value.getClass()); 811 } 812 } 813 814 /** 815 * Inserts a Boolean value into the mapping of this Bundle, replacing 816 * any existing value for the given key. Either key or value may be null. 817 * 818 * @param key a String, or null 819 * @param value a boolean 820 */ putBoolean(@ullable String key, boolean value)821 public void putBoolean(@Nullable String key, boolean value) { 822 unparcel(); 823 mMap.put(key, value); 824 } 825 826 /** 827 * Inserts a byte value into the mapping of this Bundle, replacing 828 * any existing value for the given key. 829 * 830 * @param key a String, or null 831 * @param value a byte 832 */ putByte(@ullable String key, byte value)833 void putByte(@Nullable String key, byte value) { 834 unparcel(); 835 mMap.put(key, value); 836 } 837 838 /** 839 * Inserts a char value into the mapping of this Bundle, replacing 840 * any existing value for the given key. 841 * 842 * @param key a String, or null 843 * @param value a char 844 */ putChar(@ullable String key, char value)845 void putChar(@Nullable String key, char value) { 846 unparcel(); 847 mMap.put(key, value); 848 } 849 850 /** 851 * Inserts a short value into the mapping of this Bundle, replacing 852 * any existing value for the given key. 853 * 854 * @param key a String, or null 855 * @param value a short 856 */ putShort(@ullable String key, short value)857 void putShort(@Nullable String key, short value) { 858 unparcel(); 859 mMap.put(key, value); 860 } 861 862 /** 863 * Inserts an int value into the mapping of this Bundle, replacing 864 * any existing value for the given key. 865 * 866 * @param key a String, or null 867 * @param value an int 868 */ putInt(@ullable String key, int value)869 public void putInt(@Nullable String key, int value) { 870 unparcel(); 871 mMap.put(key, value); 872 } 873 874 /** 875 * Inserts a long value into the mapping of this Bundle, replacing 876 * any existing value for the given key. 877 * 878 * @param key a String, or null 879 * @param value a long 880 */ putLong(@ullable String key, long value)881 public void putLong(@Nullable String key, long value) { 882 unparcel(); 883 mMap.put(key, value); 884 } 885 886 /** 887 * Inserts a float value into the mapping of this Bundle, replacing 888 * any existing value for the given key. 889 * 890 * @param key a String, or null 891 * @param value a float 892 */ putFloat(@ullable String key, float value)893 void putFloat(@Nullable String key, float value) { 894 unparcel(); 895 mMap.put(key, value); 896 } 897 898 /** 899 * Inserts a double value into the mapping of this Bundle, replacing 900 * any existing value for the given key. 901 * 902 * @param key a String, or null 903 * @param value a double 904 */ putDouble(@ullable String key, double value)905 public void putDouble(@Nullable String key, double value) { 906 unparcel(); 907 mMap.put(key, value); 908 } 909 910 /** 911 * Inserts a String value into the mapping of this Bundle, replacing 912 * any existing value for the given key. Either key or value may be null. 913 * 914 * @param key a String, or null 915 * @param value a String, or null 916 */ putString(@ullable String key, @Nullable String value)917 public void putString(@Nullable String key, @Nullable String value) { 918 unparcel(); 919 mMap.put(key, value); 920 } 921 922 /** 923 * Inserts a CharSequence value into the mapping of this Bundle, replacing 924 * any existing value for the given key. Either key or value may be null. 925 * 926 * @param key a String, or null 927 * @param value a CharSequence, or null 928 */ putCharSequence(@ullable String key, @Nullable CharSequence value)929 void putCharSequence(@Nullable String key, @Nullable CharSequence value) { 930 unparcel(); 931 mMap.put(key, value); 932 } 933 934 /** 935 * Inserts an ArrayList<Integer> value into the mapping of this Bundle, replacing 936 * any existing value for the given key. Either key or value may be null. 937 * 938 * @param key a String, or null 939 * @param value an ArrayList<Integer> object, or null 940 */ putIntegerArrayList(@ullable String key, @Nullable ArrayList<Integer> value)941 void putIntegerArrayList(@Nullable String key, @Nullable ArrayList<Integer> value) { 942 unparcel(); 943 mMap.put(key, value); 944 } 945 946 /** 947 * Inserts an ArrayList<String> value into the mapping of this Bundle, replacing 948 * any existing value for the given key. Either key or value may be null. 949 * 950 * @param key a String, or null 951 * @param value an ArrayList<String> object, or null 952 */ putStringArrayList(@ullable String key, @Nullable ArrayList<String> value)953 void putStringArrayList(@Nullable String key, @Nullable ArrayList<String> value) { 954 unparcel(); 955 mMap.put(key, value); 956 } 957 958 /** 959 * Inserts an ArrayList<CharSequence> value into the mapping of this Bundle, replacing 960 * any existing value for the given key. Either key or value may be null. 961 * 962 * @param key a String, or null 963 * @param value an ArrayList<CharSequence> object, or null 964 */ putCharSequenceArrayList(@ullable String key, @Nullable ArrayList<CharSequence> value)965 void putCharSequenceArrayList(@Nullable String key, @Nullable ArrayList<CharSequence> value) { 966 unparcel(); 967 mMap.put(key, value); 968 } 969 970 /** 971 * Inserts a Serializable value into the mapping of this Bundle, replacing 972 * any existing value for the given key. Either key or value may be null. 973 * 974 * @param key a String, or null 975 * @param value a Serializable object, or null 976 */ putSerializable(@ullable String key, @Nullable Serializable value)977 void putSerializable(@Nullable String key, @Nullable Serializable value) { 978 unparcel(); 979 mMap.put(key, value); 980 } 981 982 /** 983 * Inserts a boolean array value into the mapping of this Bundle, replacing 984 * any existing value for the given key. Either key or value may be null. 985 * 986 * @param key a String, or null 987 * @param value a boolean array object, or null 988 */ putBooleanArray(@ullable String key, @Nullable boolean[] value)989 public void putBooleanArray(@Nullable String key, @Nullable boolean[] value) { 990 unparcel(); 991 mMap.put(key, value); 992 } 993 994 /** 995 * Inserts a byte array value into the mapping of this Bundle, replacing 996 * any existing value for the given key. Either key or value may be null. 997 * 998 * @param key a String, or null 999 * @param value a byte array object, or null 1000 */ putByteArray(@ullable String key, @Nullable byte[] value)1001 void putByteArray(@Nullable String key, @Nullable byte[] value) { 1002 unparcel(); 1003 mMap.put(key, value); 1004 } 1005 1006 /** 1007 * Inserts a short array value into the mapping of this Bundle, replacing 1008 * any existing value for the given key. Either key or value may be null. 1009 * 1010 * @param key a String, or null 1011 * @param value a short array object, or null 1012 */ putShortArray(@ullable String key, @Nullable short[] value)1013 void putShortArray(@Nullable String key, @Nullable short[] value) { 1014 unparcel(); 1015 mMap.put(key, value); 1016 } 1017 1018 /** 1019 * Inserts a char array value into the mapping of this Bundle, replacing 1020 * any existing value for the given key. Either key or value may be null. 1021 * 1022 * @param key a String, or null 1023 * @param value a char array object, or null 1024 */ putCharArray(@ullable String key, @Nullable char[] value)1025 void putCharArray(@Nullable String key, @Nullable char[] value) { 1026 unparcel(); 1027 mMap.put(key, value); 1028 } 1029 1030 /** 1031 * Inserts an int array value into the mapping of this Bundle, replacing 1032 * any existing value for the given key. Either key or value may be null. 1033 * 1034 * @param key a String, or null 1035 * @param value an int array object, or null 1036 */ putIntArray(@ullable String key, @Nullable int[] value)1037 public void putIntArray(@Nullable String key, @Nullable int[] value) { 1038 unparcel(); 1039 mMap.put(key, value); 1040 } 1041 1042 /** 1043 * Inserts a long array value into the mapping of this Bundle, replacing 1044 * any existing value for the given key. Either key or value may be null. 1045 * 1046 * @param key a String, or null 1047 * @param value a long array object, or null 1048 */ putLongArray(@ullable String key, @Nullable long[] value)1049 public void putLongArray(@Nullable String key, @Nullable long[] value) { 1050 unparcel(); 1051 mMap.put(key, value); 1052 } 1053 1054 /** 1055 * Inserts a float array value into the mapping of this Bundle, replacing 1056 * any existing value for the given key. Either key or value may be null. 1057 * 1058 * @param key a String, or null 1059 * @param value a float array object, or null 1060 */ putFloatArray(@ullable String key, @Nullable float[] value)1061 void putFloatArray(@Nullable String key, @Nullable float[] value) { 1062 unparcel(); 1063 mMap.put(key, value); 1064 } 1065 1066 /** 1067 * Inserts a double array value into the mapping of this Bundle, replacing 1068 * any existing value for the given key. Either key or value may be null. 1069 * 1070 * @param key a String, or null 1071 * @param value a double array object, or null 1072 */ putDoubleArray(@ullable String key, @Nullable double[] value)1073 public void putDoubleArray(@Nullable String key, @Nullable double[] value) { 1074 unparcel(); 1075 mMap.put(key, value); 1076 } 1077 1078 /** 1079 * Inserts a String array value into the mapping of this Bundle, replacing 1080 * any existing value for the given key. Either key or value may be null. 1081 * 1082 * @param key a String, or null 1083 * @param value a String array object, or null 1084 */ putStringArray(@ullable String key, @Nullable String[] value)1085 public void putStringArray(@Nullable String key, @Nullable String[] value) { 1086 unparcel(); 1087 mMap.put(key, value); 1088 } 1089 1090 /** 1091 * Inserts a CharSequence array value into the mapping of this Bundle, replacing 1092 * any existing value for the given key. Either key or value may be null. 1093 * 1094 * @param key a String, or null 1095 * @param value a CharSequence array object, or null 1096 */ putCharSequenceArray(@ullable String key, @Nullable CharSequence[] value)1097 void putCharSequenceArray(@Nullable String key, @Nullable CharSequence[] value) { 1098 unparcel(); 1099 mMap.put(key, value); 1100 } 1101 1102 /** 1103 * Returns the value associated with the given key, or false if 1104 * no mapping of the desired type exists for the given key. 1105 * 1106 * @param key a String 1107 * @return a boolean value 1108 */ getBoolean(String key)1109 public boolean getBoolean(String key) { 1110 unparcel(); 1111 if (DEBUG) Log.d(TAG, "Getting boolean in " 1112 + Integer.toHexString(System.identityHashCode(this))); 1113 return getBoolean(key, false); 1114 } 1115 1116 // Log a message if the value was non-null but not of the expected type typeWarning(String key, @Nullable Object value, String className, Object defaultValue, RuntimeException e)1117 void typeWarning(String key, @Nullable Object value, String className, 1118 Object defaultValue, RuntimeException e) { 1119 StringBuilder sb = new StringBuilder(); 1120 sb.append("Key "); 1121 sb.append(key); 1122 sb.append(" expected "); 1123 sb.append(className); 1124 if (value != null) { 1125 sb.append(" but value was a "); 1126 sb.append(value.getClass().getName()); 1127 } else { 1128 sb.append(" but value was of a different type"); 1129 } 1130 sb.append(". The default value "); 1131 sb.append(defaultValue); 1132 sb.append(" was returned."); 1133 Log.w(TAG, sb.toString()); 1134 Log.w(TAG, "Attempt to cast generated internal exception:", e); 1135 } 1136 typeWarning(String key, @Nullable Object value, String className, RuntimeException e)1137 void typeWarning(String key, @Nullable Object value, String className, RuntimeException e) { 1138 typeWarning(key, value, className, "<null>", e); 1139 } 1140 typeWarning(String key, String className, RuntimeException e)1141 void typeWarning(String key, String className, RuntimeException e) { 1142 typeWarning(key, /* value */ null, className, "<null>", e); 1143 } 1144 1145 /** 1146 * Returns the value associated with the given key, or defaultValue if 1147 * no mapping of the desired type exists for the given key. 1148 * 1149 * @param key a String 1150 * @param defaultValue Value to return if key does not exist 1151 * @return a boolean value 1152 */ getBoolean(String key, boolean defaultValue)1153 public boolean getBoolean(String key, boolean defaultValue) { 1154 unparcel(); 1155 Object o = mMap.get(key); 1156 if (o == null) { 1157 return defaultValue; 1158 } 1159 try { 1160 return (Boolean) o; 1161 } catch (ClassCastException e) { 1162 typeWarning(key, o, "Boolean", defaultValue, e); 1163 return defaultValue; 1164 } 1165 } 1166 1167 /** 1168 * Returns the value associated with the given key, or (byte) 0 if 1169 * no mapping of the desired type exists for the given key. 1170 * 1171 * @param key a String 1172 * @return a byte value 1173 */ getByte(String key)1174 byte getByte(String key) { 1175 unparcel(); 1176 return getByte(key, (byte) 0); 1177 } 1178 1179 /** 1180 * Returns the value associated with the given key, or defaultValue if 1181 * no mapping of the desired type exists for the given key. 1182 * 1183 * @param key a String 1184 * @param defaultValue Value to return if key does not exist 1185 * @return a byte value 1186 */ getByte(String key, byte defaultValue)1187 Byte getByte(String key, byte defaultValue) { 1188 unparcel(); 1189 Object o = mMap.get(key); 1190 if (o == null) { 1191 return defaultValue; 1192 } 1193 try { 1194 return (Byte) o; 1195 } catch (ClassCastException e) { 1196 typeWarning(key, o, "Byte", defaultValue, e); 1197 return defaultValue; 1198 } 1199 } 1200 1201 /** 1202 * Returns the value associated with the given key, or (char) 0 if 1203 * no mapping of the desired type exists for the given key. 1204 * 1205 * @param key a String 1206 * @return a char value 1207 */ getChar(String key)1208 char getChar(String key) { 1209 unparcel(); 1210 return getChar(key, (char) 0); 1211 } 1212 1213 /** 1214 * Returns the value associated with the given key, or defaultValue if 1215 * no mapping of the desired type exists for the given key. 1216 * 1217 * @param key a String 1218 * @param defaultValue Value to return if key does not exist 1219 * @return a char value 1220 */ getChar(String key, char defaultValue)1221 char getChar(String key, char defaultValue) { 1222 unparcel(); 1223 Object o = mMap.get(key); 1224 if (o == null) { 1225 return defaultValue; 1226 } 1227 try { 1228 return (Character) o; 1229 } catch (ClassCastException e) { 1230 typeWarning(key, o, "Character", defaultValue, e); 1231 return defaultValue; 1232 } 1233 } 1234 1235 /** 1236 * Returns the value associated with the given key, or (short) 0 if 1237 * no mapping of the desired type exists for the given key. 1238 * 1239 * @param key a String 1240 * @return a short value 1241 */ getShort(String key)1242 short getShort(String key) { 1243 unparcel(); 1244 return getShort(key, (short) 0); 1245 } 1246 1247 /** 1248 * Returns the value associated with the given key, or defaultValue if 1249 * no mapping of the desired type exists for the given key. 1250 * 1251 * @param key a String 1252 * @param defaultValue Value to return if key does not exist 1253 * @return a short value 1254 */ getShort(String key, short defaultValue)1255 short getShort(String key, short defaultValue) { 1256 unparcel(); 1257 Object o = mMap.get(key); 1258 if (o == null) { 1259 return defaultValue; 1260 } 1261 try { 1262 return (Short) o; 1263 } catch (ClassCastException e) { 1264 typeWarning(key, o, "Short", defaultValue, e); 1265 return defaultValue; 1266 } 1267 } 1268 1269 /** 1270 * Returns the value associated with the given key, or 0 if 1271 * no mapping of the desired type exists for the given key. 1272 * 1273 * @param key a String 1274 * @return an int value 1275 */ getInt(String key)1276 public int getInt(String key) { 1277 unparcel(); 1278 return getInt(key, 0); 1279 } 1280 1281 /** 1282 * Returns the value associated with the given key, or defaultValue if 1283 * no mapping of the desired type exists for the given key. 1284 * 1285 * @param key a String 1286 * @param defaultValue Value to return if key does not exist 1287 * @return an int value 1288 */ getInt(String key, int defaultValue)1289 public int getInt(String key, int defaultValue) { 1290 unparcel(); 1291 Object o = mMap.get(key); 1292 if (o == null) { 1293 return defaultValue; 1294 } 1295 try { 1296 return (Integer) o; 1297 } catch (ClassCastException e) { 1298 typeWarning(key, o, "Integer", defaultValue, e); 1299 return defaultValue; 1300 } 1301 } 1302 1303 /** 1304 * Returns the value associated with the given key, or 0L if 1305 * no mapping of the desired type exists for the given key. 1306 * 1307 * @param key a String 1308 * @return a long value 1309 */ getLong(String key)1310 public long getLong(String key) { 1311 unparcel(); 1312 return getLong(key, 0L); 1313 } 1314 1315 /** 1316 * Returns the value associated with the given key, or defaultValue if 1317 * no mapping of the desired type exists for the given key. 1318 * 1319 * @param key a String 1320 * @param defaultValue Value to return if key does not exist 1321 * @return a long value 1322 */ getLong(String key, long defaultValue)1323 public long getLong(String key, long defaultValue) { 1324 unparcel(); 1325 Object o = mMap.get(key); 1326 if (o == null) { 1327 return defaultValue; 1328 } 1329 try { 1330 return (Long) o; 1331 } catch (ClassCastException e) { 1332 typeWarning(key, o, "Long", defaultValue, e); 1333 return defaultValue; 1334 } 1335 } 1336 1337 /** 1338 * Returns the value associated with the given key, or 0.0f if 1339 * no mapping of the desired type exists for the given key. 1340 * 1341 * @param key a String 1342 * @return a float value 1343 */ getFloat(String key)1344 float getFloat(String key) { 1345 unparcel(); 1346 return getFloat(key, 0.0f); 1347 } 1348 1349 /** 1350 * Returns the value associated with the given key, or defaultValue if 1351 * no mapping of the desired type exists for the given key. 1352 * 1353 * @param key a String 1354 * @param defaultValue Value to return if key does not exist 1355 * @return a float value 1356 */ getFloat(String key, float defaultValue)1357 float getFloat(String key, float defaultValue) { 1358 unparcel(); 1359 Object o = mMap.get(key); 1360 if (o == null) { 1361 return defaultValue; 1362 } 1363 try { 1364 return (Float) o; 1365 } catch (ClassCastException e) { 1366 typeWarning(key, o, "Float", defaultValue, e); 1367 return defaultValue; 1368 } 1369 } 1370 1371 /** 1372 * Returns the value associated with the given key, or 0.0 if 1373 * no mapping of the desired type exists for the given key. 1374 * 1375 * @param key a String 1376 * @return a double value 1377 */ getDouble(String key)1378 public double getDouble(String key) { 1379 unparcel(); 1380 return getDouble(key, 0.0); 1381 } 1382 1383 /** 1384 * Returns the value associated with the given key, or defaultValue if 1385 * no mapping of the desired type exists for the given key. 1386 * 1387 * @param key a String 1388 * @param defaultValue Value to return if key does not exist 1389 * @return a double value 1390 */ getDouble(String key, double defaultValue)1391 public double getDouble(String key, double defaultValue) { 1392 unparcel(); 1393 Object o = mMap.get(key); 1394 if (o == null) { 1395 return defaultValue; 1396 } 1397 try { 1398 return (Double) o; 1399 } catch (ClassCastException e) { 1400 typeWarning(key, o, "Double", defaultValue, e); 1401 return defaultValue; 1402 } 1403 } 1404 1405 /** 1406 * Returns the value associated with the given key, or null if 1407 * no mapping of the desired type exists for the given key or a null 1408 * value is explicitly associated with the key. 1409 * 1410 * @param key a String, or null 1411 * @return a String value, or null 1412 */ 1413 @Nullable getString(@ullable String key)1414 public String getString(@Nullable String key) { 1415 unparcel(); 1416 final Object o = mMap.get(key); 1417 try { 1418 return (String) o; 1419 } catch (ClassCastException e) { 1420 typeWarning(key, o, "String", e); 1421 return null; 1422 } 1423 } 1424 1425 /** 1426 * Returns the value associated with the given key, or defaultValue if 1427 * no mapping of the desired type exists for the given key or if a null 1428 * value is explicitly associated with the given key. 1429 * 1430 * @param key a String, or null 1431 * @param defaultValue Value to return if key does not exist or if a null 1432 * value is associated with the given key. 1433 * @return the String value associated with the given key, or defaultValue 1434 * if no valid String object is currently mapped to that key. 1435 */ getString(@ullable String key, String defaultValue)1436 public String getString(@Nullable String key, String defaultValue) { 1437 final String s = getString(key); 1438 return (s == null) ? defaultValue : s; 1439 } 1440 1441 /** 1442 * Returns the value associated with the given key, or null if 1443 * no mapping of the desired type exists for the given key or a null 1444 * value is explicitly associated with the key. 1445 * 1446 * @param key a String, or null 1447 * @return a CharSequence value, or null 1448 */ 1449 @Nullable getCharSequence(@ullable String key)1450 CharSequence getCharSequence(@Nullable String key) { 1451 unparcel(); 1452 final Object o = mMap.get(key); 1453 try { 1454 return (CharSequence) o; 1455 } catch (ClassCastException e) { 1456 typeWarning(key, o, "CharSequence", e); 1457 return null; 1458 } 1459 } 1460 1461 /** 1462 * Returns the value associated with the given key, or defaultValue if 1463 * no mapping of the desired type exists for the given key or if a null 1464 * value is explicitly associated with the given key. 1465 * 1466 * @param key a String, or null 1467 * @param defaultValue Value to return if key does not exist or if a null 1468 * value is associated with the given key. 1469 * @return the CharSequence value associated with the given key, or defaultValue 1470 * if no valid CharSequence object is currently mapped to that key. 1471 */ getCharSequence(@ullable String key, CharSequence defaultValue)1472 CharSequence getCharSequence(@Nullable String key, CharSequence defaultValue) { 1473 final CharSequence cs = getCharSequence(key); 1474 return (cs == null) ? defaultValue : cs; 1475 } 1476 1477 /** 1478 * Returns the value associated with the given key, or null if 1479 * no mapping of the desired type exists for the given key or a null 1480 * value is explicitly associated with the key. 1481 * 1482 * @param key a String, or null 1483 * @return a Serializable value, or null 1484 * 1485 * @deprecated Use {@link #getSerializable(String, Class)}. This method should only be used in 1486 * other deprecated APIs. 1487 */ 1488 @Deprecated 1489 @Nullable getSerializable(@ullable String key)1490 Serializable getSerializable(@Nullable String key) { 1491 unparcel(); 1492 Object o = getValue(key); 1493 if (o == null) { 1494 return null; 1495 } 1496 try { 1497 return (Serializable) o; 1498 } catch (ClassCastException e) { 1499 typeWarning(key, o, "Serializable", e); 1500 return null; 1501 } 1502 } 1503 1504 /** 1505 * Returns the value associated with the given key, or {@code null} if: 1506 * <ul> 1507 * <li>No mapping of the desired type exists for the given key. 1508 * <li>A {@code null} value is explicitly associated with the key. 1509 * <li>The object is not of type {@code clazz}. 1510 * </ul> 1511 * 1512 * @param key a String, or null 1513 * @param clazz The expected class of the returned type 1514 * @return a Serializable value, or null 1515 */ 1516 @Nullable getSerializable(@ullable String key, @NonNull Class<T> clazz)1517 <T extends Serializable> T getSerializable(@Nullable String key, @NonNull Class<T> clazz) { 1518 return get(key, clazz); 1519 } 1520 1521 1522 @SuppressWarnings("unchecked") 1523 @Nullable getArrayList(@ullable String key, @NonNull Class<? extends T> clazz)1524 <T> ArrayList<T> getArrayList(@Nullable String key, @NonNull Class<? extends T> clazz) { 1525 unparcel(); 1526 try { 1527 return getValue(key, ArrayList.class, requireNonNull(clazz)); 1528 } catch (ClassCastException | BadTypeParcelableException e) { 1529 typeWarning(key, "ArrayList<" + clazz.getCanonicalName() + ">", e); 1530 return null; 1531 } 1532 } 1533 1534 /** 1535 * Returns the value associated with the given key, or null if 1536 * no mapping of the desired type exists for the given key or a null 1537 * value is explicitly associated with the key. 1538 * 1539 * @param key a String, or null 1540 * @return an ArrayList<String> value, or null 1541 */ 1542 @Nullable getIntegerArrayList(@ullable String key)1543 ArrayList<Integer> getIntegerArrayList(@Nullable String key) { 1544 return getArrayList(key, Integer.class); 1545 } 1546 1547 /** 1548 * Returns the value associated with the given key, or null if 1549 * no mapping of the desired type exists for the given key or a null 1550 * value is explicitly associated with the key. 1551 * 1552 * @param key a String, or null 1553 * @return an ArrayList<String> value, or null 1554 */ 1555 @Nullable getStringArrayList(@ullable String key)1556 ArrayList<String> getStringArrayList(@Nullable String key) { 1557 return getArrayList(key, String.class); 1558 } 1559 1560 /** 1561 * Returns the value associated with the given key, or null if 1562 * no mapping of the desired type exists for the given key or a null 1563 * value is explicitly associated with the key. 1564 * 1565 * @param key a String, or null 1566 * @return an ArrayList<CharSequence> value, or null 1567 */ 1568 @Nullable getCharSequenceArrayList(@ullable String key)1569 ArrayList<CharSequence> getCharSequenceArrayList(@Nullable String key) { 1570 return getArrayList(key, CharSequence.class); 1571 } 1572 1573 /** 1574 * Returns the value associated with the given key, or null if 1575 * no mapping of the desired type exists for the given key or a null 1576 * value is explicitly associated with the key. 1577 * 1578 * @param key a String, or null 1579 * @return a boolean[] value, or null 1580 */ 1581 @Nullable getBooleanArray(@ullable String key)1582 public boolean[] getBooleanArray(@Nullable String key) { 1583 unparcel(); 1584 Object o = mMap.get(key); 1585 if (o == null) { 1586 return null; 1587 } 1588 try { 1589 return (boolean[]) o; 1590 } catch (ClassCastException e) { 1591 typeWarning(key, o, "byte[]", e); 1592 return null; 1593 } 1594 } 1595 1596 /** 1597 * Returns the value associated with the given key, or null if 1598 * no mapping of the desired type exists for the given key or a null 1599 * value is explicitly associated with the key. 1600 * 1601 * @param key a String, or null 1602 * @return a byte[] value, or null 1603 */ 1604 @Nullable getByteArray(@ullable String key)1605 byte[] getByteArray(@Nullable String key) { 1606 unparcel(); 1607 Object o = mMap.get(key); 1608 if (o == null) { 1609 return null; 1610 } 1611 try { 1612 return (byte[]) o; 1613 } catch (ClassCastException e) { 1614 typeWarning(key, o, "byte[]", e); 1615 return null; 1616 } 1617 } 1618 1619 /** 1620 * Returns the value associated with the given key, or null if 1621 * no mapping of the desired type exists for the given key or a null 1622 * value is explicitly associated with the key. 1623 * 1624 * @param key a String, or null 1625 * @return a short[] value, or null 1626 */ 1627 @Nullable getShortArray(@ullable String key)1628 short[] getShortArray(@Nullable String key) { 1629 unparcel(); 1630 Object o = mMap.get(key); 1631 if (o == null) { 1632 return null; 1633 } 1634 try { 1635 return (short[]) o; 1636 } catch (ClassCastException e) { 1637 typeWarning(key, o, "short[]", e); 1638 return null; 1639 } 1640 } 1641 1642 /** 1643 * Returns the value associated with the given key, or null if 1644 * no mapping of the desired type exists for the given key or a null 1645 * value is explicitly associated with the key. 1646 * 1647 * @param key a String, or null 1648 * @return a char[] value, or null 1649 */ 1650 @Nullable getCharArray(@ullable String key)1651 char[] getCharArray(@Nullable String key) { 1652 unparcel(); 1653 Object o = mMap.get(key); 1654 if (o == null) { 1655 return null; 1656 } 1657 try { 1658 return (char[]) o; 1659 } catch (ClassCastException e) { 1660 typeWarning(key, o, "char[]", e); 1661 return null; 1662 } 1663 } 1664 1665 /** 1666 * Returns the value associated with the given key, or null if 1667 * no mapping of the desired type exists for the given key or a null 1668 * value is explicitly associated with the key. 1669 * 1670 * @param key a String, or null 1671 * @return an int[] value, or null 1672 */ 1673 @Nullable getIntArray(@ullable String key)1674 public int[] getIntArray(@Nullable String key) { 1675 unparcel(); 1676 Object o = mMap.get(key); 1677 if (o == null) { 1678 return null; 1679 } 1680 try { 1681 return (int[]) o; 1682 } catch (ClassCastException e) { 1683 typeWarning(key, o, "int[]", e); 1684 return null; 1685 } 1686 } 1687 1688 /** 1689 * Returns the value associated with the given key, or null if 1690 * no mapping of the desired type exists for the given key or a null 1691 * value is explicitly associated with the key. 1692 * 1693 * @param key a String, or null 1694 * @return a long[] value, or null 1695 */ 1696 @Nullable getLongArray(@ullable String key)1697 public long[] getLongArray(@Nullable String key) { 1698 unparcel(); 1699 Object o = mMap.get(key); 1700 if (o == null) { 1701 return null; 1702 } 1703 try { 1704 return (long[]) o; 1705 } catch (ClassCastException e) { 1706 typeWarning(key, o, "long[]", e); 1707 return null; 1708 } 1709 } 1710 1711 /** 1712 * Returns the value associated with the given key, or null if 1713 * no mapping of the desired type exists for the given key or a null 1714 * value is explicitly associated with the key. 1715 * 1716 * @param key a String, or null 1717 * @return a float[] value, or null 1718 */ 1719 @Nullable getFloatArray(@ullable String key)1720 float[] getFloatArray(@Nullable String key) { 1721 unparcel(); 1722 Object o = mMap.get(key); 1723 if (o == null) { 1724 return null; 1725 } 1726 try { 1727 return (float[]) o; 1728 } catch (ClassCastException e) { 1729 typeWarning(key, o, "float[]", e); 1730 return null; 1731 } 1732 } 1733 1734 /** 1735 * Returns the value associated with the given key, or null if 1736 * no mapping of the desired type exists for the given key or a null 1737 * value is explicitly associated with the key. 1738 * 1739 * @param key a String, or null 1740 * @return a double[] value, or null 1741 */ 1742 @Nullable getDoubleArray(@ullable String key)1743 public double[] getDoubleArray(@Nullable String key) { 1744 unparcel(); 1745 Object o = mMap.get(key); 1746 if (o == null) { 1747 return null; 1748 } 1749 try { 1750 return (double[]) o; 1751 } catch (ClassCastException e) { 1752 typeWarning(key, o, "double[]", e); 1753 return null; 1754 } 1755 } 1756 1757 /** 1758 * Returns the value associated with the given key, or null if 1759 * no mapping of the desired type exists for the given key or a null 1760 * value is explicitly associated with the key. 1761 * 1762 * @param key a String, or null 1763 * @return a String[] value, or null 1764 */ 1765 @Nullable getStringArray(@ullable String key)1766 public String[] getStringArray(@Nullable String key) { 1767 unparcel(); 1768 Object o = mMap.get(key); 1769 if (o == null) { 1770 return null; 1771 } 1772 try { 1773 return (String[]) o; 1774 } catch (ClassCastException e) { 1775 typeWarning(key, o, "String[]", e); 1776 return null; 1777 } 1778 } 1779 1780 /** 1781 * Returns the value associated with the given key, or null if 1782 * no mapping of the desired type exists for the given key or a null 1783 * value is explicitly associated with the key. 1784 * 1785 * @param key a String, or null 1786 * @return a CharSequence[] value, or null 1787 */ 1788 @Nullable getCharSequenceArray(@ullable String key)1789 CharSequence[] getCharSequenceArray(@Nullable String key) { 1790 unparcel(); 1791 Object o = mMap.get(key); 1792 if (o == null) { 1793 return null; 1794 } 1795 try { 1796 return (CharSequence[]) o; 1797 } catch (ClassCastException e) { 1798 typeWarning(key, o, "CharSequence[]", e); 1799 return null; 1800 } 1801 } 1802 1803 /** 1804 * Writes the Bundle contents to a Parcel, typically in order for 1805 * it to be passed through an IBinder connection. 1806 * @param parcel The parcel to copy this bundle to. 1807 */ writeToParcelInner(Parcel parcel, int flags)1808 void writeToParcelInner(Parcel parcel, int flags) { 1809 // If the parcel has a read-write helper, we can't just copy the blob, so unparcel it first. 1810 if (parcel.hasReadWriteHelper()) { 1811 unparcel(/* itemwise */ true); 1812 } 1813 // Keep implementation in sync with writeToParcel() in 1814 // frameworks/native/libs/binder/PersistableBundle.cpp. 1815 final ArrayMap<String, Object> map; 1816 synchronized (this) { 1817 // unparcel() can race with this method and cause the parcel to recycle 1818 // at the wrong time. So synchronize access the mParcelledData's content. 1819 if (mParcelledData != null) { 1820 if (mParcelledData == NoImagePreloadHolder.EMPTY_PARCEL) { 1821 parcel.writeInt(0); 1822 } else { 1823 int length = mParcelledData.dataSize(); 1824 parcel.writeInt(length); 1825 parcel.writeInt(mParcelledByNative ? BUNDLE_MAGIC_NATIVE : BUNDLE_MAGIC); 1826 parcel.appendFrom(mParcelledData, 0, length); 1827 } 1828 return; 1829 } 1830 map = mMap; 1831 } 1832 1833 // Special case for empty bundles. 1834 if (map == null || map.size() <= 0) { 1835 parcel.writeInt(0); 1836 return; 1837 } 1838 int lengthPos = parcel.dataPosition(); 1839 parcel.writeInt(-1); // placeholder, will hold length 1840 parcel.writeInt(BUNDLE_MAGIC); 1841 1842 int startPos = parcel.dataPosition(); 1843 parcel.writeArrayMapInternal(map); 1844 int endPos = parcel.dataPosition(); 1845 1846 // Backpatch length 1847 parcel.setDataPosition(lengthPos); 1848 int length = endPos - startPos; 1849 parcel.writeInt(length); 1850 parcel.setDataPosition(endPos); 1851 } 1852 1853 /** 1854 * Reads the Parcel contents into this Bundle, typically in order for 1855 * it to be passed through an IBinder connection. 1856 * @param parcel The parcel to overwrite this bundle from. 1857 */ readFromParcelInner(Parcel parcel)1858 void readFromParcelInner(Parcel parcel) { 1859 // Keep implementation in sync with readFromParcel() in 1860 // frameworks/native/libs/binder/PersistableBundle.cpp. 1861 int length = parcel.readInt(); 1862 readFromParcelInner(parcel, length); 1863 } 1864 readFromParcelInner(Parcel parcel, int length)1865 private void readFromParcelInner(Parcel parcel, int length) { 1866 if (length < 0) { 1867 throw new RuntimeException("Bad length in parcel: " + length); 1868 } else if (length == 0) { 1869 mParcelledByNative = false; 1870 // Empty Bundle or end of data. 1871 mParcelledData = NoImagePreloadHolder.EMPTY_PARCEL; 1872 return; 1873 } else if (length % 4 != 0) { 1874 throw new IllegalStateException("Bundle length is not aligned by 4: " + length); 1875 } 1876 1877 final int magic = parcel.readInt(); 1878 final boolean isJavaBundle = magic == BUNDLE_MAGIC; 1879 final boolean isNativeBundle = magic == BUNDLE_MAGIC_NATIVE; 1880 if (!isJavaBundle && !isNativeBundle) { 1881 throw new IllegalStateException("Bad magic number for Bundle: 0x" 1882 + Integer.toHexString(magic)); 1883 } 1884 1885 if (parcel.hasReadWriteHelper()) { 1886 // If the parcel has a read-write helper, it's better to deserialize immediately 1887 // otherwise the helper would have to either maintain valid state long after the bundle 1888 // had been constructed with parcel or to make sure they trigger deserialization of the 1889 // bundle immediately; neither of which is obvious. 1890 synchronized (this) { 1891 mOwnsLazyValues = false; 1892 initializeFromParcelLocked(parcel, /*ownsParcel*/ false, isNativeBundle); 1893 } 1894 return; 1895 } 1896 1897 // Advance within this Parcel 1898 int offset = parcel.dataPosition(); 1899 parcel.setDataPosition(MathUtils.addOrThrow(offset, length)); 1900 1901 Parcel p = Parcel.obtain(); 1902 p.setDataPosition(0); 1903 p.appendFrom(parcel, offset, length); 1904 p.adoptClassCookies(parcel); 1905 if (DEBUG) Log.d(TAG, "Retrieving " + Integer.toHexString(System.identityHashCode(this)) 1906 + ": " + length + " bundle bytes starting at " + offset); 1907 p.setDataPosition(0); 1908 1909 mOwnsLazyValues = true; 1910 mParcelledByNative = isNativeBundle; 1911 mParcelledData = p; 1912 } 1913 1914 /** {@hide} */ dumpStats(IndentingPrintWriter pw, String key, Object value)1915 public static void dumpStats(IndentingPrintWriter pw, String key, Object value) { 1916 final Parcel tmp = Parcel.obtain(); 1917 tmp.writeValue(value); 1918 final int size = tmp.dataPosition(); 1919 tmp.recycle(); 1920 1921 // We only really care about logging large values 1922 if (size > 1024) { 1923 pw.println(key + " [size=" + size + "]"); 1924 if (value instanceof BaseBundle) { 1925 dumpStats(pw, (BaseBundle) value); 1926 } else if (value instanceof SparseArray) { 1927 dumpStats(pw, (SparseArray) value); 1928 } 1929 } 1930 } 1931 1932 /** {@hide} */ dumpStats(IndentingPrintWriter pw, SparseArray array)1933 public static void dumpStats(IndentingPrintWriter pw, SparseArray array) { 1934 pw.increaseIndent(); 1935 if (array == null) { 1936 pw.println("[null]"); 1937 return; 1938 } 1939 for (int i = 0; i < array.size(); i++) { 1940 dumpStats(pw, "0x" + Integer.toHexString(array.keyAt(i)), array.valueAt(i)); 1941 } 1942 pw.decreaseIndent(); 1943 } 1944 1945 /** {@hide} */ dumpStats(IndentingPrintWriter pw, BaseBundle bundle)1946 public static void dumpStats(IndentingPrintWriter pw, BaseBundle bundle) { 1947 pw.increaseIndent(); 1948 if (bundle == null) { 1949 pw.println("[null]"); 1950 return; 1951 } 1952 final ArrayMap<String, Object> map = bundle.getItemwiseMap(); 1953 for (int i = 0; i < map.size(); i++) { 1954 dumpStats(pw, map.keyAt(i), map.valueAt(i)); 1955 } 1956 pw.decreaseIndent(); 1957 } 1958 } 1959