| // Copyright 2016 The Chromium Authors |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| #ifdef UNSAFE_BUFFERS_BUILD |
| // TODO(crbug.com/390223051): Remove C-library calls to fix the errors. |
| #pragma allow_unsafe_libc_calls |
| #endif |
| |
| #ifndef MOJO_PUBLIC_CPP_BINDINGS_LIB_SERIALIZATION_H_ |
| #define MOJO_PUBLIC_CPP_BINDINGS_LIB_SERIALIZATION_H_ |
| |
| #include <type_traits> |
| |
| #include "mojo/public/cpp/bindings/array_traits_span.h" |
| #include "mojo/public/cpp/bindings/array_traits_stl.h" |
| #include "mojo/public/cpp/bindings/lib/array_serialization.h" |
| #include "mojo/public/cpp/bindings/lib/bindings_internal.h" |
| #include "mojo/public/cpp/bindings/lib/buffer.h" |
| #include "mojo/public/cpp/bindings/lib/handle_serialization.h" |
| #include "mojo/public/cpp/bindings/lib/map_serialization.h" |
| #include "mojo/public/cpp/bindings/lib/message_fragment.h" |
| #include "mojo/public/cpp/bindings/lib/string_serialization.h" |
| #include "mojo/public/cpp/bindings/lib/template_util.h" |
| #include "mojo/public/cpp/bindings/map_traits_flat_map.h" |
| #include "mojo/public/cpp/bindings/map_traits_stl.h" |
| #include "mojo/public/cpp/bindings/message.h" |
| #include "mojo/public/cpp/bindings/string_traits_stl.h" |
| #include "mojo/public/cpp/bindings/string_traits_string_piece.h" |
| |
| namespace mojo { |
| namespace internal { |
| |
| template <typename MojomType, typename EnableType = void> |
| struct MojomSerializationImplTraits; |
| |
| template <typename MojomType> |
| struct MojomSerializationImplTraits< |
| MojomType, |
| typename std::enable_if< |
| BelongsTo<MojomType, MojomTypeCategory::kStruct>::value>::type> { |
| template <typename MaybeConstUserType, typename FragmentType> |
| static void Serialize(MaybeConstUserType& input, FragmentType& fragment) { |
| mojo::internal::Serialize<MojomType>(input, fragment); |
| } |
| }; |
| |
| template <typename MojomType> |
| struct MojomSerializationImplTraits< |
| MojomType, |
| typename std::enable_if< |
| BelongsTo<MojomType, MojomTypeCategory::kUnion>::value>::type> { |
| template <typename MaybeConstUserType, typename FragmentType> |
| static void Serialize(MaybeConstUserType& input, FragmentType& fragment) { |
| mojo::internal::Serialize<MojomType>(input, fragment, false /* inline */); |
| } |
| }; |
| |
| template <typename MojomType, typename UserType> |
| mojo::Message SerializeAsMessageImpl(UserType* input) { |
| // Note that this is only called by application code serializing a structure |
| // manually (e.g. for storage). As such we don't want Mojo's soft message size |
| // limits to be applied. |
| mojo::Message message(0, 0, 0, 0, MOJO_CREATE_MESSAGE_FLAG_UNLIMITED_SIZE, |
| nullptr); |
| MessageFragment<typename MojomTypeTraits<MojomType>::Data> fragment(message); |
| MojomSerializationImplTraits<MojomType>::Serialize(*input, fragment); |
| message.SerializeHandles(/*group_controller=*/nullptr); |
| return message; |
| } |
| |
| template <typename MojomType, typename DataArrayType, typename UserType> |
| DataArrayType SerializeImpl(UserType* input) { |
| static_assert(BelongsTo<MojomType, MojomTypeCategory::kStruct>::value || |
| BelongsTo<MojomType, MojomTypeCategory::kUnion>::value, |
| "Unexpected type."); |
| Message message = SerializeAsMessageImpl<MojomType>(input); |
| uint32_t size = message.payload_num_bytes(); |
| DataArrayType result(size); |
| if (size) |
| memcpy(&result.front(), message.payload(), size); |
| return result; |
| } |
| |
| template <typename MojomType, typename UserType> |
| bool DeserializeImpl(Message& message, |
| const void* data, |
| size_t data_num_bytes, |
| UserType* output, |
| bool (*validate_func)(const void*, ValidationContext*)) { |
| static_assert(BelongsTo<MojomType, MojomTypeCategory::kStruct>::value || |
| BelongsTo<MojomType, MojomTypeCategory::kUnion>::value, |
| "Unexpected type."); |
| using DataType = typename MojomTypeTraits<MojomType>::Data; |
| |
| const void* input_buffer = data_num_bytes == 0 ? nullptr : data; |
| void* aligned_input_buffer = nullptr; |
| |
| // Validation code will insist that the input buffer is aligned, so we ensure |
| // that here. If the input data is not aligned, we (sadly) copy into an |
| // aligned buffer. In practice this should happen only rarely if ever. |
| bool need_copy = !IsAligned(input_buffer); |
| if (need_copy) { |
| aligned_input_buffer = malloc(data_num_bytes); |
| DCHECK(IsAligned(aligned_input_buffer)); |
| memcpy(aligned_input_buffer, data, data_num_bytes); |
| input_buffer = aligned_input_buffer; |
| } |
| |
| DCHECK(base::IsValueInRangeForNumericType<uint32_t>(data_num_bytes)); |
| ValidationContext validation_context(input_buffer, |
| static_cast<uint32_t>(data_num_bytes), |
| message.handles()->size(), 0); |
| bool result = false; |
| if (validate_func(input_buffer, &validation_context)) { |
| result = Deserialize<MojomType>( |
| reinterpret_cast<DataType*>(const_cast<void*>(input_buffer)), output, |
| &message); |
| } |
| |
| if (aligned_input_buffer) |
| free(aligned_input_buffer); |
| |
| return result; |
| } |
| |
| } // namespace internal |
| } // namespace mojo |
| |
| #endif // MOJO_PUBLIC_CPP_BINDINGS_LIB_SERIALIZATION_H_ |