• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
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 in sync with BUNDLE_MAGIC in frameworks/base/core/java/android/os/BaseBundle.java.
43     BUNDLE_MAGIC = 0x4C444E42,
44 };
45 
46 namespace {
47 template <typename T>
getValue(const android::String16 & key,T * out,const map<android::String16,T> & map)48 bool getValue(const android::String16& key, T* out, const map<android::String16, T>& map) {
49     const auto& it = map.find(key);
50     if (it == map.end()) return false;
51     *out = it->second;
52     return true;
53 }
54 
55 template <typename T>
getKeys(const map<android::String16,T> & map)56 set<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 }
64 }  // namespace
65 
66 namespace android {
67 
68 namespace 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     {                                                                    \
81         size_t num_erased = (map).erase(key);                            \
82         if (num_erased) {                                                \
83             ALOGE("Failed at %s:%d (%s)", __FILE__, __LINE__, __func__); \
84             return num_erased;                                           \
85          }                                                               \
86     }
87 
writeToParcel(Parcel * parcel) const88 status_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()) {
112         ALOGE("Parcel length (%zu) too large to store in 32-bit signed int", length);
113         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 
readFromParcel(const Parcel * parcel)120 status_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 
empty() const134 bool PersistableBundle::empty() const {
135     return size() == 0u;
136 }
137 
size() const138 size_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 
erase(const String16 & key)152 size_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 
putBoolean(const String16 & key,bool value)166 void PersistableBundle::putBoolean(const String16& key, bool value) {
167     erase(key);
168     mBoolMap[key] = value;
169 }
170 
putInt(const String16 & key,int32_t value)171 void PersistableBundle::putInt(const String16& key, int32_t value) {
172     erase(key);
173     mIntMap[key] = value;
174 }
175 
putLong(const String16 & key,int64_t value)176 void PersistableBundle::putLong(const String16& key, int64_t value) {
177     erase(key);
178     mLongMap[key] = value;
179 }
180 
putDouble(const String16 & key,double value)181 void PersistableBundle::putDouble(const String16& key, double value) {
182     erase(key);
183     mDoubleMap[key] = value;
184 }
185 
putString(const String16 & key,const String16 & value)186 void PersistableBundle::putString(const String16& key, const String16& value) {
187     erase(key);
188     mStringMap[key] = value;
189 }
190 
putBooleanVector(const String16 & key,const vector<bool> & value)191 void PersistableBundle::putBooleanVector(const String16& key, const vector<bool>& value) {
192     erase(key);
193     mBoolVectorMap[key] = value;
194 }
195 
putIntVector(const String16 & key,const vector<int32_t> & value)196 void PersistableBundle::putIntVector(const String16& key, const vector<int32_t>& value) {
197     erase(key);
198     mIntVectorMap[key] = value;
199 }
200 
putLongVector(const String16 & key,const vector<int64_t> & value)201 void PersistableBundle::putLongVector(const String16& key, const vector<int64_t>& value) {
202     erase(key);
203     mLongVectorMap[key] = value;
204 }
205 
putDoubleVector(const String16 & key,const vector<double> & value)206 void PersistableBundle::putDoubleVector(const String16& key, const vector<double>& value) {
207     erase(key);
208     mDoubleVectorMap[key] = value;
209 }
210 
putStringVector(const String16 & key,const vector<String16> & value)211 void PersistableBundle::putStringVector(const String16& key, const vector<String16>& value) {
212     erase(key);
213     mStringVectorMap[key] = value;
214 }
215 
putPersistableBundle(const String16 & key,const PersistableBundle & value)216 void PersistableBundle::putPersistableBundle(const String16& key, const PersistableBundle& value) {
217     erase(key);
218     mPersistableBundleMap[key] = value;
219 }
220 
getBoolean(const String16 & key,bool * out) const221 bool PersistableBundle::getBoolean(const String16& key, bool* out) const {
222     return getValue(key, out, mBoolMap);
223 }
224 
getInt(const String16 & key,int32_t * out) const225 bool PersistableBundle::getInt(const String16& key, int32_t* out) const {
226     return getValue(key, out, mIntMap);
227 }
228 
getLong(const String16 & key,int64_t * out) const229 bool PersistableBundle::getLong(const String16& key, int64_t* out) const {
230     return getValue(key, out, mLongMap);
231 }
232 
getDouble(const String16 & key,double * out) const233 bool PersistableBundle::getDouble(const String16& key, double* out) const {
234     return getValue(key, out, mDoubleMap);
235 }
236 
getString(const String16 & key,String16 * out) const237 bool PersistableBundle::getString(const String16& key, String16* out) const {
238     return getValue(key, out, mStringMap);
239 }
240 
getBooleanVector(const String16 & key,vector<bool> * out) const241 bool PersistableBundle::getBooleanVector(const String16& key, vector<bool>* out) const {
242     return getValue(key, out, mBoolVectorMap);
243 }
244 
getIntVector(const String16 & key,vector<int32_t> * out) const245 bool PersistableBundle::getIntVector(const String16& key, vector<int32_t>* out) const {
246     return getValue(key, out, mIntVectorMap);
247 }
248 
getLongVector(const String16 & key,vector<int64_t> * out) const249 bool PersistableBundle::getLongVector(const String16& key, vector<int64_t>* out) const {
250     return getValue(key, out, mLongVectorMap);
251 }
252 
getDoubleVector(const String16 & key,vector<double> * out) const253 bool PersistableBundle::getDoubleVector(const String16& key, vector<double>* out) const {
254     return getValue(key, out, mDoubleVectorMap);
255 }
256 
getStringVector(const String16 & key,vector<String16> * out) const257 bool PersistableBundle::getStringVector(const String16& key, vector<String16>* out) const {
258     return getValue(key, out, mStringVectorMap);
259 }
260 
getPersistableBundle(const String16 & key,PersistableBundle * out) const261 bool PersistableBundle::getPersistableBundle(const String16& key, PersistableBundle* out) const {
262     return getValue(key, out, mPersistableBundleMap);
263 }
264 
getBooleanKeys() const265 set<String16> PersistableBundle::getBooleanKeys() const {
266     return getKeys(mBoolMap);
267 }
268 
getIntKeys() const269 set<String16> PersistableBundle::getIntKeys() const {
270     return getKeys(mIntMap);
271 }
272 
getLongKeys() const273 set<String16> PersistableBundle::getLongKeys() const {
274     return getKeys(mLongMap);
275 }
276 
getDoubleKeys() const277 set<String16> PersistableBundle::getDoubleKeys() const {
278     return getKeys(mDoubleMap);
279 }
280 
getStringKeys() const281 set<String16> PersistableBundle::getStringKeys() const {
282     return getKeys(mStringMap);
283 }
284 
getBooleanVectorKeys() const285 set<String16> PersistableBundle::getBooleanVectorKeys() const {
286     return getKeys(mBoolVectorMap);
287 }
288 
getIntVectorKeys() const289 set<String16> PersistableBundle::getIntVectorKeys() const {
290     return getKeys(mIntVectorMap);
291 }
292 
getLongVectorKeys() const293 set<String16> PersistableBundle::getLongVectorKeys() const {
294     return getKeys(mLongVectorMap);
295 }
296 
getDoubleVectorKeys() const297 set<String16> PersistableBundle::getDoubleVectorKeys() const {
298     return getKeys(mDoubleVectorMap);
299 }
300 
getStringVectorKeys() const301 set<String16> PersistableBundle::getStringVectorKeys() const {
302     return getKeys(mStringVectorMap);
303 }
304 
getPersistableBundleKeys() const305 set<String16> PersistableBundle::getPersistableBundleKeys() const {
306     return getKeys(mPersistableBundleMap);
307 }
308 
writeToParcelInner(Parcel * parcel) const309 status_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()) {
318         ALOGE("The size of this PersistableBundle (%zu) too large to store in 32-bit signed int",
319               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 
readFromParcelInner(const Parcel * parcel,size_t length)382 status_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) {
410         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
478