• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 //===----------------------------------------------------------------------===//
2 //
3 //                     The LLVM Compiler Infrastructure
4 //
5 // This file is dual licensed under the MIT and the University of Illinois Open
6 // Source Licenses. See LICENSE.TXT for details.
7 //
8 //===----------------------------------------------------------------------===//
9 #ifndef ANY_HELPERS_H
10 #define ANY_HELPERS_H
11 
12 #include <typeinfo>
13 #include <type_traits>
14 #include <cassert>
15 
16 namespace std { namespace experimental {} }
17 
18 #include "test_macros.h"
19 #include "type_id.h"
20 
21 #if !defined(TEST_HAS_NO_RTTI)
22 #define RTTI_ASSERT(X) assert(X)
23 #else
24 #define RTTI_ASSERT(X)
25 #endif
26 
27 template <class _Tp>
28   struct IsSmallObject
29     : public std::integral_constant<bool
30         , sizeof(_Tp) <= (sizeof(void*)*3)
31           && std::alignment_of<void*>::value
32              % std::alignment_of<_Tp>::value == 0
33           && std::is_nothrow_move_constructible<_Tp>::value
34         >
35   {};
36 
37 template <class T>
containsType(std::any const & a)38 bool containsType(std::any const& a) {
39 #if !defined(TEST_HAS_NO_RTTI)
40     return a.type() == typeid(T);
41 #else
42     return a.has_value() && std::any_cast<T>(&a) != nullptr;
43 #endif
44 }
45 
46 // Return 'true' if 'Type' will be considered a small type by 'any'
47 template <class Type>
isSmallType()48 bool isSmallType() {
49     return IsSmallObject<Type>::value;
50 }
51 
52 // Assert that an object is empty. If the object used to contain an object
53 // of type 'LastType' check that it can no longer be accessed.
54 template <class LastType = int>
assertEmpty(std::any const & a)55 void assertEmpty(std::any const& a) {
56     using namespace std;
57     assert(!a.has_value());
58     RTTI_ASSERT(a.type() == typeid(void));
59     assert(any_cast<LastType const>(&a) == nullptr);
60 }
61 
62 template <class Type>
63 constexpr auto has_value_member(int) -> decltype(std::declval<Type&>().value, true)
64 { return true; }
has_value_member(long)65 template <class> constexpr bool has_value_member(long) { return false; }
66 
67 
68 // Assert that an 'any' object stores the specified 'Type' and 'value'.
69 template <class Type>
70 std::enable_if_t<has_value_member<Type>(0)>
assertContains(std::any const & a,int value)71 assertContains(std::any const& a, int value) {
72     assert(a.has_value());
73     assert(containsType<Type>(a));
74     assert(std::any_cast<Type const &>(a).value == value);
75 }
76 
77 template <class Type, class Value>
78 std::enable_if_t<!has_value_member<Type>(0)>
assertContains(std::any const & a,Value value)79 assertContains(std::any const& a, Value value) {
80     assert(a.has_value());
81     assert(containsType<Type>(a));
82     assert(std::any_cast<Type const &>(a) == value);
83 }
84 
85 
86 // Modify the value of a "test type" stored within an any to the specified
87 // 'value'.
88 template <class Type>
modifyValue(std::any & a,int value)89 void modifyValue(std::any& a, int value) {
90     using namespace std;
91     using namespace std::experimental;
92     assert(a.has_value());
93     assert(containsType<Type>(a));
94     any_cast<Type&>(a).value = value;
95 }
96 
97 // A test type that will trigger the small object optimization within 'any'.
98 template <int Dummy = 0>
99 struct small_type
100 {
101     static int count;
102     static int copied;
103     static int moved;
104     static int const_copied;
105     static int non_const_copied;
106 
resetsmall_type107     static void reset() {
108         small_type::copied = 0;
109         small_type::moved = 0;
110         small_type::const_copied = 0;
111         small_type::non_const_copied = 0;
112     }
113 
114     int value;
115 
valuesmall_type116     explicit small_type(int val = 0) : value(val) {
117         ++count;
118     }
small_typesmall_type119     explicit small_type(int, int val, int) : value(val) {
120         ++count;
121     }
small_typesmall_type122     small_type(std::initializer_list<int> il) : value(*il.begin()) {
123         ++count;
124     }
125 
small_typesmall_type126     small_type(small_type const & other) noexcept {
127         value = other.value;
128         ++count;
129         ++copied;
130         ++const_copied;
131     }
132 
small_typesmall_type133     small_type(small_type& other) noexcept {
134         value = other.value;
135         ++count;
136         ++copied;
137         ++non_const_copied;
138     }
139 
small_typesmall_type140     small_type(small_type && other) noexcept {
141         value = other.value;
142         other.value = 0;
143         ++count;
144         ++moved;
145     }
146 
~small_typesmall_type147     ~small_type() {
148         value = -1;
149         --count;
150     }
151 
152 private:
153     small_type& operator=(small_type const&) = delete;
154     small_type& operator=(small_type&&) = delete;
155 };
156 
157 template <int Dummy>
158 int small_type<Dummy>::count = 0;
159 
160 template <int Dummy>
161 int small_type<Dummy>::copied = 0;
162 
163 template <int Dummy>
164 int small_type<Dummy>::moved = 0;
165 
166 template <int Dummy>
167 int small_type<Dummy>::const_copied = 0;
168 
169 template <int Dummy>
170 int small_type<Dummy>::non_const_copied = 0;
171 
172 typedef small_type<> small;
173 typedef small_type<1> small1;
174 typedef small_type<2> small2;
175 
176 
177 // A test type that will NOT trigger the small object optimization in any.
178 template <int Dummy = 0>
179 struct large_type
180 {
181     static int count;
182     static int copied;
183     static int moved;
184     static int const_copied;
185     static int non_const_copied;
186 
resetlarge_type187     static void reset() {
188         large_type::copied = 0;
189         large_type::moved  = 0;
190         large_type::const_copied = 0;
191         large_type::non_const_copied = 0;
192     }
193 
194     int value;
195 
valuelarge_type196     large_type(int val = 0) : value(val) {
197         ++count;
198         data[0] = 0;
199     }
large_typelarge_type200     large_type(int, int val, int) : value(val) {
201         ++count;
202         data[0] = 0;
203     }
large_typelarge_type204     large_type(std::initializer_list<int> il) : value(*il.begin()) {
205         ++count;
206     }
large_typelarge_type207     large_type(large_type const & other) {
208         value = other.value;
209         ++count;
210         ++copied;
211         ++const_copied;
212     }
213 
large_typelarge_type214     large_type(large_type & other) {
215         value = other.value;
216         ++count;
217         ++copied;
218         ++non_const_copied;
219     }
220 
large_typelarge_type221     large_type(large_type && other) {
222         value = other.value;
223         other.value = 0;
224         ++count;
225         ++moved;
226     }
227 
~large_typelarge_type228     ~large_type()  {
229         value = 0;
230         --count;
231     }
232 
233 private:
234     large_type& operator=(large_type const&) = delete;
235     large_type& operator=(large_type &&) = delete;
236     int data[10];
237 };
238 
239 template <int Dummy>
240 int large_type<Dummy>::count = 0;
241 
242 template <int Dummy>
243 int large_type<Dummy>::copied = 0;
244 
245 template <int Dummy>
246 int large_type<Dummy>::moved = 0;
247 
248 template <int Dummy>
249 int large_type<Dummy>::const_copied = 0;
250 
251 template <int Dummy>
252 int large_type<Dummy>::non_const_copied = 0;
253 
254 typedef large_type<> large;
255 typedef large_type<1> large1;
256 typedef large_type<2> large2;
257 
258 // The exception type thrown by 'small_throws_on_copy', 'large_throws_on_copy'
259 // and 'throws_on_move'.
260 struct my_any_exception {};
261 
throwMyAnyExpression()262 void throwMyAnyExpression() {
263 #if !defined(TEST_HAS_NO_EXCEPTIONS)
264         throw my_any_exception();
265 #else
266         assert(false && "Exceptions are disabled");
267 #endif
268 }
269 
270 // A test type that will trigger the small object optimization within 'any'.
271 // this type throws if it is copied.
272 struct small_throws_on_copy
273 {
274     static int count;
275     static int copied;
276     static int moved;
resetsmall_throws_on_copy277     static void reset() { count = copied = moved = 0; }
278     int value;
279 
valuesmall_throws_on_copy280     explicit small_throws_on_copy(int val = 0) : value(val) {
281         ++count;
282     }
small_throws_on_copysmall_throws_on_copy283     explicit small_throws_on_copy(int, int val, int) : value(val) {
284         ++count;
285     }
small_throws_on_copysmall_throws_on_copy286     small_throws_on_copy(small_throws_on_copy const &) {
287         throwMyAnyExpression();
288     }
289 
throwsmall_throws_on_copy290     small_throws_on_copy(small_throws_on_copy && other) throw() {
291         value = other.value;
292         ++count; ++moved;
293     }
294 
~small_throws_on_copysmall_throws_on_copy295     ~small_throws_on_copy() {
296         --count;
297     }
298 private:
299     small_throws_on_copy& operator=(small_throws_on_copy const&) = delete;
300     small_throws_on_copy& operator=(small_throws_on_copy &&) = delete;
301 };
302 
303 int small_throws_on_copy::count = 0;
304 int small_throws_on_copy::copied = 0;
305 int small_throws_on_copy::moved = 0;
306 
307 
308 // A test type that will NOT trigger the small object optimization within 'any'.
309 // this type throws if it is copied.
310 struct large_throws_on_copy
311 {
312     static int count;
313     static int copied;
314     static int moved;
resetlarge_throws_on_copy315     static void reset() { count = copied = moved = 0; }
316     int value = 0;
317 
valuelarge_throws_on_copy318     explicit large_throws_on_copy(int val = 0) : value(val) {
319         data[0] = 0;
320         ++count;
321     }
large_throws_on_copylarge_throws_on_copy322     explicit large_throws_on_copy(int, int val, int) : value(val) {
323         data[0] = 0;
324         ++count;
325     }
large_throws_on_copylarge_throws_on_copy326     large_throws_on_copy(large_throws_on_copy const &) {
327          throwMyAnyExpression();
328     }
329 
throwlarge_throws_on_copy330     large_throws_on_copy(large_throws_on_copy && other) throw() {
331         value = other.value;
332         ++count; ++moved;
333     }
334 
~large_throws_on_copylarge_throws_on_copy335     ~large_throws_on_copy() {
336         --count;
337     }
338 
339 private:
340     large_throws_on_copy& operator=(large_throws_on_copy const&) = delete;
341     large_throws_on_copy& operator=(large_throws_on_copy &&) = delete;
342     int data[10];
343 };
344 
345 int large_throws_on_copy::count = 0;
346 int large_throws_on_copy::copied = 0;
347 int large_throws_on_copy::moved = 0;
348 
349 // A test type that throws when it is moved. This object will NOT trigger
350 // the small object optimization in 'any'.
351 struct throws_on_move
352 {
353     static int count;
354     static int copied;
355     static int moved;
resetthrows_on_move356     static void reset() { count = copied = moved = 0; }
357     int value;
358 
valuethrows_on_move359     explicit throws_on_move(int val = 0) : value(val) { ++count; }
throws_on_movethrows_on_move360     explicit throws_on_move(int, int val, int) : value(val) { ++count; }
throws_on_movethrows_on_move361     throws_on_move(throws_on_move const & other) {
362         value = other.value;
363         ++count; ++copied;
364     }
365 
throws_on_movethrows_on_move366     throws_on_move(throws_on_move &&) {
367         throwMyAnyExpression();
368     }
369 
~throws_on_movethrows_on_move370     ~throws_on_move() {
371         --count;
372     }
373 private:
374     throws_on_move& operator=(throws_on_move const&) = delete;
375     throws_on_move& operator=(throws_on_move &&) = delete;
376 };
377 
378 int throws_on_move::count = 0;
379 int throws_on_move::copied = 0;
380 int throws_on_move::moved = 0;
381 
382 struct small_tracked_t {
small_tracked_tsmall_tracked_t383   small_tracked_t()
384       : arg_types(&makeArgumentID<>()) {}
small_tracked_tsmall_tracked_t385   small_tracked_t(small_tracked_t const&) noexcept
386       : arg_types(&makeArgumentID<small_tracked_t const&>()) {}
small_tracked_tsmall_tracked_t387   small_tracked_t(small_tracked_t &&) noexcept
388       : arg_types(&makeArgumentID<small_tracked_t &&>()) {}
389   template <class ...Args>
small_tracked_tsmall_tracked_t390   explicit small_tracked_t(Args&&...)
391       : arg_types(&makeArgumentID<Args...>()) {}
392   template <class ...Args>
small_tracked_tsmall_tracked_t393   explicit small_tracked_t(std::initializer_list<int>, Args&&...)
394       : arg_types(&makeArgumentID<std::initializer_list<int>, Args...>()) {}
395 
396   TypeID const* arg_types;
397 };
398 static_assert(IsSmallObject<small_tracked_t>::value, "must be small");
399 
400 struct large_tracked_t {
large_tracked_tlarge_tracked_t401   large_tracked_t()
402       : arg_types(&makeArgumentID<>()) { dummy[0] = 42; }
large_tracked_tlarge_tracked_t403   large_tracked_t(large_tracked_t const&) noexcept
404       : arg_types(&makeArgumentID<large_tracked_t const&>()) {}
large_tracked_tlarge_tracked_t405   large_tracked_t(large_tracked_t &&) noexcept
406       : arg_types(&makeArgumentID<large_tracked_t &&>()) {}
407   template <class ...Args>
large_tracked_tlarge_tracked_t408   explicit large_tracked_t(Args&&...)
409       : arg_types(&makeArgumentID<Args...>()) {}
410   template <class ...Args>
large_tracked_tlarge_tracked_t411   explicit large_tracked_t(std::initializer_list<int>, Args&&...)
412       : arg_types(&makeArgumentID<std::initializer_list<int>, Args...>()) {}
413 
414   TypeID const* arg_types;
415   int dummy[10];
416 };
417 
418 static_assert(!IsSmallObject<large_tracked_t>::value, "must be small");
419 
420 
421 template <class Type, class ...Args>
assertArgsMatch(std::any const & a)422 void assertArgsMatch(std::any const& a) {
423     using namespace std;
424     using namespace std::experimental;
425     assert(a.has_value());
426     assert(containsType<Type>(a));
427     assert(any_cast<Type const &>(a).arg_types == &makeArgumentID<Args...>());
428 };
429 
430 
431 #endif
432