// Copyright (c) 2001-2010 Hartmut Kaiser // // 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_LEX_SUPPORT_FUNCTIONS_JUN_08_2009_0211PM) #define SPIRIT_LEX_SUPPORT_FUNCTIONS_JUN_08_2009_0211PM #if defined(_MSC_VER) #pragma once #endif #include #include #include #include #include #include #include /////////////////////////////////////////////////////////////////////////////// namespace boost { namespace spirit { namespace lex { /////////////////////////////////////////////////////////////////////////// // The function less() is used by the implementation of the support // function lex::less(). Its functionality is equivalent to flex' function // yyless(): it returns an iterator positioned to the nth input character // beyond the current start iterator (i.e. by assigning the return value to // the placeholder '_end' it is possible to return all but the first n // characters of the current token back to the input stream. // // This Phoenix actor is invoked whenever the function lex::less(n) is // used inside a lexer semantic action: // // lex::token_def<> identifier = "[a-zA-Z_][a-zA-Z0-9_]*"; // this->self = identifier [ _end = lex::less(4) ]; // // The example shows how to limit the length of the matched identifier to // four characters. // // Note: the function lex::less() has no effect if used on it's own, you // need to use the returned result in order to make use of its // functionality. template struct less_type { typedef mpl::true_ no_nullary; template struct result { typedef typename remove_const< typename mpl::at_c::type >::type context_type; typedef typename context_type::base_iterator_type type; }; template typename result::type eval(Env const& env) const { typename result::type it; return fusion::at_c<4>(env.args()).less(it, actor_()); } less_type(Actor const& actor) : actor_(actor) {} Actor actor_; }; // The function lex::less() is used to create a Phoenix actor allowing to // implement functionality similar to flex' function yyless(). template inline phoenix::actor::type> > less(T const& v) { typedef typename phoenix::as_actor::type actor_type; return less_type(phoenix::as_actor::convert(v)); } /////////////////////////////////////////////////////////////////////////// // The function more() is used by the implemention of the support function // lex::more(). Its functionality is equivalent to flex' function yymore(): // it tells the lexer that the next time it matches a rule, the // corresponding token should be appended onto the current token value // rather than replacing it. // // This Phoenix actor is invoked whenever the function lex::less(n) is // used inside a lexer semantic action: // // lex::token_def<> identifier = "[a-zA-Z_][a-zA-Z0-9_]*"; // this->self = identifier [ lex::more() ]; // // The example shows how prefix the next matched token with the matched // identifier. struct more_type { typedef mpl::true_ no_nullary; template struct result { typedef void type; }; template void eval(Env const& env) const { fusion::at_c<4>(env.args()).more(); } }; // The function lex::more() is used to create a Phoenix actor allowing to // implement functionality similar to flex' function yymore(). inline phoenix::actor more() { return more_type(); } /////////////////////////////////////////////////////////////////////////// template struct lookahead_type { typedef mpl::true_ no_nullary; template struct result { typedef bool type; }; template bool eval(Env const& env) const { return fusion::at_c<4>(env.args()).lookahead(actor_()); } lookahead_type(Actor const& actor) : actor_(actor) {} Actor actor_; }; template inline phoenix::actor< lookahead_type::type> > lookahead(T const& id) { typedef typename phoenix::as_actor::type actor_type; return lookahead_type(phoenix::as_actor::convert(id)); } template inline phoenix::actor< lookahead_type::type> > lookahead(token_def const& tok) { typedef typename phoenix::as_actor::type actor_type; return lookahead_type( phoenix::as_actor::convert(tok.id())); } /////////////////////////////////////////////////////////////////////////// inline BOOST_SCOPED_ENUM(pass_flags) ignore() { return pass_flags::pass_ignore; } }}} #endif