Chris@16
|
1 /*=============================================================================
|
Chris@16
|
2 Boost.Wave: A Standard compliant C++ preprocessor library
|
Chris@16
|
3
|
Chris@16
|
4 http://www.boost.org/
|
Chris@16
|
5
|
Chris@16
|
6 Copyright (c) 2001-2012 Hartmut Kaiser. Distributed under the Boost
|
Chris@16
|
7 Software License, Version 1.0. (See accompanying file
|
Chris@16
|
8 LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
|
Chris@16
|
9 =============================================================================*/
|
Chris@16
|
10
|
Chris@16
|
11 #if !defined(CPP_INTLIT_GRAMMAR_HPP_2E1E70B1_F15C_4132_8554_10A231B0D91C_INCLUDED)
|
Chris@16
|
12 #define CPP_INTLIT_GRAMMAR_HPP_2E1E70B1_F15C_4132_8554_10A231B0D91C_INCLUDED
|
Chris@16
|
13
|
Chris@16
|
14 #include <boost/wave/wave_config.hpp>
|
Chris@16
|
15
|
Chris@16
|
16 #include <boost/spirit/include/classic_core.hpp>
|
Chris@16
|
17 #include <boost/spirit/include/classic_closure.hpp>
|
Chris@16
|
18 #include <boost/spirit/include/classic_assign_actor.hpp>
|
Chris@16
|
19 #include <boost/spirit/include/classic_push_back_actor.hpp>
|
Chris@16
|
20
|
Chris@16
|
21 #include <boost/spirit/include/phoenix1_operators.hpp>
|
Chris@16
|
22 #include <boost/spirit/include/phoenix1_primitives.hpp>
|
Chris@16
|
23 #include <boost/spirit/include/phoenix1_statements.hpp>
|
Chris@16
|
24
|
Chris@16
|
25 #include <boost/wave/cpp_exceptions.hpp>
|
Chris@16
|
26 #include <boost/wave/grammars/cpp_literal_grammar_gen.hpp>
|
Chris@16
|
27
|
Chris@16
|
28 #if !defined(spirit_append_actor)
|
Chris@16
|
29 #define spirit_append_actor(actor) boost::spirit::classic::push_back_a(actor)
|
Chris@16
|
30 #define spirit_assign_actor(actor) boost::spirit::classic::assign_a(actor)
|
Chris@16
|
31 #endif // !defined(spirit_append_actor)
|
Chris@16
|
32
|
Chris@16
|
33 // this must occur after all of the includes and before any code appears
|
Chris@16
|
34 #ifdef BOOST_HAS_ABI_HEADERS
|
Chris@16
|
35 #include BOOST_ABI_PREFIX
|
Chris@16
|
36 #endif
|
Chris@16
|
37
|
Chris@16
|
38 ///////////////////////////////////////////////////////////////////////////////
|
Chris@16
|
39 //
|
Chris@16
|
40 // Reusable grammar for parsing of C++ style integer literals
|
Chris@16
|
41 //
|
Chris@16
|
42 ///////////////////////////////////////////////////////////////////////////////
|
Chris@16
|
43 namespace boost {
|
Chris@16
|
44 namespace wave {
|
Chris@16
|
45 namespace grammars {
|
Chris@16
|
46
|
Chris@16
|
47 ///////////////////////////////////////////////////////////////////////////////
|
Chris@16
|
48 namespace closures {
|
Chris@16
|
49
|
Chris@16
|
50 struct intlit_closure
|
Chris@16
|
51 : boost::spirit::classic::closure<intlit_closure, uint_literal_type>
|
Chris@16
|
52 {
|
Chris@16
|
53 member1 val;
|
Chris@16
|
54 };
|
Chris@16
|
55 }
|
Chris@16
|
56
|
Chris@16
|
57 ///////////////////////////////////////////////////////////////////////////////
|
Chris@16
|
58 // define, whether the rule's should generate some debug output
|
Chris@16
|
59 #define TRACE_INTLIT_GRAMMAR \
|
Chris@16
|
60 bool(BOOST_SPIRIT_DEBUG_FLAGS_CPP & BOOST_SPIRIT_DEBUG_FLAGS_INTLIT_GRAMMAR) \
|
Chris@16
|
61 /**/
|
Chris@16
|
62
|
Chris@16
|
63 struct intlit_grammar :
|
Chris@16
|
64 boost::spirit::classic::grammar<intlit_grammar, closures::intlit_closure::context_t>
|
Chris@16
|
65 {
|
Chris@16
|
66 intlit_grammar(bool &is_unsigned_) : is_unsigned(is_unsigned_)
|
Chris@16
|
67 {
|
Chris@16
|
68 BOOST_SPIRIT_DEBUG_TRACE_GRAMMAR_NAME(*this, "intlit_grammar",
|
Chris@16
|
69 TRACE_INTLIT_GRAMMAR);
|
Chris@16
|
70 }
|
Chris@16
|
71
|
Chris@16
|
72 template <typename ScannerT>
|
Chris@16
|
73 struct definition
|
Chris@16
|
74 {
|
Chris@16
|
75 typedef boost::spirit::classic::rule<ScannerT> rule_t;
|
Chris@16
|
76
|
Chris@16
|
77 rule_t int_lit;
|
Chris@16
|
78 boost::spirit::classic::subrule<0> sub_int_lit;
|
Chris@16
|
79 boost::spirit::classic::subrule<1> oct_lit;
|
Chris@16
|
80 boost::spirit::classic::subrule<2> hex_lit;
|
Chris@16
|
81 boost::spirit::classic::subrule<3> dec_lit;
|
Chris@16
|
82
|
Chris@16
|
83 definition(intlit_grammar const &self)
|
Chris@16
|
84 {
|
Chris@16
|
85 using namespace boost::spirit::classic;
|
Chris@16
|
86 namespace phx = phoenix;
|
Chris@16
|
87
|
Chris@16
|
88
|
Chris@16
|
89 int_lit = (
|
Chris@16
|
90 sub_int_lit =
|
Chris@16
|
91 ( ch_p('0')[self.val = 0] >> (hex_lit | oct_lit)
|
Chris@16
|
92 | dec_lit
|
Chris@16
|
93 )
|
Chris@16
|
94 >> !as_lower_d[
|
Chris@16
|
95 (ch_p('u')[phx::var(self.is_unsigned) = true] || ch_p('l'))
|
Chris@16
|
96 | (ch_p('l') || ch_p('u')[phx::var(self.is_unsigned) = true])
|
Chris@16
|
97 ]
|
Chris@16
|
98 ,
|
Chris@16
|
99
|
Chris@16
|
100 hex_lit =
|
Chris@16
|
101 (ch_p('X') | ch_p('x'))
|
Chris@16
|
102 >> uint_parser<uint_literal_type, 16>()
|
Chris@16
|
103 [
|
Chris@16
|
104 self.val = phx::arg1,
|
Chris@16
|
105 phx::var(self.is_unsigned) = true
|
Chris@16
|
106 ]
|
Chris@16
|
107 ,
|
Chris@16
|
108
|
Chris@16
|
109 oct_lit =
|
Chris@16
|
110 !uint_parser<uint_literal_type, 8>()
|
Chris@16
|
111 [
|
Chris@16
|
112 self.val = phx::arg1,
|
Chris@16
|
113 phx::var(self.is_unsigned) = true
|
Chris@16
|
114 ]
|
Chris@16
|
115 ,
|
Chris@16
|
116
|
Chris@16
|
117 dec_lit =
|
Chris@16
|
118 uint_parser<uint_literal_type, 10>()
|
Chris@16
|
119 [
|
Chris@16
|
120 self.val = phx::arg1
|
Chris@16
|
121 ]
|
Chris@16
|
122 )
|
Chris@16
|
123 ;
|
Chris@16
|
124
|
Chris@16
|
125 BOOST_SPIRIT_DEBUG_TRACE_RULE(int_lit, TRACE_INTLIT_GRAMMAR);
|
Chris@16
|
126 BOOST_SPIRIT_DEBUG_TRACE_RULE(sub_int_lit, TRACE_INTLIT_GRAMMAR);
|
Chris@16
|
127 BOOST_SPIRIT_DEBUG_TRACE_RULE(hex_lit, TRACE_INTLIT_GRAMMAR);
|
Chris@16
|
128 BOOST_SPIRIT_DEBUG_TRACE_RULE(oct_lit, TRACE_INTLIT_GRAMMAR);
|
Chris@16
|
129 BOOST_SPIRIT_DEBUG_TRACE_RULE(dec_lit, TRACE_INTLIT_GRAMMAR);
|
Chris@16
|
130 }
|
Chris@16
|
131
|
Chris@16
|
132 // start rule of this grammar
|
Chris@16
|
133 rule_t const& start() const
|
Chris@16
|
134 { return int_lit; }
|
Chris@16
|
135 };
|
Chris@16
|
136
|
Chris@16
|
137 bool &is_unsigned;
|
Chris@16
|
138 };
|
Chris@16
|
139
|
Chris@16
|
140 #undef TRACE_INTLIT_GRAMMAR
|
Chris@16
|
141
|
Chris@16
|
142 ///////////////////////////////////////////////////////////////////////////////
|
Chris@16
|
143 //
|
Chris@16
|
144 // The following function is defined here, to allow the separation of
|
Chris@16
|
145 // the compilation of the intlit_grammar from the function using it.
|
Chris@16
|
146 //
|
Chris@16
|
147 ///////////////////////////////////////////////////////////////////////////////
|
Chris@16
|
148
|
Chris@16
|
149 #if BOOST_WAVE_SEPARATE_GRAMMAR_INSTANTIATION != 0
|
Chris@16
|
150 #define BOOST_WAVE_INTLITGRAMMAR_GEN_INLINE
|
Chris@16
|
151 #else
|
Chris@16
|
152 #define BOOST_WAVE_INTLITGRAMMAR_GEN_INLINE inline
|
Chris@16
|
153 #endif
|
Chris@16
|
154
|
Chris@16
|
155 template <typename TokenT>
|
Chris@16
|
156 BOOST_WAVE_INTLITGRAMMAR_GEN_INLINE
|
Chris@16
|
157 uint_literal_type
|
Chris@16
|
158 intlit_grammar_gen<TokenT>::evaluate(TokenT const &token,
|
Chris@16
|
159 bool &is_unsigned)
|
Chris@16
|
160 {
|
Chris@16
|
161 using namespace boost::spirit::classic;
|
Chris@16
|
162
|
Chris@16
|
163 intlit_grammar g(is_unsigned);
|
Chris@16
|
164 uint_literal_type result = 0;
|
Chris@16
|
165 typename TokenT::string_type const &token_val = token.get_value();
|
Chris@16
|
166 parse_info<typename TokenT::string_type::const_iterator> hit =
|
Chris@16
|
167 parse(token_val.begin(), token_val.end(), g[spirit_assign_actor(result)]);
|
Chris@16
|
168
|
Chris@16
|
169 if (!hit.hit) {
|
Chris@16
|
170 BOOST_WAVE_THROW(preprocess_exception, ill_formed_integer_literal,
|
Chris@16
|
171 token_val.c_str(), token.get_position());
|
Chris@16
|
172 }
|
Chris@16
|
173 return result;
|
Chris@16
|
174 }
|
Chris@16
|
175
|
Chris@16
|
176 #undef BOOST_WAVE_INTLITGRAMMAR_GEN_INLINE
|
Chris@16
|
177
|
Chris@16
|
178 ///////////////////////////////////////////////////////////////////////////////
|
Chris@16
|
179 } // namespace grammars
|
Chris@16
|
180 } // namespace wave
|
Chris@16
|
181 } // namespace boost
|
Chris@16
|
182
|
Chris@16
|
183 // the suffix header occurs after all of the code
|
Chris@16
|
184 #ifdef BOOST_HAS_ABI_HEADERS
|
Chris@16
|
185 #include BOOST_ABI_SUFFIX
|
Chris@16
|
186 #endif
|
Chris@16
|
187
|
Chris@16
|
188 #endif // !defined(CPP_INTLIT_GRAMMAR_HPP_2E1E70B1_F15C_4132_8554_10A231B0D91C_INCLUDED)
|