1 // Copyright 2014 The Chromium OS Authors. All rights reserved. 2 // Use of this source code is governed by a BSD-style license that can be 3 // found in the LICENSE file. 4 5 #ifndef LIBBRILLO_BRILLO_DBUS_DATA_SERIALIZATION_H_ 6 #define LIBBRILLO_BRILLO_DBUS_DATA_SERIALIZATION_H_ 7 8 // The main functionality provided by this header file is methods to serialize 9 // native C++ data over D-Bus. This includes three major parts: 10 // - Methods to get the D-Bus signature for a given C++ type: 11 // std::string GetDBusSignature<T>(); 12 // - Methods to write arbitrary C++ data to D-Bus MessageWriter: 13 // void AppendValueToWriter(dbus::MessageWriter* writer, const T& value); 14 // void AppendValueToWriterAsVariant(dbus::MessageWriter*, const T&); 15 // - Methods to read arbitrary C++ data from D-Bus MessageReader: 16 // bool PopValueFromReader(dbus::MessageReader* reader, T* value); 17 // bool PopVariantValueFromReader(dbus::MessageReader* reader, T* value); 18 // 19 // There are a number of overloads to handle C++ equivalents of basic D-Bus 20 // types: 21 // D-Bus Type | D-Bus Signature | Native C++ type 22 // -------------------------------------------------- 23 // BYTE | y | uint8_t 24 // BOOL | b | bool 25 // INT16 | n | int16_t 26 // UINT16 | q | uint16_t 27 // INT32 | i | int32_t (int) 28 // UINT32 | u | uint32_t (unsigned) 29 // INT64 | x | int64_t 30 // UINT64 | t | uint64_t 31 // DOUBLE | d | double 32 // STRING | s | std::string 33 // OBJECT_PATH | o | dbus::ObjectPath 34 // ARRAY | aT | std::vector<T> 35 // STRUCT | (UV) | std::pair<U,V> 36 // | (UVW...) | std::tuple<U,V,W,...> 37 // DICT | a{KV} | std::map<K,V> 38 // VARIANT | v | brillo::Any 39 // UNIX_FD | h | brillo::dbus_utils::FileDescriptor (write) 40 // | | base::ScopedFD (read) 41 // SIGNATURE | g | (unsupported) 42 // 43 // Additional overloads/specialization can be provided for custom types. 44 // In order to do that, provide overloads of AppendValueToWriter() and 45 // PopValueFromReader() functions in brillo::dbus_utils namespace for the 46 // CustomType. As well as a template specialization of DBusType<> for the same 47 // CustomType. This specialization must provide three static functions: 48 // - static std::string GetSignature(); 49 // - static void Write(dbus::MessageWriter* writer, const CustomType& value); 50 // - static bool Read(dbus::MessageReader* reader, CustomType* value); 51 // See an example in DBusUtils.CustomStruct unit test in 52 // brillo/dbus/data_serialization_test.cc. 53 54 #include <map> 55 #include <memory> 56 #include <string> 57 #include <tuple> 58 #include <utility> 59 #include <vector> 60 61 #include <base/files/scoped_file.h> 62 #include <base/logging.h> 63 #include <base/files/scoped_file.h> 64 #include <brillo/brillo_export.h> 65 #include <brillo/dbus/file_descriptor.h> 66 #include <brillo/type_name_undecorate.h> 67 #include <dbus/message.h> 68 69 namespace google { 70 namespace protobuf { 71 class MessageLite; 72 } // namespace protobuf 73 } // namespace google 74 75 namespace brillo { 76 77 // Forward-declare only. Can't include any.h right now because it needs 78 // AppendValueToWriter() declared below. 79 class Any; 80 81 namespace dbus_utils { 82 83 // Base class for DBusType<T> for T not supported by D-Bus. This used to 84 // implement IsTypeSupported<> below. 85 struct Unsupported {}; 86 87 // Generic definition of DBusType<T> which will be specialized for particular 88 // types later. 89 // The second template parameter is used only in SFINAE situations to resolve 90 // class hierarchy chains for protobuf-derived classes. This type is defaulted 91 // to be 'void' in all other cases and simply ignored. 92 // See DBusType specialization for google::protobuf::MessageLite below for more 93 // detailed information. 94 template<typename T, typename = void> 95 struct DBusType : public Unsupported {}; 96 97 // A helper type trait to determine if all of the types listed in Types... are 98 // supported by D-Bus. This is a generic forward-declaration which will be 99 // specialized for different type combinations. 100 template<typename... Types> 101 struct IsTypeSupported; 102 103 // Both T and the Types... must be supported for the complete set to be 104 // supported. 105 template<typename T, typename... Types> 106 struct IsTypeSupported<T, Types...> 107 : public std::integral_constant< 108 bool, 109 IsTypeSupported<T>::value && IsTypeSupported<Types...>::value> {}; 110 111 // For a single type T, check if DBusType<T> derives from Unsupported. 112 // If it does, then the type is not supported by the D-Bus. 113 template<typename T> 114 struct IsTypeSupported<T> 115 : public std::integral_constant< 116 bool, 117 !std::is_base_of<Unsupported, DBusType<T>>::value> {}; 118 119 // Empty set is not supported. 120 template<> 121 struct IsTypeSupported<> : public std::false_type {}; 122 123 //---------------------------------------------------------------------------- 124 // AppendValueToWriter<T>(dbus::MessageWriter* writer, const T& value) 125 // Write the |value| of type T to D-Bus message. 126 // Explicitly delete the overloads for scalar types that are not supported by 127 // D-Bus. 128 void AppendValueToWriter(::dbus::MessageWriter* writer, char value) = delete; 129 void AppendValueToWriter(::dbus::MessageWriter* writer, float value) = delete; 130 131 //---------------------------------------------------------------------------- 132 // PopValueFromReader<T>(dbus::MessageWriter* writer, T* value) 133 // Reads the |value| of type T from D-Bus message. 134 // Explicitly delete the overloads for scalar types that are not supported by 135 // D-Bus. 136 void PopValueFromReader(::dbus::MessageReader* reader, char* value) = delete; 137 void PopValueFromReader(::dbus::MessageReader* reader, float* value) = delete; 138 139 //---------------------------------------------------------------------------- 140 // Get D-Bus data signature from C++ data types. 141 // Specializations of a generic GetDBusSignature<T>() provide signature strings 142 // for native C++ types. This function is available only for type supported 143 // by D-Bus. 144 template<typename T> 145 inline typename std::enable_if<IsTypeSupported<T>::value, std::string>::type 146 GetDBusSignature() { 147 return DBusType<T>::GetSignature(); 148 } 149 150 namespace details { 151 // Helper method used by the many overloads of PopValueFromReader(). 152 // If the current value in the reader is of Variant type, the method descends 153 // into the Variant and updates the |*reader_ref| with the transient 154 // |variant_reader| MessageReader instance passed in. 155 // Returns false if it fails to descend into the Variant. 156 inline bool DescendIntoVariantIfPresent(::dbus::MessageReader** reader_ref, 157 ::dbus::MessageReader* variant_reader) { 158 if ((*reader_ref)->GetDataType() != ::dbus::Message::VARIANT) 159 return true; 160 if (!(*reader_ref)->PopVariant(variant_reader)) 161 return false; 162 *reader_ref = variant_reader; 163 return true; 164 } 165 166 // Helper method to format the type string of an array. 167 // Essentially it adds "a" in front of |element_signature|. 168 inline std::string GetArrayDBusSignature(const std::string& element_signature) { 169 return DBUS_TYPE_ARRAY_AS_STRING + element_signature; 170 } 171 172 // Helper method to get a signature string for DICT_ENTRY. 173 // Returns "{KV}", where "K" and "V" are the type signatures for types 174 // KEY/VALUE. For example, GetDBusDictEntryType<std::string, int>() would return 175 // "{si}". 176 template<typename KEY, typename VALUE> 177 inline std::string GetDBusDictEntryType() { 178 return DBUS_DICT_ENTRY_BEGIN_CHAR_AS_STRING + 179 GetDBusSignature<KEY>() + GetDBusSignature<VALUE>() + 180 DBUS_DICT_ENTRY_END_CHAR_AS_STRING; 181 } 182 183 } // namespace details 184 185 //============================================================================= 186 // Specializations/overloads for AppendValueToWriter, PopValueFromReader and 187 // DBusType<T> for various C++ types that can be serialized over D-Bus. 188 189 // bool ----------------------------------------------------------------------- 190 BRILLO_EXPORT void AppendValueToWriter(::dbus::MessageWriter* writer, 191 bool value); 192 BRILLO_EXPORT bool PopValueFromReader(::dbus::MessageReader* reader, 193 bool* value); 194 195 template<> 196 struct DBusType<bool> { 197 inline static std::string GetSignature() { 198 return DBUS_TYPE_BOOLEAN_AS_STRING; 199 } 200 inline static void Write(::dbus::MessageWriter* writer, bool value) { 201 AppendValueToWriter(writer, value); 202 } 203 inline static bool Read(::dbus::MessageReader* reader, bool* value) { 204 return PopValueFromReader(reader, value); 205 } 206 }; 207 208 // uint8_t -------------------------------------------------------------------- 209 BRILLO_EXPORT void AppendValueToWriter(::dbus::MessageWriter* writer, 210 uint8_t value); 211 BRILLO_EXPORT bool PopValueFromReader(::dbus::MessageReader* reader, 212 uint8_t* value); 213 214 template<> 215 struct DBusType<uint8_t> { 216 inline static std::string GetSignature() { return DBUS_TYPE_BYTE_AS_STRING; } 217 inline static void Write(::dbus::MessageWriter* writer, uint8_t value) { 218 AppendValueToWriter(writer, value); 219 } 220 inline static bool Read(::dbus::MessageReader* reader, uint8_t* value) { 221 return PopValueFromReader(reader, value); 222 } 223 }; 224 225 // int16_t -------------------------------------------------------------------- 226 BRILLO_EXPORT void AppendValueToWriter(::dbus::MessageWriter* writer, 227 int16_t value); 228 BRILLO_EXPORT bool PopValueFromReader(::dbus::MessageReader* reader, 229 int16_t* value); 230 231 template<> 232 struct DBusType<int16_t> { 233 inline static std::string GetSignature() { return DBUS_TYPE_INT16_AS_STRING; } 234 inline static void Write(::dbus::MessageWriter* writer, int16_t value) { 235 AppendValueToWriter(writer, value); 236 } 237 inline static bool Read(::dbus::MessageReader* reader, int16_t* value) { 238 return PopValueFromReader(reader, value); 239 } 240 }; 241 242 // uint16_t ------------------------------------------------------------------- 243 BRILLO_EXPORT void AppendValueToWriter(::dbus::MessageWriter* writer, 244 uint16_t value); 245 BRILLO_EXPORT bool PopValueFromReader(::dbus::MessageReader* reader, 246 uint16_t* value); 247 248 template<> 249 struct DBusType<uint16_t> { 250 inline static std::string GetSignature() { 251 return DBUS_TYPE_UINT16_AS_STRING; 252 } 253 inline static void Write(::dbus::MessageWriter* writer, uint16_t value) { 254 AppendValueToWriter(writer, value); 255 } 256 inline static bool Read(::dbus::MessageReader* reader, uint16_t* value) { 257 return PopValueFromReader(reader, value); 258 } 259 }; 260 261 // int32_t -------------------------------------------------------------------- 262 BRILLO_EXPORT void AppendValueToWriter(::dbus::MessageWriter* writer, 263 int32_t value); 264 BRILLO_EXPORT bool PopValueFromReader(::dbus::MessageReader* reader, 265 int32_t* value); 266 267 template<> 268 struct DBusType<int32_t> { 269 inline static std::string GetSignature() { return DBUS_TYPE_INT32_AS_STRING; } 270 inline static void Write(::dbus::MessageWriter* writer, int32_t value) { 271 AppendValueToWriter(writer, value); 272 } 273 inline static bool Read(::dbus::MessageReader* reader, int32_t* value) { 274 return PopValueFromReader(reader, value); 275 } 276 }; 277 278 // uint32_t ------------------------------------------------------------------- 279 BRILLO_EXPORT void AppendValueToWriter(::dbus::MessageWriter* writer, 280 uint32_t value); 281 BRILLO_EXPORT bool PopValueFromReader(::dbus::MessageReader* reader, 282 uint32_t* value); 283 284 template<> 285 struct DBusType<uint32_t> { 286 inline static std::string GetSignature() { 287 return DBUS_TYPE_UINT32_AS_STRING; 288 } 289 inline static void Write(::dbus::MessageWriter* writer, uint32_t value) { 290 AppendValueToWriter(writer, value); 291 } 292 inline static bool Read(::dbus::MessageReader* reader, uint32_t* value) { 293 return PopValueFromReader(reader, value); 294 } 295 }; 296 297 // int64_t -------------------------------------------------------------------- 298 BRILLO_EXPORT void AppendValueToWriter(::dbus::MessageWriter* writer, 299 int64_t value); 300 BRILLO_EXPORT bool PopValueFromReader(::dbus::MessageReader* reader, 301 int64_t* value); 302 303 template<> 304 struct DBusType<int64_t> { 305 inline static std::string GetSignature() { return DBUS_TYPE_INT64_AS_STRING; } 306 inline static void Write(::dbus::MessageWriter* writer, int64_t value) { 307 AppendValueToWriter(writer, value); 308 } 309 inline static bool Read(::dbus::MessageReader* reader, int64_t* value) { 310 return PopValueFromReader(reader, value); 311 } 312 }; 313 314 // uint64_t ------------------------------------------------------------------- 315 BRILLO_EXPORT void AppendValueToWriter(::dbus::MessageWriter* writer, 316 uint64_t value); 317 BRILLO_EXPORT bool PopValueFromReader(::dbus::MessageReader* reader, 318 uint64_t* value); 319 320 template<> 321 struct DBusType<uint64_t> { 322 inline static std::string GetSignature() { 323 return DBUS_TYPE_UINT64_AS_STRING; 324 } 325 inline static void Write(::dbus::MessageWriter* writer, uint64_t value) { 326 AppendValueToWriter(writer, value); 327 } 328 inline static bool Read(::dbus::MessageReader* reader, uint64_t* value) { 329 return PopValueFromReader(reader, value); 330 } 331 }; 332 333 // double --------------------------------------------------------------------- 334 BRILLO_EXPORT void AppendValueToWriter(::dbus::MessageWriter* writer, 335 double value); 336 BRILLO_EXPORT bool PopValueFromReader(::dbus::MessageReader* reader, 337 double* value); 338 339 template<> 340 struct DBusType<double> { 341 inline static std::string GetSignature() { 342 return DBUS_TYPE_DOUBLE_AS_STRING; 343 } 344 inline static void Write(::dbus::MessageWriter* writer, double value) { 345 AppendValueToWriter(writer, value); 346 } 347 inline static bool Read(::dbus::MessageReader* reader, double* value) { 348 return PopValueFromReader(reader, value); 349 } 350 }; 351 352 // std::string ---------------------------------------------------------------- 353 BRILLO_EXPORT void AppendValueToWriter(::dbus::MessageWriter* writer, 354 const std::string& value); 355 BRILLO_EXPORT bool PopValueFromReader(::dbus::MessageReader* reader, 356 std::string* value); 357 358 template<> 359 struct DBusType<std::string> { 360 inline static std::string GetSignature() { 361 return DBUS_TYPE_STRING_AS_STRING; 362 } 363 inline static void Write(::dbus::MessageWriter* writer, 364 const std::string& value) { 365 AppendValueToWriter(writer, value); 366 } 367 inline static bool Read(::dbus::MessageReader* reader, std::string* value) { 368 return PopValueFromReader(reader, value); 369 } 370 }; 371 372 // const char* 373 BRILLO_EXPORT void AppendValueToWriter(::dbus::MessageWriter* writer, 374 const char* value); 375 376 template<> 377 struct DBusType<const char*> { 378 inline static std::string GetSignature() { 379 return DBUS_TYPE_STRING_AS_STRING; 380 } 381 inline static void Write(::dbus::MessageWriter* writer, const char* value) { 382 AppendValueToWriter(writer, value); 383 } 384 }; 385 386 // const char[] 387 template<> 388 struct DBusType<const char[]> { 389 inline static std::string GetSignature() { 390 return DBUS_TYPE_STRING_AS_STRING; 391 } 392 inline static void Write(::dbus::MessageWriter* writer, const char* value) { 393 AppendValueToWriter(writer, value); 394 } 395 }; 396 397 // dbus::ObjectPath ----------------------------------------------------------- 398 BRILLO_EXPORT void AppendValueToWriter(::dbus::MessageWriter* writer, 399 const ::dbus::ObjectPath& value); 400 BRILLO_EXPORT bool PopValueFromReader(::dbus::MessageReader* reader, 401 ::dbus::ObjectPath* value); 402 403 template <> 404 struct DBusType<::dbus::ObjectPath> { 405 inline static std::string GetSignature() { 406 return DBUS_TYPE_OBJECT_PATH_AS_STRING; 407 } 408 inline static void Write(::dbus::MessageWriter* writer, 409 const ::dbus::ObjectPath& value) { 410 AppendValueToWriter(writer, value); 411 } 412 inline static bool Read(::dbus::MessageReader* reader, 413 ::dbus::ObjectPath* value) { 414 return PopValueFromReader(reader, value); 415 } 416 }; 417 418 // brillo::dbus_utils::FileDescriptor/base::ScopedFD -------------------------- 419 BRILLO_EXPORT void AppendValueToWriter(::dbus::MessageWriter* writer, 420 const FileDescriptor& value); 421 BRILLO_EXPORT bool PopValueFromReader(::dbus::MessageReader* reader, 422 base::ScopedFD* value); 423 424 template<> 425 struct DBusType<FileDescriptor> { 426 inline static std::string GetSignature() { 427 return DBUS_TYPE_UNIX_FD_AS_STRING; 428 } 429 inline static void Write(::dbus::MessageWriter* writer, 430 const FileDescriptor& value) { 431 AppendValueToWriter(writer, value); 432 } 433 }; 434 435 template<> 436 struct DBusType<base::ScopedFD> { 437 inline static std::string GetSignature() { 438 return DBUS_TYPE_UNIX_FD_AS_STRING; 439 } 440 inline static bool Read(::dbus::MessageReader* reader, 441 base::ScopedFD* value) { 442 return PopValueFromReader(reader, value); 443 } 444 }; 445 446 // brillo::Any -------------------------------------------------------------- 447 BRILLO_EXPORT void AppendValueToWriter(::dbus::MessageWriter* writer, 448 const brillo::Any& value); 449 BRILLO_EXPORT bool PopValueFromReader(::dbus::MessageReader* reader, 450 brillo::Any* value); 451 452 template<> 453 struct DBusType<brillo::Any> { 454 inline static std::string GetSignature() { 455 return DBUS_TYPE_VARIANT_AS_STRING; 456 } 457 inline static void Write(::dbus::MessageWriter* writer, 458 const brillo::Any& value) { 459 AppendValueToWriter(writer, value); 460 } 461 inline static bool Read(::dbus::MessageReader* reader, brillo::Any* value) { 462 return PopValueFromReader(reader, value); 463 } 464 }; 465 466 // std::vector = D-Bus ARRAY. ------------------------------------------------- 467 template <typename T, typename ALLOC> 468 typename std::enable_if<IsTypeSupported<T>::value>::type AppendValueToWriter( 469 ::dbus::MessageWriter* writer, const std::vector<T, ALLOC>& value) { 470 ::dbus::MessageWriter array_writer(nullptr); 471 writer->OpenArray(GetDBusSignature<T>(), &array_writer); 472 for (const auto& element : value) { 473 // Use DBusType<T>::Write() instead of AppendValueToWriter() to delay 474 // binding to AppendValueToWriter() to the point of instantiation of this 475 // template. 476 DBusType<T>::Write(&array_writer, element); 477 } 478 writer->CloseContainer(&array_writer); 479 } 480 481 template <typename T, typename ALLOC> 482 typename std::enable_if<IsTypeSupported<T>::value, bool>::type 483 PopValueFromReader(::dbus::MessageReader* reader, 484 std::vector<T, ALLOC>* value) { 485 ::dbus::MessageReader variant_reader(nullptr); 486 ::dbus::MessageReader array_reader(nullptr); 487 if (!details::DescendIntoVariantIfPresent(&reader, &variant_reader) || 488 !reader->PopArray(&array_reader)) 489 return false; 490 value->clear(); 491 while (array_reader.HasMoreData()) { 492 T data; 493 // Use DBusType<T>::Read() instead of PopValueFromReader() to delay 494 // binding to PopValueFromReader() to the point of instantiation of this 495 // template. 496 if (!DBusType<T>::Read(&array_reader, &data)) 497 return false; 498 value->push_back(std::move(data)); 499 } 500 return true; 501 } 502 503 namespace details { 504 // DBusArrayType<> is a helper base class for DBusType<vector<T>> that provides 505 // GetSignature/Write/Read methods for T types that are supported by D-Bus 506 // and not having those methods for types that are not supported by D-Bus. 507 template<bool inner_type_supported, typename T, typename ALLOC> 508 struct DBusArrayType { 509 // Returns "aT", where "T" is the signature string for type T. 510 inline static std::string GetSignature() { 511 return GetArrayDBusSignature(GetDBusSignature<T>()); 512 } 513 inline static void Write(::dbus::MessageWriter* writer, 514 const std::vector<T, ALLOC>& value) { 515 AppendValueToWriter(writer, value); 516 } 517 inline static bool Read(::dbus::MessageReader* reader, 518 std::vector<T, ALLOC>* value) { 519 return PopValueFromReader(reader, value); 520 } 521 }; 522 523 // Explicit specialization for unsupported type T. 524 template<typename T, typename ALLOC> 525 struct DBusArrayType<false, T, ALLOC> : public Unsupported {}; 526 527 } // namespace details 528 529 template<typename T, typename ALLOC> 530 struct DBusType<std::vector<T, ALLOC>> 531 : public details::DBusArrayType<IsTypeSupported<T>::value, T, ALLOC> {}; 532 533 // std::pair = D-Bus STRUCT with two elements. -------------------------------- 534 namespace details { 535 536 // Helper class to get a D-Bus signature of a list of types. 537 // For example, TupleTraits<int32_t, bool, std::string>::GetSignature() will 538 // return "ibs". 539 template<typename... Types> 540 struct TupleTraits; 541 542 template<typename FirstType, typename... RestOfTypes> 543 struct TupleTraits<FirstType, RestOfTypes...> { 544 static std::string GetSignature() { 545 return GetDBusSignature<FirstType>() + 546 TupleTraits<RestOfTypes...>::GetSignature(); 547 } 548 }; 549 550 template<> 551 struct TupleTraits<> { 552 static std::string GetSignature() { return std::string{}; } 553 }; 554 555 } // namespace details 556 557 template<typename... Types> 558 inline std::string GetStructDBusSignature() { 559 // Returns "(T...)", where "T..." is the signature strings for types T... 560 return DBUS_STRUCT_BEGIN_CHAR_AS_STRING + 561 details::TupleTraits<Types...>::GetSignature() + 562 DBUS_STRUCT_END_CHAR_AS_STRING; 563 } 564 565 template <typename U, typename V> 566 typename std::enable_if<IsTypeSupported<U, V>::value>::type AppendValueToWriter( 567 ::dbus::MessageWriter* writer, const std::pair<U, V>& value) { 568 ::dbus::MessageWriter struct_writer(nullptr); 569 writer->OpenStruct(&struct_writer); 570 // Use DBusType<T>::Write() instead of AppendValueToWriter() to delay 571 // binding to AppendValueToWriter() to the point of instantiation of this 572 // template. 573 DBusType<U>::Write(&struct_writer, value.first); 574 DBusType<V>::Write(&struct_writer, value.second); 575 writer->CloseContainer(&struct_writer); 576 } 577 578 template <typename U, typename V> 579 typename std::enable_if<IsTypeSupported<U, V>::value, bool>::type 580 PopValueFromReader(::dbus::MessageReader* reader, std::pair<U, V>* value) { 581 ::dbus::MessageReader variant_reader(nullptr); 582 ::dbus::MessageReader struct_reader(nullptr); 583 if (!details::DescendIntoVariantIfPresent(&reader, &variant_reader) || 584 !reader->PopStruct(&struct_reader)) 585 return false; 586 // Use DBusType<T>::Read() instead of PopValueFromReader() to delay 587 // binding to PopValueFromReader() to the point of instantiation of this 588 // template. 589 return DBusType<U>::Read(&struct_reader, &value->first) && 590 DBusType<V>::Read(&struct_reader, &value->second); 591 } 592 593 namespace details { 594 595 // DBusArrayType<> is a helper base class for DBusType<pair<U, V>> that provides 596 // GetSignature/Write/Read methods for types that are supported by D-Bus 597 // and not having those methods for types that are not supported by D-Bus. 598 template<bool inner_type_supported, typename U, typename V> 599 struct DBusPairType { 600 // Returns "(UV)", where "U" and "V" are the signature strings for types U, V. 601 inline static std::string GetSignature() { 602 return GetStructDBusSignature<U, V>(); 603 } 604 inline static void Write(::dbus::MessageWriter* writer, 605 const std::pair<U, V>& value) { 606 AppendValueToWriter(writer, value); 607 } 608 inline static bool Read(::dbus::MessageReader* reader, 609 std::pair<U, V>* value) { 610 return PopValueFromReader(reader, value); 611 } 612 }; 613 614 // Either U, or V, or both are not supported by D-Bus. 615 template<typename U, typename V> 616 struct DBusPairType<false, U, V> : public Unsupported {}; 617 618 } // namespace details 619 620 template<typename U, typename V> 621 struct DBusType<std::pair<U, V>> 622 : public details::DBusPairType<IsTypeSupported<U, V>::value, U, V> {}; 623 624 // std::tuple = D-Bus STRUCT with arbitrary number of members. ---------------- 625 namespace details { 626 627 // TupleIterator<I, N, T...> is a helper class to iterate over all the elements 628 // of a tuple<T...> from index I to N. TupleIterator<>::Read and ::Write methods 629 // are called for each element of the tuple and iteration continues until I == N 630 // in which case the specialization for I==N below stops the recursion. 631 template<size_t I, size_t N, typename... T> 632 struct TupleIterator { 633 // Tuple is just a convenience alias to a tuple containing elements of type T. 634 using Tuple = std::tuple<T...>; 635 // ValueType is the type of the element at index I. 636 using ValueType = typename std::tuple_element<I, Tuple>::type; 637 638 // Write the tuple element at index I to D-Bus message. 639 static void Write(::dbus::MessageWriter* writer, const Tuple& value) { 640 // Use DBusType<T>::Write() instead of AppendValueToWriter() to delay 641 // binding to AppendValueToWriter() to the point of instantiation of this 642 // template. 643 DBusType<ValueType>::Write(writer, std::get<I>(value)); 644 TupleIterator<I + 1, N, T...>::Write(writer, value); 645 } 646 647 // Read the tuple element at index I from D-Bus message. 648 static bool Read(::dbus::MessageReader* reader, Tuple* value) { 649 // Use DBusType<T>::Read() instead of PopValueFromReader() to delay 650 // binding to PopValueFromReader() to the point of instantiation of this 651 // template. 652 return DBusType<ValueType>::Read(reader, &std::get<I>(*value)) && 653 TupleIterator<I + 1, N, T...>::Read(reader, value); 654 } 655 }; 656 657 // Specialization to end the iteration when the index reaches the last element. 658 template<size_t N, typename... T> 659 struct TupleIterator<N, N, T...> { 660 using Tuple = std::tuple<T...>; 661 static void Write(::dbus::MessageWriter* /* writer */, 662 const Tuple& /* value */) {} 663 static bool Read(::dbus::MessageReader* /* reader */, Tuple* /* value */) { 664 return true; 665 } 666 }; 667 668 } // namespace details 669 670 template <typename... T> 671 typename std::enable_if<IsTypeSupported<T...>::value>::type AppendValueToWriter( 672 ::dbus::MessageWriter* writer, const std::tuple<T...>& value) { 673 ::dbus::MessageWriter struct_writer(nullptr); 674 writer->OpenStruct(&struct_writer); 675 details::TupleIterator<0, sizeof...(T), T...>::Write(&struct_writer, value); 676 writer->CloseContainer(&struct_writer); 677 } 678 679 template <typename... T> 680 typename std::enable_if<IsTypeSupported<T...>::value, bool>::type 681 PopValueFromReader(::dbus::MessageReader* reader, std::tuple<T...>* value) { 682 ::dbus::MessageReader variant_reader(nullptr); 683 ::dbus::MessageReader struct_reader(nullptr); 684 if (!details::DescendIntoVariantIfPresent(&reader, &variant_reader) || 685 !reader->PopStruct(&struct_reader)) 686 return false; 687 return details::TupleIterator<0, sizeof...(T), T...>::Read(&struct_reader, 688 value); 689 } 690 691 namespace details { 692 693 // DBusTupleType<> is a helper base class for DBusType<tuple<T...>> that 694 // provides GetSignature/Write/Read methods for types that are supported by 695 // D-Bus and not having those methods for types that are not supported by D-Bus. 696 template<bool inner_type_supported, typename... T> 697 struct DBusTupleType { 698 // Returns "(T...)", where "T..." are the signature strings for types T... 699 inline static std::string GetSignature() { 700 return GetStructDBusSignature<T...>(); 701 } 702 inline static void Write(::dbus::MessageWriter* writer, 703 const std::tuple<T...>& value) { 704 AppendValueToWriter(writer, value); 705 } 706 inline static bool Read(::dbus::MessageReader* reader, 707 std::tuple<T...>* value) { 708 return PopValueFromReader(reader, value); 709 } 710 }; 711 712 // Some/all of types T... are not supported by D-Bus. 713 template<typename... T> 714 struct DBusTupleType<false, T...> : public Unsupported {}; 715 716 } // namespace details 717 718 template<typename... T> 719 struct DBusType<std::tuple<T...>> 720 : public details::DBusTupleType<IsTypeSupported<T...>::value, T...> {}; 721 722 // std::map = D-Bus ARRAY of DICT_ENTRY. -------------------------------------- 723 template <typename KEY, typename VALUE, typename PRED, typename ALLOC> 724 typename std::enable_if<IsTypeSupported<KEY, VALUE>::value>::type 725 AppendValueToWriter(::dbus::MessageWriter* writer, 726 const std::map<KEY, VALUE, PRED, ALLOC>& value) { 727 ::dbus::MessageWriter dict_writer(nullptr); 728 writer->OpenArray(details::GetDBusDictEntryType<KEY, VALUE>(), &dict_writer); 729 for (const auto& pair : value) { 730 ::dbus::MessageWriter entry_writer(nullptr); 731 dict_writer.OpenDictEntry(&entry_writer); 732 // Use DBusType<T>::Write() instead of AppendValueToWriter() to delay 733 // binding to AppendValueToWriter() to the point of instantiation of this 734 // template. 735 DBusType<KEY>::Write(&entry_writer, pair.first); 736 DBusType<VALUE>::Write(&entry_writer, pair.second); 737 dict_writer.CloseContainer(&entry_writer); 738 } 739 writer->CloseContainer(&dict_writer); 740 } 741 742 template <typename KEY, typename VALUE, typename PRED, typename ALLOC> 743 typename std::enable_if<IsTypeSupported<KEY, VALUE>::value, bool>::type 744 PopValueFromReader(::dbus::MessageReader* reader, 745 std::map<KEY, VALUE, PRED, ALLOC>* value) { 746 ::dbus::MessageReader variant_reader(nullptr); 747 ::dbus::MessageReader array_reader(nullptr); 748 if (!details::DescendIntoVariantIfPresent(&reader, &variant_reader) || 749 !reader->PopArray(&array_reader)) 750 return false; 751 value->clear(); 752 while (array_reader.HasMoreData()) { 753 ::dbus::MessageReader dict_entry_reader(nullptr); 754 if (!array_reader.PopDictEntry(&dict_entry_reader)) 755 return false; 756 KEY key; 757 VALUE data; 758 // Use DBusType<T>::Read() instead of PopValueFromReader() to delay 759 // binding to PopValueFromReader() to the point of instantiation of this 760 // template. 761 if (!DBusType<KEY>::Read(&dict_entry_reader, &key) || 762 !DBusType<VALUE>::Read(&dict_entry_reader, &data)) 763 return false; 764 value->emplace(std::move(key), std::move(data)); 765 } 766 return true; 767 } 768 769 namespace details { 770 771 // DBusArrayType<> is a helper base class for DBusType<map<K, V>> that provides 772 // GetSignature/Write/Read methods for T types that are supported by D-Bus 773 // and not having those methods for types that are not supported by D-Bus. 774 template<bool inner_types_supported, 775 typename KEY, 776 typename VALUE, 777 typename PRED, 778 typename ALLOC> 779 struct DBusMapType { 780 // Returns "a{KV}", where "K" and "V" are the signature strings for types 781 // KEY/VALUE. 782 inline static std::string GetSignature() { 783 return GetArrayDBusSignature(GetDBusDictEntryType<KEY, VALUE>()); 784 } 785 inline static void Write(::dbus::MessageWriter* writer, 786 const std::map<KEY, VALUE, PRED, ALLOC>& value) { 787 AppendValueToWriter(writer, value); 788 } 789 inline static bool Read(::dbus::MessageReader* reader, 790 std::map<KEY, VALUE, PRED, ALLOC>* value) { 791 return PopValueFromReader(reader, value); 792 } 793 }; 794 795 // Types KEY, VALUE or both are not supported by D-Bus. 796 template<typename KEY, typename VALUE, typename PRED, typename ALLOC> 797 struct DBusMapType<false, KEY, VALUE, PRED, ALLOC> : public Unsupported {}; 798 799 } // namespace details 800 801 template<typename KEY, typename VALUE, typename PRED, typename ALLOC> 802 struct DBusType<std::map<KEY, VALUE, PRED, ALLOC>> 803 : public details::DBusMapType<IsTypeSupported<KEY, VALUE>::value, 804 KEY, 805 VALUE, 806 PRED, 807 ALLOC> {}; 808 809 // google::protobuf::MessageLite = D-Bus ARRAY of BYTE ------------------------ 810 inline void AppendValueToWriter(::dbus::MessageWriter* writer, 811 const google::protobuf::MessageLite& value) { 812 writer->AppendProtoAsArrayOfBytes(value); 813 } 814 815 inline bool PopValueFromReader(::dbus::MessageReader* reader, 816 google::protobuf::MessageLite* value) { 817 return reader->PopArrayOfBytesAsProto(value); 818 } 819 820 // is_protobuf_t<T> is a helper type trait to determine if type T derives from 821 // google::protobuf::MessageLite. 822 template<typename T> 823 using is_protobuf = std::is_base_of<google::protobuf::MessageLite, T>; 824 825 // Specialize DBusType<T> for classes that derive from protobuf::MessageLite. 826 // Here we perform a partial specialization of DBusType<T> only for types 827 // that derive from google::protobuf::MessageLite. This is done by employing 828 // the second template parameter in DBusType and this basically relies on C++ 829 // SFINAE rules. "typename std::enable_if<is_protobuf<T>::value>::type" will 830 // evaluate to "void" for classes T that descend from MessageLite and will be 831 // an invalid construct for other types/classes which will automatically 832 // remove this particular specialization from name resolution context. 833 template<typename T> 834 struct DBusType<T, typename std::enable_if<is_protobuf<T>::value>::type> { 835 inline static std::string GetSignature() { 836 return GetDBusSignature<std::vector<uint8_t>>(); 837 } 838 inline static void Write(::dbus::MessageWriter* writer, const T& value) { 839 AppendValueToWriter(writer, value); 840 } 841 inline static bool Read(::dbus::MessageReader* reader, T* value) { 842 return PopValueFromReader(reader, value); 843 } 844 }; 845 846 //---------------------------------------------------------------------------- 847 // AppendValueToWriterAsVariant<T>(::dbus::MessageWriter* writer, const T& 848 // value) Write the |value| of type T to D-Bus message as a VARIANT. This 849 // overload is provided only if T is supported by D-Bus. 850 template <typename T> 851 typename std::enable_if<IsTypeSupported<T>::value>::type 852 AppendValueToWriterAsVariant(::dbus::MessageWriter* writer, const T& value) { 853 std::string data_type = GetDBusSignature<T>(); 854 ::dbus::MessageWriter variant_writer(nullptr); 855 writer->OpenVariant(data_type, &variant_writer); 856 // Use DBusType<T>::Write() instead of AppendValueToWriter() to delay 857 // binding to AppendValueToWriter() to the point of instantiation of this 858 // template. 859 DBusType<T>::Write(&variant_writer, value); 860 writer->CloseContainer(&variant_writer); 861 } 862 863 // Special case: do not allow to write a Variant containing a Variant. 864 // Just redirect to normal AppendValueToWriter(). 865 inline void AppendValueToWriterAsVariant(::dbus::MessageWriter* writer, 866 const brillo::Any& value) { 867 return AppendValueToWriter(writer, value); 868 } 869 870 //---------------------------------------------------------------------------- 871 // PopVariantValueFromReader<T>(::dbus::MessageWriter* writer, T* value) 872 // Reads a Variant containing the |value| of type T from D-Bus message. 873 // Note that the generic PopValueFromReader<T>(...) can do this too. 874 // This method is provided for two reasons: 875 // 1. For API symmetry with AppendValueToWriter/AppendValueToWriterAsVariant. 876 // 2. To be used when it is important to assert that the data was sent 877 // specifically as a Variant. 878 // This overload is provided only if T is supported by D-Bus. 879 template <typename T> 880 typename std::enable_if<IsTypeSupported<T>::value, bool>::type 881 PopVariantValueFromReader(::dbus::MessageReader* reader, T* value) { 882 ::dbus::MessageReader variant_reader(nullptr); 883 if (!reader->PopVariant(&variant_reader)) 884 return false; 885 // Use DBusType<T>::Read() instead of PopValueFromReader() to delay 886 // binding to PopValueFromReader() to the point of instantiation of this 887 // template. 888 return DBusType<T>::Read(&variant_reader, value); 889 } 890 891 // Special handling of request to read a Variant of Variant. 892 inline bool PopVariantValueFromReader(::dbus::MessageReader* reader, 893 Any* value) { 894 return PopValueFromReader(reader, value); 895 } 896 897 } // namespace dbus_utils 898 } // namespace brillo 899 900 #endif // LIBBRILLO_BRILLO_DBUS_DATA_SERIALIZATION_H_ 901