traits.hpp 25 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759
  1. //
  2. // Copyright (c) 2000-2002
  3. // Joerg Walter, Mathias Koch
  4. //
  5. // Distributed under the Boost Software License, Version 1.0. (See
  6. // accompanying file LICENSE_1_0.txt or copy at
  7. // http://www.boost.org/LICENSE_1_0.txt)
  8. //
  9. // The authors gratefully acknowledge the support of
  10. // GeNeSys mbH & Co. KG in producing this work.
  11. //
  12. #ifndef _BOOST_UBLAS_TRAITS_
  13. #define _BOOST_UBLAS_TRAITS_
  14. #include <iterator>
  15. #include <complex>
  16. #include <boost/config/no_tr1/cmath.hpp>
  17. #include <boost/numeric/ublas/detail/config.hpp>
  18. #include <boost/numeric/ublas/detail/iterator.hpp>
  19. #include <boost/numeric/ublas/detail/returntype_deduction.hpp>
  20. #ifdef BOOST_UBLAS_USE_INTERVAL
  21. #include <boost/numeric/interval.hpp>
  22. #endif
  23. #include <boost/type_traits.hpp>
  24. #include <complex>
  25. #include <boost/typeof/typeof.hpp>
  26. #include <boost/utility/enable_if.hpp>
  27. #include <boost/type_traits/is_float.hpp>
  28. #include <boost/type_traits/is_integral.hpp>
  29. #include <boost/type_traits/is_unsigned.hpp>
  30. #include <boost/mpl/and.hpp>
  31. // anonymous namespace to avoid ADL issues
  32. namespace {
  33. template<class T> T boost_numeric_ublas_sqrt (const T& t) {
  34. using namespace std;
  35. // we'll find either std::sqrt or else another version via ADL:
  36. return sqrt (t);
  37. }
  38. template<typename T>
  39. inline typename boost::disable_if<
  40. boost::is_unsigned<T>, T >::type
  41. boost_numeric_ublas_abs (const T &t ) {
  42. using namespace std;
  43. return abs( t );
  44. }
  45. template<typename T>
  46. inline typename boost::enable_if<
  47. boost::is_unsigned<T>, T >::type
  48. boost_numeric_ublas_abs (const T &t ) {
  49. return t;
  50. }
  51. }
  52. namespace boost { namespace numeric { namespace ublas {
  53. template<typename R, typename I>
  54. typename boost::enable_if<
  55. mpl::and_<
  56. boost::is_float<R>,
  57. boost::is_integral<I>
  58. >,
  59. std::complex<R> >::type inline operator+ (I in1, std::complex<R> const& in2 ) {
  60. return R (in1) + in2;
  61. }
  62. template<typename R, typename I>
  63. typename boost::enable_if<
  64. mpl::and_<
  65. boost::is_float<R>,
  66. boost::is_integral<I>
  67. >,
  68. std::complex<R> >::type inline operator+ (std::complex<R> const& in1, I in2) {
  69. return in1 + R (in2);
  70. }
  71. template<typename R, typename I>
  72. typename boost::enable_if<
  73. mpl::and_<
  74. boost::is_float<R>,
  75. boost::is_integral<I>
  76. >,
  77. std::complex<R> >::type inline operator- (I in1, std::complex<R> const& in2) {
  78. return R (in1) - in2;
  79. }
  80. template<typename R, typename I>
  81. typename boost::enable_if<
  82. mpl::and_<
  83. boost::is_float<R>,
  84. boost::is_integral<I>
  85. >,
  86. std::complex<R> >::type inline operator- (std::complex<R> const& in1, I in2) {
  87. return in1 - R (in2);
  88. }
  89. template<typename R, typename I>
  90. typename boost::enable_if<
  91. mpl::and_<
  92. boost::is_float<R>,
  93. boost::is_integral<I>
  94. >,
  95. std::complex<R> >::type inline operator* (I in1, std::complex<R> const& in2) {
  96. return R (in1) * in2;
  97. }
  98. template<typename R, typename I>
  99. typename boost::enable_if<
  100. mpl::and_<
  101. boost::is_float<R>,
  102. boost::is_integral<I>
  103. >,
  104. std::complex<R> >::type inline operator* (std::complex<R> const& in1, I in2) {
  105. return in1 * R(in2);
  106. }
  107. template<typename R, typename I>
  108. typename boost::enable_if<
  109. mpl::and_<
  110. boost::is_float<R>,
  111. boost::is_integral<I>
  112. >,
  113. std::complex<R> >::type inline operator/ (I in1, std::complex<R> const& in2) {
  114. return R(in1) / in2;
  115. }
  116. template<typename R, typename I>
  117. typename boost::enable_if<
  118. mpl::and_<
  119. boost::is_float<R>,
  120. boost::is_integral<I>
  121. >,
  122. std::complex<R> >::type inline operator/ (std::complex<R> const& in1, I in2) {
  123. return in1 / R (in2);
  124. }
  125. // Use Joel de Guzman's return type deduction
  126. // uBLAS assumes a common return type for all binary arithmetic operators
  127. template<class X, class Y>
  128. struct promote_traits {
  129. typedef type_deduction_detail::base_result_of<X, Y> base_type;
  130. static typename base_type::x_type x;
  131. static typename base_type::y_type y;
  132. static const std::size_t size = sizeof (
  133. type_deduction_detail::test<
  134. typename base_type::x_type
  135. , typename base_type::y_type
  136. >(x + y) // Use x+y to stand of all the arithmetic actions
  137. );
  138. static const std::size_t index = (size / sizeof (char)) - 1;
  139. typedef typename mpl::at_c<
  140. typename base_type::types, index>::type id;
  141. typedef typename id::type promote_type;
  142. };
  143. // Type traits - generic numeric properties and functions
  144. template<class T>
  145. struct type_traits;
  146. // Define properties for a generic scalar type
  147. template<class T>
  148. struct scalar_traits {
  149. typedef scalar_traits<T> self_type;
  150. typedef T value_type;
  151. typedef const T &const_reference;
  152. typedef T &reference;
  153. typedef T real_type;
  154. typedef real_type precision_type; // we do not know what type has more precision then the real_type
  155. static const unsigned plus_complexity = 1;
  156. static const unsigned multiplies_complexity = 1;
  157. static
  158. BOOST_UBLAS_INLINE
  159. real_type real (const_reference t) {
  160. return t;
  161. }
  162. static
  163. BOOST_UBLAS_INLINE
  164. real_type imag (const_reference /*t*/) {
  165. return 0;
  166. }
  167. static
  168. BOOST_UBLAS_INLINE
  169. value_type conj (const_reference t) {
  170. return t;
  171. }
  172. static
  173. BOOST_UBLAS_INLINE
  174. real_type type_abs (const_reference t) {
  175. return boost_numeric_ublas_abs (t);
  176. }
  177. static
  178. BOOST_UBLAS_INLINE
  179. value_type type_sqrt (const_reference t) {
  180. // force a type conversion back to value_type for intgral types
  181. return value_type (boost_numeric_ublas_sqrt (t));
  182. }
  183. static
  184. BOOST_UBLAS_INLINE
  185. real_type norm_1 (const_reference t) {
  186. return self_type::type_abs (t);
  187. }
  188. static
  189. BOOST_UBLAS_INLINE
  190. real_type norm_2 (const_reference t) {
  191. return self_type::type_abs (t);
  192. }
  193. static
  194. BOOST_UBLAS_INLINE
  195. real_type norm_inf (const_reference t) {
  196. return self_type::type_abs (t);
  197. }
  198. static
  199. BOOST_UBLAS_INLINE
  200. bool equals (const_reference t1, const_reference t2) {
  201. return self_type::norm_inf (t1 - t2) < BOOST_UBLAS_TYPE_CHECK_EPSILON *
  202. (std::max) ((std::max) (self_type::norm_inf (t1),
  203. self_type::norm_inf (t2)),
  204. BOOST_UBLAS_TYPE_CHECK_MIN);
  205. }
  206. };
  207. // Define default type traits, assume T is a scalar type
  208. template<class T>
  209. struct type_traits : scalar_traits <T> {
  210. typedef type_traits<T> self_type;
  211. typedef T value_type;
  212. typedef const T &const_reference;
  213. typedef T &reference;
  214. typedef T real_type;
  215. typedef real_type precision_type;
  216. static const unsigned multiplies_complexity = 1;
  217. };
  218. // Define real type traits
  219. template<>
  220. struct type_traits<float> : scalar_traits<float> {
  221. typedef type_traits<float> self_type;
  222. typedef float value_type;
  223. typedef const value_type &const_reference;
  224. typedef value_type &reference;
  225. typedef value_type real_type;
  226. typedef double precision_type;
  227. };
  228. template<>
  229. struct type_traits<double> : scalar_traits<double> {
  230. typedef type_traits<double> self_type;
  231. typedef double value_type;
  232. typedef const value_type &const_reference;
  233. typedef value_type &reference;
  234. typedef value_type real_type;
  235. typedef long double precision_type;
  236. };
  237. template<>
  238. struct type_traits<long double> : scalar_traits<long double> {
  239. typedef type_traits<long double> self_type;
  240. typedef long double value_type;
  241. typedef const value_type &const_reference;
  242. typedef value_type &reference;
  243. typedef value_type real_type;
  244. typedef value_type precision_type;
  245. };
  246. // Define properties for a generic complex type
  247. template<class T>
  248. struct complex_traits {
  249. typedef complex_traits<T> self_type;
  250. typedef T value_type;
  251. typedef const T &const_reference;
  252. typedef T &reference;
  253. typedef typename T::value_type real_type;
  254. typedef real_type precision_type; // we do not know what type has more precision then the real_type
  255. static const unsigned plus_complexity = 2;
  256. static const unsigned multiplies_complexity = 6;
  257. static
  258. BOOST_UBLAS_INLINE
  259. real_type real (const_reference t) {
  260. return std::real (t);
  261. }
  262. static
  263. BOOST_UBLAS_INLINE
  264. real_type imag (const_reference t) {
  265. return std::imag (t);
  266. }
  267. static
  268. BOOST_UBLAS_INLINE
  269. value_type conj (const_reference t) {
  270. return std::conj (t);
  271. }
  272. static
  273. BOOST_UBLAS_INLINE
  274. real_type type_abs (const_reference t) {
  275. return abs (t);
  276. }
  277. static
  278. BOOST_UBLAS_INLINE
  279. value_type type_sqrt (const_reference t) {
  280. return sqrt (t);
  281. }
  282. static
  283. BOOST_UBLAS_INLINE
  284. real_type norm_1 (const_reference t) {
  285. return self_type::type_abs (t);
  286. // original computation has been replaced because a complex number should behave like a scalar type
  287. // return type_traits<real_type>::type_abs (self_type::real (t)) +
  288. // type_traits<real_type>::type_abs (self_type::imag (t));
  289. }
  290. static
  291. BOOST_UBLAS_INLINE
  292. real_type norm_2 (const_reference t) {
  293. return self_type::type_abs (t);
  294. }
  295. static
  296. BOOST_UBLAS_INLINE
  297. real_type norm_inf (const_reference t) {
  298. return self_type::type_abs (t);
  299. // original computation has been replaced because a complex number should behave like a scalar type
  300. // return (std::max) (type_traits<real_type>::type_abs (self_type::real (t)),
  301. // type_traits<real_type>::type_abs (self_type::imag (t)));
  302. }
  303. static
  304. BOOST_UBLAS_INLINE
  305. bool equals (const_reference t1, const_reference t2) {
  306. return self_type::norm_inf (t1 - t2) < BOOST_UBLAS_TYPE_CHECK_EPSILON *
  307. (std::max) ((std::max) (self_type::norm_inf (t1),
  308. self_type::norm_inf (t2)),
  309. BOOST_UBLAS_TYPE_CHECK_MIN);
  310. }
  311. };
  312. // Define complex type traits
  313. template<>
  314. struct type_traits<std::complex<float> > : complex_traits<std::complex<float> >{
  315. typedef type_traits<std::complex<float> > self_type;
  316. typedef std::complex<float> value_type;
  317. typedef const value_type &const_reference;
  318. typedef value_type &reference;
  319. typedef float real_type;
  320. typedef std::complex<double> precision_type;
  321. };
  322. template<>
  323. struct type_traits<std::complex<double> > : complex_traits<std::complex<double> >{
  324. typedef type_traits<std::complex<double> > self_type;
  325. typedef std::complex<double> value_type;
  326. typedef const value_type &const_reference;
  327. typedef value_type &reference;
  328. typedef double real_type;
  329. typedef std::complex<long double> precision_type;
  330. };
  331. template<>
  332. struct type_traits<std::complex<long double> > : complex_traits<std::complex<long double> > {
  333. typedef type_traits<std::complex<long double> > self_type;
  334. typedef std::complex<long double> value_type;
  335. typedef const value_type &const_reference;
  336. typedef value_type &reference;
  337. typedef long double real_type;
  338. typedef value_type precision_type;
  339. };
  340. #ifdef BOOST_UBLAS_USE_INTERVAL
  341. // Define scalar interval type traits
  342. template<>
  343. struct type_traits<boost::numeric::interval<float> > : scalar_traits<boost::numeric::interval<float> > {
  344. typedef type_traits<boost::numeric::interval<float> > self_type;
  345. typedef boost::numeric::interval<float> value_type;
  346. typedef const value_type &const_reference;
  347. typedef value_type &reference;
  348. typedef value_type real_type;
  349. typedef boost::numeric::interval<double> precision_type;
  350. };
  351. template<>
  352. struct type_traits<boost::numeric::interval<double> > : scalar_traits<boost::numeric::interval<double> > {
  353. typedef type_traits<boost::numeric::interval<double> > self_type;
  354. typedef boost::numeric::interval<double> value_type;
  355. typedef const value_type &const_reference;
  356. typedef value_type &reference;
  357. typedef value_type real_type;
  358. typedef boost::numeric::interval<long double> precision_type;
  359. };
  360. template<>
  361. struct type_traits<boost::numeric::interval<long double> > : scalar_traits<boost::numeric::interval<long double> > {
  362. typedef type_traits<boost::numeric::interval<long double> > self_type;
  363. typedef boost::numeric::interval<long double> value_type;
  364. typedef const value_type &const_reference;
  365. typedef value_type &reference;
  366. typedef value_type real_type;
  367. typedef value_type precision_type;
  368. };
  369. #endif
  370. // Storage tags -- hierarchical definition of storage characteristics
  371. struct unknown_storage_tag {};
  372. struct sparse_proxy_tag: public unknown_storage_tag {};
  373. struct sparse_tag: public sparse_proxy_tag {};
  374. struct packed_proxy_tag: public sparse_proxy_tag {};
  375. struct packed_tag: public packed_proxy_tag {};
  376. struct dense_proxy_tag: public packed_proxy_tag {};
  377. struct dense_tag: public dense_proxy_tag {};
  378. template<class S1, class S2>
  379. struct storage_restrict_traits {
  380. typedef S1 storage_category;
  381. };
  382. template<>
  383. struct storage_restrict_traits<sparse_tag, dense_proxy_tag> {
  384. typedef sparse_proxy_tag storage_category;
  385. };
  386. template<>
  387. struct storage_restrict_traits<sparse_tag, packed_proxy_tag> {
  388. typedef sparse_proxy_tag storage_category;
  389. };
  390. template<>
  391. struct storage_restrict_traits<sparse_tag, sparse_proxy_tag> {
  392. typedef sparse_proxy_tag storage_category;
  393. };
  394. template<>
  395. struct storage_restrict_traits<packed_tag, dense_proxy_tag> {
  396. typedef packed_proxy_tag storage_category;
  397. };
  398. template<>
  399. struct storage_restrict_traits<packed_tag, packed_proxy_tag> {
  400. typedef packed_proxy_tag storage_category;
  401. };
  402. template<>
  403. struct storage_restrict_traits<packed_tag, sparse_proxy_tag> {
  404. typedef sparse_proxy_tag storage_category;
  405. };
  406. template<>
  407. struct storage_restrict_traits<packed_proxy_tag, sparse_proxy_tag> {
  408. typedef sparse_proxy_tag storage_category;
  409. };
  410. template<>
  411. struct storage_restrict_traits<dense_tag, dense_proxy_tag> {
  412. typedef dense_proxy_tag storage_category;
  413. };
  414. template<>
  415. struct storage_restrict_traits<dense_tag, packed_proxy_tag> {
  416. typedef packed_proxy_tag storage_category;
  417. };
  418. template<>
  419. struct storage_restrict_traits<dense_tag, sparse_proxy_tag> {
  420. typedef sparse_proxy_tag storage_category;
  421. };
  422. template<>
  423. struct storage_restrict_traits<dense_proxy_tag, packed_proxy_tag> {
  424. typedef packed_proxy_tag storage_category;
  425. };
  426. template<>
  427. struct storage_restrict_traits<dense_proxy_tag, sparse_proxy_tag> {
  428. typedef sparse_proxy_tag storage_category;
  429. };
  430. // Iterator tags -- hierarchical definition of storage characteristics
  431. struct sparse_bidirectional_iterator_tag : public std::bidirectional_iterator_tag {};
  432. struct packed_random_access_iterator_tag : public std::random_access_iterator_tag {};
  433. struct dense_random_access_iterator_tag : public packed_random_access_iterator_tag {};
  434. // Thanks to Kresimir Fresl for convincing Comeau with iterator_base_traits ;-)
  435. template<class IC>
  436. struct iterator_base_traits {};
  437. template<>
  438. struct iterator_base_traits<std::forward_iterator_tag> {
  439. template<class I, class T>
  440. struct iterator_base {
  441. typedef forward_iterator_base<std::forward_iterator_tag, I, T> type;
  442. };
  443. };
  444. template<>
  445. struct iterator_base_traits<std::bidirectional_iterator_tag> {
  446. template<class I, class T>
  447. struct iterator_base {
  448. typedef bidirectional_iterator_base<std::bidirectional_iterator_tag, I, T> type;
  449. };
  450. };
  451. template<>
  452. struct iterator_base_traits<sparse_bidirectional_iterator_tag> {
  453. template<class I, class T>
  454. struct iterator_base {
  455. typedef bidirectional_iterator_base<sparse_bidirectional_iterator_tag, I, T> type;
  456. };
  457. };
  458. template<>
  459. struct iterator_base_traits<std::random_access_iterator_tag> {
  460. template<class I, class T>
  461. struct iterator_base {
  462. typedef random_access_iterator_base<std::random_access_iterator_tag, I, T> type;
  463. };
  464. };
  465. template<>
  466. struct iterator_base_traits<packed_random_access_iterator_tag> {
  467. template<class I, class T>
  468. struct iterator_base {
  469. typedef random_access_iterator_base<packed_random_access_iterator_tag, I, T> type;
  470. };
  471. };
  472. template<>
  473. struct iterator_base_traits<dense_random_access_iterator_tag> {
  474. template<class I, class T>
  475. struct iterator_base {
  476. typedef random_access_iterator_base<dense_random_access_iterator_tag, I, T> type;
  477. };
  478. };
  479. template<class I1, class I2>
  480. struct iterator_restrict_traits {
  481. typedef I1 iterator_category;
  482. };
  483. template<>
  484. struct iterator_restrict_traits<packed_random_access_iterator_tag, sparse_bidirectional_iterator_tag> {
  485. typedef sparse_bidirectional_iterator_tag iterator_category;
  486. };
  487. template<>
  488. struct iterator_restrict_traits<sparse_bidirectional_iterator_tag, packed_random_access_iterator_tag> {
  489. typedef sparse_bidirectional_iterator_tag iterator_category;
  490. };
  491. template<>
  492. struct iterator_restrict_traits<dense_random_access_iterator_tag, sparse_bidirectional_iterator_tag> {
  493. typedef sparse_bidirectional_iterator_tag iterator_category;
  494. };
  495. template<>
  496. struct iterator_restrict_traits<sparse_bidirectional_iterator_tag, dense_random_access_iterator_tag> {
  497. typedef sparse_bidirectional_iterator_tag iterator_category;
  498. };
  499. template<>
  500. struct iterator_restrict_traits<dense_random_access_iterator_tag, packed_random_access_iterator_tag> {
  501. typedef packed_random_access_iterator_tag iterator_category;
  502. };
  503. template<>
  504. struct iterator_restrict_traits<packed_random_access_iterator_tag, dense_random_access_iterator_tag> {
  505. typedef packed_random_access_iterator_tag iterator_category;
  506. };
  507. template<class I>
  508. BOOST_UBLAS_INLINE
  509. void increment (I &it, const I &it_end, typename I::difference_type compare, packed_random_access_iterator_tag) {
  510. it += (std::min) (compare, it_end - it);
  511. }
  512. template<class I>
  513. BOOST_UBLAS_INLINE
  514. void increment (I &it, const I &/* it_end */, typename I::difference_type /* compare */, sparse_bidirectional_iterator_tag) {
  515. ++ it;
  516. }
  517. template<class I>
  518. BOOST_UBLAS_INLINE
  519. void increment (I &it, const I &it_end, typename I::difference_type compare) {
  520. increment (it, it_end, compare, typename I::iterator_category ());
  521. }
  522. template<class I>
  523. BOOST_UBLAS_INLINE
  524. void increment (I &it, const I &it_end) {
  525. #if BOOST_UBLAS_TYPE_CHECK
  526. I cit (it);
  527. while (cit != it_end) {
  528. BOOST_UBLAS_CHECK (*cit == typename I::value_type/*zero*/(), internal_logic ());
  529. ++ cit;
  530. }
  531. #endif
  532. it = it_end;
  533. }
  534. namespace detail {
  535. // specialisation which define whether a type has a trivial constructor
  536. // or not. This is used by array types.
  537. template<typename T>
  538. struct has_trivial_constructor : public boost::has_trivial_constructor<T> {};
  539. template<typename T>
  540. struct has_trivial_destructor : public boost::has_trivial_destructor<T> {};
  541. template<typename FLT>
  542. struct has_trivial_constructor<std::complex<FLT> > : public has_trivial_constructor<FLT> {};
  543. template<typename FLT>
  544. struct has_trivial_destructor<std::complex<FLT> > : public has_trivial_destructor<FLT> {};
  545. }
  546. /** \brief Traits class to extract type information from a constant matrix or vector CONTAINER.
  547. *
  548. */
  549. template < class E >
  550. struct container_view_traits {
  551. /// type of indices
  552. typedef typename E::size_type size_type;
  553. /// type of differences of indices
  554. typedef typename E::difference_type difference_type;
  555. /// storage category: \c unknown_storage_tag, \c dense_tag, \c packed_tag, ...
  556. typedef typename E::storage_category storage_category;
  557. /// type of elements
  558. typedef typename E::value_type value_type;
  559. /// const reference to an element
  560. typedef typename E::const_reference const_reference;
  561. /// type used in expressions to mark a reference to this class (usually a const container_reference<const E> or the class itself)
  562. typedef typename E::const_closure_type const_closure_type;
  563. };
  564. /** \brief Traits class to extract additional type information from a mutable matrix or vector CONTAINER.
  565. *
  566. */
  567. template < class E >
  568. struct mutable_container_traits {
  569. /// reference to an element
  570. typedef typename E::reference reference;
  571. /// type used in expressions to mark a reference to this class (usually a container_reference<E> or the class itself)
  572. typedef typename E::closure_type closure_type;
  573. };
  574. /** \brief Traits class to extract type information from a matrix or vector CONTAINER.
  575. *
  576. */
  577. template < class E >
  578. struct container_traits
  579. : container_view_traits<E>, mutable_container_traits<E> {
  580. };
  581. /** \brief Traits class to extract type information from a constant MATRIX.
  582. *
  583. */
  584. template < class MATRIX >
  585. struct matrix_view_traits : container_view_traits <MATRIX> {
  586. /// orientation of the matrix, either \c row_major_tag, \c column_major_tag or \c unknown_orientation_tag
  587. typedef typename MATRIX::orientation_category orientation_category;
  588. /// row iterator for the matrix
  589. typedef typename MATRIX::const_iterator1 const_iterator1;
  590. /// column iterator for the matrix
  591. typedef typename MATRIX::const_iterator2 const_iterator2;
  592. };
  593. /** \brief Traits class to extract additional type information from a mutable MATRIX.
  594. *
  595. */
  596. template < class MATRIX >
  597. struct mutable_matrix_traits
  598. : mutable_container_traits <MATRIX> {
  599. /// row iterator for the matrix
  600. typedef typename MATRIX::iterator1 iterator1;
  601. /// column iterator for the matrix
  602. typedef typename MATRIX::iterator2 iterator2;
  603. };
  604. /** \brief Traits class to extract type information from a MATRIX.
  605. *
  606. */
  607. template < class MATRIX >
  608. struct matrix_traits
  609. : matrix_view_traits <MATRIX>, mutable_matrix_traits <MATRIX> {
  610. };
  611. /** \brief Traits class to extract type information from a VECTOR.
  612. *
  613. */
  614. template < class VECTOR >
  615. struct vector_view_traits : container_view_traits <VECTOR> {
  616. /// iterator for the VECTOR
  617. typedef typename VECTOR::const_iterator const_iterator;
  618. /// iterator pointing to the first element
  619. static
  620. const_iterator begin(const VECTOR & v) {
  621. return v.begin();
  622. }
  623. /// iterator pointing behind the last element
  624. static
  625. const_iterator end(const VECTOR & v) {
  626. return v.end();
  627. }
  628. };
  629. /** \brief Traits class to extract type information from a VECTOR.
  630. *
  631. */
  632. template < class VECTOR >
  633. struct mutable_vector_traits : mutable_container_traits <VECTOR> {
  634. /// iterator for the VECTOR
  635. typedef typename VECTOR::iterator iterator;
  636. /// iterator pointing to the first element
  637. static
  638. iterator begin(VECTOR & v) {
  639. return v.begin();
  640. }
  641. /// iterator pointing behind the last element
  642. static
  643. iterator end(VECTOR & v) {
  644. return v.end();
  645. }
  646. };
  647. /** \brief Traits class to extract type information from a VECTOR.
  648. *
  649. */
  650. template < class VECTOR >
  651. struct vector_traits
  652. : vector_view_traits <VECTOR>, mutable_vector_traits <VECTOR> {
  653. };
  654. // Note: specializations for T[N] and T[M][N] have been moved to traits/c_array.hpp
  655. }}}
  656. #endif