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