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
|