summaryrefslogtreecommitdiff
path: root/boost/spirit/home/x3/core/action.hpp
diff options
context:
space:
mode:
Diffstat (limited to 'boost/spirit/home/x3/core/action.hpp')
-rw-r--r--boost/spirit/home/x3/core/action.hpp120
1 files changed, 120 insertions, 0 deletions
diff --git a/boost/spirit/home/x3/core/action.hpp b/boost/spirit/home/x3/core/action.hpp
new file mode 100644
index 0000000000..890933fff1
--- /dev/null
+++ b/boost/spirit/home/x3/core/action.hpp
@@ -0,0 +1,120 @@
+/*=============================================================================
+ Copyright (arg) 2001-2014 Joel de Guzman
+
+ Distributed under the Boost Software License, Version 1.0. (See accompanying
+ file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
+=============================================================================*/
+#if !defined(SPIRIT_ACTION_JANUARY_07_2007_1128AM)
+#define SPIRIT_ACTION_JANUARY_07_2007_1128AM
+
+#if defined(_MSC_VER)
+#pragma once
+#endif
+
+#include <boost/spirit/home/x3/support/context.hpp>
+#include <boost/spirit/home/x3/support/traits/attribute_of.hpp>
+#include <boost/spirit/home/x3/support/traits/make_attribute.hpp>
+#include <boost/spirit/home/x3/core/call.hpp>
+#include <boost/spirit/home/x3/nonterminal/detail/transform_attribute.hpp>
+#include <boost/range/iterator_range.hpp>
+
+namespace boost { namespace spirit { namespace x3
+{
+ struct raw_attribute_type;
+ struct parse_pass_context_tag;
+
+ template <typename Context>
+ inline bool& _pass(Context const& context)
+ {
+ return x3::get<parse_pass_context_tag>(context);
+ }
+
+ template <typename Subject, typename Action>
+ struct action : unary_parser<Subject, action<Subject, Action>>
+ {
+ typedef unary_parser<Subject, action<Subject, Action>> base_type;
+ static bool const is_pass_through_unary = true;
+ static bool const has_action = true;
+
+ action(Subject const& subject, Action f)
+ : base_type(subject), f(f) {}
+
+ template <typename Iterator, typename Context, typename RuleContext, typename Attribute>
+ bool call_action(
+ Iterator& first, Iterator const& last
+ , Context const& context, RuleContext& rcontext, Attribute& attr) const
+ {
+ bool pass = true;
+ auto action_context = make_context<parse_pass_context_tag>(pass, context);
+ call(f, first, last, action_context, rcontext, attr);
+ return pass;
+ }
+
+ template <typename Iterator, typename Context
+ , typename RuleContext, typename Attribute>
+ bool parse_main(Iterator& first, Iterator const& last
+ , Context const& context, RuleContext& rcontext, Attribute& attr) const
+ {
+ Iterator save = first;
+ if (this->subject.parse(first, last, context, rcontext, attr))
+ {
+ if (call_action(first, last, context, rcontext, attr))
+ return true;
+
+ // reset iterators if semantic action failed the match
+ // retrospectively
+ first = save;
+ }
+ return false;
+ }
+
+ // attr==raw_attribute_type, action wants iterator_range (see raw.hpp)
+ template <typename Iterator, typename Context, typename RuleContext>
+ bool parse_main(Iterator& first, Iterator const& last
+ , Context const& context, RuleContext& rcontext, raw_attribute_type&) const
+ {
+ boost::iterator_range<Iterator> rng;
+ // synthesize the attribute since one is not supplied
+ return parse_main(first, last, context, rcontext, rng);
+ }
+
+ // attr==unused, action wants attribute
+ template <typename Iterator, typename Context, typename RuleContext>
+ bool parse(Iterator& first, Iterator const& last
+ , Context const& context, RuleContext& rcontext, unused_type) const
+ {
+ typedef typename
+ traits::attribute_of<action<Subject, Action>, Context>::type
+ attribute_type;
+ typedef traits::make_attribute<attribute_type, unused_type> make_attribute;
+ typedef traits::transform_attribute<
+ typename make_attribute::type, attribute_type, parser_id>
+ transform;
+
+ // synthesize the attribute since one is not supplied
+ typename make_attribute::type made_attr = make_attribute::call(unused_type());
+ typename transform::type attr = transform::pre(made_attr);
+ return parse_main(first, last, context, rcontext, attr);
+ }
+
+ // main parse function
+ template <typename Iterator, typename Context
+ , typename RuleContext, typename Attribute>
+ bool parse(Iterator& first, Iterator const& last
+ , Context const& context, RuleContext& rcontext, Attribute& attr) const
+ {
+ return parse_main(first, last, context, rcontext, attr);
+ }
+
+ Action f;
+ };
+
+ template <typename P, typename Action>
+ inline action<typename extension::as_parser<P>::value_type, Action>
+ operator/(P const& p, Action f)
+ {
+ return {as_parser(p), f};
+ }
+}}}
+
+#endif