forked from boostorg/core
Update pointer_traits and to_address to reflect the design adopted for C++20
This commit is contained in:
@ -1,5 +1,5 @@
|
|||||||
[/
|
[/
|
||||||
Copyright 2017 Glen Joseph Fernandes
|
Copyright 2017-2018 Glen Joseph Fernandes
|
||||||
(glenjofe@gmail.com)
|
(glenjofe@gmail.com)
|
||||||
|
|
||||||
Distributed under the Boost Software License, Version 1.0.
|
Distributed under the Boost Software License, Version 1.0.
|
||||||
@ -20,8 +20,10 @@ The header <boost/core/pointer_traits.hpp> provides the class template
|
|||||||
`boost::pointer_traits` to facilitate use of pointer-like types. The C++11
|
`boost::pointer_traits` to facilitate use of pointer-like types. The C++11
|
||||||
standard library introduced `std::pointer_traits` along with an allocator
|
standard library introduced `std::pointer_traits` along with an allocator
|
||||||
model which supported pointer-like types in addition to plain raw pointers.
|
model which supported pointer-like types in addition to plain raw pointers.
|
||||||
This implementation also supports C++98, and adds additional functionality
|
This implementation also supports C++98.
|
||||||
for obtaining raw pointers from pointers.
|
|
||||||
|
It also provides the function template `boost::to_address` to obtain a raw
|
||||||
|
pointer from an object of any pointer-like type.
|
||||||
|
|
||||||
[endsect]
|
[endsect]
|
||||||
|
|
||||||
@ -54,7 +56,6 @@ namespace boost {
|
|||||||
template<class U> using rebind = typename rebind_to<U>::type;
|
template<class U> using rebind = typename rebind_to<U>::type;
|
||||||
|
|
||||||
static pointer pointer_to(``['see below]`` v);
|
static pointer pointer_to(``['see below]`` v);
|
||||||
static element_type* to_address(const pointer& v) noexcept;
|
|
||||||
};
|
};
|
||||||
|
|
||||||
template<class T> struct pointer_traits<T*> {
|
template<class T> struct pointer_traits<T*> {
|
||||||
@ -66,11 +67,13 @@ namespace boost {
|
|||||||
template<class U> using rebind = typename rebind_to<U>::type;
|
template<class U> using rebind = typename rebind_to<U>::type;
|
||||||
|
|
||||||
static pointer pointer_to(``['see below]`` v) noexcept;
|
static pointer pointer_to(``['see below]`` v) noexcept;
|
||||||
static element_type* to_address(pointer v) noexcept;
|
|
||||||
};
|
};
|
||||||
|
|
||||||
template<class T>
|
template<class T>
|
||||||
typename pointer_traits<T>::element_type* to_address(const T& v) noexcept;
|
constexpr T* to_address(T* v) noexcept;
|
||||||
|
|
||||||
|
template<class T>
|
||||||
|
auto to_address(const T& v) noexcept;
|
||||||
}
|
}
|
||||||
```
|
```
|
||||||
|
|
||||||
@ -93,39 +96,44 @@ namespace boost {
|
|||||||
|
|
||||||
[section Member functions]
|
[section Member functions]
|
||||||
|
|
||||||
[variablelist pointer_traits
|
[variablelist
|
||||||
[[`static pointer pointer_to(`['see below] `v);`]
|
[[`static pointer pointer_traits::pointer_to(`['see below] `v);`]
|
||||||
[[variablelist
|
[[variablelist
|
||||||
[[Remark]
|
[[Remark]
|
||||||
[If `element_type` is a void type, the type of `v` is unspecified; otherwise,
|
[If `element_type` is a void type, the type of `v` is unspecified; otherwise,
|
||||||
it is `element_type&`.]]
|
it is `element_type&`.]]
|
||||||
[[Returns]
|
[[Returns]
|
||||||
[A pointer to `v` obtained by calling `T::pointer_to(v)`.]]]]]
|
[A pointer to `v` obtained by calling `T::pointer_to(v)`.]]]]]
|
||||||
[[`static element_type* to_address(const pointer& v) noexcept;`]
|
[[`static pointer pointer_traits<T*>::pointer_to(`['see below] `v) noexcept;`]
|
||||||
[[variablelist
|
|
||||||
[[Requires] [`v` is not a null pointer.]]
|
|
||||||
[[Returns]
|
|
||||||
[A pointer of type `element_type*` that references the same location
|
|
||||||
as the argument.]]]]]]
|
|
||||||
|
|
||||||
[variablelist pointer_traits<T*>
|
|
||||||
[[`static pointer pointer_to(`['see below] `v) noexcept;`]
|
|
||||||
[[variablelist
|
[[variablelist
|
||||||
[[Remark]
|
[[Remark]
|
||||||
[If `element_type` is a void type, the type of `v` is unspecified; otherwise,
|
[If `element_type` is a void type, the type of `v` is unspecified; otherwise,
|
||||||
it is `element_type&`.]]
|
it is `element_type&`.]]
|
||||||
[[Returns] [The result of `boost::addressof(v)`.]]]]]
|
[[Returns] [`addressof(v)`.]]]]]]
|
||||||
|
|
||||||
|
[endsect]
|
||||||
|
|
||||||
|
[section Optional members]
|
||||||
|
|
||||||
|
[variablelist
|
||||||
[[`static element_type* to_address(pointer v) noexcept;`]
|
[[`static element_type* to_address(pointer v) noexcept;`]
|
||||||
[[variablelist [[Returns] [The value of `v`.]]]]]]
|
[[variablelist
|
||||||
|
[[Returns]
|
||||||
|
[A pointer of type `element_type*` that references the same location as the
|
||||||
|
argument `p`.]]
|
||||||
|
[[Note] [This function should be the inverse of `pointer_to`. If defined, it
|
||||||
|
customizes the behavior of the non-member function `to_address`.]]]]]]
|
||||||
|
|
||||||
[endsect]
|
[endsect]
|
||||||
|
|
||||||
[section Free functions]
|
[section Free functions]
|
||||||
|
|
||||||
[variablelist
|
[variablelist
|
||||||
[[`template<class T> typename pointer_traits<T>::element_type*
|
[[`template<class T> constexpr T* to_address(T* v) noexcept;`]
|
||||||
to_address(const T& v) noexcept;`]
|
[[variablelist [[Returns] [`v`.]]]]]
|
||||||
[[variablelist [[Returns] [`boost::pointer_traits<T>::to_address(v)`.]]]]]]
|
[[`template<class T> auto to_address(const T& v) noexcept;`]
|
||||||
|
[[variablelist [[Returns] [`pointer_traits<T>::to_address(v)` if that
|
||||||
|
expression is well-formed, otherwise `to_address(v.operator->())`.]]]]]]
|
||||||
|
|
||||||
[endsect]
|
[endsect]
|
||||||
|
|
||||||
@ -133,8 +141,8 @@ namespace boost {
|
|||||||
|
|
||||||
[section Acknowledgments]
|
[section Acknowledgments]
|
||||||
|
|
||||||
Glen Fernandes implemented `pointer_traits` with reviews and guidance from
|
Glen Fernandes implemented `pointer_traits` and `to_address` with reviews and
|
||||||
Peter Dimov.
|
guidance from Peter Dimov.
|
||||||
|
|
||||||
[endsect]
|
[endsect]
|
||||||
|
|
||||||
|
@ -17,20 +17,6 @@ Distributed under the Boost Software License, Version 1.0.
|
|||||||
|
|
||||||
namespace boost {
|
namespace boost {
|
||||||
|
|
||||||
template<class T>
|
|
||||||
struct pointer_traits;
|
|
||||||
|
|
||||||
namespace detail {
|
|
||||||
|
|
||||||
template<class U>
|
|
||||||
inline typename boost::pointer_traits<U>::element_type*
|
|
||||||
ptr_to_address(const U& v) BOOST_NOEXCEPT
|
|
||||||
{
|
|
||||||
return boost::pointer_traits<U>::to_address(v);
|
|
||||||
}
|
|
||||||
|
|
||||||
} /* detail */
|
|
||||||
|
|
||||||
#if !defined(BOOST_NO_CXX11_POINTER_TRAITS)
|
#if !defined(BOOST_NO_CXX11_POINTER_TRAITS)
|
||||||
template<class T>
|
template<class T>
|
||||||
struct pointer_traits
|
struct pointer_traits
|
||||||
@ -39,10 +25,6 @@ struct pointer_traits
|
|||||||
struct rebind_to {
|
struct rebind_to {
|
||||||
typedef typename std::pointer_traits<T>::template rebind<U> type;
|
typedef typename std::pointer_traits<T>::template rebind<U> type;
|
||||||
};
|
};
|
||||||
static typename std::pointer_traits<T>::element_type*
|
|
||||||
to_address(const T& v) BOOST_NOEXCEPT {
|
|
||||||
return detail::ptr_to_address(v.operator->());
|
|
||||||
}
|
|
||||||
};
|
};
|
||||||
|
|
||||||
template<class T>
|
template<class T>
|
||||||
@ -52,9 +34,6 @@ struct pointer_traits<T*>
|
|||||||
struct rebind_to {
|
struct rebind_to {
|
||||||
typedef U* type;
|
typedef U* type;
|
||||||
};
|
};
|
||||||
static T* to_address(T* v) BOOST_NOEXCEPT {
|
|
||||||
return v;
|
|
||||||
}
|
|
||||||
};
|
};
|
||||||
#else
|
#else
|
||||||
namespace detail {
|
namespace detail {
|
||||||
@ -111,34 +90,34 @@ struct ptr_difference<T,
|
|||||||
};
|
};
|
||||||
|
|
||||||
template<class T, class V>
|
template<class T, class V>
|
||||||
struct ptr_rebind_to;
|
struct ptr_transform;
|
||||||
|
|
||||||
#if !defined(BOOST_NO_CXX11_VARIADIC_TEMPLATES)
|
#if !defined(BOOST_NO_CXX11_VARIADIC_TEMPLATES)
|
||||||
template<template<class, class...> class T, class U, class... Args, class V>
|
template<template<class, class...> class T, class U, class... Args, class V>
|
||||||
struct ptr_rebind_to<T<U, Args...>, V> {
|
struct ptr_transform<T<U, Args...>, V> {
|
||||||
typedef T<V, Args...> type;
|
typedef T<V, Args...> type;
|
||||||
};
|
};
|
||||||
#else
|
#else
|
||||||
template<template<class> class T, class U, class V>
|
template<template<class> class T, class U, class V>
|
||||||
struct ptr_rebind_to<T<U>, V> {
|
struct ptr_transform<T<U>, V> {
|
||||||
typedef T<V> type;
|
typedef T<V> type;
|
||||||
};
|
};
|
||||||
|
|
||||||
template<template<class, class> class T, class U1, class U2, class V>
|
template<template<class, class> class T, class U1, class U2, class V>
|
||||||
struct ptr_rebind_to<T<U1, U2>, V> {
|
struct ptr_transform<T<U1, U2>, V> {
|
||||||
typedef T<V, U2> type;
|
typedef T<V, U2> type;
|
||||||
};
|
};
|
||||||
|
|
||||||
template<template<class, class, class> class T,
|
template<template<class, class, class> class T,
|
||||||
class U1, class U2, class U3, class V>
|
class U1, class U2, class U3, class V>
|
||||||
struct ptr_rebind_to<T<U1, U2, U3>, V> {
|
struct ptr_transform<T<U1, U2, U3>, V> {
|
||||||
typedef T<V, U2, U3> type;
|
typedef T<V, U2, U3> type;
|
||||||
};
|
};
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
template<class T, class U, class = void>
|
template<class T, class U, class = void>
|
||||||
struct ptr_rebind {
|
struct ptr_rebind {
|
||||||
typedef typename ptr_rebind_to<T, U>::type type;
|
typedef typename ptr_transform<T, U>::type type;
|
||||||
};
|
};
|
||||||
|
|
||||||
#if !defined(BOOST_NO_CXX11_TEMPLATE_ALIASES)
|
#if !defined(BOOST_NO_CXX11_TEMPLATE_ALIASES)
|
||||||
@ -178,9 +157,6 @@ struct pointer_traits {
|
|||||||
pointer_to(typename detail::ptr_value<element_type>::type& v) {
|
pointer_to(typename detail::ptr_value<element_type>::type& v) {
|
||||||
return pointer::pointer_to(v);
|
return pointer::pointer_to(v);
|
||||||
}
|
}
|
||||||
static element_type* to_address(const pointer& v) BOOST_NOEXCEPT {
|
|
||||||
return detail::ptr_to_address(v.operator->());
|
|
||||||
}
|
|
||||||
};
|
};
|
||||||
|
|
||||||
template<class T>
|
template<class T>
|
||||||
@ -200,26 +176,59 @@ struct pointer_traits<T*> {
|
|||||||
pointer_to(typename detail::ptr_value<T>::type& v) BOOST_NOEXCEPT {
|
pointer_to(typename detail::ptr_value<T>::type& v) BOOST_NOEXCEPT {
|
||||||
return boost::addressof(v);
|
return boost::addressof(v);
|
||||||
}
|
}
|
||||||
static T* to_address(T* v) BOOST_NOEXCEPT {
|
|
||||||
return v;
|
|
||||||
}
|
|
||||||
};
|
};
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
template<class T>
|
template<class T>
|
||||||
inline typename pointer_traits<T>::element_type*
|
BOOST_CONSTEXPR inline T*
|
||||||
to_address(const T& v) BOOST_NOEXCEPT
|
|
||||||
{
|
|
||||||
return pointer_traits<T>::to_address(v);
|
|
||||||
}
|
|
||||||
|
|
||||||
template<class T>
|
|
||||||
inline T*
|
|
||||||
to_address(T* v) BOOST_NOEXCEPT
|
to_address(T* v) BOOST_NOEXCEPT
|
||||||
{
|
{
|
||||||
return v;
|
return v;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#if !defined(BOOST_NO_CXX11_SFINAE_EXPR) && \
|
||||||
|
!defined(BOOST_NO_CXX14_RETURN_TYPE_DEDUCTION)
|
||||||
|
namespace detail {
|
||||||
|
|
||||||
|
template<class T>
|
||||||
|
inline T*
|
||||||
|
ptr_address(T* v, int) BOOST_NOEXCEPT
|
||||||
|
{
|
||||||
|
return v;
|
||||||
|
}
|
||||||
|
|
||||||
|
template<class T>
|
||||||
|
inline auto
|
||||||
|
ptr_address(const T& v, int) BOOST_NOEXCEPT
|
||||||
|
-> decltype(boost::pointer_traits<T>::to_address(v))
|
||||||
|
{
|
||||||
|
return boost::pointer_traits<T>::to_address(v);
|
||||||
|
}
|
||||||
|
|
||||||
|
template<class T>
|
||||||
|
inline auto
|
||||||
|
ptr_address(const T& v, long) BOOST_NOEXCEPT
|
||||||
|
{
|
||||||
|
return boost::detail::ptr_address(v.operator->(), 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
} /* detail */
|
||||||
|
|
||||||
|
template<class T>
|
||||||
|
inline auto
|
||||||
|
to_address(const T& v) BOOST_NOEXCEPT
|
||||||
|
{
|
||||||
|
return boost::detail::ptr_address(v, 0);
|
||||||
|
}
|
||||||
|
#else
|
||||||
|
template<class T>
|
||||||
|
inline typename pointer_traits<T>::element_type*
|
||||||
|
to_address(const T& v) BOOST_NOEXCEPT
|
||||||
|
{
|
||||||
|
return boost::to_address(v.operator->());
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
} /* boost */
|
} /* boost */
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
@ -114,7 +114,6 @@ compile-fail scoped_enum_compile_fail_conv_to_int.cpp ;
|
|||||||
|
|
||||||
run underlying_type.cpp ;
|
run underlying_type.cpp ;
|
||||||
|
|
||||||
run pointer_traits_to_address_test.cpp ;
|
|
||||||
run pointer_traits_pointer_test.cpp ;
|
run pointer_traits_pointer_test.cpp ;
|
||||||
run pointer_traits_element_type_test.cpp ;
|
run pointer_traits_element_type_test.cpp ;
|
||||||
run pointer_traits_difference_type_test.cpp ;
|
run pointer_traits_difference_type_test.cpp ;
|
||||||
|
@ -1,131 +0,0 @@
|
|||||||
/*
|
|
||||||
Copyright 2017 Glen Joseph Fernandes
|
|
||||||
(glenjofe@gmail.com)
|
|
||||||
|
|
||||||
Distributed under the Boost Software License, Version 1.0.
|
|
||||||
(http://www.boost.org/LICENSE_1_0.txt)
|
|
||||||
*/
|
|
||||||
#include <boost/core/pointer_traits.hpp>
|
|
||||||
#include <boost/core/lightweight_test.hpp>
|
|
||||||
|
|
||||||
template<class T>
|
|
||||||
class pointer {
|
|
||||||
public:
|
|
||||||
typedef typename boost::pointer_traits<T>::element_type element_type;
|
|
||||||
pointer(T value)
|
|
||||||
: value_(value) { }
|
|
||||||
T operator->() const BOOST_NOEXCEPT {
|
|
||||||
return value_;
|
|
||||||
}
|
|
||||||
private:
|
|
||||||
T value_;
|
|
||||||
};
|
|
||||||
|
|
||||||
template<class T>
|
|
||||||
class special {
|
|
||||||
public:
|
|
||||||
special(T* value)
|
|
||||||
: value_(value) { }
|
|
||||||
T* get() const BOOST_NOEXCEPT {
|
|
||||||
return value_;
|
|
||||||
}
|
|
||||||
private:
|
|
||||||
T* value_;
|
|
||||||
};
|
|
||||||
|
|
||||||
namespace boost {
|
|
||||||
|
|
||||||
template<class T>
|
|
||||||
struct pointer_traits<special<T> > {
|
|
||||||
typedef special<T> pointer;
|
|
||||||
typedef T element_type;
|
|
||||||
typedef std::ptrdiff_t difference_type;
|
|
||||||
template<class U>
|
|
||||||
struct rebind_to {
|
|
||||||
typedef special<U> type;
|
|
||||||
};
|
|
||||||
#if !defined(BOOST_NO_CXX11_TEMPLATE_ALIASES)
|
|
||||||
template<class U>
|
|
||||||
using rebind = typename rebind_to<U>::type;
|
|
||||||
#endif
|
|
||||||
template<class U>
|
|
||||||
static pointer pointer_to(U& v) BOOST_NOEXCEPT {
|
|
||||||
return pointer(&v);
|
|
||||||
}
|
|
||||||
static element_type* to_address(const pointer& v) BOOST_NOEXCEPT {
|
|
||||||
return v.get();
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
} /* boost */
|
|
||||||
|
|
||||||
int main()
|
|
||||||
{
|
|
||||||
int i = 0;
|
|
||||||
{
|
|
||||||
typedef int* type;
|
|
||||||
type p = &i;
|
|
||||||
BOOST_TEST(boost::pointer_traits<type>::to_address(p) == &i);
|
|
||||||
}
|
|
||||||
{
|
|
||||||
typedef pointer<int*> type;
|
|
||||||
type p(&i);
|
|
||||||
BOOST_TEST(boost::pointer_traits<type>::to_address(p) == &i);
|
|
||||||
}
|
|
||||||
{
|
|
||||||
typedef pointer<pointer<int*> > type;
|
|
||||||
type p(&i);
|
|
||||||
BOOST_TEST(boost::pointer_traits<type>::to_address(p) == &i);
|
|
||||||
}
|
|
||||||
{
|
|
||||||
typedef void* type;
|
|
||||||
type p = &i;
|
|
||||||
BOOST_TEST(boost::pointer_traits<type>::to_address(p) == &i);
|
|
||||||
}
|
|
||||||
{
|
|
||||||
typedef pointer<void*> type;
|
|
||||||
type p(&i);
|
|
||||||
BOOST_TEST(boost::pointer_traits<type>::to_address(p) == &i);
|
|
||||||
}
|
|
||||||
{
|
|
||||||
typedef const int* type;
|
|
||||||
type p = &i;
|
|
||||||
BOOST_TEST(boost::pointer_traits<type>::to_address(p) == &i);
|
|
||||||
}
|
|
||||||
{
|
|
||||||
typedef pointer<const int*> type;
|
|
||||||
type p(&i);
|
|
||||||
BOOST_TEST(boost::pointer_traits<type>::to_address(p) == &i);
|
|
||||||
}
|
|
||||||
{
|
|
||||||
typedef special<int> type;
|
|
||||||
type p(&i);
|
|
||||||
BOOST_TEST(boost::pointer_traits<type>::to_address(p) == &i);
|
|
||||||
}
|
|
||||||
{
|
|
||||||
typedef special<void> type;
|
|
||||||
type p(&i);
|
|
||||||
BOOST_TEST(boost::pointer_traits<type>::to_address(p) == &i);
|
|
||||||
}
|
|
||||||
{
|
|
||||||
typedef special<const int> type;
|
|
||||||
type p(&i);
|
|
||||||
BOOST_TEST(boost::pointer_traits<type>::to_address(p) == &i);
|
|
||||||
}
|
|
||||||
{
|
|
||||||
typedef pointer<special<int> > type;
|
|
||||||
type p(&i);
|
|
||||||
BOOST_TEST(boost::pointer_traits<type>::to_address(p) == &i);
|
|
||||||
}
|
|
||||||
{
|
|
||||||
typedef pointer<special<void> > type;
|
|
||||||
type p(&i);
|
|
||||||
BOOST_TEST(boost::pointer_traits<type>::to_address(p) == &i);
|
|
||||||
}
|
|
||||||
{
|
|
||||||
typedef pointer<special<const int> > type;
|
|
||||||
type p(&i);
|
|
||||||
BOOST_TEST(boost::pointer_traits<type>::to_address(p) == &i);
|
|
||||||
}
|
|
||||||
return boost::report_errors();
|
|
||||||
}
|
|
@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
Copyright 2017 Glen Joseph Fernandes
|
Copyright 2017-2018 Glen Joseph Fernandes
|
||||||
(glenjofe@gmail.com)
|
(glenjofe@gmail.com)
|
||||||
|
|
||||||
Distributed under the Boost Software License, Version 1.0.
|
Distributed under the Boost Software License, Version 1.0.
|
||||||
@ -9,123 +9,163 @@ Distributed under the Boost Software License, Version 1.0.
|
|||||||
#include <boost/core/lightweight_test.hpp>
|
#include <boost/core/lightweight_test.hpp>
|
||||||
|
|
||||||
template<class T>
|
template<class T>
|
||||||
class pointer {
|
class P1 {
|
||||||
public:
|
public:
|
||||||
typedef typename boost::pointer_traits<T>::element_type element_type;
|
explicit P1(T* p)
|
||||||
pointer(T value)
|
: p_(p) { }
|
||||||
: value_(value) { }
|
T* operator->() const BOOST_NOEXCEPT {
|
||||||
T operator->() const BOOST_NOEXCEPT {
|
return p_;
|
||||||
return value_;
|
|
||||||
}
|
}
|
||||||
private:
|
private:
|
||||||
T value_;
|
T* p_;
|
||||||
};
|
};
|
||||||
|
|
||||||
template<class T>
|
template<class T>
|
||||||
class special {
|
class P2 {
|
||||||
public:
|
public:
|
||||||
special(T* value)
|
explicit P2(T* p)
|
||||||
: value_(value) { }
|
: p_(p) { }
|
||||||
T* get() const BOOST_NOEXCEPT {
|
P1<T> operator->() const BOOST_NOEXCEPT {
|
||||||
return value_;
|
return p_;
|
||||||
}
|
}
|
||||||
private:
|
private:
|
||||||
T* value_;
|
P1<T> p_;
|
||||||
|
};
|
||||||
|
|
||||||
|
#if !defined(BOOST_NO_CXX11_SFINAE_EXPR) && \
|
||||||
|
!defined(BOOST_NO_CXX14_RETURN_TYPE_DEDUCTION)
|
||||||
|
template<class T>
|
||||||
|
class P3 {
|
||||||
|
public:
|
||||||
|
explicit P3(T* p)
|
||||||
|
: p_(p) { }
|
||||||
|
T* get() const BOOST_NOEXCEPT {
|
||||||
|
return p_;
|
||||||
|
}
|
||||||
|
private:
|
||||||
|
T* p_;
|
||||||
};
|
};
|
||||||
|
|
||||||
namespace boost {
|
namespace boost {
|
||||||
|
|
||||||
template<class T>
|
template<class T>
|
||||||
struct pointer_traits<special<T> > {
|
struct pointer_traits<P3<T> > {
|
||||||
typedef special<T> pointer;
|
static T* to_address(const P3<T>& p) BOOST_NOEXCEPT {
|
||||||
typedef T element_type;
|
return p.get();
|
||||||
typedef std::ptrdiff_t difference_type;
|
|
||||||
template<class U>
|
|
||||||
struct rebind_to {
|
|
||||||
typedef special<U> type;
|
|
||||||
};
|
|
||||||
#if !defined(BOOST_NO_CXX11_TEMPLATE_ALIASES)
|
|
||||||
template<class U>
|
|
||||||
using rebind = typename rebind_to<U>::type;
|
|
||||||
#endif
|
|
||||||
template<class U>
|
|
||||||
static pointer pointer_to(U& v) BOOST_NOEXCEPT {
|
|
||||||
return pointer(&v);
|
|
||||||
}
|
|
||||||
static element_type* to_address(const pointer& v) BOOST_NOEXCEPT {
|
|
||||||
return v.get();
|
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
} /* boost */
|
} /* boost */
|
||||||
|
|
||||||
|
template<class T>
|
||||||
|
class P4 {
|
||||||
|
public:
|
||||||
|
explicit P4(T* p)
|
||||||
|
: p_(p) { }
|
||||||
|
T* operator->() const BOOST_NOEXCEPT {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
T* get() const BOOST_NOEXCEPT {
|
||||||
|
return p_;
|
||||||
|
}
|
||||||
|
private:
|
||||||
|
int* p_;
|
||||||
|
};
|
||||||
|
|
||||||
|
namespace boost {
|
||||||
|
|
||||||
|
template<class T>
|
||||||
|
struct pointer_traits<P4<T> > {
|
||||||
|
static T* to_address(const P4<T>& p) BOOST_NOEXCEPT {
|
||||||
|
return p.get();
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
} /* boost */
|
||||||
|
|
||||||
|
#if !defined(BOOST_NO_CXX11_POINTER_TRAITS)
|
||||||
|
template<class T>
|
||||||
|
class P5 {
|
||||||
|
public:
|
||||||
|
explicit P5(T* p)
|
||||||
|
: p_(p) { }
|
||||||
|
T* get() const BOOST_NOEXCEPT {
|
||||||
|
return p_;
|
||||||
|
}
|
||||||
|
private:
|
||||||
|
T* p_;
|
||||||
|
};
|
||||||
|
|
||||||
|
namespace std {
|
||||||
|
|
||||||
|
template<class T>
|
||||||
|
struct pointer_traits<P5<T> > {
|
||||||
|
static T* to_address(const P5<T>& p) BOOST_NOEXCEPT {
|
||||||
|
return p.get();
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
} /* std */
|
||||||
|
|
||||||
|
template<class T>
|
||||||
|
class P6 {
|
||||||
|
public:
|
||||||
|
explicit P6(T* p)
|
||||||
|
: p_(p) { }
|
||||||
|
T* get() const BOOST_NOEXCEPT {
|
||||||
|
return p_;
|
||||||
|
}
|
||||||
|
private:
|
||||||
|
T* p_;
|
||||||
|
};
|
||||||
|
|
||||||
|
namespace boost {
|
||||||
|
|
||||||
|
template<class T>
|
||||||
|
struct pointer_traits<P6<T> > {
|
||||||
|
static T* to_address(const P6<T>& p) BOOST_NOEXCEPT {
|
||||||
|
return p.get();
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
} /* boost */
|
||||||
|
|
||||||
|
namespace std {
|
||||||
|
|
||||||
|
template<class T>
|
||||||
|
struct pointer_traits<P6<T> > {
|
||||||
|
static T* to_address(const P6<T>& p) BOOST_NOEXCEPT {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
} /* std */
|
||||||
|
|
||||||
|
#endif
|
||||||
|
#endif
|
||||||
|
|
||||||
int main()
|
int main()
|
||||||
{
|
{
|
||||||
int i = 0;
|
int i = 0;
|
||||||
{
|
BOOST_TEST(boost::to_address(&i) == &i);
|
||||||
typedef int* type;
|
int* p = &i;
|
||||||
type p = &i;
|
BOOST_TEST(boost::to_address(p) == &i);
|
||||||
BOOST_TEST(boost::to_address(p) == &i);
|
P1<int> p1(&i);
|
||||||
}
|
BOOST_TEST(boost::to_address(p1) == &i);
|
||||||
{
|
P2<int> p2(&i);
|
||||||
typedef pointer<int*> type;
|
BOOST_TEST(boost::to_address(p2) == &i);
|
||||||
type p(&i);
|
#if !defined(BOOST_NO_CXX11_SFINAE_EXPR) && \
|
||||||
BOOST_TEST(boost::to_address(p) == &i);
|
!defined(BOOST_NO_CXX14_RETURN_TYPE_DEDUCTION)
|
||||||
}
|
P3<int> p3(&i);
|
||||||
{
|
BOOST_TEST(boost::to_address(p3) == &i);
|
||||||
typedef pointer<pointer<int*> > type;
|
P4<int> p4(&i);
|
||||||
type p(&i);
|
BOOST_TEST(boost::to_address(p4) == &i);
|
||||||
BOOST_TEST(boost::to_address(p) == &i);
|
#if !defined(BOOST_NO_CXX11_POINTER_TRAITS)
|
||||||
}
|
P5<int> p5(&i);
|
||||||
{
|
BOOST_TEST(boost::to_address(p5) == &i);
|
||||||
typedef void* type;
|
P6<int> p6(&i);
|
||||||
type p = &i;
|
BOOST_TEST(boost::to_address(p6) == &i);
|
||||||
BOOST_TEST(boost::to_address(p) == &i);
|
#endif
|
||||||
}
|
#endif
|
||||||
{
|
|
||||||
typedef pointer<void*> type;
|
|
||||||
type p(&i);
|
|
||||||
BOOST_TEST(boost::to_address(p) == &i);
|
|
||||||
}
|
|
||||||
{
|
|
||||||
typedef const int* type;
|
|
||||||
type p = &i;
|
|
||||||
BOOST_TEST(boost::to_address(p) == &i);
|
|
||||||
}
|
|
||||||
{
|
|
||||||
typedef pointer<const int*> type;
|
|
||||||
type p(&i);
|
|
||||||
BOOST_TEST(boost::to_address(p) == &i);
|
|
||||||
}
|
|
||||||
{
|
|
||||||
typedef special<int> type;
|
|
||||||
type p(&i);
|
|
||||||
BOOST_TEST(boost::to_address(p) == &i);
|
|
||||||
}
|
|
||||||
{
|
|
||||||
typedef special<void> type;
|
|
||||||
type p(&i);
|
|
||||||
BOOST_TEST(boost::to_address(p) == &i);
|
|
||||||
}
|
|
||||||
{
|
|
||||||
typedef special<const int> type;
|
|
||||||
type p(&i);
|
|
||||||
BOOST_TEST(boost::to_address(p) == &i);
|
|
||||||
}
|
|
||||||
{
|
|
||||||
typedef pointer<special<int> > type;
|
|
||||||
type p(&i);
|
|
||||||
BOOST_TEST(boost::to_address(p) == &i);
|
|
||||||
}
|
|
||||||
{
|
|
||||||
typedef pointer<special<void> > type;
|
|
||||||
type p(&i);
|
|
||||||
BOOST_TEST(boost::to_address(p) == &i);
|
|
||||||
}
|
|
||||||
{
|
|
||||||
typedef pointer<special<const int> > type;
|
|
||||||
type p(&i);
|
|
||||||
BOOST_TEST(boost::to_address(p) == &i);
|
|
||||||
}
|
|
||||||
return boost::report_errors();
|
return boost::report_errors();
|
||||||
}
|
}
|
||||||
|
Reference in New Issue
Block a user