176 lines
		
	
	
		
			5.9 KiB
		
	
	
	
		
			Plaintext
		
	
	
	
	
	
		
		
			
		
	
	
			176 lines
		
	
	
		
			5.9 KiB
		
	
	
	
		
			Plaintext
		
	
	
	
	
	
|   | /*============================================================================= | ||
|  |     Copyright (c) 1998-2003 Joel de Guzman | ||
|  |     http://spirit.sourceforge.net/ | ||
|  | 
 | ||
|  |   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(BOOST_SPIRIT_RULE_HPP) | ||
|  | #define BOOST_SPIRIT_RULE_HPP | ||
|  | 
 | ||
|  | #include <boost/static_assert.hpp> | ||
|  | 
 | ||
|  | /////////////////////////////////////////////////////////////////////////////// | ||
|  | // | ||
|  | //  Spirit predefined maximum number of simultaneously usable different | ||
|  | //  scanner types. | ||
|  | // | ||
|  | //  This limit defines the maximum number of possible different scanner | ||
|  | //  types for which a specific rule<> may be used. If this isn't defined, a | ||
|  | //  rule<> may be used with one scanner type only (multiple scanner support | ||
|  | //  is disabled). | ||
|  | // | ||
|  | /////////////////////////////////////////////////////////////////////////////// | ||
|  | #if !defined(BOOST_SPIRIT_RULE_SCANNERTYPE_LIMIT) | ||
|  | #  define BOOST_SPIRIT_RULE_SCANNERTYPE_LIMIT 1 | ||
|  | #endif | ||
|  | 
 | ||
|  | //  Ensure a meaningful maximum number of simultaneously usable scanner types | ||
|  | BOOST_STATIC_ASSERT(BOOST_SPIRIT_RULE_SCANNERTYPE_LIMIT > 0); | ||
|  | 
 | ||
|  | #include <boost/scoped_ptr.hpp> | ||
|  | #include <boost/spirit/home/classic/namespace.hpp> | ||
|  | #include <boost/spirit/home/classic/core/non_terminal/impl/rule.ipp> | ||
|  | 
 | ||
|  | #if BOOST_SPIRIT_RULE_SCANNERTYPE_LIMIT > 1 | ||
|  | #  include <boost/preprocessor/enum_params.hpp> | ||
|  | #endif | ||
|  | 
 | ||
|  | /////////////////////////////////////////////////////////////////////////////// | ||
|  | namespace boost { namespace spirit { | ||
|  | 
 | ||
|  | BOOST_SPIRIT_CLASSIC_NAMESPACE_BEGIN | ||
|  | 
 | ||
|  | #if BOOST_SPIRIT_RULE_SCANNERTYPE_LIMIT > 1 | ||
|  | 
 | ||
|  |     /////////////////////////////////////////////////////////////////////////// | ||
|  |     // | ||
|  |     //  scanner_list (a fake scanner) | ||
|  |     // | ||
|  |     //      Typically, rules are tied to a specific scanner type and | ||
|  |     //      a particular rule cannot be used with anything else. Sometimes | ||
|  |     //      there's a need for rules that can accept more than one scanner | ||
|  |     //      type. The scanner_list<S0, ...SN> can be used as a template | ||
|  |     //      parameter to the rule class to specify up to the number of | ||
|  |     //      scanner types defined by the BOOST_SPIRIT_RULE_SCANNERTYPE_LIMIT | ||
|  |     //      constant. Example: | ||
|  |     // | ||
|  |     //          rule<scanner_list<ScannerT0, ScannerT1> > r; | ||
|  |     // | ||
|  |     //      *** This feature is available only to compilers that support | ||
|  |     //      partial template specialization. *** | ||
|  |     // | ||
|  |     /////////////////////////////////////////////////////////////////////////// | ||
|  |     template < | ||
|  |         BOOST_PP_ENUM_PARAMS( | ||
|  |             BOOST_SPIRIT_RULE_SCANNERTYPE_LIMIT, | ||
|  |             typename ScannerT | ||
|  |         ) | ||
|  |     > | ||
|  |     struct scanner_list : scanner_base {}; | ||
|  | 
 | ||
|  | #endif | ||
|  | 
 | ||
|  |     /////////////////////////////////////////////////////////////////////////// | ||
|  |     // | ||
|  |     //  rule class | ||
|  |     // | ||
|  |     //      The rule is a polymorphic parser that acts as a named place- | ||
|  |     //      holder capturing the behavior of an EBNF expression assigned to | ||
|  |     //      it. | ||
|  |     // | ||
|  |     //      The rule is a template class parameterized by: | ||
|  |     // | ||
|  |     //          1) scanner (scanner_t, see scanner.hpp), | ||
|  |     //          2) the rule's context (context_t, see parser_context.hpp) | ||
|  |     //          3) an arbitrary tag (tag_t, see parser_id.hpp) that allows | ||
|  |     //             a rule to be tagged for identification. | ||
|  |     // | ||
|  |     //      These template parameters may be specified in any order. The | ||
|  |     //      scanner will default to scanner<> when it is not specified. | ||
|  |     //      The context will default to parser_context when not specified. | ||
|  |     //      The tag will default to parser_address_tag when not specified. | ||
|  |     // | ||
|  |     //      The definition of the rule (its right hand side, RHS) held by | ||
|  |     //      the rule through a scoped_ptr. When a rule is seen in the RHS | ||
|  |     //      of an assignment or copy construction EBNF expression, the rule | ||
|  |     //      is held by the LHS rule by reference. | ||
|  |     // | ||
|  |     /////////////////////////////////////////////////////////////////////////// | ||
|  |     template < | ||
|  |         typename T0 = nil_t | ||
|  |       , typename T1 = nil_t | ||
|  |       , typename T2 = nil_t | ||
|  |     > | ||
|  |     class rule | ||
|  |         : public impl::rule_base< | ||
|  |             rule<T0, T1, T2> | ||
|  |           , rule<T0, T1, T2> const& | ||
|  |           , T0, T1, T2> | ||
|  |     { | ||
|  |     public: | ||
|  | 
 | ||
|  |         typedef rule<T0, T1, T2> self_t; | ||
|  |         typedef impl::rule_base< | ||
|  |             self_t | ||
|  |           , self_t const& | ||
|  |           , T0, T1, T2> | ||
|  |         base_t; | ||
|  | 
 | ||
|  |         typedef typename base_t::scanner_t scanner_t; | ||
|  |         typedef typename base_t::attr_t attr_t; | ||
|  |         typedef impl::abstract_parser<scanner_t, attr_t> abstract_parser_t; | ||
|  | 
 | ||
|  |         rule() : ptr() {} | ||
|  |         ~rule() {} | ||
|  | 
 | ||
|  |         rule(rule const& r) | ||
|  |         : ptr(new impl::concrete_parser<rule, scanner_t, attr_t>(r)) {} | ||
|  | 
 | ||
|  |         template <typename ParserT> | ||
|  |         rule(ParserT const& p) | ||
|  |         : ptr(new impl::concrete_parser<ParserT, scanner_t, attr_t>(p)) {} | ||
|  | 
 | ||
|  |         template <typename ParserT> | ||
|  |         rule& operator=(ParserT const& p) | ||
|  |         { | ||
|  |             ptr.reset(new impl::concrete_parser<ParserT, scanner_t, attr_t>(p)); | ||
|  |             return *this; | ||
|  |         } | ||
|  | 
 | ||
|  |         rule& operator=(rule const& r) | ||
|  |         { | ||
|  |             ptr.reset(new impl::concrete_parser<rule, scanner_t, attr_t>(r)); | ||
|  |             return *this; | ||
|  |         } | ||
|  | 
 | ||
|  |         rule<T0, T1, T2> | ||
|  |         copy() const | ||
|  |         { | ||
|  |             return rule<T0, T1, T2>(ptr.get() ? ptr->clone() : 0); | ||
|  |         } | ||
|  | 
 | ||
|  |     private: | ||
|  |         friend class impl::rule_base_access; | ||
|  | 
 | ||
|  |         abstract_parser_t* | ||
|  |         get() const | ||
|  |         { | ||
|  |             return ptr.get(); | ||
|  |         } | ||
|  | 
 | ||
|  |         rule(abstract_parser_t* ptr_) | ||
|  |         : ptr(ptr_) {} | ||
|  | 
 | ||
|  |         rule(abstract_parser_t const* ptr_) | ||
|  |         : ptr(ptr_) {} | ||
|  | 
 | ||
|  |         scoped_ptr<abstract_parser_t> ptr; | ||
|  |     }; | ||
|  | 
 | ||
|  | BOOST_SPIRIT_CLASSIC_NAMESPACE_END | ||
|  | 
 | ||
|  | }} // namespace BOOST_SPIRIT_CLASSIC_NS | ||
|  | 
 | ||
|  | #endif |