1#ifndef BOOST_TT_IS_ABSTRACT_CLASS_HPP
2#define BOOST_TT_IS_ABSTRACT_CLASS_HPP
3
4#if defined(_MSC_VER)
5# pragma once
6#endif
7
8/////////1/////////2/////////3/////////4/////////5/////////6/////////7/////////8
9// is_abstract_class.hpp:
10//
11// (C) Copyright 2002 Rani Sharoni (rani_sharoni@hotmail.com) and Robert Ramey
12// Use, modification and distribution is subject to the Boost Software
13// License, Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at
14// http://www.boost.org/LICENSE_1_0.txt)
15//
16// See http://www.boost.org for updates, documentation, and revision history.
17//
18
19// Compile type discovery whether given type is abstract class or not.
20//
21// Requires DR 337 to be supported by compiler
22// (http://www.open-std.org/jtc1/sc22/wg21/docs/cwg_active.html#337).
23//
24//
25// Believed (Jan 2004) to work on:
26// - GCC 3.4
27// - VC++ 7.1
28// - compilers with new EDG frontend (Intel C++ 7, Comeau 4.3.2)
29//
30// Doesn't work on:
31// - VC++6, VC++7.0 and less
32// - GCC 3.3.X and less
33// - Borland C++ 6 and less
34//
35//
36// History:
37// - Originally written by Rani Sharoni, see
38// http://groups.google.com/groups?selm=df893da6.0207110613.75b2fe90%40posting.google.com
39// At this time supported by EDG (Intel C++ 7, Comeau 4.3.2) and VC7.1.
40// - Adapted and added into Boost.Serialization library by Robert Ramey
41// (starting with submission #10).
42// - Jan 2004: GCC 3.4 fixed to support DR337 (Giovanni Bajo).
43// - Jan 2004: modified to be part of Boost.TypeTraits (Pavel Vozenilek).
44// - Nov 2004: Christoph Ludwig found that the implementation did not work with
45// template types and gcc-3.4 or VC7.1, fix due to Christoph Ludwig
46// and John Maddock.
47// - Dec 2004: Added new config macro BOOST_NO_IS_ABSTRACT which causes the template
48// to degrade gracefully, rather than trash the compiler (John Maddock).
49//
50
51#include <boost/type_traits/intrinsics.hpp>
52#include <boost/type_traits/integral_constant.hpp>
53#ifndef BOOST_IS_ABSTRACT
54#include <boost/static_assert.hpp>
55#include <boost/type_traits/detail/yes_no_type.hpp>
56#include <boost/type_traits/is_class.hpp>
57#ifdef BOOST_NO_IS_ABSTRACT
58#include <boost/type_traits/is_polymorphic.hpp>
59#endif
60#endif
61
62namespace boost {
63
64namespace detail{
65
66#ifdef BOOST_IS_ABSTRACT
67template <class T>
68struct is_abstract_imp
69{
70 BOOST_STATIC_CONSTANT(bool, value = BOOST_IS_ABSTRACT(T));
71};
72#elif !defined(BOOST_NO_IS_ABSTRACT)
73template<class T>
74struct is_abstract_imp2
75{
76 // Deduction fails if T is void, function type,
77 // reference type (14.8.2/2)or an abstract class type
78 // according to review status issue #337
79 //
80 template<class U>
81 static type_traits::no_type check_sig(U (*)[1]);
82 template<class U>
83 static type_traits::yes_type check_sig(...);
84 //
85 // T must be a complete type, further if T is a template then
86 // it must be instantiated in order for us to get the right answer:
87 //
88 BOOST_STATIC_ASSERT(sizeof(T) != 0);
89
90 // GCC2 won't even parse this template if we embed the computation
91 // of s1 in the computation of value.
92#ifdef __GNUC__
93 BOOST_STATIC_CONSTANT(std::size_t, s1 = sizeof(is_abstract_imp2<T>::template check_sig<T>(0)));
94#else
95#if BOOST_WORKAROUND(BOOST_MSVC_FULL_VER, >= 140050000)
96#pragma warning(push)
97#pragma warning(disable:6334)
98#endif
99 BOOST_STATIC_CONSTANT(std::size_t, s1 = sizeof(check_sig<T>(0)));
100#if BOOST_WORKAROUND(BOOST_MSVC_FULL_VER, >= 140050000)
101#pragma warning(pop)
102#endif
103#endif
104
105 BOOST_STATIC_CONSTANT(bool, value =
106 (s1 == sizeof(type_traits::yes_type)));
107};
108
109template <bool v>
110struct is_abstract_select
111{
112 template <class T>
113 struct rebind
114 {
115 typedef is_abstract_imp2<T> type;
116 };
117};
118template <>
119struct is_abstract_select<false>
120{
121 template <class T>
122 struct rebind
123 {
124 typedef false_type type;
125 };
126};
127
128template <class T>
129struct is_abstract_imp
130{
131 typedef is_abstract_select< ::boost::is_class<T>::value> selector;
132 typedef typename selector::template rebind<T> binder;
133 typedef typename binder::type type;
134
135 BOOST_STATIC_CONSTANT(bool, value = type::value);
136};
137
138#endif
139}
140
141#ifndef BOOST_NO_IS_ABSTRACT
142template <class T> struct is_abstract : public integral_constant<bool, ::boost::detail::is_abstract_imp<T>::value> {};
143#else
144template <class T> struct is_abstract : public integral_constant<bool, ::boost::detail::is_polymorphic_imp<T>::value> {};
145#endif
146
147} // namespace boost
148
149#endif //BOOST_TT_IS_ABSTRACT_CLASS_HPP
150