// Copyright 2013 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#ifndef MOJO_PUBLIC_CPP_BINDINGS_LIB_SERIALIZATION_UTIL_H_
#define MOJO_PUBLIC_CPP_BINDINGS_LIB_SERIALIZATION_UTIL_H_
#include <stddef.h>
#include <stdint.h>
#include <queue>
#include "base/logging.h"
#include "base/macros.h"
#include "mojo/public/cpp/bindings/lib/bindings_internal.h"
#include "mojo/public/cpp/bindings/lib/serialization_context.h"
namespace mojo {
namespace internal {
template <typename T>
struct HasIsNullMethod {
template <typename U>
static char Test(decltype(U::IsNull)*);
template <typename U>
static int Test(...);
static const bool value = sizeof(Test<T>(0)) == sizeof(char);
private:
EnsureTypeIsComplete<T> check_t_;
};
template <
typename Traits,
typename UserType,
typename std::enable_if<HasIsNullMethod<Traits>::value>::type* = nullptr>
bool CallIsNullIfExists(const UserType& input) {
return Traits::IsNull(input);
}
template <
typename Traits,
typename UserType,
typename std::enable_if<!HasIsNullMethod<Traits>::value>::type* = nullptr>
bool CallIsNullIfExists(const UserType& input) {
return false;
}
template <typename T>
struct HasSetToNullMethod {
template <typename U>
static char Test(decltype(U::SetToNull)*);
template <typename U>
static int Test(...);
static const bool value = sizeof(Test<T>(0)) == sizeof(char);
private:
EnsureTypeIsComplete<T> check_t_;
};
template <
typename Traits,
typename UserType,
typename std::enable_if<HasSetToNullMethod<Traits>::value>::type* = nullptr>
bool CallSetToNullIfExists(UserType* output) {
Traits::SetToNull(output);
return true;
}
template <typename Traits,
typename UserType,
typename std::enable_if<!HasSetToNullMethod<Traits>::value>::type* =
nullptr>
bool CallSetToNullIfExists(UserType* output) {
LOG(ERROR) << "A null value is received. But the Struct/Array/StringTraits "
<< "class doesn't define a SetToNull() function and therefore is "
<< "unable to deserialize the value.";
return false;
}
template <typename T>
struct HasSetUpContextMethod {
template <typename U>
static char Test(decltype(U::SetUpContext)*);
template <typename U>
static int Test(...);
static const bool value = sizeof(Test<T>(0)) == sizeof(char);
private:
EnsureTypeIsComplete<T> check_t_;
};
template <typename Traits,
bool has_context = HasSetUpContextMethod<Traits>::value>
struct CustomContextHelper;
template <typename Traits>
struct CustomContextHelper<Traits, true> {
template <typename MaybeConstUserType>
static void* SetUp(MaybeConstUserType& input, SerializationContext* context) {
void* custom_context = Traits::SetUpContext(input);
if (!context->custom_contexts)
context->custom_contexts.reset(new std::queue<void*>());
context->custom_contexts->push(custom_context);
return custom_context;
}
static void* GetNext(SerializationContext* context) {
void* custom_context = context->custom_contexts->front();
context->custom_contexts->pop();
return custom_context;
}
template <typename MaybeConstUserType>
static void TearDown(MaybeConstUserType& input, void* custom_context) {
Traits::TearDownContext(input, custom_context);
}
};
template <typename Traits>
struct CustomContextHelper<Traits, false> {
template <typename MaybeConstUserType>
static void* SetUp(MaybeConstUserType& input, SerializationContext* context) {
return nullptr;
}
static void* GetNext(SerializationContext* context) { return nullptr; }
template <typename MaybeConstUserType>
static void TearDown(MaybeConstUserType& input, void* custom_context) {
DCHECK(!custom_context);
}
};
template <typename ReturnType, typename ParamType, typename InputUserType>
ReturnType CallWithContext(ReturnType (*f)(ParamType, void*),
InputUserType&& input,
void* context) {
return f(std::forward<InputUserType>(input), context);
}
template <typename ReturnType, typename ParamType, typename InputUserType>
ReturnType CallWithContext(ReturnType (*f)(ParamType),
InputUserType&& input,
void* context) {
return f(std::forward<InputUserType>(input));
}
template <typename T, typename MaybeConstUserType>
struct HasGetBeginMethod {
template <typename U>
static char Test(decltype(U::GetBegin(std::declval<MaybeConstUserType&>()))*);
template <typename U>
static int Test(...);
static const bool value = sizeof(Test<T>(0)) == sizeof(char);
private:
EnsureTypeIsComplete<T> check_t_;
};
template <
typename Traits,
typename MaybeConstUserType,
typename std::enable_if<
HasGetBeginMethod<Traits, MaybeConstUserType>::value>::type* = nullptr>
decltype(Traits::GetBegin(std::declval<MaybeConstUserType&>()))
CallGetBeginIfExists(MaybeConstUserType& input) {
return Traits::GetBegin(input);
}
template <
typename Traits,
typename MaybeConstUserType,
typename std::enable_if<
!HasGetBeginMethod<Traits, MaybeConstUserType>::value>::type* = nullptr>
size_t CallGetBeginIfExists(MaybeConstUserType& input) {
return 0;
}
template <typename T, typename MaybeConstUserType>
struct HasGetDataMethod {
template <typename U>
static char Test(decltype(U::GetData(std::declval<MaybeConstUserType&>()))*);
template <typename U>
static int Test(...);
static const bool value = sizeof(Test<T>(0)) == sizeof(char);
private:
EnsureTypeIsComplete<T> check_t_;
};
template <
typename Traits,
typename MaybeConstUserType,
typename std::enable_if<
HasGetDataMethod<Traits, MaybeConstUserType>::value>::type* = nullptr>
decltype(Traits::GetData(std::declval<MaybeConstUserType&>()))
CallGetDataIfExists(MaybeConstUserType& input) {
return Traits::GetData(input);
}
template <
typename Traits,
typename MaybeConstUserType,
typename std::enable_if<
!HasGetDataMethod<Traits, MaybeConstUserType>::value>::type* = nullptr>
void* CallGetDataIfExists(MaybeConstUserType& input) {
return nullptr;
}
} // namespace internal
} // namespace mojo
#endif // MOJO_PUBLIC_CPP_BINDINGS_LIB_SERIALIZATION_UTIL_H_