Chris@16
|
1 // - casts.hpp -- BLambda Library -------------
|
Chris@16
|
2 //
|
Chris@16
|
3 // Copyright (C) 2000 Gary Powell (powellg@amazon.com)
|
Chris@16
|
4 // Copyright (C) 1999, 2000 Jaakko Jarvi (jaakko.jarvi@cs.utu.fi)
|
Chris@16
|
5 //
|
Chris@16
|
6 // Distributed under the Boost Software License, Version 1.0. (See
|
Chris@16
|
7 // accompanying file LICENSE_1_0.txt or copy at
|
Chris@16
|
8 // http://www.boost.org/LICENSE_1_0.txt)
|
Chris@16
|
9 //
|
Chris@16
|
10 // For more information, see http://www.boost.org
|
Chris@16
|
11
|
Chris@16
|
12 // -----------------------------------------------
|
Chris@16
|
13
|
Chris@16
|
14 #if !defined(BOOST_LAMBDA_CASTS_HPP)
|
Chris@16
|
15 #define BOOST_LAMBDA_CASTS_HPP
|
Chris@16
|
16
|
Chris@16
|
17 #include "boost/lambda/detail/suppress_unused.hpp"
|
Chris@16
|
18 #include "boost/lambda/core.hpp"
|
Chris@16
|
19
|
Chris@16
|
20 #include <typeinfo>
|
Chris@16
|
21
|
Chris@16
|
22 namespace boost {
|
Chris@16
|
23 namespace lambda {
|
Chris@16
|
24
|
Chris@16
|
25 template<class Act, class Args>
|
Chris@16
|
26 struct return_type_N;
|
Chris@16
|
27
|
Chris@16
|
28 template<class T> class cast_action;
|
Chris@16
|
29
|
Chris@16
|
30 template<class T> class static_cast_action;
|
Chris@16
|
31 template<class T> class dynamic_cast_action;
|
Chris@16
|
32 template<class T> class const_cast_action;
|
Chris@16
|
33 template<class T> class reinterpret_cast_action;
|
Chris@16
|
34
|
Chris@16
|
35 class typeid_action;
|
Chris@16
|
36 class sizeof_action;
|
Chris@16
|
37
|
Chris@16
|
38 // Cast actions
|
Chris@16
|
39
|
Chris@16
|
40 template<class T> class cast_action<static_cast_action<T> >
|
Chris@16
|
41 {
|
Chris@16
|
42 public:
|
Chris@16
|
43 template<class RET, class Arg1>
|
Chris@16
|
44 static RET apply(Arg1 &a1) {
|
Chris@16
|
45 return static_cast<RET>(a1);
|
Chris@16
|
46 }
|
Chris@16
|
47 };
|
Chris@16
|
48
|
Chris@16
|
49 template<class T> class cast_action<dynamic_cast_action<T> > {
|
Chris@16
|
50 public:
|
Chris@16
|
51 template<class RET, class Arg1>
|
Chris@16
|
52 static RET apply(Arg1 &a1) {
|
Chris@16
|
53 return dynamic_cast<RET>(a1);
|
Chris@16
|
54 }
|
Chris@16
|
55 };
|
Chris@16
|
56
|
Chris@16
|
57 template<class T> class cast_action<const_cast_action<T> > {
|
Chris@16
|
58 public:
|
Chris@16
|
59 template<class RET, class Arg1>
|
Chris@16
|
60 static RET apply(Arg1 &a1) {
|
Chris@16
|
61 return const_cast<RET>(a1);
|
Chris@16
|
62 }
|
Chris@16
|
63 };
|
Chris@16
|
64
|
Chris@16
|
65 template<class T> class cast_action<reinterpret_cast_action<T> > {
|
Chris@16
|
66 public:
|
Chris@16
|
67 template<class RET, class Arg1>
|
Chris@16
|
68 static RET apply(Arg1 &a1) {
|
Chris@16
|
69 return reinterpret_cast<RET>(a1);
|
Chris@16
|
70 }
|
Chris@16
|
71 };
|
Chris@16
|
72
|
Chris@16
|
73 // typeid action
|
Chris@16
|
74 class typeid_action {
|
Chris@16
|
75 public:
|
Chris@16
|
76 template<class RET, class Arg1>
|
Chris@16
|
77 static RET apply(Arg1 &a1) {
|
Chris@16
|
78 detail::suppress_unused_variable_warnings(a1);
|
Chris@16
|
79 return typeid(a1);
|
Chris@16
|
80 }
|
Chris@16
|
81 };
|
Chris@16
|
82
|
Chris@16
|
83 // sizeof action
|
Chris@16
|
84 class sizeof_action
|
Chris@16
|
85 {
|
Chris@16
|
86 public:
|
Chris@16
|
87 template<class RET, class Arg1>
|
Chris@16
|
88 static RET apply(Arg1 &a1) {
|
Chris@16
|
89 return sizeof(a1);
|
Chris@16
|
90 }
|
Chris@16
|
91 };
|
Chris@16
|
92
|
Chris@16
|
93
|
Chris@16
|
94 // return types of casting lambda_functors (all "T" type.)
|
Chris@16
|
95
|
Chris@16
|
96 template<template <class> class cast_type, class T, class A>
|
Chris@16
|
97 struct return_type_N<cast_action< cast_type<T> >, A> {
|
Chris@16
|
98 typedef T type;
|
Chris@16
|
99 };
|
Chris@16
|
100
|
Chris@16
|
101 // return type of typeid_action
|
Chris@16
|
102 template<class A>
|
Chris@16
|
103 struct return_type_N<typeid_action, A> {
|
Chris@16
|
104 typedef std::type_info const & type;
|
Chris@16
|
105 };
|
Chris@16
|
106
|
Chris@16
|
107 // return type of sizeof_action
|
Chris@16
|
108
|
Chris@16
|
109 template<class A>
|
Chris@16
|
110 struct return_type_N<sizeof_action, A> {
|
Chris@16
|
111 typedef std::size_t type;
|
Chris@16
|
112 };
|
Chris@16
|
113
|
Chris@16
|
114
|
Chris@16
|
115 // the four cast & typeid overloads.
|
Chris@16
|
116 // casts can take ordinary variables (not just lambda functors)
|
Chris@16
|
117
|
Chris@16
|
118 // static_cast
|
Chris@16
|
119 template <class T, class Arg1>
|
Chris@16
|
120 inline const lambda_functor<
|
Chris@16
|
121 lambda_functor_base<
|
Chris@16
|
122 action<1, cast_action<static_cast_action<T> > >,
|
Chris@16
|
123 tuple<typename const_copy_argument <const Arg1>::type>
|
Chris@16
|
124 >
|
Chris@16
|
125 >
|
Chris@16
|
126 ll_static_cast(const Arg1& a1) {
|
Chris@16
|
127 return
|
Chris@16
|
128 lambda_functor_base<
|
Chris@16
|
129 action<1, cast_action<static_cast_action<T> > >,
|
Chris@16
|
130 tuple<typename const_copy_argument <const Arg1>::type>
|
Chris@16
|
131 >
|
Chris@16
|
132 ( tuple<typename const_copy_argument <const Arg1>::type>(a1));
|
Chris@16
|
133 }
|
Chris@16
|
134
|
Chris@16
|
135 // dynamic_cast
|
Chris@16
|
136 template <class T, class Arg1>
|
Chris@16
|
137 inline const lambda_functor<
|
Chris@16
|
138 lambda_functor_base<
|
Chris@16
|
139 action<1, cast_action<dynamic_cast_action<T> > >,
|
Chris@16
|
140 tuple<typename const_copy_argument <const Arg1>::type>
|
Chris@16
|
141 >
|
Chris@16
|
142 >
|
Chris@16
|
143 ll_dynamic_cast(const Arg1& a1) {
|
Chris@16
|
144 return
|
Chris@16
|
145 lambda_functor_base<
|
Chris@16
|
146 action<1, cast_action<dynamic_cast_action<T> > >,
|
Chris@16
|
147 tuple<typename const_copy_argument <const Arg1>::type>
|
Chris@16
|
148 >
|
Chris@16
|
149 ( tuple<typename const_copy_argument <const Arg1>::type>(a1));
|
Chris@16
|
150 }
|
Chris@16
|
151
|
Chris@16
|
152 // const_cast
|
Chris@16
|
153 template <class T, class Arg1>
|
Chris@16
|
154 inline const lambda_functor<
|
Chris@16
|
155 lambda_functor_base<
|
Chris@16
|
156 action<1, cast_action<const_cast_action<T> > >,
|
Chris@16
|
157 tuple<typename const_copy_argument <const Arg1>::type>
|
Chris@16
|
158 >
|
Chris@16
|
159 >
|
Chris@16
|
160 ll_const_cast(const Arg1& a1) {
|
Chris@16
|
161 return
|
Chris@16
|
162 lambda_functor_base<
|
Chris@16
|
163 action<1, cast_action<const_cast_action<T> > >,
|
Chris@16
|
164 tuple<typename const_copy_argument <const Arg1>::type>
|
Chris@16
|
165 >
|
Chris@16
|
166 ( tuple<typename const_copy_argument <const Arg1>::type>(a1));
|
Chris@16
|
167 }
|
Chris@16
|
168
|
Chris@16
|
169 // reinterpret_cast
|
Chris@16
|
170 template <class T, class Arg1>
|
Chris@16
|
171 inline const lambda_functor<
|
Chris@16
|
172 lambda_functor_base<
|
Chris@16
|
173 action<1, cast_action<reinterpret_cast_action<T> > >,
|
Chris@16
|
174 tuple<typename const_copy_argument <const Arg1>::type>
|
Chris@16
|
175 >
|
Chris@16
|
176 >
|
Chris@16
|
177 ll_reinterpret_cast(const Arg1& a1) {
|
Chris@16
|
178 return
|
Chris@16
|
179 lambda_functor_base<
|
Chris@16
|
180 action<1, cast_action<reinterpret_cast_action<T> > >,
|
Chris@16
|
181 tuple<typename const_copy_argument <const Arg1>::type>
|
Chris@16
|
182 >
|
Chris@16
|
183 ( tuple<typename const_copy_argument <const Arg1>::type>(a1));
|
Chris@16
|
184 }
|
Chris@16
|
185
|
Chris@16
|
186 // typeid
|
Chris@16
|
187 // can be applied to a normal variable as well (can refer to a polymorphic
|
Chris@16
|
188 // class object)
|
Chris@16
|
189 template <class Arg1>
|
Chris@16
|
190 inline const lambda_functor<
|
Chris@16
|
191 lambda_functor_base<
|
Chris@16
|
192 action<1, typeid_action>,
|
Chris@16
|
193 tuple<typename const_copy_argument <const Arg1>::type>
|
Chris@16
|
194 >
|
Chris@16
|
195 >
|
Chris@16
|
196 ll_typeid(const Arg1& a1) {
|
Chris@16
|
197 return
|
Chris@16
|
198 lambda_functor_base<
|
Chris@16
|
199 action<1, typeid_action>,
|
Chris@16
|
200 tuple<typename const_copy_argument <const Arg1>::type>
|
Chris@16
|
201 >
|
Chris@16
|
202 ( tuple<typename const_copy_argument <const Arg1>::type>(a1));
|
Chris@16
|
203 }
|
Chris@16
|
204
|
Chris@16
|
205 // sizeof(expression)
|
Chris@16
|
206 // Always takes a lambda expression (if not, built in sizeof will do)
|
Chris@16
|
207 template <class Arg1>
|
Chris@16
|
208 inline const lambda_functor<
|
Chris@16
|
209 lambda_functor_base<
|
Chris@16
|
210 action<1, sizeof_action>,
|
Chris@16
|
211 tuple<lambda_functor<Arg1> >
|
Chris@16
|
212 >
|
Chris@16
|
213 >
|
Chris@16
|
214 ll_sizeof(const lambda_functor<Arg1>& a1) {
|
Chris@16
|
215 return
|
Chris@16
|
216 lambda_functor_base<
|
Chris@16
|
217 action<1, sizeof_action>,
|
Chris@16
|
218 tuple<lambda_functor<Arg1> >
|
Chris@16
|
219 >
|
Chris@16
|
220 ( tuple<lambda_functor<Arg1> >(a1));
|
Chris@16
|
221 }
|
Chris@16
|
222
|
Chris@16
|
223 } // namespace lambda
|
Chris@16
|
224 } // namespace boost
|
Chris@16
|
225
|
Chris@16
|
226 #endif
|