156 lines
3.4 KiB
C++
156 lines
3.4 KiB
C++
/*
|
|
Copyright 2018 Glen Joseph Fernandes
|
|
(glenjofe@gmail.com)
|
|
|
|
Distributed under the Boost Software License, Version 1.0.
|
|
(http://www.boost.org/LICENSE_1_0.txt)
|
|
*/
|
|
#ifndef BOOST_CORE_EMPTY_VALUE_HPP
|
|
#define BOOST_CORE_EMPTY_VALUE_HPP
|
|
|
|
#include <boost/config.hpp>
|
|
#if !defined(BOOST_NO_CXX11_RVALUE_REFERENCES)
|
|
#include <utility>
|
|
#endif
|
|
|
|
#if defined(BOOST_GCC_VERSION) && (BOOST_GCC_VERSION >= 40700)
|
|
#define BOOST_DETAIL_EMPTY_VALUE_BASE
|
|
#elif defined(BOOST_INTEL) && defined(_MSC_VER) && (_MSC_VER >= 1800)
|
|
#define BOOST_DETAIL_EMPTY_VALUE_BASE
|
|
#elif defined(BOOST_MSVC) && (BOOST_MSVC >= 1800)
|
|
#define BOOST_DETAIL_EMPTY_VALUE_BASE
|
|
#elif defined(BOOST_CLANG) && !defined(__CUDACC__)
|
|
#if __has_feature(is_empty) && __has_feature(is_final)
|
|
#define BOOST_DETAIL_EMPTY_VALUE_BASE
|
|
#endif
|
|
#endif
|
|
|
|
#if defined(_MSC_VER)
|
|
#pragma warning(push)
|
|
#pragma warning(disable:4510)
|
|
#endif
|
|
|
|
namespace boost {
|
|
|
|
template<class T>
|
|
struct use_empty_value_base {
|
|
enum {
|
|
#if defined(BOOST_DETAIL_EMPTY_VALUE_BASE)
|
|
value = __is_empty(T) && !__is_final(T)
|
|
#else
|
|
value = false
|
|
#endif
|
|
};
|
|
};
|
|
|
|
struct empty_init_t { };
|
|
|
|
namespace empty_ {
|
|
|
|
template<class T, unsigned N = 0,
|
|
bool E = boost::use_empty_value_base<T>::value>
|
|
class empty_value {
|
|
public:
|
|
typedef T type;
|
|
|
|
#if !defined(BOOST_NO_CXX11_DEFAULTED_FUNCTIONS)
|
|
empty_value() = default;
|
|
#else
|
|
empty_value() { }
|
|
#endif
|
|
|
|
empty_value(boost::empty_init_t)
|
|
: value_() { }
|
|
|
|
#if !defined(BOOST_NO_CXX11_RVALUE_REFERENCES)
|
|
#if !defined(BOOST_NO_CXX11_VARIADIC_TEMPLATES)
|
|
template<class U, class... Args>
|
|
empty_value(boost::empty_init_t, U&& value, Args&&... args)
|
|
: value_(std::forward<U>(value), std::forward<Args>(args)...) { }
|
|
#else
|
|
template<class U>
|
|
empty_value(boost::empty_init_t, U&& value)
|
|
: value_(std::forward<U>(value)) { }
|
|
#endif
|
|
#else
|
|
template<class U>
|
|
empty_value(boost::empty_init_t, const U& value)
|
|
: value_(value) { }
|
|
|
|
template<class U>
|
|
empty_value(boost::empty_init_t, U& value)
|
|
: value_(value) { }
|
|
#endif
|
|
|
|
const T& get() const BOOST_NOEXCEPT {
|
|
return value_;
|
|
}
|
|
|
|
T& get() BOOST_NOEXCEPT {
|
|
return value_;
|
|
}
|
|
|
|
private:
|
|
T value_;
|
|
};
|
|
|
|
#if !defined(BOOST_NO_TEMPLATE_PARTIAL_SPECIALIZATION)
|
|
template<class T, unsigned N>
|
|
class empty_value<T, N, true>
|
|
: T {
|
|
public:
|
|
typedef T type;
|
|
|
|
#if !defined(BOOST_NO_CXX11_DEFAULTED_FUNCTIONS)
|
|
empty_value() = default;
|
|
#else
|
|
empty_value() { }
|
|
#endif
|
|
|
|
empty_value(boost::empty_init_t)
|
|
: T() { }
|
|
|
|
#if !defined(BOOST_NO_CXX11_RVALUE_REFERENCES)
|
|
#if !defined(BOOST_NO_CXX11_VARIADIC_TEMPLATES)
|
|
template<class U, class... Args>
|
|
empty_value(boost::empty_init_t, U&& value, Args&&... args)
|
|
: T(std::forward<U>(value), std::forward<Args>(args)...) { }
|
|
#else
|
|
template<class U>
|
|
empty_value(boost::empty_init_t, U&& value)
|
|
: T(std::forward<U>(value)) { }
|
|
#endif
|
|
#else
|
|
template<class U>
|
|
empty_value(boost::empty_init_t, const U& value)
|
|
: T(value) { }
|
|
|
|
template<class U>
|
|
empty_value(boost::empty_init_t, U& value)
|
|
: T(value) { }
|
|
#endif
|
|
|
|
const T& get() const BOOST_NOEXCEPT {
|
|
return *this;
|
|
}
|
|
|
|
T& get() BOOST_NOEXCEPT {
|
|
return *this;
|
|
}
|
|
};
|
|
#endif
|
|
|
|
} /* empty_ */
|
|
|
|
using empty_::empty_value;
|
|
|
|
BOOST_INLINE_CONSTEXPR empty_init_t empty_init = empty_init_t();
|
|
|
|
} /* boost */
|
|
|
|
#if defined(_MSC_VER)
|
|
#pragma warning(pop)
|
|
#endif
|
|
|
|
#endif
|