offset_ptr.hpp 29 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749
  1. //////////////////////////////////////////////////////////////////////////////
  2. //
  3. // (C) Copyright Ion Gaztanaga 2005-2015. Distributed under the Boost
  4. // Software License, Version 1.0. (See accompanying file
  5. // LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
  6. //
  7. // See http://www.boost.org/libs/interprocess for documentation.
  8. //
  9. //////////////////////////////////////////////////////////////////////////////
  10. #ifndef BOOST_INTERPROCESS_OFFSET_PTR_HPP
  11. #define BOOST_INTERPROCESS_OFFSET_PTR_HPP
  12. #ifndef BOOST_CONFIG_HPP
  13. # include <boost/config.hpp>
  14. #endif
  15. #
  16. #if defined(BOOST_HAS_PRAGMA_ONCE)
  17. # pragma once
  18. #endif
  19. #include <boost/interprocess/detail/config_begin.hpp>
  20. #include <boost/interprocess/detail/workaround.hpp>
  21. #include <boost/interprocess/interprocess_fwd.hpp>
  22. #include <boost/interprocess/detail/utilities.hpp>
  23. #include <boost/interprocess/detail/cast_tags.hpp>
  24. #include <boost/interprocess/detail/mpl.hpp>
  25. #include <boost/container/detail/type_traits.hpp> //alignment_of, aligned_storage
  26. #include <boost/assert.hpp>
  27. #include <iosfwd>
  28. //!\file
  29. //!Describes a smart pointer that stores the offset between this pointer and
  30. //!target pointee, called offset_ptr.
  31. namespace boost {
  32. #if !defined(BOOST_INTERPROCESS_DOXYGEN_INVOKED)
  33. //Predeclarations
  34. template <class T>
  35. struct has_trivial_destructor;
  36. #endif //#if !defined(BOOST_INTERPROCESS_DOXYGEN_INVOKED)
  37. namespace interprocess {
  38. #if !defined(BOOST_INTERPROCESS_DOXYGEN_INVOKED)
  39. namespace ipcdetail {
  40. template<class OffsetType, std::size_t OffsetAlignment>
  41. union offset_ptr_internal
  42. {
  43. BOOST_STATIC_ASSERT(sizeof(OffsetType) >= sizeof(uintptr_t));
  44. explicit offset_ptr_internal(OffsetType off)
  45. : m_offset(off)
  46. {}
  47. OffsetType m_offset; //Distance between this object and pointee address
  48. typename ::boost::container::container_detail::aligned_storage
  49. < sizeof(OffsetType)//for offset_type_alignment m_offset will be enough
  50. , (OffsetAlignment == offset_type_alignment) ? 1u : OffsetAlignment
  51. >::type alignment_helper;
  52. };
  53. //Note: using the address of a local variable to point to another address
  54. //is not standard conforming and this can be optimized-away by the compiler.
  55. //Non-inlining is a method to remain illegal but correct
  56. //Undef BOOST_INTERPROCESS_OFFSET_PTR_INLINE_XXX if your compiler can inline
  57. //this code without breaking the library
  58. ////////////////////////////////////////////////////////////////////////
  59. //
  60. // offset_ptr_to_raw_pointer
  61. //
  62. ////////////////////////////////////////////////////////////////////////
  63. #define BOOST_INTERPROCESS_OFFSET_PTR_BRANCHLESS_TO_PTR
  64. BOOST_INTERPROCESS_FORCEINLINE void * offset_ptr_to_raw_pointer(const volatile void *this_ptr, uintptr_t offset)
  65. {
  66. typedef pointer_uintptr_caster<void*> caster_t;
  67. #ifndef BOOST_INTERPROCESS_OFFSET_PTR_BRANCHLESS_TO_PTR
  68. if(offset == 1){
  69. return 0;
  70. }
  71. else{
  72. return caster_t(caster_t(this_ptr).uintptr() + offset).pointer();
  73. }
  74. #else
  75. uintptr_t mask = offset == 1;
  76. --mask;
  77. uintptr_t target_offset = caster_t(this_ptr).uintptr() + offset;
  78. target_offset &= mask;
  79. return caster_t(target_offset).pointer();
  80. #endif
  81. }
  82. ////////////////////////////////////////////////////////////////////////
  83. //
  84. // offset_ptr_to_offset
  85. //
  86. ////////////////////////////////////////////////////////////////////////
  87. #define BOOST_INTERPROCESS_OFFSET_PTR_BRANCHLESS_TO_OFF
  88. BOOST_INTERPROCESS_FORCEINLINE uintptr_t offset_ptr_to_offset(const volatile void *ptr, const volatile void *this_ptr)
  89. {
  90. typedef pointer_uintptr_caster<void*> caster_t;
  91. #ifndef BOOST_INTERPROCESS_OFFSET_PTR_BRANCHLESS_TO_OFF
  92. //offset == 1 && ptr != 0 is not legal for this pointer
  93. if(!ptr){
  94. return 1;
  95. }
  96. else{
  97. uintptr_t offset = caster_t(ptr).uintptr() - caster_t(this_ptr).uintptr();
  98. BOOST_ASSERT(offset != 1);
  99. return offset;
  100. }
  101. #else
  102. //const uintptr_t other = -uintptr_t(ptr != 0);
  103. //const uintptr_t offset = (caster_t(ptr).uintptr() - caster_t(this_ptr).uintptr()) & other;
  104. //return offset + uintptr_t(!other);
  105. //
  106. uintptr_t offset = caster_t(ptr).uintptr() - caster_t(this_ptr).uintptr();
  107. --offset;
  108. uintptr_t mask = uintptr_t(ptr == 0);
  109. --mask;
  110. offset &= mask;
  111. return ++offset;
  112. #endif
  113. }
  114. ////////////////////////////////////////////////////////////////////////
  115. //
  116. // offset_ptr_to_offset_from_other
  117. //
  118. ////////////////////////////////////////////////////////////////////////
  119. #define BOOST_INTERPROCESS_OFFSET_PTR_BRANCHLESS_TO_OFF_FROM_OTHER
  120. BOOST_INTERPROCESS_FORCEINLINE uintptr_t offset_ptr_to_offset_from_other
  121. (const volatile void *this_ptr, const volatile void *other_ptr, uintptr_t other_offset)
  122. {
  123. typedef pointer_uintptr_caster<void*> caster_t;
  124. #ifndef BOOST_INTERPROCESS_OFFSET_PTR_BRANCHLESS_TO_OFF_FROM_OTHER
  125. if(other_offset == 1){
  126. return 1;
  127. }
  128. else{
  129. uintptr_t offset = caster_t(other_ptr).uintptr() - caster_t(this_ptr).uintptr() + other_offset;
  130. BOOST_ASSERT(offset != 1);
  131. return offset;
  132. }
  133. #else
  134. uintptr_t mask = other_offset == 1;
  135. --mask;
  136. uintptr_t offset = caster_t(other_ptr).uintptr() - caster_t(this_ptr).uintptr();
  137. offset &= mask;
  138. return offset + other_offset;
  139. //uintptr_t mask = -uintptr_t(other_offset != 1);
  140. //uintptr_t offset = caster_t(other_ptr).uintptr() - caster_t(this_ptr).uintptr();
  141. //offset &= mask;
  142. //return offset + other_offset;
  143. #endif
  144. }
  145. ////////////////////////////////////////////////////////////////////////
  146. //
  147. // Let's assume cast to void and cv cast don't change any target address
  148. //
  149. ////////////////////////////////////////////////////////////////////////
  150. template<class From, class To>
  151. struct offset_ptr_maintains_address
  152. {
  153. static const bool value = ipcdetail::is_cv_same<From, To>::value
  154. || ipcdetail::is_cv_same<void, To>::value
  155. || ipcdetail::is_cv_same<char, To>::value
  156. ;
  157. };
  158. template<class From, class To, class Ret = void>
  159. struct enable_if_convertible_equal_address
  160. : enable_if_c< is_convertible<From*, To*>::value
  161. && offset_ptr_maintains_address<From, To>::value
  162. , Ret>
  163. {};
  164. template<class From, class To, class Ret = void>
  165. struct enable_if_convertible_unequal_address
  166. : enable_if_c< is_convertible<From*, To*>::value
  167. && !offset_ptr_maintains_address<From, To>::value
  168. , Ret>
  169. {};
  170. } //namespace ipcdetail {
  171. #endif //#ifndef BOOST_INTERPROCESS_DOXYGEN_INVOKED
  172. //!A smart pointer that stores the offset between between the pointer and the
  173. //!the object it points. This allows offset allows special properties, since
  174. //!the pointer is independent from the address of the pointee, if the
  175. //!pointer and the pointee are still separated by the same offset. This feature
  176. //!converts offset_ptr in a smart pointer that can be placed in shared memory and
  177. //!memory mapped files mapped in different addresses in every process.
  178. //!
  179. //! \tparam PointedType The type of the pointee.
  180. //! \tparam DifferenceType A signed integer type that can represent the arithmetic operations on the pointer
  181. //! \tparam OffsetType An unsigned integer type that can represent the
  182. //! distance between two pointers reinterpret_cast-ed as unsigned integers. This type
  183. //! should be at least of the same size of std::uintptr_t. In some systems it's possible to communicate
  184. //! between 32 and 64 bit processes using 64 bit offsets.
  185. //! \tparam OffsetAlignment Alignment of the OffsetType stored inside. In some systems might be necessary
  186. //! to align it to 64 bits in order to communicate 32 and 64 bit processes using 64 bit offsets.
  187. //!
  188. //!<b>Note</b>: offset_ptr uses implementation defined properties, present in most platforms, for
  189. //!performance reasons:
  190. //! - Assumes that uintptr_t representation of nullptr is (uintptr_t)zero.
  191. //! - Assumes that incrementing a uintptr_t obtained from a pointer is equivalent
  192. //! to incrementing the pointer and then converting it back to uintptr_t.
  193. template <class PointedType, class DifferenceType, class OffsetType, std::size_t OffsetAlignment>
  194. class offset_ptr
  195. {
  196. #if !defined(BOOST_INTERPROCESS_DOXYGEN_INVOKED)
  197. typedef offset_ptr<PointedType, DifferenceType, OffsetType, OffsetAlignment> self_t;
  198. void unspecified_bool_type_func() const {}
  199. typedef void (self_t::*unspecified_bool_type)() const;
  200. #endif //#ifndef BOOST_INTERPROCESS_DOXYGEN_INVOKED
  201. public:
  202. typedef PointedType element_type;
  203. typedef PointedType * pointer;
  204. typedef typename ipcdetail::
  205. add_reference<PointedType>::type reference;
  206. typedef typename ipcdetail::
  207. remove_volatile<typename ipcdetail::
  208. remove_const<PointedType>::type
  209. >::type value_type;
  210. typedef DifferenceType difference_type;
  211. typedef std::random_access_iterator_tag iterator_category;
  212. typedef OffsetType offset_type;
  213. public: //Public Functions
  214. //!Default constructor (null pointer).
  215. //!Never throws.
  216. BOOST_INTERPROCESS_FORCEINLINE offset_ptr() BOOST_NOEXCEPT
  217. : internal(1)
  218. {}
  219. //!Constructor from raw pointer (allows "0" pointer conversion).
  220. //!Never throws.
  221. BOOST_INTERPROCESS_FORCEINLINE offset_ptr(pointer ptr) BOOST_NOEXCEPT
  222. : internal(static_cast<OffsetType>(ipcdetail::offset_ptr_to_offset(ptr, this)))
  223. {}
  224. //!Constructor from other pointer.
  225. //!Never throws.
  226. template <class T>
  227. BOOST_INTERPROCESS_FORCEINLINE offset_ptr( T *ptr
  228. , typename ipcdetail::enable_if< ipcdetail::is_convertible<T*, PointedType*> >::type * = 0) BOOST_NOEXCEPT
  229. : internal(static_cast<OffsetType>
  230. (ipcdetail::offset_ptr_to_offset(static_cast<PointedType*>(ptr), this)))
  231. {}
  232. //!Constructor from other offset_ptr
  233. //!Never throws.
  234. BOOST_INTERPROCESS_FORCEINLINE offset_ptr(const offset_ptr& ptr) BOOST_NOEXCEPT
  235. : internal(static_cast<OffsetType>
  236. (ipcdetail::offset_ptr_to_offset_from_other(this, &ptr, ptr.internal.m_offset)))
  237. {}
  238. //!Constructor from other offset_ptr. If pointers of pointee types are
  239. //!convertible, offset_ptrs will be convertibles. Never throws.
  240. template<class T2>
  241. BOOST_INTERPROCESS_FORCEINLINE offset_ptr( const offset_ptr<T2, DifferenceType, OffsetType, OffsetAlignment> &ptr
  242. #ifndef BOOST_INTERPROCESS_DOXYGEN_INVOKED
  243. , typename ipcdetail::enable_if_convertible_equal_address<T2, PointedType>::type* = 0
  244. #endif
  245. ) BOOST_NOEXCEPT
  246. : internal(static_cast<OffsetType>
  247. (ipcdetail::offset_ptr_to_offset_from_other(this, &ptr, ptr.get_offset())))
  248. {}
  249. #ifndef BOOST_INTERPROCESS_DOXYGEN_INVOKED
  250. //!Constructor from other offset_ptr. If pointers of pointee types are
  251. //!convertible, offset_ptrs will be convertibles. Never throws.
  252. template<class T2>
  253. BOOST_INTERPROCESS_FORCEINLINE offset_ptr( const offset_ptr<T2, DifferenceType, OffsetType, OffsetAlignment> &ptr
  254. , typename ipcdetail::enable_if_convertible_unequal_address<T2, PointedType>::type* = 0) BOOST_NOEXCEPT
  255. : internal(static_cast<OffsetType>
  256. (ipcdetail::offset_ptr_to_offset(static_cast<PointedType*>(ptr.get()), this)))
  257. {}
  258. #endif
  259. //!Emulates static_cast operator.
  260. //!Never throws.
  261. template<class T2, class P2, class O2, std::size_t A2>
  262. BOOST_INTERPROCESS_FORCEINLINE offset_ptr(const offset_ptr<T2, P2, O2, A2> & r, ipcdetail::static_cast_tag) BOOST_NOEXCEPT
  263. : internal(static_cast<OffsetType>
  264. (ipcdetail::offset_ptr_to_offset(static_cast<PointedType*>(r.get()), this)))
  265. {}
  266. //!Emulates const_cast operator.
  267. //!Never throws.
  268. template<class T2, class P2, class O2, std::size_t A2>
  269. BOOST_INTERPROCESS_FORCEINLINE offset_ptr(const offset_ptr<T2, P2, O2, A2> & r, ipcdetail::const_cast_tag) BOOST_NOEXCEPT
  270. : internal(static_cast<OffsetType>
  271. (ipcdetail::offset_ptr_to_offset(const_cast<PointedType*>(r.get()), this)))
  272. {}
  273. //!Emulates dynamic_cast operator.
  274. //!Never throws.
  275. template<class T2, class P2, class O2, std::size_t A2>
  276. BOOST_INTERPROCESS_FORCEINLINE offset_ptr(const offset_ptr<T2, P2, O2, A2> & r, ipcdetail::dynamic_cast_tag) BOOST_NOEXCEPT
  277. : internal(static_cast<OffsetType>
  278. (ipcdetail::offset_ptr_to_offset(dynamic_cast<PointedType*>(r.get()), this)))
  279. {}
  280. //!Emulates reinterpret_cast operator.
  281. //!Never throws.
  282. template<class T2, class P2, class O2, std::size_t A2>
  283. BOOST_INTERPROCESS_FORCEINLINE offset_ptr(const offset_ptr<T2, P2, O2, A2> & r, ipcdetail::reinterpret_cast_tag) BOOST_NOEXCEPT
  284. : internal(static_cast<OffsetType>
  285. (ipcdetail::offset_ptr_to_offset(reinterpret_cast<PointedType*>(r.get()), this)))
  286. {}
  287. //!Obtains raw pointer from offset.
  288. //!Never throws.
  289. BOOST_INTERPROCESS_FORCEINLINE pointer get() const BOOST_NOEXCEPT
  290. { return (pointer)ipcdetail::offset_ptr_to_raw_pointer(this, this->internal.m_offset); }
  291. BOOST_INTERPROCESS_FORCEINLINE offset_type get_offset() const BOOST_NOEXCEPT
  292. { return this->internal.m_offset; }
  293. //!Pointer-like -> operator. It can return 0 pointer.
  294. //!Never throws.
  295. BOOST_INTERPROCESS_FORCEINLINE pointer operator->() const BOOST_NOEXCEPT
  296. { return this->get(); }
  297. //!Dereferencing operator, if it is a null offset_ptr behavior
  298. //! is undefined. Never throws.
  299. BOOST_INTERPROCESS_FORCEINLINE reference operator* () const BOOST_NOEXCEPT
  300. {
  301. pointer p = this->get();
  302. reference r = *p;
  303. return r;
  304. }
  305. //!Indexing operator.
  306. //!Never throws.
  307. BOOST_INTERPROCESS_FORCEINLINE reference operator[](difference_type idx) const BOOST_NOEXCEPT
  308. { return this->get()[idx]; }
  309. //!Assignment from pointer (saves extra conversion).
  310. //!Never throws.
  311. BOOST_INTERPROCESS_FORCEINLINE offset_ptr& operator= (pointer from) BOOST_NOEXCEPT
  312. {
  313. this->internal.m_offset =
  314. static_cast<OffsetType>(ipcdetail::offset_ptr_to_offset(from, this));
  315. return *this;
  316. }
  317. //!Assignment from other offset_ptr.
  318. //!Never throws.
  319. BOOST_INTERPROCESS_FORCEINLINE offset_ptr& operator= (const offset_ptr & ptr) BOOST_NOEXCEPT
  320. {
  321. this->internal.m_offset =
  322. static_cast<OffsetType>(ipcdetail::offset_ptr_to_offset_from_other(this, &ptr, ptr.internal.m_offset));
  323. return *this;
  324. }
  325. //!Assignment from related offset_ptr. If pointers of pointee types
  326. //! are assignable, offset_ptrs will be assignable. Never throws.
  327. template<class T2> BOOST_INTERPROCESS_FORCEINLINE
  328. #ifndef BOOST_INTERPROCESS_DOXYGEN_INVOKED
  329. typename ipcdetail::enable_if_c
  330. < ipcdetail::is_convertible<T2*, PointedType*>::value, offset_ptr&>::type
  331. #else
  332. offset_ptr&
  333. #endif
  334. operator= (const offset_ptr<T2, DifferenceType, OffsetType, OffsetAlignment> &ptr) BOOST_NOEXCEPT
  335. {
  336. this->assign(ptr, ipcdetail::bool_<ipcdetail::offset_ptr_maintains_address<T2, PointedType>::value>());
  337. return *this;
  338. }
  339. public:
  340. //!offset_ptr += difference_type.
  341. //!Never throws.
  342. BOOST_INTERPROCESS_FORCEINLINE offset_ptr &operator+= (difference_type offset) BOOST_NOEXCEPT
  343. { this->inc_offset(offset * sizeof (PointedType)); return *this; }
  344. //!offset_ptr -= difference_type.
  345. //!Never throws.
  346. BOOST_INTERPROCESS_FORCEINLINE offset_ptr &operator-= (difference_type offset) BOOST_NOEXCEPT
  347. { this->dec_offset(offset * sizeof (PointedType)); return *this; }
  348. //!++offset_ptr.
  349. //!Never throws.
  350. BOOST_INTERPROCESS_FORCEINLINE offset_ptr& operator++ (void) BOOST_NOEXCEPT
  351. { this->inc_offset(sizeof (PointedType)); return *this; }
  352. //!offset_ptr++.
  353. //!Never throws.
  354. BOOST_INTERPROCESS_FORCEINLINE offset_ptr operator++ (int) BOOST_NOEXCEPT
  355. {
  356. offset_ptr tmp(*this);
  357. this->inc_offset(sizeof (PointedType));
  358. return tmp;
  359. }
  360. //!--offset_ptr.
  361. //!Never throws.
  362. BOOST_INTERPROCESS_FORCEINLINE offset_ptr& operator-- (void) BOOST_NOEXCEPT
  363. { this->dec_offset(sizeof (PointedType)); return *this; }
  364. //!offset_ptr--.
  365. //!Never throws.
  366. BOOST_INTERPROCESS_FORCEINLINE offset_ptr operator-- (int) BOOST_NOEXCEPT
  367. {
  368. offset_ptr tmp(*this);
  369. this->dec_offset(sizeof (PointedType));
  370. return tmp;
  371. }
  372. //!safe bool conversion operator.
  373. //!Never throws.
  374. #if defined(BOOST_NO_CXX11_EXPLICIT_CONVERSION_OPERATORS)
  375. BOOST_INTERPROCESS_FORCEINLINE operator unspecified_bool_type() const BOOST_NOEXCEPT
  376. { return this->internal.m_offset != 1? &self_t::unspecified_bool_type_func : 0; }
  377. #else
  378. explicit operator bool() const BOOST_NOEXCEPT
  379. { return this->internal.m_offset != 1; }
  380. #endif
  381. //!Not operator. Not needed in theory, but improves portability.
  382. //!Never throws
  383. BOOST_INTERPROCESS_FORCEINLINE bool operator! () const BOOST_NOEXCEPT
  384. { return this->internal.m_offset == 1; }
  385. //!Compatibility with pointer_traits
  386. //!
  387. template <class U>
  388. struct rebind
  389. { typedef offset_ptr<U, DifferenceType, OffsetType, OffsetAlignment> other; };
  390. //!Compatibility with pointer_traits
  391. //!
  392. BOOST_INTERPROCESS_FORCEINLINE static offset_ptr pointer_to(reference r) BOOST_NOEXCEPT
  393. { return offset_ptr(&r); }
  394. //!difference_type + offset_ptr
  395. //!operation
  396. BOOST_INTERPROCESS_FORCEINLINE friend offset_ptr operator+(difference_type diff, offset_ptr right) BOOST_NOEXCEPT
  397. { right += diff; return right; }
  398. //!offset_ptr + difference_type
  399. //!operation
  400. BOOST_INTERPROCESS_FORCEINLINE friend offset_ptr operator+(offset_ptr left, difference_type diff) BOOST_NOEXCEPT
  401. { left += diff; return left; }
  402. //!offset_ptr - diff
  403. //!operation
  404. BOOST_INTERPROCESS_FORCEINLINE friend offset_ptr operator-(offset_ptr left, difference_type diff) BOOST_NOEXCEPT
  405. { left -= diff; return left; }
  406. //!offset_ptr - diff
  407. //!operation
  408. BOOST_INTERPROCESS_FORCEINLINE friend offset_ptr operator-(difference_type diff, offset_ptr right) BOOST_NOEXCEPT
  409. { right -= diff; return right; }
  410. //!offset_ptr - offset_ptr
  411. //!operation
  412. BOOST_INTERPROCESS_FORCEINLINE friend difference_type operator-(const offset_ptr &pt, const offset_ptr &pt2) BOOST_NOEXCEPT
  413. { return difference_type(pt.get()- pt2.get()); }
  414. //Comparison
  415. BOOST_INTERPROCESS_FORCEINLINE friend bool operator== (const offset_ptr &pt1, const offset_ptr &pt2) BOOST_NOEXCEPT
  416. { return pt1.get() == pt2.get(); }
  417. BOOST_INTERPROCESS_FORCEINLINE friend bool operator!= (const offset_ptr &pt1, const offset_ptr &pt2) BOOST_NOEXCEPT
  418. { return pt1.get() != pt2.get(); }
  419. BOOST_INTERPROCESS_FORCEINLINE friend bool operator<(const offset_ptr &pt1, const offset_ptr &pt2) BOOST_NOEXCEPT
  420. { return pt1.get() < pt2.get(); }
  421. BOOST_INTERPROCESS_FORCEINLINE friend bool operator<=(const offset_ptr &pt1, const offset_ptr &pt2) BOOST_NOEXCEPT
  422. { return pt1.get() <= pt2.get(); }
  423. BOOST_INTERPROCESS_FORCEINLINE friend bool operator>(const offset_ptr &pt1, const offset_ptr &pt2) BOOST_NOEXCEPT
  424. { return pt1.get() > pt2.get(); }
  425. BOOST_INTERPROCESS_FORCEINLINE friend bool operator>=(const offset_ptr &pt1, const offset_ptr &pt2) BOOST_NOEXCEPT
  426. { return pt1.get() >= pt2.get(); }
  427. //Comparison to raw ptr to support literal 0
  428. BOOST_INTERPROCESS_FORCEINLINE friend bool operator== (pointer pt1, const offset_ptr &pt2) BOOST_NOEXCEPT
  429. { return pt1 == pt2.get(); }
  430. BOOST_INTERPROCESS_FORCEINLINE friend bool operator!= (pointer pt1, const offset_ptr &pt2) BOOST_NOEXCEPT
  431. { return pt1 != pt2.get(); }
  432. BOOST_INTERPROCESS_FORCEINLINE friend bool operator<(pointer pt1, const offset_ptr &pt2) BOOST_NOEXCEPT
  433. { return pt1 < pt2.get(); }
  434. BOOST_INTERPROCESS_FORCEINLINE friend bool operator<=(pointer pt1, const offset_ptr &pt2) BOOST_NOEXCEPT
  435. { return pt1 <= pt2.get(); }
  436. BOOST_INTERPROCESS_FORCEINLINE friend bool operator>(pointer pt1, const offset_ptr &pt2) BOOST_NOEXCEPT
  437. { return pt1 > pt2.get(); }
  438. BOOST_INTERPROCESS_FORCEINLINE friend bool operator>=(pointer pt1, const offset_ptr &pt2) BOOST_NOEXCEPT
  439. { return pt1 >= pt2.get(); }
  440. //Comparison
  441. BOOST_INTERPROCESS_FORCEINLINE friend bool operator== (const offset_ptr &pt1, pointer pt2) BOOST_NOEXCEPT
  442. { return pt1.get() == pt2; }
  443. BOOST_INTERPROCESS_FORCEINLINE friend bool operator!= (const offset_ptr &pt1, pointer pt2) BOOST_NOEXCEPT
  444. { return pt1.get() != pt2; }
  445. BOOST_INTERPROCESS_FORCEINLINE friend bool operator<(const offset_ptr &pt1, pointer pt2) BOOST_NOEXCEPT
  446. { return pt1.get() < pt2; }
  447. BOOST_INTERPROCESS_FORCEINLINE friend bool operator<=(const offset_ptr &pt1, pointer pt2) BOOST_NOEXCEPT
  448. { return pt1.get() <= pt2; }
  449. BOOST_INTERPROCESS_FORCEINLINE friend bool operator>(const offset_ptr &pt1, pointer pt2) BOOST_NOEXCEPT
  450. { return pt1.get() > pt2; }
  451. BOOST_INTERPROCESS_FORCEINLINE friend bool operator>=(const offset_ptr &pt1, pointer pt2) BOOST_NOEXCEPT
  452. { return pt1.get() >= pt2; }
  453. BOOST_INTERPROCESS_FORCEINLINE friend void swap(offset_ptr &left, offset_ptr &right) BOOST_NOEXCEPT
  454. {
  455. pointer ptr = right.get();
  456. right = left;
  457. left = ptr;
  458. }
  459. private:
  460. template<class T2>
  461. BOOST_INTERPROCESS_FORCEINLINE void assign(const offset_ptr<T2, DifferenceType, OffsetType, OffsetAlignment> &ptr, ipcdetail::bool_<true>) BOOST_NOEXCEPT
  462. { //no need to pointer adjustment
  463. this->internal.m_offset =
  464. static_cast<OffsetType>(ipcdetail::offset_ptr_to_offset_from_other(this, &ptr, ptr.get_offset()));
  465. }
  466. template<class T2>
  467. BOOST_INTERPROCESS_FORCEINLINE void assign(const offset_ptr<T2, DifferenceType, OffsetType, OffsetAlignment> &ptr, ipcdetail::bool_<false>) BOOST_NOEXCEPT
  468. { //we must convert to raw before calculating the offset
  469. this->internal.m_offset =
  470. static_cast<OffsetType>(ipcdetail::offset_ptr_to_offset(static_cast<PointedType*>(ptr.get()), this));
  471. }
  472. #if !defined(BOOST_INTERPROCESS_DOXYGEN_INVOKED)
  473. BOOST_INTERPROCESS_FORCEINLINE void inc_offset(DifferenceType bytes) BOOST_NOEXCEPT
  474. { internal.m_offset += bytes; }
  475. BOOST_INTERPROCESS_FORCEINLINE void dec_offset(DifferenceType bytes) BOOST_NOEXCEPT
  476. { internal.m_offset -= bytes; }
  477. ipcdetail::offset_ptr_internal<OffsetType, OffsetAlignment> internal;
  478. public:
  479. BOOST_INTERPROCESS_FORCEINLINE const OffsetType &priv_offset() const BOOST_NOEXCEPT
  480. { return internal.m_offset; }
  481. BOOST_INTERPROCESS_FORCEINLINE OffsetType &priv_offset() BOOST_NOEXCEPT
  482. { return internal.m_offset; }
  483. #endif //#ifndef BOOST_INTERPROCESS_DOXYGEN_INVOKED
  484. };
  485. //!operator<<
  486. //!for offset ptr
  487. template<class E, class T, class W, class X, class Y, std::size_t Z>
  488. inline std::basic_ostream<E, T> & operator<<
  489. (std::basic_ostream<E, T> & os, offset_ptr<W, X, Y, Z> const & p)
  490. { return os << p.get_offset(); }
  491. //!operator>>
  492. //!for offset ptr
  493. template<class E, class T, class W, class X, class Y, std::size_t Z>
  494. inline std::basic_istream<E, T> & operator>>
  495. (std::basic_istream<E, T> & is, offset_ptr<W, X, Y, Z> & p)
  496. { return is >> p.get_offset(); }
  497. //!Simulation of static_cast between pointers. Never throws.
  498. template<class T1, class P1, class O1, std::size_t A1, class T2, class P2, class O2, std::size_t A2>
  499. BOOST_INTERPROCESS_FORCEINLINE boost::interprocess::offset_ptr<T1, P1, O1, A1>
  500. static_pointer_cast(const boost::interprocess::offset_ptr<T2, P2, O2, A2> & r) BOOST_NOEXCEPT
  501. {
  502. return boost::interprocess::offset_ptr<T1, P1, O1, A1>
  503. (r, boost::interprocess::ipcdetail::static_cast_tag());
  504. }
  505. //!Simulation of const_cast between pointers. Never throws.
  506. template<class T1, class P1, class O1, std::size_t A1, class T2, class P2, class O2, std::size_t A2>
  507. BOOST_INTERPROCESS_FORCEINLINE boost::interprocess::offset_ptr<T1, P1, O1, A1>
  508. const_pointer_cast(const boost::interprocess::offset_ptr<T2, P2, O2, A2> & r) BOOST_NOEXCEPT
  509. {
  510. return boost::interprocess::offset_ptr<T1, P1, O1, A1>
  511. (r, boost::interprocess::ipcdetail::const_cast_tag());
  512. }
  513. //!Simulation of dynamic_cast between pointers. Never throws.
  514. template<class T1, class P1, class O1, std::size_t A1, class T2, class P2, class O2, std::size_t A2>
  515. BOOST_INTERPROCESS_FORCEINLINE boost::interprocess::offset_ptr<T1, P1, O1, A1>
  516. dynamic_pointer_cast(const boost::interprocess::offset_ptr<T2, P2, O2, A2> & r) BOOST_NOEXCEPT
  517. {
  518. return boost::interprocess::offset_ptr<T1, P1, O1, A1>
  519. (r, boost::interprocess::ipcdetail::dynamic_cast_tag());
  520. }
  521. //!Simulation of reinterpret_cast between pointers. Never throws.
  522. template<class T1, class P1, class O1, std::size_t A1, class T2, class P2, class O2, std::size_t A2>
  523. BOOST_INTERPROCESS_FORCEINLINE boost::interprocess::offset_ptr<T1, P1, O1, A1>
  524. reinterpret_pointer_cast(const boost::interprocess::offset_ptr<T2, P2, O2, A2> & r) BOOST_NOEXCEPT
  525. {
  526. return boost::interprocess::offset_ptr<T1, P1, O1, A1>
  527. (r, boost::interprocess::ipcdetail::reinterpret_cast_tag());
  528. }
  529. } //namespace interprocess {
  530. #if !defined(BOOST_INTERPROCESS_DOXYGEN_INVOKED)
  531. ///has_trivial_destructor<> == true_type specialization for optimizations
  532. template <class T, class P, class O, std::size_t A>
  533. struct has_trivial_destructor< ::boost::interprocess::offset_ptr<T, P, O, A> >
  534. {
  535. static const bool value = true;
  536. };
  537. namespace move_detail {
  538. ///has_trivial_destructor<> == true_type specialization for optimizations
  539. template <class T, class P, class O, std::size_t A>
  540. struct is_trivially_destructible< ::boost::interprocess::offset_ptr<T, P, O, A> >
  541. {
  542. static const bool value = true;
  543. };
  544. } //namespace move_detail {
  545. namespace interprocess {
  546. //!to_raw_pointer() enables boost::mem_fn to recognize offset_ptr.
  547. //!Never throws.
  548. template <class T, class P, class O, std::size_t A>
  549. BOOST_INTERPROCESS_FORCEINLINE T * to_raw_pointer(boost::interprocess::offset_ptr<T, P, O, A> const & p) BOOST_NOEXCEPT
  550. { return ipcdetail::to_raw_pointer(p); }
  551. } //namespace interprocess
  552. #endif //#ifndef BOOST_INTERPROCESS_DOXYGEN_INVOKED
  553. } //namespace boost {
  554. #if !defined(BOOST_INTERPROCESS_DOXYGEN_INVOKED)
  555. namespace boost{
  556. //This is to support embedding a bit in the pointer
  557. //for intrusive containers, saving space
  558. namespace intrusive {
  559. //Predeclaration to avoid including header
  560. template<class VoidPointer, std::size_t N>
  561. struct max_pointer_plus_bits;
  562. template<std::size_t OffsetAlignment, class P, class O, std::size_t A>
  563. struct max_pointer_plus_bits<boost::interprocess::offset_ptr<void, P, O, A>, OffsetAlignment>
  564. {
  565. //The offset ptr can embed one bit less than the alignment since it
  566. //uses offset == 1 to store the null pointer.
  567. static const std::size_t value = ::boost::interprocess::ipcdetail::ls_zeros<OffsetAlignment>::value - 1;
  568. };
  569. //Predeclaration
  570. template<class Pointer, std::size_t NumBits>
  571. struct pointer_plus_bits;
  572. template<class T, class P, class O, std::size_t A, std::size_t NumBits>
  573. struct pointer_plus_bits<boost::interprocess::offset_ptr<T, P, O, A>, NumBits>
  574. {
  575. typedef boost::interprocess::offset_ptr<T, P, O, A> pointer;
  576. //Bits are stored in the lower bits of the pointer except the LSB,
  577. //because this bit is used to represent the null pointer.
  578. static const uintptr_t Mask = ((uintptr_t(1) << uintptr_t(NumBits)) - uintptr_t(1)) << uintptr_t(1);
  579. BOOST_STATIC_ASSERT(0 ==(Mask&1));
  580. //We must ALWAYS take argument "n" by reference as a copy of a null pointer
  581. //with a bit (e.g. offset == 3) would be incorrectly copied and interpreted as non-null.
  582. BOOST_INTERPROCESS_FORCEINLINE static pointer get_pointer(const pointer &n) BOOST_NOEXCEPT
  583. {
  584. pointer p;
  585. O const tmp_off = n.priv_offset() & O(~Mask);
  586. p.priv_offset() = boost::interprocess::ipcdetail::offset_ptr_to_offset_from_other(&p, &n, tmp_off);
  587. return p;
  588. }
  589. BOOST_INTERPROCESS_FORCEINLINE static void set_pointer(pointer &n, const pointer &p) BOOST_NOEXCEPT
  590. {
  591. BOOST_ASSERT(0 == (get_bits)(p));
  592. O const stored_bits = O(n.priv_offset() & Mask);
  593. n = p;
  594. n.priv_offset() |= stored_bits;
  595. }
  596. BOOST_INTERPROCESS_FORCEINLINE static std::size_t get_bits(const pointer &n) BOOST_NOEXCEPT
  597. {
  598. return std::size_t((n.priv_offset() & Mask) >> 1u);
  599. }
  600. BOOST_INTERPROCESS_FORCEINLINE static void set_bits(pointer &n, std::size_t const b) BOOST_NOEXCEPT
  601. {
  602. BOOST_ASSERT(b < (std::size_t(1) << NumBits));
  603. O tmp = n.priv_offset();
  604. tmp &= O(~Mask);
  605. tmp |= O(b << 1u);
  606. n.priv_offset() = tmp;
  607. }
  608. };
  609. } //namespace intrusive
  610. //Predeclaration
  611. template<class T, class U>
  612. struct pointer_to_other;
  613. //Backwards compatibility with pointer_to_other
  614. template <class PointedType, class DifferenceType, class OffsetType, std::size_t OffsetAlignment, class U>
  615. struct pointer_to_other
  616. < ::boost::interprocess::offset_ptr<PointedType, DifferenceType, OffsetType, OffsetAlignment>, U >
  617. {
  618. typedef ::boost::interprocess::offset_ptr<U, DifferenceType, OffsetType, OffsetAlignment> type;
  619. };
  620. } //namespace boost{
  621. #endif //#ifndef BOOST_INTERPROCESS_DOXYGEN_INVOKED
  622. #include <boost/interprocess/detail/config_end.hpp>
  623. #endif //#ifndef BOOST_INTERPROCESS_OFFSET_PTR_HPP