chain.hpp 22 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598
  1. // (C) Copyright 2008 CodeRage, LLC (turkanis at coderage dot com)
  2. // (C) Copyright 2003-2007 Jonathan Turkanis
  3. // Distributed under the Boost Software License, Version 1.0. (See accompanying
  4. // file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt.)
  5. // See http://www.boost.org/libs/iostreams for documentation.
  6. #ifndef BOOST_IOSTREAMS_DETAIL_CHAIN_HPP_INCLUDED
  7. #define BOOST_IOSTREAMS_DETAIL_CHAIN_HPP_INCLUDED
  8. #if defined(_MSC_VER) && (_MSC_VER >= 1020)
  9. # pragma once
  10. #endif
  11. #include <boost/assert.hpp>
  12. #include <exception>
  13. #include <functional> // unary_function.
  14. #include <iterator> // advance.
  15. #include <list>
  16. #include <memory> // allocator, auto_ptr.
  17. #include <typeinfo>
  18. #include <stdexcept> // logic_error, out_of_range.
  19. #include <boost/checked_delete.hpp>
  20. #include <boost/config.hpp> // BOOST_MSVC, template friends,
  21. #include <boost/detail/workaround.hpp> // BOOST_NESTED_TEMPLATE
  22. #include <boost/iostreams/constants.hpp>
  23. #include <boost/iostreams/detail/access_control.hpp>
  24. #include <boost/iostreams/detail/char_traits.hpp>
  25. #include <boost/iostreams/detail/push.hpp>
  26. #include <boost/iostreams/detail/streambuf.hpp> // pubsync.
  27. #include <boost/iostreams/detail/wrap_unwrap.hpp>
  28. #include <boost/iostreams/device/null.hpp>
  29. #include <boost/iostreams/positioning.hpp>
  30. #include <boost/iostreams/traits.hpp> // is_filter.
  31. #include <boost/iostreams/stream_buffer.hpp>
  32. #include <boost/next_prior.hpp>
  33. #include <boost/shared_ptr.hpp>
  34. #include <boost/static_assert.hpp>
  35. #include <boost/throw_exception.hpp>
  36. #include <boost/type_traits/is_convertible.hpp>
  37. #include <boost/type.hpp>
  38. #include <boost/iostreams/detail/execute.hpp> // VC6.5 requires this
  39. #if BOOST_WORKAROUND(BOOST_MSVC, < 1310) // #include order
  40. # include <boost/mpl/int.hpp>
  41. #endif
  42. // Sometimes type_info objects must be compared by name. Borrowed from
  43. // Boost.Python and Boost.Function.
  44. #if (defined(__GNUC__) && __GNUC__ >= 3) || \
  45. defined(_AIX) || \
  46. (defined(__sgi) && defined(__host_mips)) || \
  47. (defined(linux) && defined(__INTEL_COMPILER) && defined(__ICC)) \
  48. /**/
  49. # include <cstring>
  50. # define BOOST_IOSTREAMS_COMPARE_TYPE_ID(X,Y) \
  51. (std::strcmp((X).name(),(Y).name()) == 0)
  52. #else
  53. # define BOOST_IOSTREAMS_COMPARE_TYPE_ID(X,Y) ((X)==(Y))
  54. #endif
  55. // Deprecated
  56. #define BOOST_IOSTREAMS_COMPONENT_TYPE(chain, index) \
  57. chain.component_type( index ) \
  58. /**/
  59. #if !BOOST_WORKAROUND(BOOST_MSVC, < 1310)
  60. # define BOOST_IOSTREAMS_COMPONENT(chain, index, target) \
  61. chain.component< target >( index ) \
  62. /**/
  63. #else
  64. # define BOOST_IOSTREAMS_COMPONENT(chain, index, target) \
  65. chain.component( index, ::boost::type< target >() ) \
  66. /**/
  67. #endif
  68. namespace boost { namespace iostreams {
  69. //--------------Definition of chain and wchain--------------------------------//
  70. namespace detail {
  71. template<typename Chain> class chain_client;
  72. //
  73. // Concept name: Chain.
  74. // Description: Represents a chain of stream buffers which provides access
  75. // to the first buffer in the chain and sends notifications when the
  76. // streambufs are added to or removed from chain.
  77. // Refines: Closable device with mode equal to typename Chain::mode.
  78. // Models: chain, converting_chain.
  79. // Example:
  80. //
  81. // class chain {
  82. // public:
  83. // typedef xxx chain_type;
  84. // typedef xxx client_type;
  85. // typedef xxx mode;
  86. // bool is_complete() const; // Ready for i/o.
  87. // template<typename T>
  88. // void push( const T& t, // Adds a stream buffer to
  89. // streamsize, // chain, based on t, with
  90. // streamsize ); // given buffer and putback
  91. // // buffer sizes. Pass -1 to
  92. // // request default size.
  93. // protected:
  94. // void register_client(client_type* client); // Associate client.
  95. // void notify(); // Notify client.
  96. // };
  97. //
  98. //
  99. // Description: Represents a chain of filters with an optional device at the
  100. // end.
  101. // Template parameters:
  102. // Self - A class deriving from the current instantiation of this template.
  103. // This is an example of the Curiously Recurring Template Pattern.
  104. // Ch - The character type.
  105. // Tr - The character traits type.
  106. // Alloc - The allocator type.
  107. // Mode - A mode tag.
  108. //
  109. template<typename Self, typename Ch, typename Tr, typename Alloc, typename Mode>
  110. class chain_base {
  111. public:
  112. typedef Ch char_type;
  113. BOOST_IOSTREAMS_STREAMBUF_TYPEDEFS(Tr)
  114. typedef Alloc allocator_type;
  115. typedef Mode mode;
  116. struct category
  117. : Mode,
  118. device_tag
  119. { };
  120. typedef chain_client<Self> client_type;
  121. friend class chain_client<Self>;
  122. private:
  123. typedef linked_streambuf<Ch> streambuf_type;
  124. typedef std::list<streambuf_type*> list_type;
  125. typedef chain_base<Self, Ch, Tr, Alloc, Mode> my_type;
  126. protected:
  127. chain_base() : pimpl_(new chain_impl) { }
  128. chain_base(const chain_base& rhs): pimpl_(rhs.pimpl_) { }
  129. public:
  130. // dual_use is a pseudo-mode to facilitate filter writing,
  131. // not a genuine mode.
  132. BOOST_STATIC_ASSERT((!is_convertible<mode, dual_use>::value));
  133. //----------Buffer sizing-------------------------------------------------//
  134. // Sets the size of the buffer created for the devices to be added to this
  135. // chain. Does not affect the size of the buffer for devices already
  136. // added.
  137. void set_device_buffer_size(std::streamsize n)
  138. { pimpl_->device_buffer_size_ = n; }
  139. // Sets the size of the buffer created for the filters to be added
  140. // to this chain. Does not affect the size of the buffer for filters already
  141. // added.
  142. void set_filter_buffer_size(std::streamsize n)
  143. { pimpl_->filter_buffer_size_ = n; }
  144. // Sets the size of the putback buffer for filters and devices to be added
  145. // to this chain. Does not affect the size of the buffer for filters or
  146. // devices already added.
  147. void set_pback_size(std::streamsize n)
  148. { pimpl_->pback_size_ = n; }
  149. //----------Device interface----------------------------------------------//
  150. std::streamsize read(char_type* s, std::streamsize n);
  151. std::streamsize write(const char_type* s, std::streamsize n);
  152. std::streampos seek(stream_offset off, BOOST_IOS::seekdir way);
  153. //----------Direct component access---------------------------------------//
  154. const std::type_info& component_type(int n) const
  155. {
  156. if (static_cast<size_type>(n) >= size())
  157. boost::throw_exception(std::out_of_range("bad chain offset"));
  158. return (*boost::next(list().begin(), n))->component_type();
  159. }
  160. #if !BOOST_WORKAROUND(BOOST_MSVC, < 1310)
  161. // Deprecated.
  162. template<int N>
  163. const std::type_info& component_type() const { return component_type(N); }
  164. template<typename T>
  165. T* component(int n) const { return component(n, boost::type<T>()); }
  166. // Deprecated.
  167. template<int N, typename T>
  168. T* component() const { return component<T>(N); }
  169. #endif
  170. #if !BOOST_WORKAROUND(BOOST_MSVC, < 1310)
  171. private:
  172. #endif
  173. template<typename T>
  174. T* component(int n, boost::type<T>) const
  175. {
  176. if (static_cast<size_type>(n) >= size())
  177. boost::throw_exception(std::out_of_range("bad chain offset"));
  178. streambuf_type* link = *boost::next(list().begin(), n);
  179. if (BOOST_IOSTREAMS_COMPARE_TYPE_ID(link->component_type(), typeid(T)))
  180. return static_cast<T*>(link->component_impl());
  181. else
  182. return 0;
  183. }
  184. public:
  185. //----------Container-like interface--------------------------------------//
  186. typedef typename list_type::size_type size_type;
  187. streambuf_type& front() { return *list().front(); }
  188. BOOST_IOSTREAMS_DEFINE_PUSH(push, mode, char_type, push_impl)
  189. void pop();
  190. bool empty() const { return list().empty(); }
  191. size_type size() const { return list().size(); }
  192. void reset();
  193. //----------Additional i/o functions--------------------------------------//
  194. // Returns true if this chain is non-empty and its final link
  195. // is a source or sink, i.e., if it is ready to perform i/o.
  196. bool is_complete() const;
  197. bool auto_close() const;
  198. void set_auto_close(bool close);
  199. bool sync() { return front().BOOST_IOSTREAMS_PUBSYNC() != -1; }
  200. bool strict_sync();
  201. private:
  202. template<typename T>
  203. void push_impl(const T& t, std::streamsize buffer_size = -1,
  204. std::streamsize pback_size = -1)
  205. {
  206. typedef typename iostreams::category_of<T>::type category;
  207. typedef typename unwrap_ios<T>::type component_type;
  208. typedef stream_buffer<
  209. component_type,
  210. BOOST_IOSTREAMS_CHAR_TRAITS(char_type),
  211. Alloc, Mode
  212. > streambuf_t;
  213. typedef typename list_type::iterator iterator;
  214. BOOST_STATIC_ASSERT((is_convertible<category, Mode>::value));
  215. if (is_complete())
  216. boost::throw_exception(std::logic_error("chain complete"));
  217. streambuf_type* prev = !empty() ? list().back() : 0;
  218. buffer_size =
  219. buffer_size != -1 ?
  220. buffer_size :
  221. iostreams::optimal_buffer_size(t);
  222. pback_size =
  223. pback_size != -1 ?
  224. pback_size :
  225. pimpl_->pback_size_;
  226. std::auto_ptr<streambuf_t>
  227. buf(new streambuf_t(t, buffer_size, pback_size));
  228. list().push_back(buf.get());
  229. buf.release();
  230. if (is_device<component_type>::value) {
  231. pimpl_->flags_ |= f_complete | f_open;
  232. for ( iterator first = list().begin(),
  233. last = list().end();
  234. first != last;
  235. ++first )
  236. {
  237. (*first)->set_needs_close();
  238. }
  239. }
  240. if (prev) prev->set_next(list().back());
  241. notify();
  242. }
  243. list_type& list() { return pimpl_->links_; }
  244. const list_type& list() const { return pimpl_->links_; }
  245. void register_client(client_type* client) { pimpl_->client_ = client; }
  246. void notify() { if (pimpl_->client_) pimpl_->client_->notify(); }
  247. //----------Nested classes------------------------------------------------//
  248. static void close(streambuf_type* b, BOOST_IOS::openmode m)
  249. {
  250. if (m == BOOST_IOS::out && is_convertible<Mode, output>::value)
  251. b->BOOST_IOSTREAMS_PUBSYNC();
  252. b->close(m);
  253. }
  254. static void set_next(streambuf_type* b, streambuf_type* next)
  255. { b->set_next(next); }
  256. static void set_auto_close(streambuf_type* b, bool close)
  257. { b->set_auto_close(close); }
  258. struct closer : public std::unary_function<streambuf_type*, void> {
  259. closer(BOOST_IOS::openmode m) : mode_(m) { }
  260. void operator() (streambuf_type* b)
  261. {
  262. close(b, mode_);
  263. }
  264. BOOST_IOS::openmode mode_;
  265. };
  266. friend struct closer;
  267. enum flags {
  268. f_complete = 1,
  269. f_open = 2,
  270. f_auto_close = 4
  271. };
  272. struct chain_impl {
  273. chain_impl()
  274. : client_(0), device_buffer_size_(default_device_buffer_size),
  275. filter_buffer_size_(default_filter_buffer_size),
  276. pback_size_(default_pback_buffer_size),
  277. flags_(f_auto_close)
  278. { }
  279. ~chain_impl()
  280. {
  281. try { close(); } catch (...) { }
  282. try { reset(); } catch (...) { }
  283. }
  284. void close()
  285. {
  286. if ((flags_ & f_open) != 0) {
  287. flags_ &= ~f_open;
  288. stream_buffer< basic_null_device<Ch, Mode> > null;
  289. if ((flags_ & f_complete) == 0) {
  290. null.open(basic_null_device<Ch, Mode>());
  291. set_next(links_.back(), &null);
  292. }
  293. links_.front()->BOOST_IOSTREAMS_PUBSYNC();
  294. try {
  295. boost::iostreams::detail::execute_foreach(
  296. links_.rbegin(), links_.rend(),
  297. closer(BOOST_IOS::in)
  298. );
  299. } catch (...) {
  300. try {
  301. boost::iostreams::detail::execute_foreach(
  302. links_.begin(), links_.end(),
  303. closer(BOOST_IOS::out)
  304. );
  305. } catch (...) { }
  306. throw;
  307. }
  308. boost::iostreams::detail::execute_foreach(
  309. links_.begin(), links_.end(),
  310. closer(BOOST_IOS::out)
  311. );
  312. }
  313. }
  314. void reset()
  315. {
  316. typedef typename list_type::iterator iterator;
  317. for ( iterator first = links_.begin(),
  318. last = links_.end();
  319. first != last;
  320. ++first )
  321. {
  322. if ( (flags_ & f_complete) == 0 ||
  323. (flags_ & f_auto_close) == 0 )
  324. {
  325. set_auto_close(*first, false);
  326. }
  327. streambuf_type* buf = 0;
  328. std::swap(buf, *first);
  329. delete buf;
  330. }
  331. links_.clear();
  332. flags_ &= ~f_complete;
  333. flags_ &= ~f_open;
  334. }
  335. list_type links_;
  336. client_type* client_;
  337. std::streamsize device_buffer_size_,
  338. filter_buffer_size_,
  339. pback_size_;
  340. int flags_;
  341. };
  342. friend struct chain_impl;
  343. //----------Member data---------------------------------------------------//
  344. private:
  345. shared_ptr<chain_impl> pimpl_;
  346. };
  347. } // End namespace detail.
  348. //
  349. // Macro: BOOST_IOSTREAMS_DECL_CHAIN(name, category)
  350. // Description: Defines a template derived from chain_base appropriate for a
  351. // particular i/o category. The template has the following parameters:
  352. // Ch - The character type.
  353. // Tr - The character traits type.
  354. // Alloc - The allocator type.
  355. // Macro parameters:
  356. // name_ - The name of the template to be defined.
  357. // category_ - The i/o category of the template to be defined.
  358. //
  359. #define BOOST_IOSTREAMS_DECL_CHAIN(name_, default_char_) \
  360. template< typename Mode, typename Ch = default_char_, \
  361. typename Tr = BOOST_IOSTREAMS_CHAR_TRAITS(Ch), \
  362. typename Alloc = std::allocator<Ch> > \
  363. class name_ : public boost::iostreams::detail::chain_base< \
  364. name_<Mode, Ch, Tr, Alloc>, \
  365. Ch, Tr, Alloc, Mode \
  366. > \
  367. { \
  368. public: \
  369. struct category : device_tag, Mode { }; \
  370. typedef Mode mode; \
  371. private: \
  372. typedef boost::iostreams::detail::chain_base< \
  373. name_<Mode, Ch, Tr, Alloc>, \
  374. Ch, Tr, Alloc, Mode \
  375. > base_type; \
  376. public: \
  377. typedef Ch char_type; \
  378. typedef Tr traits_type; \
  379. typedef typename traits_type::int_type int_type; \
  380. typedef typename traits_type::off_type off_type; \
  381. name_() { } \
  382. name_(const name_& rhs) : base_type(rhs) { } \
  383. name_& operator=(const name_& rhs) \
  384. { base_type::operator=(rhs); return *this; } \
  385. }; \
  386. /**/
  387. BOOST_IOSTREAMS_DECL_CHAIN(chain, char)
  388. BOOST_IOSTREAMS_DECL_CHAIN(wchain, wchar_t)
  389. #undef BOOST_IOSTREAMS_DECL_CHAIN
  390. //--------------Definition of chain_client------------------------------------//
  391. namespace detail {
  392. //
  393. // Template name: chain_client
  394. // Description: Class whose instances provide access to an underlying chain
  395. // using an interface similar to the chains.
  396. // Subclasses: the various stream and stream buffer templates.
  397. //
  398. template<typename Chain>
  399. class chain_client {
  400. public:
  401. typedef Chain chain_type;
  402. typedef typename chain_type::char_type char_type;
  403. typedef typename chain_type::traits_type traits_type;
  404. typedef typename chain_type::size_type size_type;
  405. typedef typename chain_type::mode mode;
  406. chain_client(chain_type* chn = 0) : chain_(chn ) { }
  407. chain_client(chain_client* client) : chain_(client->chain_) { }
  408. virtual ~chain_client() { }
  409. const std::type_info& component_type(int n) const
  410. { return chain_->component_type(n); }
  411. #if !BOOST_WORKAROUND(BOOST_MSVC, < 1310)
  412. // Deprecated.
  413. template<int N>
  414. const std::type_info& component_type() const
  415. { return chain_->BOOST_NESTED_TEMPLATE component_type<N>(); }
  416. template<typename T>
  417. T* component(int n) const
  418. { return chain_->BOOST_NESTED_TEMPLATE component<T>(n); }
  419. // Deprecated.
  420. template<int N, typename T>
  421. T* component() const
  422. { return chain_->BOOST_NESTED_TEMPLATE component<N, T>(); }
  423. #else
  424. template<typename T>
  425. T* component(int n, boost::type<T> t) const
  426. { return chain_->component(n, t); }
  427. #endif
  428. bool is_complete() const { return chain_->is_complete(); }
  429. bool auto_close() const { return chain_->auto_close(); }
  430. void set_auto_close(bool close) { chain_->set_auto_close(close); }
  431. bool strict_sync() { return chain_->strict_sync(); }
  432. void set_device_buffer_size(std::streamsize n)
  433. { chain_->set_device_buffer_size(n); }
  434. void set_filter_buffer_size(std::streamsize n)
  435. { chain_->set_filter_buffer_size(n); }
  436. void set_pback_size(std::streamsize n) { chain_->set_pback_size(n); }
  437. BOOST_IOSTREAMS_DEFINE_PUSH(push, mode, char_type, push_impl)
  438. void pop() { chain_->pop(); }
  439. bool empty() const { return chain_->empty(); }
  440. size_type size() { return chain_->size(); }
  441. void reset() { chain_->reset(); }
  442. // Returns a copy of the underlying chain.
  443. chain_type filters() { return *chain_; }
  444. chain_type filters() const { return *chain_; }
  445. protected:
  446. template<typename T>
  447. void push_impl(const T& t BOOST_IOSTREAMS_PUSH_PARAMS())
  448. { chain_->push(t BOOST_IOSTREAMS_PUSH_ARGS()); }
  449. chain_type& ref() { return *chain_; }
  450. void set_chain(chain_type* c)
  451. { chain_ = c; chain_->register_client(this); }
  452. #if !defined(BOOST_NO_MEMBER_TEMPLATE_FRIENDS) && \
  453. (!BOOST_WORKAROUND(__BORLANDC__, < 0x600))
  454. template<typename S, typename C, typename T, typename A, typename M>
  455. friend class chain_base;
  456. #else
  457. public:
  458. #endif
  459. virtual void notify() { }
  460. private:
  461. chain_type* chain_;
  462. };
  463. //--------------Implementation of chain_base----------------------------------//
  464. template<typename Self, typename Ch, typename Tr, typename Alloc, typename Mode>
  465. inline std::streamsize chain_base<Self, Ch, Tr, Alloc, Mode>::read
  466. (char_type* s, std::streamsize n)
  467. { return iostreams::read(*list().front(), s, n); }
  468. template<typename Self, typename Ch, typename Tr, typename Alloc, typename Mode>
  469. inline std::streamsize chain_base<Self, Ch, Tr, Alloc, Mode>::write
  470. (const char_type* s, std::streamsize n)
  471. { return iostreams::write(*list().front(), s, n); }
  472. template<typename Self, typename Ch, typename Tr, typename Alloc, typename Mode>
  473. inline std::streampos chain_base<Self, Ch, Tr, Alloc, Mode>::seek
  474. (stream_offset off, BOOST_IOS::seekdir way)
  475. { return iostreams::seek(*list().front(), off, way); }
  476. template<typename Self, typename Ch, typename Tr, typename Alloc, typename Mode>
  477. void chain_base<Self, Ch, Tr, Alloc, Mode>::reset()
  478. {
  479. using namespace std;
  480. pimpl_->close();
  481. pimpl_->reset();
  482. }
  483. template<typename Self, typename Ch, typename Tr, typename Alloc, typename Mode>
  484. bool chain_base<Self, Ch, Tr, Alloc, Mode>::is_complete() const
  485. {
  486. return (pimpl_->flags_ & f_complete) != 0;
  487. }
  488. template<typename Self, typename Ch, typename Tr, typename Alloc, typename Mode>
  489. bool chain_base<Self, Ch, Tr, Alloc, Mode>::auto_close() const
  490. {
  491. return (pimpl_->flags_ & f_auto_close) != 0;
  492. }
  493. template<typename Self, typename Ch, typename Tr, typename Alloc, typename Mode>
  494. void chain_base<Self, Ch, Tr, Alloc, Mode>::set_auto_close(bool close)
  495. {
  496. pimpl_->flags_ =
  497. (pimpl_->flags_ & ~f_auto_close) |
  498. (close ? f_auto_close : 0);
  499. }
  500. template<typename Self, typename Ch, typename Tr, typename Alloc, typename Mode>
  501. bool chain_base<Self, Ch, Tr, Alloc, Mode>::strict_sync()
  502. {
  503. typedef typename list_type::iterator iterator;
  504. bool result = true;
  505. for ( iterator first = list().begin(),
  506. last = list().end();
  507. first != last;
  508. ++first )
  509. {
  510. bool s = (*first)->strict_sync();
  511. result = result && s;
  512. }
  513. return result;
  514. }
  515. template<typename Self, typename Ch, typename Tr, typename Alloc, typename Mode>
  516. void chain_base<Self, Ch, Tr, Alloc, Mode>::pop()
  517. {
  518. BOOST_ASSERT(!empty());
  519. if (auto_close())
  520. pimpl_->close();
  521. streambuf_type* buf = 0;
  522. std::swap(buf, list().back());
  523. buf->set_auto_close(false);
  524. buf->set_next(0);
  525. delete buf;
  526. list().pop_back();
  527. pimpl_->flags_ &= ~f_complete;
  528. if (auto_close() || list().empty())
  529. pimpl_->flags_ &= ~f_open;
  530. }
  531. } // End namespace detail.
  532. } } // End namespaces iostreams, boost.
  533. #endif // #ifndef BOOST_IOSTREAMS_DETAIL_CHAIN_HPP_INCLUDED