245 lines
8.0 KiB
Plaintext
245 lines
8.0 KiB
Plaintext
|
|
||
|
// (C) Copyright Rani Sharoni 2003.
|
||
|
// Use, modification and distribution are subject to the Boost Software License,
|
||
|
// Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at
|
||
|
// http://www.boost.org/LICENSE_1_0.txt).
|
||
|
//
|
||
|
// See http://www.boost.org/libs/type_traits for most recent version including documentation.
|
||
|
|
||
|
#ifndef BOOST_TT_IS_BASE_AND_DERIVED_HPP_INCLUDED
|
||
|
#define BOOST_TT_IS_BASE_AND_DERIVED_HPP_INCLUDED
|
||
|
|
||
|
#include <boost/type_traits/intrinsics.hpp>
|
||
|
#include <boost/type_traits/integral_constant.hpp>
|
||
|
#ifndef BOOST_IS_BASE_OF
|
||
|
#include <boost/type_traits/is_class.hpp>
|
||
|
#include <boost/type_traits/is_same.hpp>
|
||
|
#include <boost/type_traits/is_convertible.hpp>
|
||
|
#include <boost/config.hpp>
|
||
|
#include <boost/static_assert.hpp>
|
||
|
#endif
|
||
|
#include <boost/type_traits/remove_cv.hpp>
|
||
|
#include <boost/type_traits/is_same.hpp>
|
||
|
|
||
|
namespace boost {
|
||
|
|
||
|
namespace detail {
|
||
|
|
||
|
#ifndef BOOST_IS_BASE_OF
|
||
|
#if !BOOST_WORKAROUND(__BORLANDC__, BOOST_TESTED_AT(0x581)) \
|
||
|
&& !BOOST_WORKAROUND(__SUNPRO_CC , <= 0x540) \
|
||
|
&& !BOOST_WORKAROUND(__EDG_VERSION__, <= 243) \
|
||
|
&& !BOOST_WORKAROUND(__DMC__, BOOST_TESTED_AT(0x840))
|
||
|
|
||
|
// The EDG version number is a lower estimate.
|
||
|
// It is not currently known which EDG version
|
||
|
// exactly fixes the problem.
|
||
|
|
||
|
/*************************************************************************
|
||
|
|
||
|
This version detects ambiguous base classes and private base classes
|
||
|
correctly, and was devised by Rani Sharoni.
|
||
|
|
||
|
Explanation by Terje Slettebo and Rani Sharoni.
|
||
|
|
||
|
Let's take the multiple base class below as an example, and the following
|
||
|
will also show why there's not a problem with private or ambiguous base
|
||
|
class:
|
||
|
|
||
|
struct B {};
|
||
|
struct B1 : B {};
|
||
|
struct B2 : B {};
|
||
|
struct D : private B1, private B2 {};
|
||
|
|
||
|
is_base_and_derived<B, D>::value;
|
||
|
|
||
|
First, some terminology:
|
||
|
|
||
|
SC - Standard conversion
|
||
|
UDC - User-defined conversion
|
||
|
|
||
|
A user-defined conversion sequence consists of an SC, followed by an UDC,
|
||
|
followed by another SC. Either SC may be the identity conversion.
|
||
|
|
||
|
When passing the default-constructed Host object to the overloaded check_sig()
|
||
|
functions (initialization 8.5/14/4/3), we have several viable implicit
|
||
|
conversion sequences:
|
||
|
|
||
|
For "static no_type check_sig(B const volatile *, int)" we have the conversion
|
||
|
sequences:
|
||
|
|
||
|
C -> C const (SC - Qualification Adjustment) -> B const volatile* (UDC)
|
||
|
C -> D const volatile* (UDC) -> B1 const volatile* / B2 const volatile* ->
|
||
|
B const volatile* (SC - Conversion)
|
||
|
|
||
|
For "static yes_type check_sig(D const volatile *, T)" we have the conversion
|
||
|
sequence:
|
||
|
|
||
|
C -> D const volatile* (UDC)
|
||
|
|
||
|
According to 13.3.3.1/4, in context of user-defined conversion only the
|
||
|
standard conversion sequence is considered when selecting the best viable
|
||
|
function, so it only considers up to the user-defined conversion. For the
|
||
|
first function this means choosing between C -> C const and C -> C, and it
|
||
|
chooses the latter, because it's a proper subset (13.3.3.2/3/2) of the
|
||
|
former. Therefore, we have:
|
||
|
|
||
|
C -> D const volatile* (UDC) -> B1 const volatile* / B2 const volatile* ->
|
||
|
B const volatile* (SC - Conversion)
|
||
|
C -> D const volatile* (UDC)
|
||
|
|
||
|
Here, the principle of the "shortest subsequence" applies again, and it
|
||
|
chooses C -> D const volatile*. This shows that it doesn't even need to
|
||
|
consider the multiple paths to B, or accessibility, as that possibility is
|
||
|
eliminated before it could possibly cause ambiguity or access violation.
|
||
|
|
||
|
If D is not derived from B, it has to choose between C -> C const -> B const
|
||
|
volatile* for the first function, and C -> D const volatile* for the second
|
||
|
function, which are just as good (both requires a UDC, 13.3.3.2), had it not
|
||
|
been for the fact that "static no_type check_sig(B const volatile *, int)" is
|
||
|
not templated, which makes C -> C const -> B const volatile* the best choice
|
||
|
(13.3.3/1/4), resulting in "no".
|
||
|
|
||
|
Also, if Host::operator B const volatile* hadn't been const, the two
|
||
|
conversion sequences for "static no_type check_sig(B const volatile *, int)", in
|
||
|
the case where D is derived from B, would have been ambiguous.
|
||
|
|
||
|
See also
|
||
|
http://groups.google.com/groups?selm=df893da6.0301280859.522081f7%40posting.
|
||
|
google.com and links therein.
|
||
|
|
||
|
*************************************************************************/
|
||
|
|
||
|
template <typename B, typename D>
|
||
|
struct bd_helper
|
||
|
{
|
||
|
//
|
||
|
// This VC7.1 specific workaround stops the compiler from generating
|
||
|
// an internal compiler error when compiling with /vmg (thanks to
|
||
|
// Aleksey Gurtovoy for figuring out the workaround).
|
||
|
//
|
||
|
#if !BOOST_WORKAROUND(BOOST_MSVC, == 1310)
|
||
|
template <typename T>
|
||
|
static type_traits::yes_type check_sig(D const volatile *, T);
|
||
|
static type_traits::no_type check_sig(B const volatile *, int);
|
||
|
#else
|
||
|
static type_traits::yes_type check_sig(D const volatile *, long);
|
||
|
static type_traits::no_type check_sig(B const volatile * const&, int);
|
||
|
#endif
|
||
|
};
|
||
|
|
||
|
template<typename B, typename D>
|
||
|
struct is_base_and_derived_impl2
|
||
|
{
|
||
|
#if BOOST_WORKAROUND(BOOST_MSVC_FULL_VER, >= 140050000)
|
||
|
#pragma warning(push)
|
||
|
#pragma warning(disable:6334)
|
||
|
#endif
|
||
|
//
|
||
|
// May silently do the wrong thing with incomplete types
|
||
|
// unless we trap them here:
|
||
|
//
|
||
|
BOOST_STATIC_ASSERT(sizeof(B) != 0);
|
||
|
BOOST_STATIC_ASSERT(sizeof(D) != 0);
|
||
|
|
||
|
struct Host
|
||
|
{
|
||
|
#if !BOOST_WORKAROUND(BOOST_MSVC, == 1310)
|
||
|
operator B const volatile *() const;
|
||
|
#else
|
||
|
operator B const volatile * const&() const;
|
||
|
#endif
|
||
|
operator D const volatile *();
|
||
|
};
|
||
|
|
||
|
BOOST_STATIC_CONSTANT(bool, value =
|
||
|
sizeof(bd_helper<B,D>::check_sig(Host(), 0)) == sizeof(type_traits::yes_type));
|
||
|
#if BOOST_WORKAROUND(BOOST_MSVC_FULL_VER, >= 140050000)
|
||
|
#pragma warning(pop)
|
||
|
#endif
|
||
|
};
|
||
|
|
||
|
#else
|
||
|
|
||
|
//
|
||
|
// broken version:
|
||
|
//
|
||
|
template<typename B, typename D>
|
||
|
struct is_base_and_derived_impl2
|
||
|
{
|
||
|
BOOST_STATIC_CONSTANT(bool, value =
|
||
|
(::boost::is_convertible<D*,B*>::value));
|
||
|
};
|
||
|
|
||
|
#define BOOST_BROKEN_IS_BASE_AND_DERIVED
|
||
|
|
||
|
#endif
|
||
|
|
||
|
template <typename B, typename D>
|
||
|
struct is_base_and_derived_impl3
|
||
|
{
|
||
|
BOOST_STATIC_CONSTANT(bool, value = false);
|
||
|
};
|
||
|
|
||
|
template <bool ic1, bool ic2, bool iss>
|
||
|
struct is_base_and_derived_select
|
||
|
{
|
||
|
template <class T, class U>
|
||
|
struct rebind
|
||
|
{
|
||
|
typedef is_base_and_derived_impl3<T,U> type;
|
||
|
};
|
||
|
};
|
||
|
|
||
|
template <>
|
||
|
struct is_base_and_derived_select<true,true,false>
|
||
|
{
|
||
|
template <class T, class U>
|
||
|
struct rebind
|
||
|
{
|
||
|
typedef is_base_and_derived_impl2<T,U> type;
|
||
|
};
|
||
|
};
|
||
|
|
||
|
template <typename B, typename D>
|
||
|
struct is_base_and_derived_impl
|
||
|
{
|
||
|
typedef typename remove_cv<B>::type ncvB;
|
||
|
typedef typename remove_cv<D>::type ncvD;
|
||
|
|
||
|
typedef is_base_and_derived_select<
|
||
|
::boost::is_class<B>::value,
|
||
|
::boost::is_class<D>::value,
|
||
|
::boost::is_same<ncvB,ncvD>::value> selector;
|
||
|
typedef typename selector::template rebind<ncvB,ncvD> binder;
|
||
|
typedef typename binder::type bound_type;
|
||
|
|
||
|
BOOST_STATIC_CONSTANT(bool, value = bound_type::value);
|
||
|
};
|
||
|
#else
|
||
|
template <typename B, typename D>
|
||
|
struct is_base_and_derived_impl
|
||
|
{
|
||
|
typedef typename remove_cv<B>::type ncvB;
|
||
|
typedef typename remove_cv<D>::type ncvD;
|
||
|
|
||
|
BOOST_STATIC_CONSTANT(bool, value = (BOOST_IS_BASE_OF(B,D) && ! ::boost::is_same<ncvB,ncvD>::value));
|
||
|
};
|
||
|
#endif
|
||
|
} // namespace detail
|
||
|
|
||
|
template <class Base, class Derived> struct is_base_and_derived
|
||
|
: public integral_constant<bool, (::boost::detail::is_base_and_derived_impl<Base, Derived>::value)> {};
|
||
|
|
||
|
template <class Base, class Derived> struct is_base_and_derived<Base&, Derived> : public false_type{};
|
||
|
template <class Base, class Derived> struct is_base_and_derived<Base, Derived&> : public false_type{};
|
||
|
template <class Base, class Derived> struct is_base_and_derived<Base&, Derived&> : public false_type{};
|
||
|
|
||
|
#if BOOST_WORKAROUND(__CODEGEARC__, BOOST_TESTED_AT(0x610))
|
||
|
template <class Base> struct is_base_and_derived<Base, Base> : public true_type{};
|
||
|
#endif
|
||
|
|
||
|
} // namespace boost
|
||
|
|
||
|
#endif // BOOST_TT_IS_BASE_AND_DERIVED_HPP_INCLUDED
|