blob: 8b6bda3d14198a95dcc29d4d8c7b2fa25bdac152 [file]
//
//
// Copyright 2015 gRPC authors.
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
//
//
#ifndef GRPCPP_IMPL_SERIALIZATION_TRAITS_H
#define GRPCPP_IMPL_SERIALIZATION_TRAITS_H
#include <grpc/event_engine/memory_allocator.h>
#include <type_traits>
#include <utility>
namespace grpc {
/// Defines how to serialize and deserialize some type.
///
/// Used for hooking different message serialization API's into GRPC.
/// Each SerializationTraits<Message> implementation must provide the
/// following functions:
/// 1. static Status Serialize(const Message& msg,
/// ByteBuffer* buffer,
/// bool* own_buffer);
/// OR
/// static Status Serialize(const Message& msg,
/// grpc_byte_buffer** buffer,
/// bool* own_buffer);
/// The former is preferred; the latter is deprecated
///
/// 2. static Status Deserialize(ByteBuffer* buffer,
/// Message* msg);
/// OR
/// static Status Deserialize(grpc_byte_buffer* buffer,
/// Message* msg);
/// The former is preferred; the latter is deprecated
///
/// Serialize is required to convert message to a ByteBuffer, and
/// return that byte buffer through *buffer. *own_buffer should
/// be set to true if the caller owns said byte buffer, or false if
/// ownership is retained elsewhere.
///
/// Deserialize is required to convert buffer into the message stored at
/// msg. max_receive_message_size is passed in as a bound on the maximum
/// number of message bytes Deserialize should accept.
///
/// Both functions return a Status, allowing them to explain what went
/// wrong if required.
template <class Message,
class UnusedButHereForPartialTemplateSpecialization = void>
class SerializationTraits;
namespace impl {
// Helper trait to robustly check if the allocator-aware Serialize exists.
template <typename Message, typename BufferPtr, typename = void>
struct has_allocator_serialize : std::false_type {};
template <typename Message, typename BufferPtr>
struct has_allocator_serialize<
Message, BufferPtr,
std::void_t<decltype(SerializationTraits<Message>::Serialize(
std::declval<grpc_event_engine::experimental::MemoryAllocator*>(),
std::declval<const Message&>(), std::declval<BufferPtr>(),
std::declval<bool*>()))>> : std::true_type {};
template <typename Message, typename BufferPtr>
inline constexpr bool has_allocator_serialize_v =
has_allocator_serialize<Message, BufferPtr>::value;
// Primary template for SerializeDispatch, now switched by a bool.
template <typename Message, typename BufferPtr,
bool = has_allocator_serialize_v<Message, BufferPtr>>
struct SerializeDispatch;
// Specialization for when the allocator-aware Serialize *exists*.
template <typename Message, typename BufferPtr>
struct SerializeDispatch<Message, BufferPtr, true> {
static auto Serialize(
grpc_event_engine::experimental::MemoryAllocator* allocator,
const Message& msg, BufferPtr buffer, bool* own_buffer) {
return SerializationTraits<Message>::Serialize(allocator, msg, buffer,
own_buffer);
}
};
// Specialization for when the allocator-aware Serialize *does not* exist.
template <typename Message, typename BufferPtr>
struct SerializeDispatch<Message, BufferPtr, false> {
static auto Serialize(
grpc_event_engine::experimental::MemoryAllocator* /*allocator*/,
const Message& msg, BufferPtr buffer, bool* own_buffer) {
// Fallback to the old Serialize, ignoring the allocator.
return SerializationTraits<Message>::Serialize(msg, buffer, own_buffer);
}
};
} // namespace impl
template <typename Message, typename BufferPtr>
auto Serialize(grpc_event_engine::experimental::MemoryAllocator* allocator,
const Message& msg, BufferPtr buffer, bool* own_buffer) {
return impl::SerializeDispatch<Message, BufferPtr>::Serialize(
allocator, msg, buffer, own_buffer);
}
template <typename BufferPtr, typename Message>
auto Deserialize(BufferPtr buffer, Message* msg) {
return SerializationTraits<Message>::Deserialize(buffer, msg);
}
} // namespace grpc
#endif // GRPCPP_IMPL_SERIALIZATION_TRAITS_H