traits.hpp 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391
  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. //
  7. // Contains metafunctions char_type_of, category_of and mode_of used for
  8. // deducing the i/o category and i/o mode of a model of Filter or Device.
  9. //
  10. // Also contains several utility metafunctions, functions and macros.
  11. //
  12. #ifndef BOOST_IOSTREAMS_IO_TRAITS_HPP_INCLUDED
  13. #define BOOST_IOSTREAMS_IO_TRAITS_HPP_INCLUDED
  14. #if defined(_MSC_VER) && (_MSC_VER >= 1020)
  15. # pragma once
  16. #endif
  17. #include <iosfwd> // stream types, char_traits.
  18. #include <boost/config.hpp> // partial spec, deduced typename.
  19. #include <boost/detail/workaround.hpp>
  20. #include <boost/iostreams/categories.hpp>
  21. #include <boost/iostreams/detail/bool_trait_def.hpp>
  22. #include <boost/iostreams/detail/config/wide_streams.hpp>
  23. #include <boost/iostreams/detail/is_iterator_range.hpp>
  24. #include <boost/iostreams/detail/select.hpp>
  25. #include <boost/iostreams/detail/select_by_size.hpp>
  26. #include <boost/iostreams/detail/wrap_unwrap.hpp>
  27. #include <boost/iostreams/traits_fwd.hpp>
  28. #include <boost/mpl/bool.hpp>
  29. #include <boost/mpl/eval_if.hpp>
  30. #include <boost/mpl/identity.hpp>
  31. #include <boost/mpl/int.hpp>
  32. #include <boost/mpl/or.hpp>
  33. #if !BOOST_WORKAROUND(BOOST_MSVC, <= 1300)
  34. # include <boost/range/iterator_range.hpp>
  35. # include <boost/range/value_type.hpp>
  36. #endif // #if BOOST_WORKAROUND(BOOST_MSVC, <= 1300)
  37. #include <boost/ref.hpp>
  38. #include <boost/type_traits/is_convertible.hpp>
  39. // Must come last.
  40. #include <boost/iostreams/detail/config/disable_warnings.hpp>
  41. namespace boost { namespace iostreams {
  42. //----------Definitions of predicates for streams and stream buffers----------//
  43. #ifndef BOOST_IOSTREAMS_NO_STREAM_TEMPLATES //--------------------------------//
  44. BOOST_IOSTREAMS_BOOL_TRAIT_DEF(is_istream, std::basic_istream, 2)
  45. BOOST_IOSTREAMS_BOOL_TRAIT_DEF(is_ostream, std::basic_ostream, 2)
  46. BOOST_IOSTREAMS_BOOL_TRAIT_DEF(is_iostream, std::basic_iostream, 2)
  47. BOOST_IOSTREAMS_BOOL_TRAIT_DEF(is_streambuf, std::basic_streambuf, 2)
  48. BOOST_IOSTREAMS_BOOL_TRAIT_DEF(is_ifstream, std::basic_ifstream, 2)
  49. BOOST_IOSTREAMS_BOOL_TRAIT_DEF(is_ofstream, std::basic_ofstream, 2)
  50. BOOST_IOSTREAMS_BOOL_TRAIT_DEF(is_fstream, std::basic_fstream, 2)
  51. BOOST_IOSTREAMS_BOOL_TRAIT_DEF(is_filebuf, std::basic_filebuf, 2)
  52. BOOST_IOSTREAMS_BOOL_TRAIT_DEF(is_istringstream, std::basic_istringstream, 3)
  53. BOOST_IOSTREAMS_BOOL_TRAIT_DEF(is_ostringstream, std::basic_ostringstream, 3)
  54. BOOST_IOSTREAMS_BOOL_TRAIT_DEF(is_stringstream, std::basic_stringstream, 3)
  55. BOOST_IOSTREAMS_BOOL_TRAIT_DEF(is_stringbuf, std::basic_stringbuf, 3)
  56. #else // #ifndef BOOST_IOSTREAMS_NO_STREAM_TEMPLATES //-----------------------//
  57. BOOST_IOSTREAMS_BOOL_TRAIT_DEF(is_istream, std::istream, 0)
  58. BOOST_IOSTREAMS_BOOL_TRAIT_DEF(is_ostream, std::ostream, 0)
  59. BOOST_IOSTREAMS_BOOL_TRAIT_DEF(is_iostream, std::iostream, 0)
  60. BOOST_IOSTREAMS_BOOL_TRAIT_DEF(is_streambuf, std::streambuf, 0)
  61. #endif // #ifndef BOOST_IOSTREAMS_NO_STREAM_TEMPLATES //----------------------//
  62. template<typename T>
  63. struct is_std_io
  64. : mpl::or_< is_istream<T>, is_ostream<T>, is_streambuf<T> >
  65. { };
  66. template<typename T>
  67. struct is_std_file_device
  68. : mpl::or_<
  69. is_ifstream<T>,
  70. is_ofstream<T>,
  71. is_fstream<T>,
  72. is_filebuf<T>
  73. >
  74. { };
  75. template<typename T>
  76. struct is_std_string_device
  77. : mpl::or_<
  78. is_istringstream<T>,
  79. is_ostringstream<T>,
  80. is_stringstream<T>,
  81. is_stringbuf<T>
  82. >
  83. { };
  84. template<typename Device, typename Tr, typename Alloc>
  85. struct stream;
  86. template<typename T, typename Tr, typename Alloc, typename Mode>
  87. class stream_buffer;
  88. template< typename Mode, typename Ch, typename Tr,
  89. typename Alloc, typename Access >
  90. class filtering_stream;
  91. template< typename Mode, typename Ch, typename Tr,
  92. typename Alloc, typename Access >
  93. class wfiltering_stream;
  94. template< typename Mode, typename Ch, typename Tr,
  95. typename Alloc, typename Access >
  96. class filtering_streambuf;
  97. template< typename Mode, typename Ch, typename Tr,
  98. typename Alloc, typename Access >
  99. class filtering_wstreambuf;
  100. namespace detail {
  101. template<typename T, typename Tr>
  102. class linked_streambuf;
  103. BOOST_IOSTREAMS_BOOL_TRAIT_DEF( is_boost_stream,
  104. boost::iostreams::stream,
  105. 3 )
  106. BOOST_IOSTREAMS_BOOL_TRAIT_DEF( is_boost_stream_buffer,
  107. boost::iostreams::stream_buffer,
  108. 4 )
  109. BOOST_IOSTREAMS_BOOL_TRAIT_DEF( is_filtering_stream_impl,
  110. boost::iostreams::filtering_stream,
  111. 5 )
  112. BOOST_IOSTREAMS_BOOL_TRAIT_DEF( is_filtering_wstream_impl,
  113. boost::iostreams::wfiltering_stream,
  114. 5 )
  115. BOOST_IOSTREAMS_BOOL_TRAIT_DEF( is_filtering_streambuf_impl,
  116. boost::iostreams::filtering_streambuf,
  117. 5 )
  118. BOOST_IOSTREAMS_BOOL_TRAIT_DEF( is_filtering_wstreambuf_impl,
  119. boost::iostreams::filtering_wstreambuf,
  120. 5 )
  121. BOOST_IOSTREAMS_BOOL_TRAIT_DEF(is_linked, linked_streambuf, 2)
  122. template<typename T>
  123. struct is_filtering_stream
  124. : mpl::or_<
  125. is_filtering_stream_impl<T>,
  126. is_filtering_wstream_impl<T>
  127. >
  128. { };
  129. template<typename T>
  130. struct is_filtering_streambuf
  131. : mpl::or_<
  132. is_filtering_streambuf_impl<T>,
  133. is_filtering_wstreambuf_impl<T>
  134. >
  135. { };
  136. template<typename T>
  137. struct is_boost
  138. : mpl::or_<
  139. is_boost_stream<T>,
  140. is_boost_stream_buffer<T>,
  141. is_filtering_stream<T>,
  142. is_filtering_streambuf<T>
  143. >
  144. { };
  145. } // End namespace detail.
  146. //------------------Definitions of char_type_of-------------------------------//
  147. namespace detail {
  148. template<typename T>
  149. struct member_char_type { typedef typename T::char_type type; };
  150. } // End namespace detail.
  151. #ifndef BOOST_NO_TEMPLATE_PARTIAL_SPECIALIZATION //---------------------------//
  152. # ifndef BOOST_IOSTREAMS_NO_STREAM_TEMPLATES //-------------------------------//
  153. template<typename T>
  154. struct char_type_of
  155. : detail::member_char_type<
  156. typename detail::unwrapped_type<T>::type
  157. >
  158. { };
  159. # else // # ifndef BOOST_IOSTREAMS_NO_STREAM_TEMPLATES //---------------------//
  160. template<typename T>
  161. struct char_type_of {
  162. typedef typename detail::unwrapped_type<T>::type U;
  163. typedef typename
  164. mpl::eval_if<
  165. is_std_io<U>,
  166. mpl::identity<char>,
  167. detail::member_char_type<U>
  168. >::type type;
  169. };
  170. # endif // # ifndef BOOST_IOSTREAMS_NO_STREAM_TEMPLATES //--------------------//
  171. template<typename Iter>
  172. struct char_type_of< iterator_range<Iter> > {
  173. typedef typename iterator_value<Iter>::type type;
  174. };
  175. #else // #ifndef BOOST_NO_TEMPLATE_PARTIAL_SPECIALIZATION //------------------//
  176. template<typename T>
  177. struct char_type_of {
  178. template<typename U>
  179. struct get_value_type {
  180. #if !BOOST_WORKAROUND(BOOST_MSVC, <= 1300)
  181. typedef typename range_value<U>::type type;
  182. #endif // #if BOOST_WORKAROUND(BOOST_MSVC, <= 1300)
  183. };
  184. typedef typename
  185. mpl::eval_if<
  186. is_iterator_range<T>,
  187. get_value_type<T>,
  188. detail::member_char_type<
  189. BOOST_DEDUCED_TYPENAME detail::unwrapped_type<T>::type
  190. >
  191. >::type type;
  192. };
  193. #endif // #ifndef BOOST_NO_TEMPLATE_PARTIAL_SPECIALIZATION //-----------------//
  194. //------------------Definitions of category_of--------------------------------//
  195. namespace detail {
  196. template<typename T>
  197. struct member_category { typedef typename T::category type; };
  198. } // End namespace detail.
  199. template<typename T>
  200. struct category_of {
  201. template<typename U>
  202. struct member_category {
  203. typedef typename U::category type;
  204. };
  205. typedef typename detail::unwrapped_type<T>::type U;
  206. typedef typename
  207. mpl::eval_if<
  208. mpl::and_<
  209. is_std_io<U>,
  210. mpl::not_< detail::is_boost<U> >
  211. >,
  212. iostreams::select< // Disambiguation for Tru64
  213. is_filebuf<U>, filebuf_tag,
  214. is_ifstream<U>, ifstream_tag,
  215. is_ofstream<U>, ofstream_tag,
  216. is_fstream<U>, fstream_tag,
  217. is_stringbuf<U>, stringbuf_tag,
  218. is_istringstream<U>, istringstream_tag,
  219. is_ostringstream<U>, ostringstream_tag,
  220. is_stringstream<U>, stringstream_tag,
  221. is_streambuf<U>, generic_streambuf_tag,
  222. is_iostream<U>, generic_iostream_tag,
  223. is_istream<U>, generic_istream_tag,
  224. is_ostream<U>, generic_ostream_tag
  225. >,
  226. detail::member_category<U>
  227. >::type type;
  228. };
  229. // Partial specialization for reference wrappers
  230. #ifndef BOOST_NO_TEMPLATE_PARTIAL_SPECIALIZATION //---------------------------//
  231. template<typename T>
  232. struct category_of< reference_wrapper<T> >
  233. : category_of<T>
  234. { };
  235. #endif // #ifndef BOOST_NO_TEMPLATE_PARTIAL_SPECIALIZATION //-----------------//
  236. //------------------Definition of get_category--------------------------------//
  237. //
  238. // Returns an object of type category_of<T>::type.
  239. //
  240. template<typename T>
  241. inline typename category_of<T>::type get_category(const T&)
  242. { typedef typename category_of<T>::type category; return category(); }
  243. //------------------Definition of int_type_of---------------------------------//
  244. template<typename T>
  245. struct int_type_of {
  246. #ifndef BOOST_IOSTREAMS_NO_STREAM_TEMPLATES
  247. typedef std::char_traits<
  248. BOOST_DEDUCED_TYPENAME char_type_of<T>::type
  249. > traits_type;
  250. typedef typename traits_type::int_type type;
  251. #else
  252. typedef int type;
  253. #endif
  254. };
  255. //------------------Definition of mode_of-------------------------------------//
  256. namespace detail {
  257. template<int N> struct io_mode_impl;
  258. #define BOOST_IOSTREAMS_MODE_HELPER(tag_, id_) \
  259. case_<id_> io_mode_impl_helper(tag_); \
  260. template<> struct io_mode_impl<id_> { typedef tag_ type; }; \
  261. /**/
  262. BOOST_IOSTREAMS_MODE_HELPER(input, 1)
  263. BOOST_IOSTREAMS_MODE_HELPER(output, 2)
  264. BOOST_IOSTREAMS_MODE_HELPER(bidirectional, 3)
  265. BOOST_IOSTREAMS_MODE_HELPER(input_seekable, 4)
  266. BOOST_IOSTREAMS_MODE_HELPER(output_seekable, 5)
  267. BOOST_IOSTREAMS_MODE_HELPER(seekable, 6)
  268. BOOST_IOSTREAMS_MODE_HELPER(dual_seekable, 7)
  269. BOOST_IOSTREAMS_MODE_HELPER(bidirectional_seekable, 8)
  270. BOOST_IOSTREAMS_MODE_HELPER(dual_use, 9)
  271. #undef BOOST_IOSTREAMS_MODE_HELPER
  272. template<typename T>
  273. struct io_mode_id {
  274. typedef typename category_of<T>::type category;
  275. BOOST_SELECT_BY_SIZE(int, value, detail::io_mode_impl_helper(category()));
  276. };
  277. } // End namespace detail.
  278. template<typename T> // Borland 5.6.4 requires this circumlocution.
  279. struct mode_of : detail::io_mode_impl< detail::io_mode_id<T>::value > { };
  280. // Partial specialization for reference wrappers
  281. #ifndef BOOST_NO_TEMPLATE_PARTIAL_SPECIALIZATION //---------------------------//
  282. template<typename T>
  283. struct mode_of< reference_wrapper<T> >
  284. : mode_of<T>
  285. { };
  286. #endif // #ifndef BOOST_NO_TEMPLATE_PARTIAL_SPECIALIZATION //-----------------//
  287. //------------------Definition of is_device, is_filter and is_direct----------//
  288. namespace detail {
  289. template<typename T, typename Tag>
  290. struct has_trait_impl {
  291. typedef typename category_of<T>::type category;
  292. BOOST_STATIC_CONSTANT(bool, value = (is_convertible<category, Tag>::value));
  293. };
  294. template<typename T, typename Tag>
  295. struct has_trait
  296. : mpl::bool_<has_trait_impl<T, Tag>::value>
  297. { };
  298. } // End namespace detail.
  299. template<typename T>
  300. struct is_device : detail::has_trait<T, device_tag> { };
  301. template<typename T>
  302. struct is_filter : detail::has_trait<T, filter_tag> { };
  303. template<typename T>
  304. struct is_direct : detail::has_trait<T, direct_tag> { };
  305. //------------------Definition of BOOST_IOSTREAMS_STREAMBUF_TYPEDEFS----------//
  306. #define BOOST_IOSTREAMS_STREAMBUF_TYPEDEFS(Tr) \
  307. typedef Tr traits_type; \
  308. typedef typename traits_type::int_type int_type; \
  309. typedef typename traits_type::off_type off_type; \
  310. typedef typename traits_type::pos_type pos_type; \
  311. /**/
  312. } } // End namespaces iostreams, boost.
  313. #include <boost/iostreams/detail/config/enable_warnings.hpp>
  314. #endif // #ifndef BOOST_IOSTREAMS_IO_TRAITS_HPP_INCLUDED