select.ipp 4.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120
  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. result_t result(t[::phoenix::tuple_index<index>()].parse(scan));
  61. if (result) {
  62. return scan.create_match(result.length(), TupleT::length - N,
  63. save, scan.first);
  64. }
  65. scan.first = save; // reset the input stream
  66. return parse_tuple_element<N-1, ResultT, TupleT, BehaviourT>::
  67. do_(t, scan);
  68. }
  69. };
  70. template <typename ResultT, typename TupleT, typename BehaviourT>
  71. struct parse_tuple_element<1, ResultT, TupleT, BehaviourT> {
  72. BOOST_STATIC_CONSTANT(int, index = (TupleT::length - 1));
  73. template <typename ScannerT>
  74. static ResultT
  75. do_(TupleT const &t, ScannerT const &scan)
  76. {
  77. typedef typename ::phoenix::tuple_element<index, TupleT>::type parser_t;
  78. typedef typename ScannerT::iterator_t iterator_t;
  79. typedef typename parser_result<parser_t, ScannerT>::type result_t;
  80. iterator_t save(scan.first);
  81. result_t result(t[::phoenix::tuple_index<index>()].parse(scan));
  82. if (result) {
  83. return scan.create_match(result.length(), TupleT::length - 1,
  84. save, scan.first);
  85. }
  86. scan.first = save; // reset the input stream
  87. return select_match_gen<ResultT, BehaviourT>::do_(scan);
  88. }
  89. };
  90. ///////////////////////////////////////////////////////////////////////////////
  91. } // namespace impl
  92. BOOST_SPIRIT_CLASSIC_NAMESPACE_END
  93. }} // namespace boost::spirit
  94. #endif // BOOST_SPIRIT_SELECT_IPP