williamr@2
|
1 |
#ifndef BOOST_SERIALIZATION_TRACKING_HPP
|
williamr@2
|
2 |
#define BOOST_SERIALIZATION_TRACKING_HPP
|
williamr@2
|
3 |
|
williamr@2
|
4 |
// MS compatible compilers support #pragma once
|
williamr@2
|
5 |
#if defined(_MSC_VER) && (_MSC_VER >= 1020)
|
williamr@2
|
6 |
# pragma once
|
williamr@2
|
7 |
#endif
|
williamr@2
|
8 |
|
williamr@2
|
9 |
/////////1/////////2/////////3/////////4/////////5/////////6/////////7/////////8
|
williamr@2
|
10 |
// tracking.hpp:
|
williamr@2
|
11 |
|
williamr@2
|
12 |
// (C) Copyright 2002 Robert Ramey - http://www.rrsd.com .
|
williamr@2
|
13 |
// Use, modification and distribution is subject to the Boost Software
|
williamr@2
|
14 |
// License, Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at
|
williamr@2
|
15 |
// http://www.boost.org/LICENSE_1_0.txt)
|
williamr@2
|
16 |
|
williamr@2
|
17 |
// See http://www.boost.org for updates, documentation, and revision history.
|
williamr@2
|
18 |
|
williamr@2
|
19 |
#include <boost/config.hpp>
|
williamr@2
|
20 |
#include <boost/static_assert.hpp>
|
williamr@2
|
21 |
#include <boost/mpl/eval_if.hpp>
|
williamr@2
|
22 |
#include <boost/mpl/identity.hpp>
|
williamr@2
|
23 |
#include <boost/mpl/int.hpp>
|
williamr@2
|
24 |
#include <boost/mpl/equal_to.hpp>
|
williamr@2
|
25 |
#include <boost/mpl/greater.hpp>
|
williamr@2
|
26 |
#include <boost/mpl/integral_c_tag.hpp>
|
williamr@2
|
27 |
|
williamr@2
|
28 |
#include <boost/type_traits/is_base_and_derived.hpp>
|
williamr@2
|
29 |
#include <boost/type_traits/is_pointer.hpp>
|
williamr@2
|
30 |
#include <boost/serialization/level.hpp>
|
williamr@2
|
31 |
#include <boost/serialization/tracking_enum.hpp>
|
williamr@2
|
32 |
//#include <boost/serialization/traits.hpp>
|
williamr@2
|
33 |
|
williamr@2
|
34 |
namespace boost {
|
williamr@2
|
35 |
namespace serialization {
|
williamr@2
|
36 |
|
williamr@2
|
37 |
struct basic_traits;
|
williamr@2
|
38 |
|
williamr@2
|
39 |
// default tracking level
|
williamr@2
|
40 |
template<class T>
|
williamr@2
|
41 |
struct tracking_level {
|
williamr@2
|
42 |
template<class U>
|
williamr@2
|
43 |
struct traits_class_tracking {
|
williamr@2
|
44 |
typedef BOOST_DEDUCED_TYPENAME U::tracking type;
|
williamr@2
|
45 |
};
|
williamr@2
|
46 |
typedef mpl::integral_c_tag tag;
|
williamr@2
|
47 |
// note: at least one compiler complained w/o the full qualification
|
williamr@2
|
48 |
// on basic traits below
|
williamr@2
|
49 |
typedef
|
williamr@2
|
50 |
BOOST_DEDUCED_TYPENAME mpl::eval_if<
|
williamr@2
|
51 |
is_base_and_derived<boost::serialization::basic_traits, T>,
|
williamr@2
|
52 |
traits_class_tracking<T>,
|
williamr@2
|
53 |
//else
|
williamr@2
|
54 |
BOOST_DEDUCED_TYPENAME mpl::eval_if<
|
williamr@2
|
55 |
is_pointer<T>,
|
williamr@2
|
56 |
// pointers are not tracked by default
|
williamr@2
|
57 |
mpl::int_<track_never>,
|
williamr@2
|
58 |
//else
|
williamr@2
|
59 |
BOOST_DEDUCED_TYPENAME mpl::eval_if<
|
williamr@2
|
60 |
// for primitives
|
williamr@2
|
61 |
BOOST_DEDUCED_TYPENAME mpl::equal_to<
|
williamr@2
|
62 |
implementation_level<T>,
|
williamr@2
|
63 |
mpl::int_<primitive_type>
|
williamr@2
|
64 |
>,
|
williamr@2
|
65 |
// is never
|
williamr@2
|
66 |
mpl::int_<track_never>,
|
williamr@2
|
67 |
// otherwise its selective
|
williamr@2
|
68 |
mpl::int_<track_selectivly>
|
williamr@2
|
69 |
> > >::type type;
|
williamr@2
|
70 |
BOOST_STATIC_CONSTANT(int, value = tracking_level::type::value);
|
williamr@2
|
71 |
};
|
williamr@2
|
72 |
|
williamr@2
|
73 |
|
williamr@2
|
74 |
template<class T, enum tracking_type L>
|
williamr@2
|
75 |
inline bool operator>=(tracking_level<T> t, enum tracking_type l)
|
williamr@2
|
76 |
{
|
williamr@2
|
77 |
return t.value >= (int)l;
|
williamr@2
|
78 |
}
|
williamr@2
|
79 |
|
williamr@2
|
80 |
} // namespace serialization
|
williamr@2
|
81 |
} // namespace boost
|
williamr@2
|
82 |
|
williamr@2
|
83 |
|
williamr@2
|
84 |
// The STATIC_ASSERT is prevents one from setting tracking for a primitive type.
|
williamr@2
|
85 |
// This almost HAS to be an error. Doing this will effect serialization of all
|
williamr@2
|
86 |
// char's in your program which is almost certainly what you don't want to do.
|
williamr@2
|
87 |
// If you want to track all instances of a given primitive type, You'll have to
|
williamr@2
|
88 |
// wrap it in your own type so its not a primitive anymore. Then it will compile
|
williamr@2
|
89 |
// without problem.
|
williamr@2
|
90 |
#define BOOST_CLASS_TRACKING(T, E) \
|
williamr@2
|
91 |
namespace boost { \
|
williamr@2
|
92 |
namespace serialization { \
|
williamr@2
|
93 |
template<> \
|
williamr@2
|
94 |
struct tracking_level< T > \
|
williamr@2
|
95 |
{ \
|
williamr@2
|
96 |
typedef mpl::integral_c_tag tag; \
|
williamr@2
|
97 |
typedef mpl::int_< E> type; \
|
williamr@2
|
98 |
BOOST_STATIC_CONSTANT( \
|
williamr@2
|
99 |
int, \
|
williamr@2
|
100 |
value = tracking_level::type::value \
|
williamr@2
|
101 |
); \
|
williamr@2
|
102 |
/* tracking for a class */ \
|
williamr@2
|
103 |
BOOST_STATIC_ASSERT(( \
|
williamr@2
|
104 |
mpl::greater< \
|
williamr@2
|
105 |
/* that is a prmitive */ \
|
williamr@2
|
106 |
implementation_level< T >, \
|
williamr@2
|
107 |
mpl::int_<primitive_type> \
|
williamr@2
|
108 |
>::value \
|
williamr@2
|
109 |
)); \
|
williamr@2
|
110 |
}; \
|
williamr@2
|
111 |
}}
|
williamr@2
|
112 |
|
williamr@2
|
113 |
#endif // BOOST_SERIALIZATION_TRACKING_HPP
|