function_template.hpp 39 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190
  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. // Note: this header is a header template and must NOT have multiple-inclusion
  9. // protection.
  10. #include <boost/function/detail/prologue.hpp>
  11. #include <boost/detail/no_exceptions_support.hpp>
  12. #if defined(BOOST_MSVC)
  13. # pragma warning( push )
  14. # pragma warning( disable : 4127 ) // "conditional expression is constant"
  15. #endif
  16. #define BOOST_FUNCTION_TEMPLATE_PARMS BOOST_PP_ENUM_PARAMS(BOOST_FUNCTION_NUM_ARGS, typename T)
  17. #define BOOST_FUNCTION_TEMPLATE_ARGS BOOST_PP_ENUM_PARAMS(BOOST_FUNCTION_NUM_ARGS, T)
  18. #define BOOST_FUNCTION_PARM(J,I,D) BOOST_PP_CAT(T,I) BOOST_PP_CAT(a,I)
  19. #define BOOST_FUNCTION_PARMS BOOST_PP_ENUM(BOOST_FUNCTION_NUM_ARGS,BOOST_FUNCTION_PARM,BOOST_PP_EMPTY)
  20. #ifdef BOOST_NO_CXX11_RVALUE_REFERENCES
  21. # define BOOST_FUNCTION_ARGS BOOST_PP_ENUM_PARAMS(BOOST_FUNCTION_NUM_ARGS, a)
  22. #else
  23. # include <boost/move/utility_core.hpp>
  24. # define BOOST_FUNCTION_ARG(J,I,D) ::boost::forward< BOOST_PP_CAT(T,I) >(BOOST_PP_CAT(a,I))
  25. # define BOOST_FUNCTION_ARGS BOOST_PP_ENUM(BOOST_FUNCTION_NUM_ARGS,BOOST_FUNCTION_ARG,BOOST_PP_EMPTY)
  26. #endif
  27. #define BOOST_FUNCTION_ARG_TYPE(J,I,D) \
  28. typedef BOOST_PP_CAT(T,I) BOOST_PP_CAT(BOOST_PP_CAT(arg, BOOST_PP_INC(I)),_type);
  29. #define BOOST_FUNCTION_ARG_TYPES BOOST_PP_REPEAT(BOOST_FUNCTION_NUM_ARGS,BOOST_FUNCTION_ARG_TYPE,BOOST_PP_EMPTY)
  30. // Comma if nonzero number of arguments
  31. #if BOOST_FUNCTION_NUM_ARGS == 0
  32. # define BOOST_FUNCTION_COMMA
  33. #else
  34. # define BOOST_FUNCTION_COMMA ,
  35. #endif // BOOST_FUNCTION_NUM_ARGS > 0
  36. // Class names used in this version of the code
  37. #define BOOST_FUNCTION_FUNCTION BOOST_JOIN(function,BOOST_FUNCTION_NUM_ARGS)
  38. #define BOOST_FUNCTION_FUNCTION_INVOKER \
  39. BOOST_JOIN(function_invoker,BOOST_FUNCTION_NUM_ARGS)
  40. #define BOOST_FUNCTION_VOID_FUNCTION_INVOKER \
  41. BOOST_JOIN(void_function_invoker,BOOST_FUNCTION_NUM_ARGS)
  42. #define BOOST_FUNCTION_FUNCTION_OBJ_INVOKER \
  43. BOOST_JOIN(function_obj_invoker,BOOST_FUNCTION_NUM_ARGS)
  44. #define BOOST_FUNCTION_VOID_FUNCTION_OBJ_INVOKER \
  45. BOOST_JOIN(void_function_obj_invoker,BOOST_FUNCTION_NUM_ARGS)
  46. #define BOOST_FUNCTION_FUNCTION_REF_INVOKER \
  47. BOOST_JOIN(function_ref_invoker,BOOST_FUNCTION_NUM_ARGS)
  48. #define BOOST_FUNCTION_VOID_FUNCTION_REF_INVOKER \
  49. BOOST_JOIN(void_function_ref_invoker,BOOST_FUNCTION_NUM_ARGS)
  50. #define BOOST_FUNCTION_MEMBER_INVOKER \
  51. BOOST_JOIN(function_mem_invoker,BOOST_FUNCTION_NUM_ARGS)
  52. #define BOOST_FUNCTION_VOID_MEMBER_INVOKER \
  53. BOOST_JOIN(function_void_mem_invoker,BOOST_FUNCTION_NUM_ARGS)
  54. #define BOOST_FUNCTION_GET_FUNCTION_INVOKER \
  55. BOOST_JOIN(get_function_invoker,BOOST_FUNCTION_NUM_ARGS)
  56. #define BOOST_FUNCTION_GET_FUNCTION_OBJ_INVOKER \
  57. BOOST_JOIN(get_function_obj_invoker,BOOST_FUNCTION_NUM_ARGS)
  58. #define BOOST_FUNCTION_GET_FUNCTION_REF_INVOKER \
  59. BOOST_JOIN(get_function_ref_invoker,BOOST_FUNCTION_NUM_ARGS)
  60. #define BOOST_FUNCTION_GET_MEMBER_INVOKER \
  61. BOOST_JOIN(get_member_invoker,BOOST_FUNCTION_NUM_ARGS)
  62. #define BOOST_FUNCTION_GET_INVOKER \
  63. BOOST_JOIN(get_invoker,BOOST_FUNCTION_NUM_ARGS)
  64. #define BOOST_FUNCTION_VTABLE BOOST_JOIN(basic_vtable,BOOST_FUNCTION_NUM_ARGS)
  65. #ifndef BOOST_NO_VOID_RETURNS
  66. # define BOOST_FUNCTION_VOID_RETURN_TYPE void
  67. # define BOOST_FUNCTION_RETURN(X) X
  68. #else
  69. # define BOOST_FUNCTION_VOID_RETURN_TYPE boost::detail::function::unusable
  70. # define BOOST_FUNCTION_RETURN(X) X; return BOOST_FUNCTION_VOID_RETURN_TYPE ()
  71. #endif
  72. namespace boost {
  73. namespace detail {
  74. namespace function {
  75. template<
  76. typename FunctionPtr,
  77. typename R BOOST_FUNCTION_COMMA
  78. BOOST_FUNCTION_TEMPLATE_PARMS
  79. >
  80. struct BOOST_FUNCTION_FUNCTION_INVOKER
  81. {
  82. static R invoke(function_buffer& function_ptr BOOST_FUNCTION_COMMA
  83. BOOST_FUNCTION_PARMS)
  84. {
  85. FunctionPtr f = reinterpret_cast<FunctionPtr>(function_ptr.func_ptr);
  86. return f(BOOST_FUNCTION_ARGS);
  87. }
  88. };
  89. template<
  90. typename FunctionPtr,
  91. typename R BOOST_FUNCTION_COMMA
  92. BOOST_FUNCTION_TEMPLATE_PARMS
  93. >
  94. struct BOOST_FUNCTION_VOID_FUNCTION_INVOKER
  95. {
  96. static BOOST_FUNCTION_VOID_RETURN_TYPE
  97. invoke(function_buffer& function_ptr BOOST_FUNCTION_COMMA
  98. BOOST_FUNCTION_PARMS)
  99. {
  100. FunctionPtr f = reinterpret_cast<FunctionPtr>(function_ptr.func_ptr);
  101. BOOST_FUNCTION_RETURN(f(BOOST_FUNCTION_ARGS));
  102. }
  103. };
  104. template<
  105. typename FunctionObj,
  106. typename R BOOST_FUNCTION_COMMA
  107. BOOST_FUNCTION_TEMPLATE_PARMS
  108. >
  109. struct BOOST_FUNCTION_FUNCTION_OBJ_INVOKER
  110. {
  111. static R invoke(function_buffer& function_obj_ptr BOOST_FUNCTION_COMMA
  112. BOOST_FUNCTION_PARMS)
  113. {
  114. FunctionObj* f;
  115. if (function_allows_small_object_optimization<FunctionObj>::value)
  116. f = reinterpret_cast<FunctionObj*>(&function_obj_ptr.data);
  117. else
  118. f = reinterpret_cast<FunctionObj*>(function_obj_ptr.obj_ptr);
  119. return (*f)(BOOST_FUNCTION_ARGS);
  120. }
  121. };
  122. template<
  123. typename FunctionObj,
  124. typename R BOOST_FUNCTION_COMMA
  125. BOOST_FUNCTION_TEMPLATE_PARMS
  126. >
  127. struct BOOST_FUNCTION_VOID_FUNCTION_OBJ_INVOKER
  128. {
  129. static BOOST_FUNCTION_VOID_RETURN_TYPE
  130. invoke(function_buffer& function_obj_ptr BOOST_FUNCTION_COMMA
  131. BOOST_FUNCTION_PARMS)
  132. {
  133. FunctionObj* f;
  134. if (function_allows_small_object_optimization<FunctionObj>::value)
  135. f = reinterpret_cast<FunctionObj*>(&function_obj_ptr.data);
  136. else
  137. f = reinterpret_cast<FunctionObj*>(function_obj_ptr.obj_ptr);
  138. BOOST_FUNCTION_RETURN((*f)(BOOST_FUNCTION_ARGS));
  139. }
  140. };
  141. template<
  142. typename FunctionObj,
  143. typename R BOOST_FUNCTION_COMMA
  144. BOOST_FUNCTION_TEMPLATE_PARMS
  145. >
  146. struct BOOST_FUNCTION_FUNCTION_REF_INVOKER
  147. {
  148. static R invoke(function_buffer& function_obj_ptr BOOST_FUNCTION_COMMA
  149. BOOST_FUNCTION_PARMS)
  150. {
  151. FunctionObj* f =
  152. reinterpret_cast<FunctionObj*>(function_obj_ptr.obj_ptr);
  153. return (*f)(BOOST_FUNCTION_ARGS);
  154. }
  155. };
  156. template<
  157. typename FunctionObj,
  158. typename R BOOST_FUNCTION_COMMA
  159. BOOST_FUNCTION_TEMPLATE_PARMS
  160. >
  161. struct BOOST_FUNCTION_VOID_FUNCTION_REF_INVOKER
  162. {
  163. static BOOST_FUNCTION_VOID_RETURN_TYPE
  164. invoke(function_buffer& function_obj_ptr BOOST_FUNCTION_COMMA
  165. BOOST_FUNCTION_PARMS)
  166. {
  167. FunctionObj* f =
  168. reinterpret_cast<FunctionObj*>(function_obj_ptr.obj_ptr);
  169. BOOST_FUNCTION_RETURN((*f)(BOOST_FUNCTION_ARGS));
  170. }
  171. };
  172. #if BOOST_FUNCTION_NUM_ARGS > 0
  173. /* Handle invocation of member pointers. */
  174. template<
  175. typename MemberPtr,
  176. typename R BOOST_FUNCTION_COMMA
  177. BOOST_FUNCTION_TEMPLATE_PARMS
  178. >
  179. struct BOOST_FUNCTION_MEMBER_INVOKER
  180. {
  181. static R invoke(function_buffer& function_obj_ptr BOOST_FUNCTION_COMMA
  182. BOOST_FUNCTION_PARMS)
  183. {
  184. MemberPtr* f =
  185. reinterpret_cast<MemberPtr*>(&function_obj_ptr.data);
  186. return boost::mem_fn(*f)(BOOST_FUNCTION_ARGS);
  187. }
  188. };
  189. template<
  190. typename MemberPtr,
  191. typename R BOOST_FUNCTION_COMMA
  192. BOOST_FUNCTION_TEMPLATE_PARMS
  193. >
  194. struct BOOST_FUNCTION_VOID_MEMBER_INVOKER
  195. {
  196. static BOOST_FUNCTION_VOID_RETURN_TYPE
  197. invoke(function_buffer& function_obj_ptr BOOST_FUNCTION_COMMA
  198. BOOST_FUNCTION_PARMS)
  199. {
  200. MemberPtr* f =
  201. reinterpret_cast<MemberPtr*>(&function_obj_ptr.data);
  202. BOOST_FUNCTION_RETURN(boost::mem_fn(*f)(BOOST_FUNCTION_ARGS));
  203. }
  204. };
  205. #endif
  206. template<
  207. typename FunctionPtr,
  208. typename R BOOST_FUNCTION_COMMA
  209. BOOST_FUNCTION_TEMPLATE_PARMS
  210. >
  211. struct BOOST_FUNCTION_GET_FUNCTION_INVOKER
  212. {
  213. typedef typename mpl::if_c<(is_void<R>::value),
  214. BOOST_FUNCTION_VOID_FUNCTION_INVOKER<
  215. FunctionPtr,
  216. R BOOST_FUNCTION_COMMA
  217. BOOST_FUNCTION_TEMPLATE_ARGS
  218. >,
  219. BOOST_FUNCTION_FUNCTION_INVOKER<
  220. FunctionPtr,
  221. R BOOST_FUNCTION_COMMA
  222. BOOST_FUNCTION_TEMPLATE_ARGS
  223. >
  224. >::type type;
  225. };
  226. template<
  227. typename FunctionObj,
  228. typename R BOOST_FUNCTION_COMMA
  229. BOOST_FUNCTION_TEMPLATE_PARMS
  230. >
  231. struct BOOST_FUNCTION_GET_FUNCTION_OBJ_INVOKER
  232. {
  233. typedef typename mpl::if_c<(is_void<R>::value),
  234. BOOST_FUNCTION_VOID_FUNCTION_OBJ_INVOKER<
  235. FunctionObj,
  236. R BOOST_FUNCTION_COMMA
  237. BOOST_FUNCTION_TEMPLATE_ARGS
  238. >,
  239. BOOST_FUNCTION_FUNCTION_OBJ_INVOKER<
  240. FunctionObj,
  241. R BOOST_FUNCTION_COMMA
  242. BOOST_FUNCTION_TEMPLATE_ARGS
  243. >
  244. >::type type;
  245. };
  246. template<
  247. typename FunctionObj,
  248. typename R BOOST_FUNCTION_COMMA
  249. BOOST_FUNCTION_TEMPLATE_PARMS
  250. >
  251. struct BOOST_FUNCTION_GET_FUNCTION_REF_INVOKER
  252. {
  253. typedef typename mpl::if_c<(is_void<R>::value),
  254. BOOST_FUNCTION_VOID_FUNCTION_REF_INVOKER<
  255. FunctionObj,
  256. R BOOST_FUNCTION_COMMA
  257. BOOST_FUNCTION_TEMPLATE_ARGS
  258. >,
  259. BOOST_FUNCTION_FUNCTION_REF_INVOKER<
  260. FunctionObj,
  261. R BOOST_FUNCTION_COMMA
  262. BOOST_FUNCTION_TEMPLATE_ARGS
  263. >
  264. >::type type;
  265. };
  266. #if BOOST_FUNCTION_NUM_ARGS > 0
  267. /* Retrieve the appropriate invoker for a member pointer. */
  268. template<
  269. typename MemberPtr,
  270. typename R BOOST_FUNCTION_COMMA
  271. BOOST_FUNCTION_TEMPLATE_PARMS
  272. >
  273. struct BOOST_FUNCTION_GET_MEMBER_INVOKER
  274. {
  275. typedef typename mpl::if_c<(is_void<R>::value),
  276. BOOST_FUNCTION_VOID_MEMBER_INVOKER<
  277. MemberPtr,
  278. R BOOST_FUNCTION_COMMA
  279. BOOST_FUNCTION_TEMPLATE_ARGS
  280. >,
  281. BOOST_FUNCTION_MEMBER_INVOKER<
  282. MemberPtr,
  283. R BOOST_FUNCTION_COMMA
  284. BOOST_FUNCTION_TEMPLATE_ARGS
  285. >
  286. >::type type;
  287. };
  288. #endif
  289. /* Given the tag returned by get_function_tag, retrieve the
  290. actual invoker that will be used for the given function
  291. object.
  292. Each specialization contains an "apply" nested class template
  293. that accepts the function object, return type, function
  294. argument types, and allocator. The resulting "apply" class
  295. contains two typedefs, "invoker_type" and "manager_type",
  296. which correspond to the invoker and manager types. */
  297. template<typename Tag>
  298. struct BOOST_FUNCTION_GET_INVOKER { };
  299. /* Retrieve the invoker for a function pointer. */
  300. template<>
  301. struct BOOST_FUNCTION_GET_INVOKER<function_ptr_tag>
  302. {
  303. template<typename FunctionPtr,
  304. typename R BOOST_FUNCTION_COMMA BOOST_FUNCTION_TEMPLATE_PARMS>
  305. struct apply
  306. {
  307. typedef typename BOOST_FUNCTION_GET_FUNCTION_INVOKER<
  308. FunctionPtr,
  309. R BOOST_FUNCTION_COMMA
  310. BOOST_FUNCTION_TEMPLATE_ARGS
  311. >::type
  312. invoker_type;
  313. typedef functor_manager<FunctionPtr> manager_type;
  314. };
  315. template<typename FunctionPtr,
  316. typename R BOOST_FUNCTION_COMMA BOOST_FUNCTION_TEMPLATE_PARMS,
  317. typename Allocator>
  318. struct apply_a
  319. {
  320. typedef typename BOOST_FUNCTION_GET_FUNCTION_INVOKER<
  321. FunctionPtr,
  322. R BOOST_FUNCTION_COMMA
  323. BOOST_FUNCTION_TEMPLATE_ARGS
  324. >::type
  325. invoker_type;
  326. typedef functor_manager<FunctionPtr> manager_type;
  327. };
  328. };
  329. #if BOOST_FUNCTION_NUM_ARGS > 0
  330. /* Retrieve the invoker for a member pointer. */
  331. template<>
  332. struct BOOST_FUNCTION_GET_INVOKER<member_ptr_tag>
  333. {
  334. template<typename MemberPtr,
  335. typename R BOOST_FUNCTION_COMMA BOOST_FUNCTION_TEMPLATE_PARMS>
  336. struct apply
  337. {
  338. typedef typename BOOST_FUNCTION_GET_MEMBER_INVOKER<
  339. MemberPtr,
  340. R BOOST_FUNCTION_COMMA
  341. BOOST_FUNCTION_TEMPLATE_ARGS
  342. >::type
  343. invoker_type;
  344. typedef functor_manager<MemberPtr> manager_type;
  345. };
  346. template<typename MemberPtr,
  347. typename R BOOST_FUNCTION_COMMA BOOST_FUNCTION_TEMPLATE_PARMS,
  348. typename Allocator>
  349. struct apply_a
  350. {
  351. typedef typename BOOST_FUNCTION_GET_MEMBER_INVOKER<
  352. MemberPtr,
  353. R BOOST_FUNCTION_COMMA
  354. BOOST_FUNCTION_TEMPLATE_ARGS
  355. >::type
  356. invoker_type;
  357. typedef functor_manager<MemberPtr> manager_type;
  358. };
  359. };
  360. #endif
  361. /* Retrieve the invoker for a function object. */
  362. template<>
  363. struct BOOST_FUNCTION_GET_INVOKER<function_obj_tag>
  364. {
  365. template<typename FunctionObj,
  366. typename R BOOST_FUNCTION_COMMA BOOST_FUNCTION_TEMPLATE_PARMS>
  367. struct apply
  368. {
  369. typedef typename BOOST_FUNCTION_GET_FUNCTION_OBJ_INVOKER<
  370. FunctionObj,
  371. R BOOST_FUNCTION_COMMA
  372. BOOST_FUNCTION_TEMPLATE_ARGS
  373. >::type
  374. invoker_type;
  375. typedef functor_manager<FunctionObj> manager_type;
  376. };
  377. template<typename FunctionObj,
  378. typename R BOOST_FUNCTION_COMMA BOOST_FUNCTION_TEMPLATE_PARMS,
  379. typename Allocator>
  380. struct apply_a
  381. {
  382. typedef typename BOOST_FUNCTION_GET_FUNCTION_OBJ_INVOKER<
  383. FunctionObj,
  384. R BOOST_FUNCTION_COMMA
  385. BOOST_FUNCTION_TEMPLATE_ARGS
  386. >::type
  387. invoker_type;
  388. typedef functor_manager_a<FunctionObj, Allocator> manager_type;
  389. };
  390. };
  391. /* Retrieve the invoker for a reference to a function object. */
  392. template<>
  393. struct BOOST_FUNCTION_GET_INVOKER<function_obj_ref_tag>
  394. {
  395. template<typename RefWrapper,
  396. typename R BOOST_FUNCTION_COMMA BOOST_FUNCTION_TEMPLATE_PARMS>
  397. struct apply
  398. {
  399. typedef typename BOOST_FUNCTION_GET_FUNCTION_REF_INVOKER<
  400. typename RefWrapper::type,
  401. R BOOST_FUNCTION_COMMA
  402. BOOST_FUNCTION_TEMPLATE_ARGS
  403. >::type
  404. invoker_type;
  405. typedef reference_manager<typename RefWrapper::type> manager_type;
  406. };
  407. template<typename RefWrapper,
  408. typename R BOOST_FUNCTION_COMMA BOOST_FUNCTION_TEMPLATE_PARMS,
  409. typename Allocator>
  410. struct apply_a
  411. {
  412. typedef typename BOOST_FUNCTION_GET_FUNCTION_REF_INVOKER<
  413. typename RefWrapper::type,
  414. R BOOST_FUNCTION_COMMA
  415. BOOST_FUNCTION_TEMPLATE_ARGS
  416. >::type
  417. invoker_type;
  418. typedef reference_manager<typename RefWrapper::type> manager_type;
  419. };
  420. };
  421. /**
  422. * vtable for a specific boost::function instance. This
  423. * structure must be an aggregate so that we can use static
  424. * initialization in boost::function's assign_to and assign_to_a
  425. * members. It therefore cannot have any constructors,
  426. * destructors, base classes, etc.
  427. */
  428. template<typename R BOOST_FUNCTION_COMMA BOOST_FUNCTION_TEMPLATE_PARMS>
  429. struct BOOST_FUNCTION_VTABLE
  430. {
  431. #ifndef BOOST_NO_VOID_RETURNS
  432. typedef R result_type;
  433. #else
  434. typedef typename function_return_type<R>::type result_type;
  435. #endif // BOOST_NO_VOID_RETURNS
  436. typedef result_type (*invoker_type)(function_buffer&
  437. BOOST_FUNCTION_COMMA
  438. BOOST_FUNCTION_TEMPLATE_ARGS);
  439. template<typename F>
  440. bool assign_to(F f, function_buffer& functor) const
  441. {
  442. typedef typename get_function_tag<F>::type tag;
  443. return assign_to(f, functor, tag());
  444. }
  445. template<typename F,typename Allocator>
  446. bool assign_to_a(F f, function_buffer& functor, Allocator a) const
  447. {
  448. typedef typename get_function_tag<F>::type tag;
  449. return assign_to_a(f, functor, a, tag());
  450. }
  451. void clear(function_buffer& functor) const
  452. {
  453. if (base.manager)
  454. base.manager(functor, functor, destroy_functor_tag);
  455. }
  456. private:
  457. // Function pointers
  458. template<typename FunctionPtr>
  459. bool
  460. assign_to(FunctionPtr f, function_buffer& functor, function_ptr_tag) const
  461. {
  462. this->clear(functor);
  463. if (f) {
  464. // should be a reinterpret cast, but some compilers insist
  465. // on giving cv-qualifiers to free functions
  466. functor.func_ptr = reinterpret_cast<void (*)()>(f);
  467. return true;
  468. } else {
  469. return false;
  470. }
  471. }
  472. template<typename FunctionPtr,typename Allocator>
  473. bool
  474. assign_to_a(FunctionPtr f, function_buffer& functor, Allocator, function_ptr_tag) const
  475. {
  476. return assign_to(f,functor,function_ptr_tag());
  477. }
  478. // Member pointers
  479. #if BOOST_FUNCTION_NUM_ARGS > 0
  480. template<typename MemberPtr>
  481. bool assign_to(MemberPtr f, function_buffer& functor, member_ptr_tag) const
  482. {
  483. // DPG TBD: Add explicit support for member function
  484. // objects, so we invoke through mem_fn() but we retain the
  485. // right target_type() values.
  486. if (f) {
  487. this->assign_to(boost::mem_fn(f), functor);
  488. return true;
  489. } else {
  490. return false;
  491. }
  492. }
  493. template<typename MemberPtr,typename Allocator>
  494. bool assign_to_a(MemberPtr f, function_buffer& functor, Allocator a, member_ptr_tag) const
  495. {
  496. // DPG TBD: Add explicit support for member function
  497. // objects, so we invoke through mem_fn() but we retain the
  498. // right target_type() values.
  499. if (f) {
  500. this->assign_to_a(boost::mem_fn(f), functor, a);
  501. return true;
  502. } else {
  503. return false;
  504. }
  505. }
  506. #endif // BOOST_FUNCTION_NUM_ARGS > 0
  507. // Function objects
  508. // Assign to a function object using the small object optimization
  509. template<typename FunctionObj>
  510. void
  511. assign_functor(FunctionObj f, function_buffer& functor, mpl::true_) const
  512. {
  513. new (reinterpret_cast<void*>(&functor.data)) FunctionObj(f);
  514. }
  515. template<typename FunctionObj,typename Allocator>
  516. void
  517. assign_functor_a(FunctionObj f, function_buffer& functor, Allocator, mpl::true_) const
  518. {
  519. assign_functor(f,functor,mpl::true_());
  520. }
  521. // Assign to a function object allocated on the heap.
  522. template<typename FunctionObj>
  523. void
  524. assign_functor(FunctionObj f, function_buffer& functor, mpl::false_) const
  525. {
  526. functor.obj_ptr = new FunctionObj(f);
  527. }
  528. template<typename FunctionObj,typename Allocator>
  529. void
  530. assign_functor_a(FunctionObj f, function_buffer& functor, Allocator a, mpl::false_) const
  531. {
  532. typedef functor_wrapper<FunctionObj,Allocator> functor_wrapper_type;
  533. typedef typename Allocator::template rebind<functor_wrapper_type>::other
  534. wrapper_allocator_type;
  535. typedef typename wrapper_allocator_type::pointer wrapper_allocator_pointer_type;
  536. wrapper_allocator_type wrapper_allocator(a);
  537. wrapper_allocator_pointer_type copy = wrapper_allocator.allocate(1);
  538. wrapper_allocator.construct(copy, functor_wrapper_type(f,a));
  539. functor_wrapper_type* new_f = static_cast<functor_wrapper_type*>(copy);
  540. functor.obj_ptr = new_f;
  541. }
  542. template<typename FunctionObj>
  543. bool
  544. assign_to(FunctionObj f, function_buffer& functor, function_obj_tag) const
  545. {
  546. if (!boost::detail::function::has_empty_target(boost::addressof(f))) {
  547. assign_functor(f, functor,
  548. mpl::bool_<(function_allows_small_object_optimization<FunctionObj>::value)>());
  549. return true;
  550. } else {
  551. return false;
  552. }
  553. }
  554. template<typename FunctionObj,typename Allocator>
  555. bool
  556. assign_to_a(FunctionObj f, function_buffer& functor, Allocator a, function_obj_tag) const
  557. {
  558. if (!boost::detail::function::has_empty_target(boost::addressof(f))) {
  559. assign_functor_a(f, functor, a,
  560. mpl::bool_<(function_allows_small_object_optimization<FunctionObj>::value)>());
  561. return true;
  562. } else {
  563. return false;
  564. }
  565. }
  566. // Reference to a function object
  567. template<typename FunctionObj>
  568. bool
  569. assign_to(const reference_wrapper<FunctionObj>& f,
  570. function_buffer& functor, function_obj_ref_tag) const
  571. {
  572. functor.obj_ref.obj_ptr = (void *)(f.get_pointer());
  573. functor.obj_ref.is_const_qualified = is_const<FunctionObj>::value;
  574. functor.obj_ref.is_volatile_qualified = is_volatile<FunctionObj>::value;
  575. return true;
  576. }
  577. template<typename FunctionObj,typename Allocator>
  578. bool
  579. assign_to_a(const reference_wrapper<FunctionObj>& f,
  580. function_buffer& functor, Allocator, function_obj_ref_tag) const
  581. {
  582. return assign_to(f,functor,function_obj_ref_tag());
  583. }
  584. public:
  585. vtable_base base;
  586. invoker_type invoker;
  587. };
  588. } // end namespace function
  589. } // end namespace detail
  590. template<
  591. typename R BOOST_FUNCTION_COMMA
  592. BOOST_FUNCTION_TEMPLATE_PARMS
  593. >
  594. class BOOST_FUNCTION_FUNCTION : public function_base
  595. #if BOOST_FUNCTION_NUM_ARGS == 1
  596. , public std::unary_function<T0,R>
  597. #elif BOOST_FUNCTION_NUM_ARGS == 2
  598. , public std::binary_function<T0,T1,R>
  599. #endif
  600. {
  601. public:
  602. #ifndef BOOST_NO_VOID_RETURNS
  603. typedef R result_type;
  604. #else
  605. typedef typename boost::detail::function::function_return_type<R>::type
  606. result_type;
  607. #endif // BOOST_NO_VOID_RETURNS
  608. private:
  609. typedef boost::detail::function::BOOST_FUNCTION_VTABLE<
  610. R BOOST_FUNCTION_COMMA BOOST_FUNCTION_TEMPLATE_ARGS>
  611. vtable_type;
  612. vtable_type* get_vtable() const {
  613. return reinterpret_cast<vtable_type*>(
  614. reinterpret_cast<std::size_t>(vtable) & ~static_cast<std::size_t>(0x01));
  615. }
  616. struct clear_type {};
  617. public:
  618. BOOST_STATIC_CONSTANT(int, args = BOOST_FUNCTION_NUM_ARGS);
  619. // add signature for boost::lambda
  620. template<typename Args>
  621. struct sig
  622. {
  623. typedef result_type type;
  624. };
  625. #if BOOST_FUNCTION_NUM_ARGS == 1
  626. typedef T0 argument_type;
  627. #elif BOOST_FUNCTION_NUM_ARGS == 2
  628. typedef T0 first_argument_type;
  629. typedef T1 second_argument_type;
  630. #endif
  631. BOOST_STATIC_CONSTANT(int, arity = BOOST_FUNCTION_NUM_ARGS);
  632. BOOST_FUNCTION_ARG_TYPES
  633. typedef BOOST_FUNCTION_FUNCTION self_type;
  634. BOOST_FUNCTION_FUNCTION() : function_base() { }
  635. // MSVC chokes if the following two constructors are collapsed into
  636. // one with a default parameter.
  637. template<typename Functor>
  638. BOOST_FUNCTION_FUNCTION(Functor BOOST_FUNCTION_TARGET_FIX(const &) f
  639. #ifndef BOOST_NO_SFINAE
  640. ,typename boost::enable_if_c<
  641. !(is_integral<Functor>::value),
  642. int>::type = 0
  643. #endif // BOOST_NO_SFINAE
  644. ) :
  645. function_base()
  646. {
  647. this->assign_to(f);
  648. }
  649. template<typename Functor,typename Allocator>
  650. BOOST_FUNCTION_FUNCTION(Functor BOOST_FUNCTION_TARGET_FIX(const &) f, Allocator a
  651. #ifndef BOOST_NO_SFINAE
  652. ,typename boost::enable_if_c<
  653. !(is_integral<Functor>::value),
  654. int>::type = 0
  655. #endif // BOOST_NO_SFINAE
  656. ) :
  657. function_base()
  658. {
  659. this->assign_to_a(f,a);
  660. }
  661. #ifndef BOOST_NO_SFINAE
  662. BOOST_FUNCTION_FUNCTION(clear_type*) : function_base() { }
  663. #else
  664. BOOST_FUNCTION_FUNCTION(int zero) : function_base()
  665. {
  666. BOOST_ASSERT(zero == 0);
  667. }
  668. #endif
  669. BOOST_FUNCTION_FUNCTION(const BOOST_FUNCTION_FUNCTION& f) : function_base()
  670. {
  671. this->assign_to_own(f);
  672. }
  673. #ifndef BOOST_NO_CXX11_RVALUE_REFERENCES
  674. BOOST_FUNCTION_FUNCTION(BOOST_FUNCTION_FUNCTION&& f) : function_base()
  675. {
  676. this->move_assign(f);
  677. }
  678. #endif
  679. ~BOOST_FUNCTION_FUNCTION() { clear(); }
  680. result_type operator()(BOOST_FUNCTION_PARMS) const
  681. {
  682. if (this->empty())
  683. boost::throw_exception(bad_function_call());
  684. return get_vtable()->invoker
  685. (this->functor BOOST_FUNCTION_COMMA BOOST_FUNCTION_ARGS);
  686. }
  687. // The distinction between when to use BOOST_FUNCTION_FUNCTION and
  688. // when to use self_type is obnoxious. MSVC cannot handle self_type as
  689. // the return type of these assignment operators, but Borland C++ cannot
  690. // handle BOOST_FUNCTION_FUNCTION as the type of the temporary to
  691. // construct.
  692. template<typename Functor>
  693. #ifndef BOOST_NO_SFINAE
  694. typename boost::enable_if_c<
  695. !(is_integral<Functor>::value),
  696. BOOST_FUNCTION_FUNCTION&>::type
  697. #else
  698. BOOST_FUNCTION_FUNCTION&
  699. #endif
  700. operator=(Functor BOOST_FUNCTION_TARGET_FIX(const &) f)
  701. {
  702. this->clear();
  703. BOOST_TRY {
  704. this->assign_to(f);
  705. } BOOST_CATCH (...) {
  706. vtable = 0;
  707. BOOST_RETHROW;
  708. }
  709. BOOST_CATCH_END
  710. return *this;
  711. }
  712. template<typename Functor,typename Allocator>
  713. void assign(Functor BOOST_FUNCTION_TARGET_FIX(const &) f, Allocator a)
  714. {
  715. this->clear();
  716. BOOST_TRY{
  717. this->assign_to_a(f,a);
  718. } BOOST_CATCH (...) {
  719. vtable = 0;
  720. BOOST_RETHROW;
  721. }
  722. BOOST_CATCH_END
  723. }
  724. #ifndef BOOST_NO_SFINAE
  725. BOOST_FUNCTION_FUNCTION& operator=(clear_type*)
  726. {
  727. this->clear();
  728. return *this;
  729. }
  730. #else
  731. BOOST_FUNCTION_FUNCTION& operator=(int zero)
  732. {
  733. BOOST_ASSERT(zero == 0);
  734. this->clear();
  735. return *this;
  736. }
  737. #endif
  738. // Assignment from another BOOST_FUNCTION_FUNCTION
  739. BOOST_FUNCTION_FUNCTION& operator=(const BOOST_FUNCTION_FUNCTION& f)
  740. {
  741. if (&f == this)
  742. return *this;
  743. this->clear();
  744. BOOST_TRY {
  745. this->assign_to_own(f);
  746. } BOOST_CATCH (...) {
  747. vtable = 0;
  748. BOOST_RETHROW;
  749. }
  750. BOOST_CATCH_END
  751. return *this;
  752. }
  753. #ifndef BOOST_NO_CXX11_RVALUE_REFERENCES
  754. // Move assignment from another BOOST_FUNCTION_FUNCTION
  755. BOOST_FUNCTION_FUNCTION& operator=(BOOST_FUNCTION_FUNCTION&& f)
  756. {
  757. if (&f == this)
  758. return *this;
  759. this->clear();
  760. BOOST_TRY {
  761. this->move_assign(f);
  762. } BOOST_CATCH (...) {
  763. vtable = 0;
  764. BOOST_RETHROW;
  765. }
  766. BOOST_CATCH_END
  767. return *this;
  768. }
  769. #endif
  770. void swap(BOOST_FUNCTION_FUNCTION& other)
  771. {
  772. if (&other == this)
  773. return;
  774. BOOST_FUNCTION_FUNCTION tmp;
  775. tmp.move_assign(*this);
  776. this->move_assign(other);
  777. other.move_assign(tmp);
  778. }
  779. // Clear out a target, if there is one
  780. void clear()
  781. {
  782. if (vtable) {
  783. if (!this->has_trivial_copy_and_destroy())
  784. get_vtable()->clear(this->functor);
  785. vtable = 0;
  786. }
  787. }
  788. #if (defined __SUNPRO_CC) && (__SUNPRO_CC <= 0x530) && !(defined BOOST_NO_COMPILER_CONFIG)
  789. // Sun C++ 5.3 can't handle the safe_bool idiom, so don't use it
  790. operator bool () const { return !this->empty(); }
  791. #else
  792. private:
  793. struct dummy {
  794. void nonnull() {}
  795. };
  796. typedef void (dummy::*safe_bool)();
  797. public:
  798. operator safe_bool () const
  799. { return (this->empty())? 0 : &dummy::nonnull; }
  800. bool operator!() const
  801. { return this->empty(); }
  802. #endif
  803. private:
  804. void assign_to_own(const BOOST_FUNCTION_FUNCTION& f)
  805. {
  806. if (!f.empty()) {
  807. this->vtable = f.vtable;
  808. if (this->has_trivial_copy_and_destroy())
  809. this->functor = f.functor;
  810. else
  811. get_vtable()->base.manager(f.functor, this->functor,
  812. boost::detail::function::clone_functor_tag);
  813. }
  814. }
  815. template<typename Functor>
  816. void assign_to(Functor f)
  817. {
  818. using boost::detail::function::vtable_base;
  819. typedef typename boost::detail::function::get_function_tag<Functor>::type tag;
  820. typedef boost::detail::function::BOOST_FUNCTION_GET_INVOKER<tag> get_invoker;
  821. typedef typename get_invoker::
  822. template apply<Functor, R BOOST_FUNCTION_COMMA
  823. BOOST_FUNCTION_TEMPLATE_ARGS>
  824. handler_type;
  825. typedef typename handler_type::invoker_type invoker_type;
  826. typedef typename handler_type::manager_type manager_type;
  827. // Note: it is extremely important that this initialization use
  828. // static initialization. Otherwise, we will have a race
  829. // condition here in multi-threaded code. See
  830. // http://thread.gmane.org/gmane.comp.lib.boost.devel/164902/.
  831. static const vtable_type stored_vtable =
  832. { { &manager_type::manage }, &invoker_type::invoke };
  833. if (stored_vtable.assign_to(f, functor)) {
  834. std::size_t value = reinterpret_cast<std::size_t>(&stored_vtable.base);
  835. // coverity[pointless_expression]: suppress coverity warnings on apparant if(const).
  836. if (boost::has_trivial_copy_constructor<Functor>::value &&
  837. boost::has_trivial_destructor<Functor>::value &&
  838. boost::detail::function::function_allows_small_object_optimization<Functor>::value)
  839. value |= static_cast<std::size_t>(0x01);
  840. vtable = reinterpret_cast<boost::detail::function::vtable_base *>(value);
  841. } else
  842. vtable = 0;
  843. }
  844. template<typename Functor,typename Allocator>
  845. void assign_to_a(Functor f,Allocator a)
  846. {
  847. using boost::detail::function::vtable_base;
  848. typedef typename boost::detail::function::get_function_tag<Functor>::type tag;
  849. typedef boost::detail::function::BOOST_FUNCTION_GET_INVOKER<tag> get_invoker;
  850. typedef typename get_invoker::
  851. template apply_a<Functor, R BOOST_FUNCTION_COMMA
  852. BOOST_FUNCTION_TEMPLATE_ARGS,
  853. Allocator>
  854. handler_type;
  855. typedef typename handler_type::invoker_type invoker_type;
  856. typedef typename handler_type::manager_type manager_type;
  857. // Note: it is extremely important that this initialization use
  858. // static initialization. Otherwise, we will have a race
  859. // condition here in multi-threaded code. See
  860. // http://thread.gmane.org/gmane.comp.lib.boost.devel/164902/.
  861. static const vtable_type stored_vtable =
  862. { { &manager_type::manage }, &invoker_type::invoke };
  863. if (stored_vtable.assign_to_a(f, functor, a)) {
  864. std::size_t value = reinterpret_cast<std::size_t>(&stored_vtable.base);
  865. // coverity[pointless_expression]: suppress coverity warnings on apparant if(const).
  866. if (boost::has_trivial_copy_constructor<Functor>::value &&
  867. boost::has_trivial_destructor<Functor>::value &&
  868. boost::detail::function::function_allows_small_object_optimization<Functor>::value)
  869. value |= static_cast<std::size_t>(0x01);
  870. vtable = reinterpret_cast<boost::detail::function::vtable_base *>(value);
  871. } else
  872. vtable = 0;
  873. }
  874. // Moves the value from the specified argument to *this. If the argument
  875. // has its function object allocated on the heap, move_assign will pass
  876. // its buffer to *this, and set the argument's buffer pointer to NULL.
  877. void move_assign(BOOST_FUNCTION_FUNCTION& f)
  878. {
  879. if (&f == this)
  880. return;
  881. BOOST_TRY {
  882. if (!f.empty()) {
  883. this->vtable = f.vtable;
  884. if (this->has_trivial_copy_and_destroy())
  885. this->functor = f.functor;
  886. else
  887. get_vtable()->base.manager(f.functor, this->functor,
  888. boost::detail::function::move_functor_tag);
  889. f.vtable = 0;
  890. } else {
  891. clear();
  892. }
  893. } BOOST_CATCH (...) {
  894. vtable = 0;
  895. BOOST_RETHROW;
  896. }
  897. BOOST_CATCH_END
  898. }
  899. };
  900. template<typename R BOOST_FUNCTION_COMMA BOOST_FUNCTION_TEMPLATE_PARMS>
  901. inline void swap(BOOST_FUNCTION_FUNCTION<
  902. R BOOST_FUNCTION_COMMA
  903. BOOST_FUNCTION_TEMPLATE_ARGS
  904. >& f1,
  905. BOOST_FUNCTION_FUNCTION<
  906. R BOOST_FUNCTION_COMMA
  907. BOOST_FUNCTION_TEMPLATE_ARGS
  908. >& f2)
  909. {
  910. f1.swap(f2);
  911. }
  912. // Poison comparisons between boost::function objects of the same type.
  913. template<typename R BOOST_FUNCTION_COMMA BOOST_FUNCTION_TEMPLATE_PARMS>
  914. void operator==(const BOOST_FUNCTION_FUNCTION<
  915. R BOOST_FUNCTION_COMMA
  916. BOOST_FUNCTION_TEMPLATE_ARGS>&,
  917. const BOOST_FUNCTION_FUNCTION<
  918. R BOOST_FUNCTION_COMMA
  919. BOOST_FUNCTION_TEMPLATE_ARGS>&);
  920. template<typename R BOOST_FUNCTION_COMMA BOOST_FUNCTION_TEMPLATE_PARMS>
  921. void operator!=(const BOOST_FUNCTION_FUNCTION<
  922. R BOOST_FUNCTION_COMMA
  923. BOOST_FUNCTION_TEMPLATE_ARGS>&,
  924. const BOOST_FUNCTION_FUNCTION<
  925. R BOOST_FUNCTION_COMMA
  926. BOOST_FUNCTION_TEMPLATE_ARGS>& );
  927. #if !defined(BOOST_FUNCTION_NO_FUNCTION_TYPE_SYNTAX)
  928. #if BOOST_FUNCTION_NUM_ARGS == 0
  929. #define BOOST_FUNCTION_PARTIAL_SPEC R (void)
  930. #else
  931. #define BOOST_FUNCTION_PARTIAL_SPEC R (BOOST_PP_ENUM_PARAMS(BOOST_FUNCTION_NUM_ARGS,T))
  932. #endif
  933. template<typename R BOOST_FUNCTION_COMMA
  934. BOOST_FUNCTION_TEMPLATE_PARMS>
  935. class function<BOOST_FUNCTION_PARTIAL_SPEC>
  936. : public BOOST_FUNCTION_FUNCTION<R BOOST_FUNCTION_COMMA BOOST_FUNCTION_TEMPLATE_ARGS>
  937. {
  938. typedef BOOST_FUNCTION_FUNCTION<R BOOST_FUNCTION_COMMA BOOST_FUNCTION_TEMPLATE_ARGS> base_type;
  939. typedef function self_type;
  940. struct clear_type {};
  941. public:
  942. function() : base_type() {}
  943. template<typename Functor>
  944. function(Functor f
  945. #ifndef BOOST_NO_SFINAE
  946. ,typename boost::enable_if_c<
  947. !(is_integral<Functor>::value),
  948. int>::type = 0
  949. #endif
  950. ) :
  951. base_type(f)
  952. {
  953. }
  954. template<typename Functor,typename Allocator>
  955. function(Functor f, Allocator a
  956. #ifndef BOOST_NO_SFINAE
  957. ,typename boost::enable_if_c<
  958. !(is_integral<Functor>::value),
  959. int>::type = 0
  960. #endif
  961. ) :
  962. base_type(f,a)
  963. {
  964. }
  965. #ifndef BOOST_NO_SFINAE
  966. function(clear_type*) : base_type() {}
  967. #endif
  968. function(const self_type& f) : base_type(static_cast<const base_type&>(f)){}
  969. function(const base_type& f) : base_type(static_cast<const base_type&>(f)){}
  970. #ifndef BOOST_NO_CXX11_RVALUE_REFERENCES
  971. // Move constructors
  972. function(self_type&& f): base_type(static_cast<base_type&&>(f)){}
  973. function(base_type&& f): base_type(static_cast<base_type&&>(f)){}
  974. #endif
  975. self_type& operator=(const self_type& f)
  976. {
  977. self_type(f).swap(*this);
  978. return *this;
  979. }
  980. #ifndef BOOST_NO_CXX11_RVALUE_REFERENCES
  981. self_type& operator=(self_type&& f)
  982. {
  983. self_type(static_cast<self_type&&>(f)).swap(*this);
  984. return *this;
  985. }
  986. #endif
  987. template<typename Functor>
  988. #ifndef BOOST_NO_SFINAE
  989. typename boost::enable_if_c<
  990. !(is_integral<Functor>::value),
  991. self_type&>::type
  992. #else
  993. self_type&
  994. #endif
  995. operator=(Functor f)
  996. {
  997. self_type(f).swap(*this);
  998. return *this;
  999. }
  1000. #ifndef BOOST_NO_SFINAE
  1001. self_type& operator=(clear_type*)
  1002. {
  1003. this->clear();
  1004. return *this;
  1005. }
  1006. #endif
  1007. self_type& operator=(const base_type& f)
  1008. {
  1009. self_type(f).swap(*this);
  1010. return *this;
  1011. }
  1012. #ifndef BOOST_NO_CXX11_RVALUE_REFERENCES
  1013. self_type& operator=(base_type&& f)
  1014. {
  1015. self_type(static_cast<base_type&&>(f)).swap(*this);
  1016. return *this;
  1017. }
  1018. #endif
  1019. };
  1020. #undef BOOST_FUNCTION_PARTIAL_SPEC
  1021. #endif // have partial specialization
  1022. } // end namespace boost
  1023. // Cleanup after ourselves...
  1024. #undef BOOST_FUNCTION_VTABLE
  1025. #undef BOOST_FUNCTION_COMMA
  1026. #undef BOOST_FUNCTION_FUNCTION
  1027. #undef BOOST_FUNCTION_FUNCTION_INVOKER
  1028. #undef BOOST_FUNCTION_VOID_FUNCTION_INVOKER
  1029. #undef BOOST_FUNCTION_FUNCTION_OBJ_INVOKER
  1030. #undef BOOST_FUNCTION_VOID_FUNCTION_OBJ_INVOKER
  1031. #undef BOOST_FUNCTION_FUNCTION_REF_INVOKER
  1032. #undef BOOST_FUNCTION_VOID_FUNCTION_REF_INVOKER
  1033. #undef BOOST_FUNCTION_MEMBER_INVOKER
  1034. #undef BOOST_FUNCTION_VOID_MEMBER_INVOKER
  1035. #undef BOOST_FUNCTION_GET_FUNCTION_INVOKER
  1036. #undef BOOST_FUNCTION_GET_FUNCTION_OBJ_INVOKER
  1037. #undef BOOST_FUNCTION_GET_FUNCTION_REF_INVOKER
  1038. #undef BOOST_FUNCTION_GET_MEM_FUNCTION_INVOKER
  1039. #undef BOOST_FUNCTION_GET_INVOKER
  1040. #undef BOOST_FUNCTION_TEMPLATE_PARMS
  1041. #undef BOOST_FUNCTION_TEMPLATE_ARGS
  1042. #undef BOOST_FUNCTION_PARMS
  1043. #undef BOOST_FUNCTION_PARM
  1044. #ifdef BOOST_FUNCTION_ARG
  1045. # undef BOOST_FUNCTION_ARG
  1046. #endif
  1047. #undef BOOST_FUNCTION_ARGS
  1048. #undef BOOST_FUNCTION_ARG_TYPE
  1049. #undef BOOST_FUNCTION_ARG_TYPES
  1050. #undef BOOST_FUNCTION_VOID_RETURN_TYPE
  1051. #undef BOOST_FUNCTION_RETURN
  1052. #if defined(BOOST_MSVC)
  1053. # pragma warning( pop )
  1054. #endif