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