1 /* 2 * Copyright (C) 2015 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 #define LOG_TAG "PersistableBundle" 18 19 #include <binder/PersistableBundle.h> 20 #include <private/binder/ParcelValTypes.h> 21 22 #include <limits> 23 24 #include <binder/IBinder.h> 25 #include <binder/Parcel.h> 26 #include <log/log.h> 27 #include <utils/Errors.h> 28 29 using android::BAD_TYPE; 30 using android::BAD_VALUE; 31 using android::NO_ERROR; 32 using android::Parcel; 33 using android::sp; 34 using android::status_t; 35 using android::UNEXPECTED_NULL; 36 using std::map; 37 using std::set; 38 using std::vector; 39 using namespace ::android::binder; 40 41 enum { 42 // Keep them in sync with BUNDLE_MAGIC* in frameworks/base/core/java/android/os/BaseBundle.java. 43 BUNDLE_MAGIC = 0x4C444E42, 44 BUNDLE_MAGIC_NATIVE = 0x4C444E44, 45 }; 46 47 namespace { 48 template <typename T> 49 bool getValue(const android::String16& key, T* out, const map<android::String16, T>& map) { 50 const auto& it = map.find(key); 51 if (it == map.end()) return false; 52 *out = it->second; 53 return true; 54 } 55 56 template <typename T> 57 set<android::String16> getKeys(const map<android::String16, T>& map) { 58 if (map.empty()) return set<android::String16>(); 59 set<android::String16> keys; 60 for (const auto& key_value_pair : map) { 61 keys.emplace(key_value_pair.first); 62 } 63 return keys; 64 } 65 } // namespace 66 67 namespace android { 68 69 namespace os { 70 71 #define RETURN_IF_FAILED(calledOnce) \ 72 { \ 73 status_t returnStatus = calledOnce; \ 74 if (returnStatus) { \ 75 ALOGE("Failed at %s:%d (%s)", __FILE__, __LINE__, __func__); \ 76 return returnStatus; \ 77 } \ 78 } 79 80 #define RETURN_IF_ENTRY_ERASED(map, key) \ 81 { \ 82 size_t num_erased = (map).erase(key); \ 83 if (num_erased) { \ 84 ALOGE("Failed at %s:%d (%s)", __FILE__, __LINE__, __func__); \ 85 return num_erased; \ 86 } \ 87 } 88 89 status_t PersistableBundle::writeToParcel(Parcel* parcel) const { 90 /* 91 * Keep implementation in sync with writeToParcelInner() in 92 * frameworks/base/core/java/android/os/BaseBundle.java. 93 */ 94 95 // Special case for empty bundles. 96 if (empty()) { 97 RETURN_IF_FAILED(parcel->writeInt32(0)); 98 return NO_ERROR; 99 } 100 101 size_t length_pos = parcel->dataPosition(); 102 RETURN_IF_FAILED(parcel->writeInt32(1)); // dummy, will hold length 103 RETURN_IF_FAILED(parcel->writeInt32(BUNDLE_MAGIC_NATIVE)); 104 105 size_t start_pos = parcel->dataPosition(); 106 RETURN_IF_FAILED(writeToParcelInner(parcel)); 107 size_t end_pos = parcel->dataPosition(); 108 109 // Backpatch length. This length value includes the length header. 110 parcel->setDataPosition(length_pos); 111 size_t length = end_pos - start_pos; 112 if (length > std::numeric_limits<int32_t>::max()) { 113 ALOGE("Parcel length (%zu) too large to store in 32-bit signed int", length); 114 return BAD_VALUE; 115 } 116 RETURN_IF_FAILED(parcel->writeInt32(static_cast<int32_t>(length))); 117 parcel->setDataPosition(end_pos); 118 return NO_ERROR; 119 } 120 121 status_t PersistableBundle::readFromParcel(const Parcel* parcel) { 122 /* 123 * Keep implementation in sync with readFromParcelInner() in 124 * frameworks/base/core/java/android/os/BaseBundle.java. 125 */ 126 int32_t length = parcel->readInt32(); 127 if (length < 0) { 128 ALOGE("Bad length in parcel: %d", length); 129 return UNEXPECTED_NULL; 130 } 131 132 return readFromParcelInner(parcel, static_cast<size_t>(length)); 133 } 134 135 bool PersistableBundle::empty() const { 136 return size() == 0u; 137 } 138 139 size_t PersistableBundle::size() const { 140 return (mBoolMap.size() + 141 mIntMap.size() + 142 mLongMap.size() + 143 mDoubleMap.size() + 144 mStringMap.size() + 145 mBoolVectorMap.size() + 146 mIntVectorMap.size() + 147 mLongVectorMap.size() + 148 mDoubleVectorMap.size() + 149 mStringVectorMap.size() + 150 mPersistableBundleMap.size()); 151 } 152 153 size_t PersistableBundle::erase(const String16& key) { 154 RETURN_IF_ENTRY_ERASED(mBoolMap, key); 155 RETURN_IF_ENTRY_ERASED(mIntMap, key); 156 RETURN_IF_ENTRY_ERASED(mLongMap, key); 157 RETURN_IF_ENTRY_ERASED(mDoubleMap, key); 158 RETURN_IF_ENTRY_ERASED(mStringMap, key); 159 RETURN_IF_ENTRY_ERASED(mBoolVectorMap, key); 160 RETURN_IF_ENTRY_ERASED(mIntVectorMap, key); 161 RETURN_IF_ENTRY_ERASED(mLongVectorMap, key); 162 RETURN_IF_ENTRY_ERASED(mDoubleVectorMap, key); 163 RETURN_IF_ENTRY_ERASED(mStringVectorMap, key); 164 return mPersistableBundleMap.erase(key); 165 } 166 167 void PersistableBundle::putBoolean(const String16& key, bool value) { 168 erase(key); 169 mBoolMap[key] = value; 170 } 171 172 void PersistableBundle::putInt(const String16& key, int32_t value) { 173 erase(key); 174 mIntMap[key] = value; 175 } 176 177 void PersistableBundle::putLong(const String16& key, int64_t value) { 178 erase(key); 179 mLongMap[key] = value; 180 } 181 182 void PersistableBundle::putDouble(const String16& key, double value) { 183 erase(key); 184 mDoubleMap[key] = value; 185 } 186 187 void PersistableBundle::putString(const String16& key, const String16& value) { 188 erase(key); 189 mStringMap[key] = value; 190 } 191 192 void PersistableBundle::putBooleanVector(const String16& key, const vector<bool>& value) { 193 erase(key); 194 mBoolVectorMap[key] = value; 195 } 196 197 void PersistableBundle::putIntVector(const String16& key, const vector<int32_t>& value) { 198 erase(key); 199 mIntVectorMap[key] = value; 200 } 201 202 void PersistableBundle::putLongVector(const String16& key, const vector<int64_t>& value) { 203 erase(key); 204 mLongVectorMap[key] = value; 205 } 206 207 void PersistableBundle::putDoubleVector(const String16& key, const vector<double>& value) { 208 erase(key); 209 mDoubleVectorMap[key] = value; 210 } 211 212 void PersistableBundle::putStringVector(const String16& key, const vector<String16>& value) { 213 erase(key); 214 mStringVectorMap[key] = value; 215 } 216 217 void PersistableBundle::putPersistableBundle(const String16& key, const PersistableBundle& value) { 218 erase(key); 219 mPersistableBundleMap[key] = value; 220 } 221 222 bool PersistableBundle::getBoolean(const String16& key, bool* out) const { 223 return getValue(key, out, mBoolMap); 224 } 225 226 bool PersistableBundle::getInt(const String16& key, int32_t* out) const { 227 return getValue(key, out, mIntMap); 228 } 229 230 bool PersistableBundle::getLong(const String16& key, int64_t* out) const { 231 return getValue(key, out, mLongMap); 232 } 233 234 bool PersistableBundle::getDouble(const String16& key, double* out) const { 235 return getValue(key, out, mDoubleMap); 236 } 237 238 bool PersistableBundle::getString(const String16& key, String16* out) const { 239 return getValue(key, out, mStringMap); 240 } 241 242 bool PersistableBundle::getBooleanVector(const String16& key, vector<bool>* out) const { 243 return getValue(key, out, mBoolVectorMap); 244 } 245 246 bool PersistableBundle::getIntVector(const String16& key, vector<int32_t>* out) const { 247 return getValue(key, out, mIntVectorMap); 248 } 249 250 bool PersistableBundle::getLongVector(const String16& key, vector<int64_t>* out) const { 251 return getValue(key, out, mLongVectorMap); 252 } 253 254 bool PersistableBundle::getDoubleVector(const String16& key, vector<double>* out) const { 255 return getValue(key, out, mDoubleVectorMap); 256 } 257 258 bool PersistableBundle::getStringVector(const String16& key, vector<String16>* out) const { 259 return getValue(key, out, mStringVectorMap); 260 } 261 262 bool PersistableBundle::getPersistableBundle(const String16& key, PersistableBundle* out) const { 263 return getValue(key, out, mPersistableBundleMap); 264 } 265 266 set<String16> PersistableBundle::getBooleanKeys() const { 267 return getKeys(mBoolMap); 268 } 269 270 set<String16> PersistableBundle::getIntKeys() const { 271 return getKeys(mIntMap); 272 } 273 274 set<String16> PersistableBundle::getLongKeys() const { 275 return getKeys(mLongMap); 276 } 277 278 set<String16> PersistableBundle::getDoubleKeys() const { 279 return getKeys(mDoubleMap); 280 } 281 282 set<String16> PersistableBundle::getStringKeys() const { 283 return getKeys(mStringMap); 284 } 285 286 set<String16> PersistableBundle::getBooleanVectorKeys() const { 287 return getKeys(mBoolVectorMap); 288 } 289 290 set<String16> PersistableBundle::getIntVectorKeys() const { 291 return getKeys(mIntVectorMap); 292 } 293 294 set<String16> PersistableBundle::getLongVectorKeys() const { 295 return getKeys(mLongVectorMap); 296 } 297 298 set<String16> PersistableBundle::getDoubleVectorKeys() const { 299 return getKeys(mDoubleVectorMap); 300 } 301 302 set<String16> PersistableBundle::getStringVectorKeys() const { 303 return getKeys(mStringVectorMap); 304 } 305 306 set<String16> PersistableBundle::getPersistableBundleKeys() const { 307 return getKeys(mPersistableBundleMap); 308 } 309 310 status_t PersistableBundle::writeToParcelInner(Parcel* parcel) const { 311 /* 312 * To keep this implementation in sync with writeArrayMapInternal() in 313 * frameworks/base/core/java/android/os/Parcel.java, the number of key 314 * value pairs must be written into the parcel before writing the key-value 315 * pairs themselves. 316 */ 317 size_t num_entries = size(); 318 if (num_entries > std::numeric_limits<int32_t>::max()) { 319 ALOGE("The size of this PersistableBundle (%zu) too large to store in 32-bit signed int", 320 num_entries); 321 return BAD_VALUE; 322 } 323 RETURN_IF_FAILED(parcel->writeInt32(static_cast<int32_t>(num_entries))); 324 325 for (const auto& key_val_pair : mBoolMap) { 326 RETURN_IF_FAILED(parcel->writeString16(key_val_pair.first)); 327 RETURN_IF_FAILED(parcel->writeInt32(VAL_BOOLEAN)); 328 RETURN_IF_FAILED(parcel->writeBool(key_val_pair.second)); 329 } 330 for (const auto& key_val_pair : mIntMap) { 331 RETURN_IF_FAILED(parcel->writeString16(key_val_pair.first)); 332 RETURN_IF_FAILED(parcel->writeInt32(VAL_INTEGER)); 333 RETURN_IF_FAILED(parcel->writeInt32(key_val_pair.second)); 334 } 335 for (const auto& key_val_pair : mLongMap) { 336 RETURN_IF_FAILED(parcel->writeString16(key_val_pair.first)); 337 RETURN_IF_FAILED(parcel->writeInt32(VAL_LONG)); 338 RETURN_IF_FAILED(parcel->writeInt64(key_val_pair.second)); 339 } 340 for (const auto& key_val_pair : mDoubleMap) { 341 RETURN_IF_FAILED(parcel->writeString16(key_val_pair.first)); 342 RETURN_IF_FAILED(parcel->writeInt32(VAL_DOUBLE)); 343 RETURN_IF_FAILED(parcel->writeDouble(key_val_pair.second)); 344 } 345 for (const auto& key_val_pair : mStringMap) { 346 RETURN_IF_FAILED(parcel->writeString16(key_val_pair.first)); 347 RETURN_IF_FAILED(parcel->writeInt32(VAL_STRING)); 348 RETURN_IF_FAILED(parcel->writeString16(key_val_pair.second)); 349 } 350 for (const auto& key_val_pair : mBoolVectorMap) { 351 RETURN_IF_FAILED(parcel->writeString16(key_val_pair.first)); 352 RETURN_IF_FAILED(parcel->writeInt32(VAL_BOOLEANARRAY)); 353 RETURN_IF_FAILED(parcel->writeBoolVector(key_val_pair.second)); 354 } 355 for (const auto& key_val_pair : mIntVectorMap) { 356 RETURN_IF_FAILED(parcel->writeString16(key_val_pair.first)); 357 RETURN_IF_FAILED(parcel->writeInt32(VAL_INTARRAY)); 358 RETURN_IF_FAILED(parcel->writeInt32Vector(key_val_pair.second)); 359 } 360 for (const auto& key_val_pair : mLongVectorMap) { 361 RETURN_IF_FAILED(parcel->writeString16(key_val_pair.first)); 362 RETURN_IF_FAILED(parcel->writeInt32(VAL_LONGARRAY)); 363 RETURN_IF_FAILED(parcel->writeInt64Vector(key_val_pair.second)); 364 } 365 for (const auto& key_val_pair : mDoubleVectorMap) { 366 RETURN_IF_FAILED(parcel->writeString16(key_val_pair.first)); 367 RETURN_IF_FAILED(parcel->writeInt32(VAL_DOUBLEARRAY)); 368 RETURN_IF_FAILED(parcel->writeDoubleVector(key_val_pair.second)); 369 } 370 for (const auto& key_val_pair : mStringVectorMap) { 371 RETURN_IF_FAILED(parcel->writeString16(key_val_pair.first)); 372 RETURN_IF_FAILED(parcel->writeInt32(VAL_STRINGARRAY)); 373 RETURN_IF_FAILED(parcel->writeString16Vector(key_val_pair.second)); 374 } 375 for (const auto& key_val_pair : mPersistableBundleMap) { 376 RETURN_IF_FAILED(parcel->writeString16(key_val_pair.first)); 377 RETURN_IF_FAILED(parcel->writeInt32(VAL_PERSISTABLEBUNDLE)); 378 RETURN_IF_FAILED(key_val_pair.second.writeToParcel(parcel)); 379 } 380 return NO_ERROR; 381 } 382 383 status_t PersistableBundle::readFromParcelInner(const Parcel* parcel, size_t length) { 384 /* 385 * Note: we don't actually use length for anything other than an empty PersistableBundle 386 * check, since we do not actually need to copy in an entire Parcel, unlike in the Java 387 * implementation. 388 */ 389 if (length == 0) { 390 // Empty PersistableBundle or end of data. 391 return NO_ERROR; 392 } 393 394 int32_t magic; 395 RETURN_IF_FAILED(parcel->readInt32(&magic)); 396 if (magic != BUNDLE_MAGIC && magic != BUNDLE_MAGIC_NATIVE) { 397 ALOGE("Bad magic number for PersistableBundle: 0x%08x", magic); 398 return BAD_VALUE; 399 } 400 401 /* 402 * To keep this implementation in sync with unparcel() in 403 * frameworks/base/core/java/android/os/BaseBundle.java, the number of 404 * key-value pairs must be read from the parcel before reading the key-value 405 * pairs themselves. 406 */ 407 int32_t num_entries; 408 RETURN_IF_FAILED(parcel->readInt32(&num_entries)); 409 410 for (; num_entries > 0; --num_entries) { 411 String16 key; 412 int32_t value_type; 413 RETURN_IF_FAILED(parcel->readString16(&key)); 414 RETURN_IF_FAILED(parcel->readInt32(&value_type)); 415 416 /* 417 * We assume that both the C++ and Java APIs ensure that all keys in a PersistableBundle 418 * are unique. 419 */ 420 switch (value_type) { 421 case VAL_STRING: { 422 RETURN_IF_FAILED(parcel->readString16(&mStringMap[key])); 423 break; 424 } 425 case VAL_INTEGER: { 426 RETURN_IF_FAILED(parcel->readInt32(&mIntMap[key])); 427 break; 428 } 429 case VAL_LONG: { 430 RETURN_IF_FAILED(parcel->readInt64(&mLongMap[key])); 431 break; 432 } 433 case VAL_DOUBLE: { 434 RETURN_IF_FAILED(parcel->readDouble(&mDoubleMap[key])); 435 break; 436 } 437 case VAL_BOOLEAN: { 438 RETURN_IF_FAILED(parcel->readBool(&mBoolMap[key])); 439 break; 440 } 441 case VAL_STRINGARRAY: { 442 RETURN_IF_FAILED(parcel->readString16Vector(&mStringVectorMap[key])); 443 break; 444 } 445 case VAL_INTARRAY: { 446 RETURN_IF_FAILED(parcel->readInt32Vector(&mIntVectorMap[key])); 447 break; 448 } 449 case VAL_LONGARRAY: { 450 RETURN_IF_FAILED(parcel->readInt64Vector(&mLongVectorMap[key])); 451 break; 452 } 453 case VAL_BOOLEANARRAY: { 454 RETURN_IF_FAILED(parcel->readBoolVector(&mBoolVectorMap[key])); 455 break; 456 } 457 case VAL_PERSISTABLEBUNDLE: { 458 RETURN_IF_FAILED(mPersistableBundleMap[key].readFromParcel(parcel)); 459 break; 460 } 461 case VAL_DOUBLEARRAY: { 462 RETURN_IF_FAILED(parcel->readDoubleVector(&mDoubleVectorMap[key])); 463 break; 464 } 465 default: { 466 ALOGE("Unrecognized type: %d", value_type); 467 return BAD_TYPE; 468 break; 469 } 470 } 471 } 472 473 return NO_ERROR; 474 } 475 476 } // namespace os 477 478 } // namespace android 479