1 #include <iostream>
2 #include <utility>
3 #include <cstring>
4 #include <boost/mpl/identity.hpp>
5
6 namespace boost { namespace spirit { namespace x3
7 {
8 template <typename Derived>
9 struct parser
10 {
derivedboost::spirit::x3::parser11 Derived const& derived() const
12 {
13 return *static_cast<Derived const*>(this);
14 }
15 };
16
17 template <typename Char>
18 struct char_parser : parser<char_parser<Char>>
19 {
char_parserboost::spirit::x3::char_parser20 char_parser(Char ch) : ch(ch) {}
21
22 template <typename Iterator, typename Context>
parseboost::spirit::x3::char_parser23 bool parse(Iterator& first, Iterator last, Context const& ctx) const
24 {
25 if (first != last && *first == ch)
26 {
27 ++first;
28 return true;
29 }
30 return false;
31 }
32
33 Char ch;
34 };
35
36 template <typename Char>
char_(Char ch)37 inline char_parser<Char> char_(Char ch)
38 {
39 return char_parser<Char>(ch);
40 };
41
42 template <typename Left, typename Right>
43 struct sequence_parser : parser<sequence_parser<Left, Right>>
44 {
sequence_parserboost::spirit::x3::sequence_parser45 sequence_parser(Left left, Right right)
46 : left(left), right(right) {}
47
48 template <typename Iterator, typename Context>
parseboost::spirit::x3::sequence_parser49 bool parse(Iterator& first, Iterator last, Context const& ctx) const
50 {
51 return left.parse(first, last, ctx)
52 && right.parse(first, last, ctx);
53 }
54
55 Left left;
56 Right right;
57 };
58
59 template <typename Left, typename Right>
operator >>(parser<Left> const & left,parser<Right> const & right)60 inline sequence_parser<Left, Right> operator>>(
61 parser<Left> const& left, parser<Right> const& right)
62 {
63 return sequence_parser<Left, Right>(
64 left.derived(), right.derived());
65 }
66
67 template <typename Left, typename Right>
68 struct alternative_parser : parser<alternative_parser<Left, Right>>
69 {
alternative_parserboost::spirit::x3::alternative_parser70 alternative_parser(Left left, Right right)
71 : left(left), right(right) {}
72
73 template <typename Iterator, typename Context>
parseboost::spirit::x3::alternative_parser74 bool parse(Iterator& first, Iterator last, Context const& ctx) const
75 {
76 if (left.parse(first, last, ctx))
77 return true;
78 return right.parse(first, last, ctx);
79 }
80
81 Left left;
82 Right right;
83 };
84
85 template <typename Left, typename Right>
operator |(parser<Left> const & left,parser<Right> const & right)86 inline alternative_parser<Left, Right> operator|(
87 parser<Left> const& left, parser<Right> const& right)
88 {
89 return alternative_parser<Left, Right>(
90 left.derived(), right.derived());
91 }
92
93 template <typename ID, typename T, typename NextContext>
94 struct context
95 {
contextboost::spirit::x3::context96 context(T const& val, NextContext const& next_ctx)
97 : val(val), next_ctx(next_ctx) {}
98
getboost::spirit::x3::context99 T const& get(mpl::identity<ID>) const
100 {
101 return val;
102 }
103
104 template <typename Identity>
105 decltype(std::declval<NextContext>().get(Identity()))
getboost::spirit::x3::context106 get(Identity id) const
107 {
108 return next_ctx.get(id);
109 }
110
111 T const& val;
112 NextContext const& next_ctx;
113 };
114
115 struct empty_context
116 {
117 struct undefined {};
118 template <typename ID>
getboost::spirit::x3::empty_context119 undefined get(ID) const
120 {
121 return undefined();
122 }
123 };
124
125 template <typename ID, typename RHS>
126 struct rule_definition : parser<rule_definition<ID, RHS>>
127 {
rule_definitionboost::spirit::x3::rule_definition128 rule_definition(RHS rhs)
129 : rhs(rhs) {}
130
131 template <typename Iterator, typename Context>
parseboost::spirit::x3::rule_definition132 bool parse(Iterator& first, Iterator last, Context const& ctx) const
133 {
134 context<ID, RHS, Context> this_ctx(rhs, ctx);
135 return rhs.parse(first, last, this_ctx);
136 }
137
138 RHS rhs;
139 };
140
141 template <typename ID>
142 struct rule : parser<rule<ID>>
143 {
144 template <typename Derived>
145 rule_definition<ID, Derived>
operator =boost::spirit::x3::rule146 operator=(parser<Derived> const& definition) const
147 {
148 return rule_definition<ID, Derived>(definition.derived());
149 }
150
151 template <typename Iterator, typename Context>
parseboost::spirit::x3::rule152 bool parse(Iterator& first, Iterator last, Context const& ctx) const
153 {
154 return ctx.get(mpl::identity<ID>()).parse(first, last, ctx);
155 }
156 };
157
158 template <typename Iterator, typename Derived>
parse(parser<Derived> const & p,Iterator & first,Iterator last)159 inline bool parse(parser<Derived> const& p, Iterator& first, Iterator last)
160 {
161 empty_context ctx;
162 return p.derived().parse(first, last, ctx);
163 }
164
165 }}}
166
167 ///////////////////////////////////////////////////////////////////////////////
168 // test code
169
170 template <typename Parser>
test_parse(Parser const & p,char const * in)171 bool test_parse(Parser const& p, char const* in)
172 {
173 return parse(p, in, in + std::strlen(in));
174 }
175
176 namespace parser
177 {
178 using namespace boost::spirit::x3;
179
180 namespace g_definition
181 {
182 auto const x = rule<class x>();
183 auto const ax = char_('a') >> x;
184
185 auto const g =
186 x = char_('x') | ax;
187 }
188 using g_definition::g;
189 }
190
main()191 int main()
192 {
193
194 { // a non-recursive parser
195 using namespace boost::spirit::x3;
196
197 auto abc = char_('a') >> char_('b') >> char_('c');
198 std::cout << test_parse(abc, "abc") << std::endl;
199 std::cout << test_parse(abc, "abx") << std::endl;
200 std::cout << "==========================================" << std::endl;
201 }
202
203 { // a recursive rule
204 using namespace boost::spirit::x3;
205
206 auto const x = rule<class x>();
207 auto const ax = char_('a') >> x;
208 auto const start = (x = char_('x') | ax);
209
210 std::cout << test_parse(start, "x") << std::endl;
211 std::cout << test_parse(start, "ax") << std::endl;
212 std::cout << test_parse(start, "aaaaax") << std::endl;
213 std::cout << test_parse(start, "aaz") << std::endl;
214 std::cout << "==========================================" << std::endl;
215 }
216
217 { // a grammar ( gcc and clang only: see http://www.open-std.org/jtc1/sc22/wg21/docs/papers/2013/n3582.html )
218
219 using namespace boost::spirit::x3;
220 auto g = []()
221 {
222 rule<class x> x;
223 auto ax = char_('a') >> x;
224 return x = char_('x') | ax;
225
226 }();
227
228 std::cout << test_parse(g, "x") << std::endl;
229 std::cout << test_parse(g, "ax") << std::endl;
230 std::cout << test_parse(g, "aaaaax") << std::endl;
231 std::cout << test_parse(g, "aaz") << std::endl;
232 std::cout << "==========================================" << std::endl;
233 }
234
235 { // another grammar using namespaces (standard c++, see grammar g definition above in namespace parser.)
236 using parser::g;
237
238 std::cout << test_parse(g, "x") << std::endl;
239 std::cout << test_parse(g, "ax") << std::endl;
240 std::cout << test_parse(g, "aaaaax") << std::endl;
241 std::cout << test_parse(g, "aaz") << std::endl;
242 std::cout << "==========================================" << std::endl;
243 }
244
245 return 0;
246 }
247
248