select.ipp 4.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122
  1. /*=============================================================================
  2. Copyright (c) 2003 Hartmut Kaiser
  3. http://spirit.sourceforge.net/
  4. Use, modification and distribution is subject to the Boost Software
  5. License, Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at
  6. http://www.boost.org/LICENSE_1_0.txt)
  7. =============================================================================*/
  8. #ifndef BOOST_SPIRIT_SELECT_IPP
  9. #define BOOST_SPIRIT_SELECT_IPP
  10. #include <boost/spirit/home/classic/core/parser.hpp>
  11. #include <boost/spirit/home/classic/core/composite/composite.hpp>
  12. #include <boost/spirit/home/classic/meta/as_parser.hpp>
  13. ///////////////////////////////////////////////////////////////////////////////
  14. namespace boost { namespace spirit {
  15. BOOST_SPIRIT_CLASSIC_NAMESPACE_BEGIN
  16. ///////////////////////////////////////////////////////////////////////////////
  17. namespace impl {
  18. ///////////////////////////////////////////////////////////////////////////////
  19. template <typename ParserT>
  20. struct as_embedded_parser : public as_parser<ParserT>
  21. {
  22. typedef typename as_parser<ParserT>::type::derived_t::embed_t type;
  23. };
  24. ///////////////////////////////////////////////////////////////////////////////
  25. // no implementation here to catch unknown BehaviourT template arguments
  26. template <typename ResultT, typename BehaviourT>
  27. struct select_match_gen;
  28. // implementation for the select_default_no_fail behaviour
  29. template <typename ResultT>
  30. struct select_match_gen<ResultT, select_default_no_fail> {
  31. template <typename ScannerT>
  32. static ResultT
  33. do_ (ScannerT const &scan)
  34. {
  35. return scan.create_match(0, -1, scan.first, scan.first);
  36. }
  37. };
  38. // implementation for the select_default_fail behaviour
  39. template <typename ResultT>
  40. struct select_match_gen<ResultT, select_default_fail> {
  41. template <typename ScannerT>
  42. static ResultT
  43. do_ (ScannerT const &scan)
  44. {
  45. return scan.no_match();
  46. }
  47. };
  48. ///////////////////////////////////////////////////////////////////////////////
  49. template <int N, typename ResultT, typename TupleT, typename BehaviourT>
  50. struct parse_tuple_element {
  51. BOOST_STATIC_CONSTANT(int, index = (TupleT::length - N));
  52. template <typename ScannerT>
  53. static ResultT
  54. do_(TupleT const &t, ScannerT const &scan)
  55. {
  56. typedef typename ::phoenix::tuple_element<index, TupleT>::type parser_t;
  57. typedef typename ScannerT::iterator_t iterator_t;
  58. typedef typename parser_result<parser_t, ScannerT>::type result_t;
  59. iterator_t save(scan.first);
  60. ::phoenix::tuple_index<index> const idx;
  61. result_t result(t[idx].parse(scan));
  62. if (result) {
  63. return scan.create_match(result.length(), TupleT::length - N,
  64. save, scan.first);
  65. }
  66. scan.first = save; // reset the input stream
  67. return parse_tuple_element<N-1, ResultT, TupleT, BehaviourT>::
  68. do_(t, scan);
  69. }
  70. };
  71. template <typename ResultT, typename TupleT, typename BehaviourT>
  72. struct parse_tuple_element<1, ResultT, TupleT, BehaviourT> {
  73. BOOST_STATIC_CONSTANT(int, index = (TupleT::length - 1));
  74. template <typename ScannerT>
  75. static ResultT
  76. do_(TupleT const &t, ScannerT const &scan)
  77. {
  78. typedef typename ::phoenix::tuple_element<index, TupleT>::type parser_t;
  79. typedef typename ScannerT::iterator_t iterator_t;
  80. typedef typename parser_result<parser_t, ScannerT>::type result_t;
  81. iterator_t save(scan.first);
  82. ::phoenix::tuple_index<index> const idx;
  83. result_t result(t[idx].parse(scan));
  84. if (result) {
  85. return scan.create_match(result.length(), TupleT::length - 1,
  86. save, scan.first);
  87. }
  88. scan.first = save; // reset the input stream
  89. return select_match_gen<ResultT, BehaviourT>::do_(scan);
  90. }
  91. };
  92. ///////////////////////////////////////////////////////////////////////////////
  93. } // namespace impl
  94. BOOST_SPIRIT_CLASSIC_NAMESPACE_END
  95. }} // namespace boost::spirit
  96. #endif // BOOST_SPIRIT_SELECT_IPP