function_base.hpp 31 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892
  1. // Boost.Function library
  2. // Copyright Douglas Gregor 2001-2006
  3. // Copyright Emil Dotchevski 2007
  4. // Use, modification and distribution is subject to the Boost Software License, Version 1.0.
  5. // (See accompanying file LICENSE_1_0.txt or copy at
  6. // http://www.boost.org/LICENSE_1_0.txt)
  7. // For more information, see http://www.boost.org
  8. #ifndef BOOST_FUNCTION_BASE_HEADER
  9. #define BOOST_FUNCTION_BASE_HEADER
  10. #include <stdexcept>
  11. #include <string>
  12. #include <memory>
  13. #include <new>
  14. #include <boost/config.hpp>
  15. #include <boost/detail/sp_typeinfo.hpp>
  16. #include <boost/assert.hpp>
  17. #include <boost/integer.hpp>
  18. #include <boost/type_traits/has_trivial_copy.hpp>
  19. #include <boost/type_traits/has_trivial_destructor.hpp>
  20. #include <boost/type_traits/is_const.hpp>
  21. #include <boost/type_traits/is_integral.hpp>
  22. #include <boost/type_traits/is_volatile.hpp>
  23. #include <boost/type_traits/composite_traits.hpp>
  24. #include <boost/ref.hpp>
  25. #include <boost/mpl/if.hpp>
  26. #include <boost/detail/workaround.hpp>
  27. #include <boost/type_traits/alignment_of.hpp>
  28. #ifndef BOOST_NO_SFINAE
  29. # include "boost/utility/enable_if.hpp"
  30. #else
  31. # include "boost/mpl/bool.hpp"
  32. #endif
  33. #include <boost/function_equal.hpp>
  34. #include <boost/function/function_fwd.hpp>
  35. #if defined(BOOST_MSVC)
  36. # pragma warning( push )
  37. # pragma warning( disable : 4793 ) // complaint about native code generation
  38. # pragma warning( disable : 4127 ) // "conditional expression is constant"
  39. #endif
  40. // Define BOOST_FUNCTION_STD_NS to the namespace that contains type_info.
  41. #ifdef BOOST_NO_STD_TYPEINFO
  42. // Embedded VC++ does not have type_info in namespace std
  43. # define BOOST_FUNCTION_STD_NS
  44. #else
  45. # define BOOST_FUNCTION_STD_NS std
  46. #endif
  47. // Borrowed from Boost.Python library: determines the cases where we
  48. // need to use std::type_info::name to compare instead of operator==.
  49. #if defined( BOOST_NO_TYPEID )
  50. # define BOOST_FUNCTION_COMPARE_TYPE_ID(X,Y) ((X)==(Y))
  51. #elif defined(__GNUC__) \
  52. || defined(_AIX) \
  53. || ( defined(__sgi) && defined(__host_mips))
  54. # include <cstring>
  55. # define BOOST_FUNCTION_COMPARE_TYPE_ID(X,Y) \
  56. (std::strcmp((X).name(),(Y).name()) == 0)
  57. # else
  58. # define BOOST_FUNCTION_COMPARE_TYPE_ID(X,Y) ((X)==(Y))
  59. #endif
  60. #if defined(__ICL) && __ICL <= 600 || defined(__MWERKS__) && __MWERKS__ < 0x2406 && !defined(BOOST_STRICT_CONFIG)
  61. # define BOOST_FUNCTION_TARGET_FIX(x) x
  62. #else
  63. # define BOOST_FUNCTION_TARGET_FIX(x)
  64. #endif // __ICL etc
  65. # define BOOST_FUNCTION_ENABLE_IF_NOT_INTEGRAL(Functor,Type) \
  66. typename ::boost::enable_if_c< \
  67. !(::boost::is_integral<Functor>::value), \
  68. Type>::type
  69. namespace boost {
  70. namespace detail {
  71. namespace function {
  72. class X;
  73. /**
  74. * A buffer used to store small function objects in
  75. * boost::function. It is a union containing function pointers,
  76. * object pointers, and a structure that resembles a bound
  77. * member function pointer.
  78. */
  79. union function_buffer
  80. {
  81. // For pointers to function objects
  82. mutable void* obj_ptr;
  83. // For pointers to std::type_info objects
  84. struct type_t {
  85. // (get_functor_type_tag, check_functor_type_tag).
  86. const detail::sp_typeinfo* type;
  87. // Whether the type is const-qualified.
  88. bool const_qualified;
  89. // Whether the type is volatile-qualified.
  90. bool volatile_qualified;
  91. } type;
  92. // For function pointers of all kinds
  93. mutable void (*func_ptr)();
  94. // For bound member pointers
  95. struct bound_memfunc_ptr_t {
  96. void (X::*memfunc_ptr)(int);
  97. void* obj_ptr;
  98. } bound_memfunc_ptr;
  99. // For references to function objects. We explicitly keep
  100. // track of the cv-qualifiers on the object referenced.
  101. struct obj_ref_t {
  102. mutable void* obj_ptr;
  103. bool is_const_qualified;
  104. bool is_volatile_qualified;
  105. } obj_ref;
  106. // To relax aliasing constraints
  107. mutable char data;
  108. };
  109. /**
  110. * The unusable class is a placeholder for unused function arguments
  111. * It is also completely unusable except that it constructable from
  112. * anything. This helps compilers without partial specialization to
  113. * handle Boost.Function objects returning void.
  114. */
  115. struct unusable
  116. {
  117. unusable() {}
  118. template<typename T> unusable(const T&) {}
  119. };
  120. /* Determine the return type. This supports compilers that do not support
  121. * void returns or partial specialization by silently changing the return
  122. * type to "unusable".
  123. */
  124. template<typename T> struct function_return_type { typedef T type; };
  125. template<>
  126. struct function_return_type<void>
  127. {
  128. typedef unusable type;
  129. };
  130. // The operation type to perform on the given functor/function pointer
  131. enum functor_manager_operation_type {
  132. clone_functor_tag,
  133. move_functor_tag,
  134. destroy_functor_tag,
  135. check_functor_type_tag,
  136. get_functor_type_tag
  137. };
  138. // Tags used to decide between different types of functions
  139. struct function_ptr_tag {};
  140. struct function_obj_tag {};
  141. struct member_ptr_tag {};
  142. struct function_obj_ref_tag {};
  143. template<typename F>
  144. class get_function_tag
  145. {
  146. typedef typename mpl::if_c<(is_pointer<F>::value),
  147. function_ptr_tag,
  148. function_obj_tag>::type ptr_or_obj_tag;
  149. typedef typename mpl::if_c<(is_member_pointer<F>::value),
  150. member_ptr_tag,
  151. ptr_or_obj_tag>::type ptr_or_obj_or_mem_tag;
  152. typedef typename mpl::if_c<(is_reference_wrapper<F>::value),
  153. function_obj_ref_tag,
  154. ptr_or_obj_or_mem_tag>::type or_ref_tag;
  155. public:
  156. typedef or_ref_tag type;
  157. };
  158. // The trivial manager does nothing but return the same pointer (if we
  159. // are cloning) or return the null pointer (if we are deleting).
  160. template<typename F>
  161. struct reference_manager
  162. {
  163. static inline void
  164. manage(const function_buffer& in_buffer, function_buffer& out_buffer,
  165. functor_manager_operation_type op)
  166. {
  167. switch (op) {
  168. case clone_functor_tag:
  169. out_buffer.obj_ref = in_buffer.obj_ref;
  170. return;
  171. case move_functor_tag:
  172. out_buffer.obj_ref = in_buffer.obj_ref;
  173. in_buffer.obj_ref.obj_ptr = 0;
  174. return;
  175. case destroy_functor_tag:
  176. out_buffer.obj_ref.obj_ptr = 0;
  177. return;
  178. case check_functor_type_tag:
  179. {
  180. const detail::sp_typeinfo& check_type
  181. = *out_buffer.type.type;
  182. // Check whether we have the same type. We can add
  183. // cv-qualifiers, but we can't take them away.
  184. if (BOOST_FUNCTION_COMPARE_TYPE_ID(check_type, BOOST_SP_TYPEID(F))
  185. && (!in_buffer.obj_ref.is_const_qualified
  186. || out_buffer.type.const_qualified)
  187. && (!in_buffer.obj_ref.is_volatile_qualified
  188. || out_buffer.type.volatile_qualified))
  189. out_buffer.obj_ptr = in_buffer.obj_ref.obj_ptr;
  190. else
  191. out_buffer.obj_ptr = 0;
  192. }
  193. return;
  194. case get_functor_type_tag:
  195. out_buffer.type.type = &BOOST_SP_TYPEID(F);
  196. out_buffer.type.const_qualified = in_buffer.obj_ref.is_const_qualified;
  197. out_buffer.type.volatile_qualified = in_buffer.obj_ref.is_volatile_qualified;
  198. return;
  199. }
  200. }
  201. };
  202. /**
  203. * Determine if boost::function can use the small-object
  204. * optimization with the function object type F.
  205. */
  206. template<typename F>
  207. struct function_allows_small_object_optimization
  208. {
  209. BOOST_STATIC_CONSTANT
  210. (bool,
  211. value = ((sizeof(F) <= sizeof(function_buffer) &&
  212. (alignment_of<function_buffer>::value
  213. % alignment_of<F>::value == 0))));
  214. };
  215. template <typename F,typename A>
  216. struct functor_wrapper: public F, public A
  217. {
  218. functor_wrapper( F f, A a ):
  219. F(f),
  220. A(a)
  221. {
  222. }
  223. functor_wrapper(const functor_wrapper& f) :
  224. F(static_cast<const F&>(f)),
  225. A(static_cast<const A&>(f))
  226. {
  227. }
  228. };
  229. /**
  230. * The functor_manager class contains a static function "manage" which
  231. * can clone or destroy the given function/function object pointer.
  232. */
  233. template<typename Functor>
  234. struct functor_manager_common
  235. {
  236. typedef Functor functor_type;
  237. // Function pointers
  238. static inline void
  239. manage_ptr(const function_buffer& in_buffer, function_buffer& out_buffer,
  240. functor_manager_operation_type op)
  241. {
  242. if (op == clone_functor_tag)
  243. out_buffer.func_ptr = in_buffer.func_ptr;
  244. else if (op == move_functor_tag) {
  245. out_buffer.func_ptr = in_buffer.func_ptr;
  246. in_buffer.func_ptr = 0;
  247. } else if (op == destroy_functor_tag)
  248. out_buffer.func_ptr = 0;
  249. else if (op == check_functor_type_tag) {
  250. const boost::detail::sp_typeinfo& check_type
  251. = *out_buffer.type.type;
  252. if (BOOST_FUNCTION_COMPARE_TYPE_ID(check_type, BOOST_SP_TYPEID(Functor)))
  253. out_buffer.obj_ptr = &in_buffer.func_ptr;
  254. else
  255. out_buffer.obj_ptr = 0;
  256. } else /* op == get_functor_type_tag */ {
  257. out_buffer.type.type = &BOOST_SP_TYPEID(Functor);
  258. out_buffer.type.const_qualified = false;
  259. out_buffer.type.volatile_qualified = false;
  260. }
  261. }
  262. // Function objects that fit in the small-object buffer.
  263. static inline void
  264. manage_small(const function_buffer& in_buffer, function_buffer& out_buffer,
  265. functor_manager_operation_type op)
  266. {
  267. if (op == clone_functor_tag || op == move_functor_tag) {
  268. const functor_type* in_functor =
  269. reinterpret_cast<const functor_type*>(&in_buffer.data);
  270. new (reinterpret_cast<void*>(&out_buffer.data)) functor_type(*in_functor);
  271. if (op == move_functor_tag) {
  272. functor_type* f = reinterpret_cast<functor_type*>(&in_buffer.data);
  273. (void)f; // suppress warning about the value of f not being used (MSVC)
  274. f->~Functor();
  275. }
  276. } else if (op == destroy_functor_tag) {
  277. // Some compilers (Borland, vc6, ...) are unhappy with ~functor_type.
  278. functor_type* f = reinterpret_cast<functor_type*>(&out_buffer.data);
  279. (void)f; // suppress warning about the value of f not being used (MSVC)
  280. f->~Functor();
  281. } else if (op == check_functor_type_tag) {
  282. const detail::sp_typeinfo& check_type
  283. = *out_buffer.type.type;
  284. if (BOOST_FUNCTION_COMPARE_TYPE_ID(check_type, BOOST_SP_TYPEID(Functor)))
  285. out_buffer.obj_ptr = &in_buffer.data;
  286. else
  287. out_buffer.obj_ptr = 0;
  288. } else /* op == get_functor_type_tag */ {
  289. out_buffer.type.type = &BOOST_SP_TYPEID(Functor);
  290. out_buffer.type.const_qualified = false;
  291. out_buffer.type.volatile_qualified = false;
  292. }
  293. }
  294. };
  295. template<typename Functor>
  296. struct functor_manager
  297. {
  298. private:
  299. typedef Functor functor_type;
  300. // Function pointers
  301. static inline void
  302. manager(const function_buffer& in_buffer, function_buffer& out_buffer,
  303. functor_manager_operation_type op, function_ptr_tag)
  304. {
  305. functor_manager_common<Functor>::manage_ptr(in_buffer,out_buffer,op);
  306. }
  307. // Function objects that fit in the small-object buffer.
  308. static inline void
  309. manager(const function_buffer& in_buffer, function_buffer& out_buffer,
  310. functor_manager_operation_type op, mpl::true_)
  311. {
  312. functor_manager_common<Functor>::manage_small(in_buffer,out_buffer,op);
  313. }
  314. // Function objects that require heap allocation
  315. static inline void
  316. manager(const function_buffer& in_buffer, function_buffer& out_buffer,
  317. functor_manager_operation_type op, mpl::false_)
  318. {
  319. if (op == clone_functor_tag) {
  320. // Clone the functor
  321. // GCC 2.95.3 gets the CV qualifiers wrong here, so we
  322. // can't do the static_cast that we should do.
  323. // jewillco: Changing this to static_cast because GCC 2.95.3 is
  324. // obsolete.
  325. const functor_type* f =
  326. static_cast<const functor_type*>(in_buffer.obj_ptr);
  327. functor_type* new_f = new functor_type(*f);
  328. out_buffer.obj_ptr = new_f;
  329. } else if (op == move_functor_tag) {
  330. out_buffer.obj_ptr = in_buffer.obj_ptr;
  331. in_buffer.obj_ptr = 0;
  332. } else if (op == destroy_functor_tag) {
  333. /* Cast from the void pointer to the functor pointer type */
  334. functor_type* f =
  335. static_cast<functor_type*>(out_buffer.obj_ptr);
  336. delete f;
  337. out_buffer.obj_ptr = 0;
  338. } else if (op == check_functor_type_tag) {
  339. const detail::sp_typeinfo& check_type
  340. = *out_buffer.type.type;
  341. if (BOOST_FUNCTION_COMPARE_TYPE_ID(check_type, BOOST_SP_TYPEID(Functor)))
  342. out_buffer.obj_ptr = in_buffer.obj_ptr;
  343. else
  344. out_buffer.obj_ptr = 0;
  345. } else /* op == get_functor_type_tag */ {
  346. out_buffer.type.type = &BOOST_SP_TYPEID(Functor);
  347. out_buffer.type.const_qualified = false;
  348. out_buffer.type.volatile_qualified = false;
  349. }
  350. }
  351. // For function objects, we determine whether the function
  352. // object can use the small-object optimization buffer or
  353. // whether we need to allocate it on the heap.
  354. static inline void
  355. manager(const function_buffer& in_buffer, function_buffer& out_buffer,
  356. functor_manager_operation_type op, function_obj_tag)
  357. {
  358. manager(in_buffer, out_buffer, op,
  359. mpl::bool_<(function_allows_small_object_optimization<functor_type>::value)>());
  360. }
  361. // For member pointers, we use the small-object optimization buffer.
  362. static inline void
  363. manager(const function_buffer& in_buffer, function_buffer& out_buffer,
  364. functor_manager_operation_type op, member_ptr_tag)
  365. {
  366. manager(in_buffer, out_buffer, op, mpl::true_());
  367. }
  368. public:
  369. /* Dispatch to an appropriate manager based on whether we have a
  370. function pointer or a function object pointer. */
  371. static inline void
  372. manage(const function_buffer& in_buffer, function_buffer& out_buffer,
  373. functor_manager_operation_type op)
  374. {
  375. typedef typename get_function_tag<functor_type>::type tag_type;
  376. switch (op) {
  377. case get_functor_type_tag:
  378. out_buffer.type.type = &BOOST_SP_TYPEID(functor_type);
  379. out_buffer.type.const_qualified = false;
  380. out_buffer.type.volatile_qualified = false;
  381. return;
  382. default:
  383. manager(in_buffer, out_buffer, op, tag_type());
  384. return;
  385. }
  386. }
  387. };
  388. template<typename Functor, typename Allocator>
  389. struct functor_manager_a
  390. {
  391. private:
  392. typedef Functor functor_type;
  393. // Function pointers
  394. static inline void
  395. manager(const function_buffer& in_buffer, function_buffer& out_buffer,
  396. functor_manager_operation_type op, function_ptr_tag)
  397. {
  398. functor_manager_common<Functor>::manage_ptr(in_buffer,out_buffer,op);
  399. }
  400. // Function objects that fit in the small-object buffer.
  401. static inline void
  402. manager(const function_buffer& in_buffer, function_buffer& out_buffer,
  403. functor_manager_operation_type op, mpl::true_)
  404. {
  405. functor_manager_common<Functor>::manage_small(in_buffer,out_buffer,op);
  406. }
  407. // Function objects that require heap allocation
  408. static inline void
  409. manager(const function_buffer& in_buffer, function_buffer& out_buffer,
  410. functor_manager_operation_type op, mpl::false_)
  411. {
  412. typedef functor_wrapper<Functor,Allocator> functor_wrapper_type;
  413. typedef typename Allocator::template rebind<functor_wrapper_type>::other
  414. wrapper_allocator_type;
  415. typedef typename wrapper_allocator_type::pointer wrapper_allocator_pointer_type;
  416. if (op == clone_functor_tag) {
  417. // Clone the functor
  418. // GCC 2.95.3 gets the CV qualifiers wrong here, so we
  419. // can't do the static_cast that we should do.
  420. const functor_wrapper_type* f =
  421. static_cast<const functor_wrapper_type*>(in_buffer.obj_ptr);
  422. wrapper_allocator_type wrapper_allocator(static_cast<Allocator const &>(*f));
  423. wrapper_allocator_pointer_type copy = wrapper_allocator.allocate(1);
  424. wrapper_allocator.construct(copy, *f);
  425. // Get back to the original pointer type
  426. functor_wrapper_type* new_f = static_cast<functor_wrapper_type*>(copy);
  427. out_buffer.obj_ptr = new_f;
  428. } else if (op == move_functor_tag) {
  429. out_buffer.obj_ptr = in_buffer.obj_ptr;
  430. in_buffer.obj_ptr = 0;
  431. } else if (op == destroy_functor_tag) {
  432. /* Cast from the void pointer to the functor_wrapper_type */
  433. functor_wrapper_type* victim =
  434. static_cast<functor_wrapper_type*>(in_buffer.obj_ptr);
  435. wrapper_allocator_type wrapper_allocator(static_cast<Allocator const &>(*victim));
  436. wrapper_allocator.destroy(victim);
  437. wrapper_allocator.deallocate(victim,1);
  438. out_buffer.obj_ptr = 0;
  439. } else if (op == check_functor_type_tag) {
  440. const detail::sp_typeinfo& check_type
  441. = *out_buffer.type.type;
  442. if (BOOST_FUNCTION_COMPARE_TYPE_ID(check_type, BOOST_SP_TYPEID(Functor)))
  443. out_buffer.obj_ptr = in_buffer.obj_ptr;
  444. else
  445. out_buffer.obj_ptr = 0;
  446. } else /* op == get_functor_type_tag */ {
  447. out_buffer.type.type = &BOOST_SP_TYPEID(Functor);
  448. out_buffer.type.const_qualified = false;
  449. out_buffer.type.volatile_qualified = false;
  450. }
  451. }
  452. // For function objects, we determine whether the function
  453. // object can use the small-object optimization buffer or
  454. // whether we need to allocate it on the heap.
  455. static inline void
  456. manager(const function_buffer& in_buffer, function_buffer& out_buffer,
  457. functor_manager_operation_type op, function_obj_tag)
  458. {
  459. manager(in_buffer, out_buffer, op,
  460. mpl::bool_<(function_allows_small_object_optimization<functor_type>::value)>());
  461. }
  462. public:
  463. /* Dispatch to an appropriate manager based on whether we have a
  464. function pointer or a function object pointer. */
  465. static inline void
  466. manage(const function_buffer& in_buffer, function_buffer& out_buffer,
  467. functor_manager_operation_type op)
  468. {
  469. typedef typename get_function_tag<functor_type>::type tag_type;
  470. switch (op) {
  471. case get_functor_type_tag:
  472. out_buffer.type.type = &BOOST_SP_TYPEID(functor_type);
  473. out_buffer.type.const_qualified = false;
  474. out_buffer.type.volatile_qualified = false;
  475. return;
  476. default:
  477. manager(in_buffer, out_buffer, op, tag_type());
  478. return;
  479. }
  480. }
  481. };
  482. // A type that is only used for comparisons against zero
  483. struct useless_clear_type {};
  484. #ifdef BOOST_NO_SFINAE
  485. // These routines perform comparisons between a Boost.Function
  486. // object and an arbitrary function object (when the last
  487. // parameter is mpl::bool_<false>) or against zero (when the
  488. // last parameter is mpl::bool_<true>). They are only necessary
  489. // for compilers that don't support SFINAE.
  490. template<typename Function, typename Functor>
  491. bool
  492. compare_equal(const Function& f, const Functor&, int, mpl::bool_<true>)
  493. { return f.empty(); }
  494. template<typename Function, typename Functor>
  495. bool
  496. compare_not_equal(const Function& f, const Functor&, int,
  497. mpl::bool_<true>)
  498. { return !f.empty(); }
  499. template<typename Function, typename Functor>
  500. bool
  501. compare_equal(const Function& f, const Functor& g, long,
  502. mpl::bool_<false>)
  503. {
  504. if (const Functor* fp = f.template target<Functor>())
  505. return function_equal(*fp, g);
  506. else return false;
  507. }
  508. template<typename Function, typename Functor>
  509. bool
  510. compare_equal(const Function& f, const reference_wrapper<Functor>& g,
  511. int, mpl::bool_<false>)
  512. {
  513. if (const Functor* fp = f.template target<Functor>())
  514. return fp == g.get_pointer();
  515. else return false;
  516. }
  517. template<typename Function, typename Functor>
  518. bool
  519. compare_not_equal(const Function& f, const Functor& g, long,
  520. mpl::bool_<false>)
  521. {
  522. if (const Functor* fp = f.template target<Functor>())
  523. return !function_equal(*fp, g);
  524. else return true;
  525. }
  526. template<typename Function, typename Functor>
  527. bool
  528. compare_not_equal(const Function& f,
  529. const reference_wrapper<Functor>& g, int,
  530. mpl::bool_<false>)
  531. {
  532. if (const Functor* fp = f.template target<Functor>())
  533. return fp != g.get_pointer();
  534. else return true;
  535. }
  536. #endif // BOOST_NO_SFINAE
  537. /**
  538. * Stores the "manager" portion of the vtable for a
  539. * boost::function object.
  540. */
  541. struct vtable_base
  542. {
  543. void (*manager)(const function_buffer& in_buffer,
  544. function_buffer& out_buffer,
  545. functor_manager_operation_type op);
  546. };
  547. } // end namespace function
  548. } // end namespace detail
  549. /**
  550. * The function_base class contains the basic elements needed for the
  551. * function1, function2, function3, etc. classes. It is common to all
  552. * functions (and as such can be used to tell if we have one of the
  553. * functionN objects).
  554. */
  555. class function_base
  556. {
  557. public:
  558. function_base() : vtable(0) { }
  559. /** Determine if the function is empty (i.e., has no target). */
  560. bool empty() const { return !vtable; }
  561. /** Retrieve the type of the stored function object, or BOOST_SP_TYPEID(void)
  562. if this is empty. */
  563. const detail::sp_typeinfo& target_type() const
  564. {
  565. if (!vtable) return BOOST_SP_TYPEID(void);
  566. detail::function::function_buffer type;
  567. get_vtable()->manager(functor, type, detail::function::get_functor_type_tag);
  568. return *type.type.type;
  569. }
  570. template<typename Functor>
  571. Functor* target()
  572. {
  573. if (!vtable) return 0;
  574. detail::function::function_buffer type_result;
  575. type_result.type.type = &BOOST_SP_TYPEID(Functor);
  576. type_result.type.const_qualified = is_const<Functor>::value;
  577. type_result.type.volatile_qualified = is_volatile<Functor>::value;
  578. get_vtable()->manager(functor, type_result,
  579. detail::function::check_functor_type_tag);
  580. return static_cast<Functor*>(type_result.obj_ptr);
  581. }
  582. template<typename Functor>
  583. const Functor* target() const
  584. {
  585. if (!vtable) return 0;
  586. detail::function::function_buffer type_result;
  587. type_result.type.type = &BOOST_SP_TYPEID(Functor);
  588. type_result.type.const_qualified = true;
  589. type_result.type.volatile_qualified = is_volatile<Functor>::value;
  590. get_vtable()->manager(functor, type_result,
  591. detail::function::check_functor_type_tag);
  592. // GCC 2.95.3 gets the CV qualifiers wrong here, so we
  593. // can't do the static_cast that we should do.
  594. return static_cast<const Functor*>(type_result.obj_ptr);
  595. }
  596. template<typename F>
  597. bool contains(const F& f) const
  598. {
  599. if (const F* fp = this->template target<F>())
  600. {
  601. return function_equal(*fp, f);
  602. } else {
  603. return false;
  604. }
  605. }
  606. #if defined(__GNUC__) && __GNUC__ == 3 && __GNUC_MINOR__ <= 3
  607. // GCC 3.3 and newer cannot copy with the global operator==, due to
  608. // problems with instantiation of function return types before it
  609. // has been verified that the argument types match up.
  610. template<typename Functor>
  611. BOOST_FUNCTION_ENABLE_IF_NOT_INTEGRAL(Functor, bool)
  612. operator==(Functor g) const
  613. {
  614. if (const Functor* fp = target<Functor>())
  615. return function_equal(*fp, g);
  616. else return false;
  617. }
  618. template<typename Functor>
  619. BOOST_FUNCTION_ENABLE_IF_NOT_INTEGRAL(Functor, bool)
  620. operator!=(Functor g) const
  621. {
  622. if (const Functor* fp = target<Functor>())
  623. return !function_equal(*fp, g);
  624. else return true;
  625. }
  626. #endif
  627. public: // should be protected, but GCC 2.95.3 will fail to allow access
  628. detail::function::vtable_base* get_vtable() const {
  629. return reinterpret_cast<detail::function::vtable_base*>(
  630. reinterpret_cast<std::size_t>(vtable) & ~static_cast<std::size_t>(0x01));
  631. }
  632. bool has_trivial_copy_and_destroy() const {
  633. return reinterpret_cast<std::size_t>(vtable) & 0x01;
  634. }
  635. detail::function::vtable_base* vtable;
  636. mutable detail::function::function_buffer functor;
  637. };
  638. /**
  639. * The bad_function_call exception class is thrown when a boost::function
  640. * object is invoked
  641. */
  642. class bad_function_call : public std::runtime_error
  643. {
  644. public:
  645. bad_function_call() : std::runtime_error("call to empty boost::function") {}
  646. };
  647. #ifndef BOOST_NO_SFINAE
  648. inline bool operator==(const function_base& f,
  649. detail::function::useless_clear_type*)
  650. {
  651. return f.empty();
  652. }
  653. inline bool operator!=(const function_base& f,
  654. detail::function::useless_clear_type*)
  655. {
  656. return !f.empty();
  657. }
  658. inline bool operator==(detail::function::useless_clear_type*,
  659. const function_base& f)
  660. {
  661. return f.empty();
  662. }
  663. inline bool operator!=(detail::function::useless_clear_type*,
  664. const function_base& f)
  665. {
  666. return !f.empty();
  667. }
  668. #endif
  669. #ifdef BOOST_NO_SFINAE
  670. // Comparisons between boost::function objects and arbitrary function objects
  671. template<typename Functor>
  672. inline bool operator==(const function_base& f, Functor g)
  673. {
  674. typedef mpl::bool_<(is_integral<Functor>::value)> integral;
  675. return detail::function::compare_equal(f, g, 0, integral());
  676. }
  677. template<typename Functor>
  678. inline bool operator==(Functor g, const function_base& f)
  679. {
  680. typedef mpl::bool_<(is_integral<Functor>::value)> integral;
  681. return detail::function::compare_equal(f, g, 0, integral());
  682. }
  683. template<typename Functor>
  684. inline bool operator!=(const function_base& f, Functor g)
  685. {
  686. typedef mpl::bool_<(is_integral<Functor>::value)> integral;
  687. return detail::function::compare_not_equal(f, g, 0, integral());
  688. }
  689. template<typename Functor>
  690. inline bool operator!=(Functor g, const function_base& f)
  691. {
  692. typedef mpl::bool_<(is_integral<Functor>::value)> integral;
  693. return detail::function::compare_not_equal(f, g, 0, integral());
  694. }
  695. #else
  696. # if !(defined(__GNUC__) && __GNUC__ == 3 && __GNUC_MINOR__ <= 3)
  697. // Comparisons between boost::function objects and arbitrary function
  698. // objects. GCC 3.3 and before has an obnoxious bug that prevents this
  699. // from working.
  700. template<typename Functor>
  701. BOOST_FUNCTION_ENABLE_IF_NOT_INTEGRAL(Functor, bool)
  702. operator==(const function_base& f, Functor g)
  703. {
  704. if (const Functor* fp = f.template target<Functor>())
  705. return function_equal(*fp, g);
  706. else return false;
  707. }
  708. template<typename Functor>
  709. BOOST_FUNCTION_ENABLE_IF_NOT_INTEGRAL(Functor, bool)
  710. operator==(Functor g, const function_base& f)
  711. {
  712. if (const Functor* fp = f.template target<Functor>())
  713. return function_equal(g, *fp);
  714. else return false;
  715. }
  716. template<typename Functor>
  717. BOOST_FUNCTION_ENABLE_IF_NOT_INTEGRAL(Functor, bool)
  718. operator!=(const function_base& f, Functor g)
  719. {
  720. if (const Functor* fp = f.template target<Functor>())
  721. return !function_equal(*fp, g);
  722. else return true;
  723. }
  724. template<typename Functor>
  725. BOOST_FUNCTION_ENABLE_IF_NOT_INTEGRAL(Functor, bool)
  726. operator!=(Functor g, const function_base& f)
  727. {
  728. if (const Functor* fp = f.template target<Functor>())
  729. return !function_equal(g, *fp);
  730. else return true;
  731. }
  732. # endif
  733. template<typename Functor>
  734. BOOST_FUNCTION_ENABLE_IF_NOT_INTEGRAL(Functor, bool)
  735. operator==(const function_base& f, reference_wrapper<Functor> g)
  736. {
  737. if (const Functor* fp = f.template target<Functor>())
  738. return fp == g.get_pointer();
  739. else return false;
  740. }
  741. template<typename Functor>
  742. BOOST_FUNCTION_ENABLE_IF_NOT_INTEGRAL(Functor, bool)
  743. operator==(reference_wrapper<Functor> g, const function_base& f)
  744. {
  745. if (const Functor* fp = f.template target<Functor>())
  746. return g.get_pointer() == fp;
  747. else return false;
  748. }
  749. template<typename Functor>
  750. BOOST_FUNCTION_ENABLE_IF_NOT_INTEGRAL(Functor, bool)
  751. operator!=(const function_base& f, reference_wrapper<Functor> g)
  752. {
  753. if (const Functor* fp = f.template target<Functor>())
  754. return fp != g.get_pointer();
  755. else return true;
  756. }
  757. template<typename Functor>
  758. BOOST_FUNCTION_ENABLE_IF_NOT_INTEGRAL(Functor, bool)
  759. operator!=(reference_wrapper<Functor> g, const function_base& f)
  760. {
  761. if (const Functor* fp = f.template target<Functor>())
  762. return g.get_pointer() != fp;
  763. else return true;
  764. }
  765. #endif // Compiler supporting SFINAE
  766. namespace detail {
  767. namespace function {
  768. inline bool has_empty_target(const function_base* f)
  769. {
  770. return f->empty();
  771. }
  772. #if BOOST_WORKAROUND(BOOST_MSVC, <= 1310)
  773. inline bool has_empty_target(const void*)
  774. {
  775. return false;
  776. }
  777. #else
  778. inline bool has_empty_target(...)
  779. {
  780. return false;
  781. }
  782. #endif
  783. } // end namespace function
  784. } // end namespace detail
  785. } // end namespace boost
  786. #undef BOOST_FUNCTION_ENABLE_IF_NOT_INTEGRAL
  787. #undef BOOST_FUNCTION_COMPARE_TYPE_ID
  788. #if defined(BOOST_MSVC)
  789. # pragma warning( pop )
  790. #endif
  791. #endif // BOOST_FUNCTION_BASE_HEADER