annotate generator/parser/rpp/pp-macro-expander.h @ 52:09a0f1d048f2

update parser to that from jambi 4.5, attemt to fix building with gcc 4.4
author eldar
date Mon, 18 May 2009 15:23:28 +0000
parents e78566595089
children
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
rev   line source
1
e78566595089 initial import
mandel
parents:
diff changeset
1 /****************************************************************************
e78566595089 initial import
mandel
parents:
diff changeset
2 **
52
09a0f1d048f2 update parser to that from jambi 4.5, attemt to fix building with gcc 4.4
eldar
parents: 1
diff changeset
3 ** Copyright (C) 1992-2009 Nokia. All rights reserved.
1
e78566595089 initial import
mandel
parents:
diff changeset
4 ** Copyright 2005 Roberto Raggi <roberto@kdevelop.org>
e78566595089 initial import
mandel
parents:
diff changeset
5 **
e78566595089 initial import
mandel
parents:
diff changeset
6 ** This file is part of Qt Jambi.
e78566595089 initial import
mandel
parents:
diff changeset
7 **
52
09a0f1d048f2 update parser to that from jambi 4.5, attemt to fix building with gcc 4.4
eldar
parents: 1
diff changeset
8 ** Commercial Usage
09a0f1d048f2 update parser to that from jambi 4.5, attemt to fix building with gcc 4.4
eldar
parents: 1
diff changeset
9 Licensees holding valid Qt Commercial licenses may use this file in
09a0f1d048f2 update parser to that from jambi 4.5, attemt to fix building with gcc 4.4
eldar
parents: 1
diff changeset
10 accordance with the Qt Commercial License Agreement provided with the
09a0f1d048f2 update parser to that from jambi 4.5, attemt to fix building with gcc 4.4
eldar
parents: 1
diff changeset
11 Software or, alternatively, in accordance with the terms contained in
09a0f1d048f2 update parser to that from jambi 4.5, attemt to fix building with gcc 4.4
eldar
parents: 1
diff changeset
12 a written agreement between you and Nokia.
09a0f1d048f2 update parser to that from jambi 4.5, attemt to fix building with gcc 4.4
eldar
parents: 1
diff changeset
13
09a0f1d048f2 update parser to that from jambi 4.5, attemt to fix building with gcc 4.4
eldar
parents: 1
diff changeset
14 GNU Lesser General Public License Usage
09a0f1d048f2 update parser to that from jambi 4.5, attemt to fix building with gcc 4.4
eldar
parents: 1
diff changeset
15 Alternatively, this file may be used under the terms of the GNU Lesser
09a0f1d048f2 update parser to that from jambi 4.5, attemt to fix building with gcc 4.4
eldar
parents: 1
diff changeset
16 General Public License version 2.1 as published by the Free Software
09a0f1d048f2 update parser to that from jambi 4.5, attemt to fix building with gcc 4.4
eldar
parents: 1
diff changeset
17 Foundation and appearing in the file LICENSE.LGPL included in the
09a0f1d048f2 update parser to that from jambi 4.5, attemt to fix building with gcc 4.4
eldar
parents: 1
diff changeset
18 packaging of this file. Please review the following information to
09a0f1d048f2 update parser to that from jambi 4.5, attemt to fix building with gcc 4.4
eldar
parents: 1
diff changeset
19 ensure the GNU Lesser General Public License version 2.1 requirements
09a0f1d048f2 update parser to that from jambi 4.5, attemt to fix building with gcc 4.4
eldar
parents: 1
diff changeset
20 will be met: http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html.
09a0f1d048f2 update parser to that from jambi 4.5, attemt to fix building with gcc 4.4
eldar
parents: 1
diff changeset
21
09a0f1d048f2 update parser to that from jambi 4.5, attemt to fix building with gcc 4.4
eldar
parents: 1
diff changeset
22 In addition, as a special exception, Nokia gives you certain
09a0f1d048f2 update parser to that from jambi 4.5, attemt to fix building with gcc 4.4
eldar
parents: 1
diff changeset
23 additional rights. These rights are described in the Nokia Qt LGPL
09a0f1d048f2 update parser to that from jambi 4.5, attemt to fix building with gcc 4.4
eldar
parents: 1
diff changeset
24 Exception version 1.0, included in the file LGPL_EXCEPTION.txt in this
09a0f1d048f2 update parser to that from jambi 4.5, attemt to fix building with gcc 4.4
eldar
parents: 1
diff changeset
25 package.
09a0f1d048f2 update parser to that from jambi 4.5, attemt to fix building with gcc 4.4
eldar
parents: 1
diff changeset
26
09a0f1d048f2 update parser to that from jambi 4.5, attemt to fix building with gcc 4.4
eldar
parents: 1
diff changeset
27 GNU General Public License Usage
09a0f1d048f2 update parser to that from jambi 4.5, attemt to fix building with gcc 4.4
eldar
parents: 1
diff changeset
28 Alternatively, this file may be used under the terms of the GNU
09a0f1d048f2 update parser to that from jambi 4.5, attemt to fix building with gcc 4.4
eldar
parents: 1
diff changeset
29 General Public License version 3.0 as published by the Free Software
09a0f1d048f2 update parser to that from jambi 4.5, attemt to fix building with gcc 4.4
eldar
parents: 1
diff changeset
30 Foundation and appearing in the file LICENSE.GPL included in the
09a0f1d048f2 update parser to that from jambi 4.5, attemt to fix building with gcc 4.4
eldar
parents: 1
diff changeset
31 packaging of this file. Please review the following information to
09a0f1d048f2 update parser to that from jambi 4.5, attemt to fix building with gcc 4.4
eldar
parents: 1
diff changeset
32 ensure the GNU General Public License version 3.0 requirements will be
09a0f1d048f2 update parser to that from jambi 4.5, attemt to fix building with gcc 4.4
eldar
parents: 1
diff changeset
33 met: http://www.gnu.org/copyleft/gpl.html.
09a0f1d048f2 update parser to that from jambi 4.5, attemt to fix building with gcc 4.4
eldar
parents: 1
diff changeset
34
09a0f1d048f2 update parser to that from jambi 4.5, attemt to fix building with gcc 4.4
eldar
parents: 1
diff changeset
35 If you are unsure which license is appropriate for your use, please
09a0f1d048f2 update parser to that from jambi 4.5, attemt to fix building with gcc 4.4
eldar
parents: 1
diff changeset
36 contact the sales department at qt-sales@nokia.com.
1
e78566595089 initial import
mandel
parents:
diff changeset
37
e78566595089 initial import
mandel
parents:
diff changeset
38 **
e78566595089 initial import
mandel
parents:
diff changeset
39 ** This file is provided AS IS with NO WARRANTY OF ANY KIND, INCLUDING THE
e78566595089 initial import
mandel
parents:
diff changeset
40 ** WARRANTY OF DESIGN, MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE.
e78566595089 initial import
mandel
parents:
diff changeset
41 **
e78566595089 initial import
mandel
parents:
diff changeset
42 ****************************************************************************/
e78566595089 initial import
mandel
parents:
diff changeset
43
e78566595089 initial import
mandel
parents:
diff changeset
44 #ifndef PP_MACRO_EXPANDER_H
e78566595089 initial import
mandel
parents:
diff changeset
45 #define PP_MACRO_EXPANDER_H
e78566595089 initial import
mandel
parents:
diff changeset
46
e78566595089 initial import
mandel
parents:
diff changeset
47 namespace rpp {
e78566595089 initial import
mandel
parents:
diff changeset
48
e78566595089 initial import
mandel
parents:
diff changeset
49 struct pp_frame
e78566595089 initial import
mandel
parents:
diff changeset
50 {
e78566595089 initial import
mandel
parents:
diff changeset
51 pp_macro *expanding_macro;
e78566595089 initial import
mandel
parents:
diff changeset
52 std::vector<std::string> *actuals;
e78566595089 initial import
mandel
parents:
diff changeset
53
e78566595089 initial import
mandel
parents:
diff changeset
54 pp_frame (pp_macro *__expanding_macro, std::vector<std::string> *__actuals):
e78566595089 initial import
mandel
parents:
diff changeset
55 expanding_macro (__expanding_macro), actuals (__actuals) {}
e78566595089 initial import
mandel
parents:
diff changeset
56 };
e78566595089 initial import
mandel
parents:
diff changeset
57
e78566595089 initial import
mandel
parents:
diff changeset
58 class pp_macro_expander
e78566595089 initial import
mandel
parents:
diff changeset
59 {
e78566595089 initial import
mandel
parents:
diff changeset
60 pp_environment &env;
e78566595089 initial import
mandel
parents:
diff changeset
61 pp_frame *frame;
e78566595089 initial import
mandel
parents:
diff changeset
62
e78566595089 initial import
mandel
parents:
diff changeset
63 pp_skip_number skip_number;
e78566595089 initial import
mandel
parents:
diff changeset
64 pp_skip_identifier skip_identifier;
e78566595089 initial import
mandel
parents:
diff changeset
65 pp_skip_string_literal skip_string_literal;
e78566595089 initial import
mandel
parents:
diff changeset
66 pp_skip_char_literal skip_char_literal;
e78566595089 initial import
mandel
parents:
diff changeset
67 pp_skip_argument skip_argument;
e78566595089 initial import
mandel
parents:
diff changeset
68 pp_skip_comment_or_divop skip_comment_or_divop;
e78566595089 initial import
mandel
parents:
diff changeset
69 pp_skip_blanks skip_blanks;
e78566595089 initial import
mandel
parents:
diff changeset
70 pp_skip_whitespaces skip_whitespaces;
e78566595089 initial import
mandel
parents:
diff changeset
71
e78566595089 initial import
mandel
parents:
diff changeset
72 std::string const *resolve_formal (pp_fast_string const *__name)
e78566595089 initial import
mandel
parents:
diff changeset
73 {
e78566595089 initial import
mandel
parents:
diff changeset
74 assert (__name != 0);
e78566595089 initial import
mandel
parents:
diff changeset
75
e78566595089 initial import
mandel
parents:
diff changeset
76 if (! frame)
e78566595089 initial import
mandel
parents:
diff changeset
77 return 0;
e78566595089 initial import
mandel
parents:
diff changeset
78
e78566595089 initial import
mandel
parents:
diff changeset
79 assert (frame->expanding_macro != 0);
e78566595089 initial import
mandel
parents:
diff changeset
80
e78566595089 initial import
mandel
parents:
diff changeset
81 std::vector<pp_fast_string const *> const formals = frame->expanding_macro->formals;
e78566595089 initial import
mandel
parents:
diff changeset
82 for (std::size_t index = 0; index < formals.size(); ++index)
e78566595089 initial import
mandel
parents:
diff changeset
83 {
e78566595089 initial import
mandel
parents:
diff changeset
84 pp_fast_string const *formal = formals[index];
e78566595089 initial import
mandel
parents:
diff changeset
85
e78566595089 initial import
mandel
parents:
diff changeset
86 if (*formal != *__name)
e78566595089 initial import
mandel
parents:
diff changeset
87 continue;
e78566595089 initial import
mandel
parents:
diff changeset
88
e78566595089 initial import
mandel
parents:
diff changeset
89 else if (frame->actuals && index < frame->actuals->size())
e78566595089 initial import
mandel
parents:
diff changeset
90 return &(*frame->actuals)[index];
e78566595089 initial import
mandel
parents:
diff changeset
91
e78566595089 initial import
mandel
parents:
diff changeset
92 else
e78566595089 initial import
mandel
parents:
diff changeset
93 assert (0); // internal error?
e78566595089 initial import
mandel
parents:
diff changeset
94 }
e78566595089 initial import
mandel
parents:
diff changeset
95
e78566595089 initial import
mandel
parents:
diff changeset
96 return 0;
e78566595089 initial import
mandel
parents:
diff changeset
97 }
e78566595089 initial import
mandel
parents:
diff changeset
98
e78566595089 initial import
mandel
parents:
diff changeset
99 public: // attributes
e78566595089 initial import
mandel
parents:
diff changeset
100 int lines;
e78566595089 initial import
mandel
parents:
diff changeset
101 int generated_lines;
e78566595089 initial import
mandel
parents:
diff changeset
102
e78566595089 initial import
mandel
parents:
diff changeset
103 public:
e78566595089 initial import
mandel
parents:
diff changeset
104 pp_macro_expander (pp_environment &__env, pp_frame *__frame = 0):
e78566595089 initial import
mandel
parents:
diff changeset
105 env (__env), frame (__frame), lines (0), generated_lines (0) {}
e78566595089 initial import
mandel
parents:
diff changeset
106
e78566595089 initial import
mandel
parents:
diff changeset
107 template <typename _InputIterator, typename _OutputIterator>
e78566595089 initial import
mandel
parents:
diff changeset
108 _InputIterator operator () (_InputIterator __first, _InputIterator __last, _OutputIterator __result)
e78566595089 initial import
mandel
parents:
diff changeset
109 {
e78566595089 initial import
mandel
parents:
diff changeset
110 generated_lines = 0;
e78566595089 initial import
mandel
parents:
diff changeset
111 __first = skip_blanks (__first, __last);
e78566595089 initial import
mandel
parents:
diff changeset
112 lines = skip_blanks.lines;
e78566595089 initial import
mandel
parents:
diff changeset
113
e78566595089 initial import
mandel
parents:
diff changeset
114 while (__first != __last)
e78566595089 initial import
mandel
parents:
diff changeset
115 {
e78566595089 initial import
mandel
parents:
diff changeset
116 if (*__first == '\n')
e78566595089 initial import
mandel
parents:
diff changeset
117 {
e78566595089 initial import
mandel
parents:
diff changeset
118 *__result++ = *__first;
e78566595089 initial import
mandel
parents:
diff changeset
119 ++lines;
e78566595089 initial import
mandel
parents:
diff changeset
120
e78566595089 initial import
mandel
parents:
diff changeset
121 __first = skip_blanks (++__first, __last);
e78566595089 initial import
mandel
parents:
diff changeset
122 lines += skip_blanks.lines;
e78566595089 initial import
mandel
parents:
diff changeset
123
e78566595089 initial import
mandel
parents:
diff changeset
124 if (__first != __last && *__first == '#')
e78566595089 initial import
mandel
parents:
diff changeset
125 break;
e78566595089 initial import
mandel
parents:
diff changeset
126 }
e78566595089 initial import
mandel
parents:
diff changeset
127 else if (*__first == '#')
e78566595089 initial import
mandel
parents:
diff changeset
128 {
e78566595089 initial import
mandel
parents:
diff changeset
129 __first = skip_blanks (++__first, __last);
e78566595089 initial import
mandel
parents:
diff changeset
130 lines += skip_blanks.lines;
e78566595089 initial import
mandel
parents:
diff changeset
131
e78566595089 initial import
mandel
parents:
diff changeset
132 _InputIterator end_id = skip_identifier (__first, __last);
e78566595089 initial import
mandel
parents:
diff changeset
133
e78566595089 initial import
mandel
parents:
diff changeset
134 // ### rewrite: not safe
e78566595089 initial import
mandel
parents:
diff changeset
135 char name_buffer[512], *cp = name_buffer;
e78566595089 initial import
mandel
parents:
diff changeset
136 std::copy (__first, end_id, cp);
e78566595089 initial import
mandel
parents:
diff changeset
137 std::size_t name_size = end_id - __first;
e78566595089 initial import
mandel
parents:
diff changeset
138 name_buffer[name_size] = '\0';
e78566595089 initial import
mandel
parents:
diff changeset
139
e78566595089 initial import
mandel
parents:
diff changeset
140 pp_fast_string fast_name (name_buffer, name_size);
e78566595089 initial import
mandel
parents:
diff changeset
141
e78566595089 initial import
mandel
parents:
diff changeset
142 if (std::string const *actual = resolve_formal (&fast_name))
e78566595089 initial import
mandel
parents:
diff changeset
143 {
e78566595089 initial import
mandel
parents:
diff changeset
144 *__result++ = '\"';
e78566595089 initial import
mandel
parents:
diff changeset
145
e78566595089 initial import
mandel
parents:
diff changeset
146 for (std::string::const_iterator it = skip_whitespaces (actual->begin (), actual->end ());
e78566595089 initial import
mandel
parents:
diff changeset
147 it != actual->end (); ++it)
e78566595089 initial import
mandel
parents:
diff changeset
148 {
e78566595089 initial import
mandel
parents:
diff changeset
149 if (*it == '"')
e78566595089 initial import
mandel
parents:
diff changeset
150 {
e78566595089 initial import
mandel
parents:
diff changeset
151 *__result++ = '\\';
e78566595089 initial import
mandel
parents:
diff changeset
152 *__result++ = *it;
e78566595089 initial import
mandel
parents:
diff changeset
153 }
e78566595089 initial import
mandel
parents:
diff changeset
154
e78566595089 initial import
mandel
parents:
diff changeset
155 else if (*it == '\n')
e78566595089 initial import
mandel
parents:
diff changeset
156 {
e78566595089 initial import
mandel
parents:
diff changeset
157 *__result++ = '"';
e78566595089 initial import
mandel
parents:
diff changeset
158 *__result++ = '\n';
e78566595089 initial import
mandel
parents:
diff changeset
159 *__result++ = '"';
e78566595089 initial import
mandel
parents:
diff changeset
160 }
e78566595089 initial import
mandel
parents:
diff changeset
161
e78566595089 initial import
mandel
parents:
diff changeset
162 else
e78566595089 initial import
mandel
parents:
diff changeset
163 *__result++ = *it;
e78566595089 initial import
mandel
parents:
diff changeset
164 }
e78566595089 initial import
mandel
parents:
diff changeset
165
e78566595089 initial import
mandel
parents:
diff changeset
166 *__result++ = '\"';
e78566595089 initial import
mandel
parents:
diff changeset
167 __first = end_id;
e78566595089 initial import
mandel
parents:
diff changeset
168 }
e78566595089 initial import
mandel
parents:
diff changeset
169 else
e78566595089 initial import
mandel
parents:
diff changeset
170 *__result++ = '#'; // ### warning message?
e78566595089 initial import
mandel
parents:
diff changeset
171 }
e78566595089 initial import
mandel
parents:
diff changeset
172 else if (*__first == '\"')
e78566595089 initial import
mandel
parents:
diff changeset
173 {
e78566595089 initial import
mandel
parents:
diff changeset
174 _InputIterator next_pos = skip_string_literal (__first, __last);
e78566595089 initial import
mandel
parents:
diff changeset
175 lines += skip_string_literal.lines;
e78566595089 initial import
mandel
parents:
diff changeset
176 std::copy (__first, next_pos, __result);
e78566595089 initial import
mandel
parents:
diff changeset
177 __first = next_pos;
e78566595089 initial import
mandel
parents:
diff changeset
178 }
e78566595089 initial import
mandel
parents:
diff changeset
179 else if (*__first == '\'')
e78566595089 initial import
mandel
parents:
diff changeset
180 {
e78566595089 initial import
mandel
parents:
diff changeset
181 _InputIterator next_pos = skip_char_literal (__first, __last);
e78566595089 initial import
mandel
parents:
diff changeset
182 lines += skip_char_literal.lines;
e78566595089 initial import
mandel
parents:
diff changeset
183 std::copy (__first, next_pos, __result);
e78566595089 initial import
mandel
parents:
diff changeset
184 __first = next_pos;
e78566595089 initial import
mandel
parents:
diff changeset
185 }
e78566595089 initial import
mandel
parents:
diff changeset
186 else if (_PP_internal::comment_p (__first, __last))
e78566595089 initial import
mandel
parents:
diff changeset
187 {
e78566595089 initial import
mandel
parents:
diff changeset
188 __first = skip_comment_or_divop (__first, __last);
e78566595089 initial import
mandel
parents:
diff changeset
189 int n = skip_comment_or_divop.lines;
e78566595089 initial import
mandel
parents:
diff changeset
190 lines += n;
e78566595089 initial import
mandel
parents:
diff changeset
191
e78566595089 initial import
mandel
parents:
diff changeset
192 while (n-- > 0)
e78566595089 initial import
mandel
parents:
diff changeset
193 *__result++ = '\n';
e78566595089 initial import
mandel
parents:
diff changeset
194 }
e78566595089 initial import
mandel
parents:
diff changeset
195 else if (pp_isspace (*__first))
e78566595089 initial import
mandel
parents:
diff changeset
196 {
e78566595089 initial import
mandel
parents:
diff changeset
197 for (; __first != __last; ++__first)
e78566595089 initial import
mandel
parents:
diff changeset
198 {
e78566595089 initial import
mandel
parents:
diff changeset
199 if (*__first == '\n' || !pp_isspace (*__first))
e78566595089 initial import
mandel
parents:
diff changeset
200 break;
e78566595089 initial import
mandel
parents:
diff changeset
201 }
e78566595089 initial import
mandel
parents:
diff changeset
202
e78566595089 initial import
mandel
parents:
diff changeset
203 *__result = ' ';
e78566595089 initial import
mandel
parents:
diff changeset
204 }
e78566595089 initial import
mandel
parents:
diff changeset
205 else if (pp_isdigit (*__first))
e78566595089 initial import
mandel
parents:
diff changeset
206 {
e78566595089 initial import
mandel
parents:
diff changeset
207 _InputIterator next_pos = skip_number (__first, __last);
e78566595089 initial import
mandel
parents:
diff changeset
208 lines += skip_number.lines;
e78566595089 initial import
mandel
parents:
diff changeset
209 std::copy (__first, next_pos, __result);
e78566595089 initial import
mandel
parents:
diff changeset
210 __first = next_pos;
e78566595089 initial import
mandel
parents:
diff changeset
211 }
e78566595089 initial import
mandel
parents:
diff changeset
212 else if (pp_isalpha (*__first) || *__first == '_')
e78566595089 initial import
mandel
parents:
diff changeset
213 {
e78566595089 initial import
mandel
parents:
diff changeset
214 _InputIterator name_begin = __first;
e78566595089 initial import
mandel
parents:
diff changeset
215 _InputIterator name_end = skip_identifier (__first, __last);
e78566595089 initial import
mandel
parents:
diff changeset
216 __first = name_end; // advance
e78566595089 initial import
mandel
parents:
diff changeset
217
e78566595089 initial import
mandel
parents:
diff changeset
218 // search for the paste token
e78566595089 initial import
mandel
parents:
diff changeset
219 _InputIterator next = skip_blanks (__first, __last);
e78566595089 initial import
mandel
parents:
diff changeset
220 if (next != __last && *next == '#')
e78566595089 initial import
mandel
parents:
diff changeset
221 {
e78566595089 initial import
mandel
parents:
diff changeset
222 ++next;
e78566595089 initial import
mandel
parents:
diff changeset
223 if (next != __last && *next == '#')
e78566595089 initial import
mandel
parents:
diff changeset
224 __first = skip_blanks(++next, __last);
e78566595089 initial import
mandel
parents:
diff changeset
225 }
e78566595089 initial import
mandel
parents:
diff changeset
226
e78566595089 initial import
mandel
parents:
diff changeset
227 // ### rewrite: not safe
e78566595089 initial import
mandel
parents:
diff changeset
228
e78566595089 initial import
mandel
parents:
diff changeset
229 std::ptrdiff_t name_size;
e78566595089 initial import
mandel
parents:
diff changeset
230 #if defined(__SUNPRO_CC)
e78566595089 initial import
mandel
parents:
diff changeset
231 std::distance (name_begin, name_end, name_size);
e78566595089 initial import
mandel
parents:
diff changeset
232 #else
e78566595089 initial import
mandel
parents:
diff changeset
233 name_size = std::distance (name_begin, name_end);
e78566595089 initial import
mandel
parents:
diff changeset
234 #endif
e78566595089 initial import
mandel
parents:
diff changeset
235 assert (name_size >= 0 && name_size < 512);
e78566595089 initial import
mandel
parents:
diff changeset
236
e78566595089 initial import
mandel
parents:
diff changeset
237 char name_buffer[512], *cp = name_buffer;
e78566595089 initial import
mandel
parents:
diff changeset
238 std::size_t __size = name_end - name_begin;
e78566595089 initial import
mandel
parents:
diff changeset
239 std::copy (name_begin, name_end, cp);
e78566595089 initial import
mandel
parents:
diff changeset
240 name_buffer[__size] = '\0';
e78566595089 initial import
mandel
parents:
diff changeset
241
e78566595089 initial import
mandel
parents:
diff changeset
242 pp_fast_string fast_name (name_buffer, name_size);
e78566595089 initial import
mandel
parents:
diff changeset
243
e78566595089 initial import
mandel
parents:
diff changeset
244 if (std::string const *actual = resolve_formal (&fast_name))
e78566595089 initial import
mandel
parents:
diff changeset
245 {
e78566595089 initial import
mandel
parents:
diff changeset
246 std::copy (actual->begin (), actual->end (), __result);
e78566595089 initial import
mandel
parents:
diff changeset
247 continue;
e78566595089 initial import
mandel
parents:
diff changeset
248 }
e78566595089 initial import
mandel
parents:
diff changeset
249
e78566595089 initial import
mandel
parents:
diff changeset
250 static bool hide_next = false; // ### remove me
e78566595089 initial import
mandel
parents:
diff changeset
251
e78566595089 initial import
mandel
parents:
diff changeset
252 pp_macro *macro = env.resolve (name_buffer, name_size);
e78566595089 initial import
mandel
parents:
diff changeset
253 if (! macro || macro->hidden || hide_next)
e78566595089 initial import
mandel
parents:
diff changeset
254 {
e78566595089 initial import
mandel
parents:
diff changeset
255 hide_next = ! strcmp (name_buffer, "defined");
e78566595089 initial import
mandel
parents:
diff changeset
256
e78566595089 initial import
mandel
parents:
diff changeset
257 if (__size == 8 && name_buffer [0] == '_' && name_buffer [1] == '_')
e78566595089 initial import
mandel
parents:
diff changeset
258 {
e78566595089 initial import
mandel
parents:
diff changeset
259 if (! strcmp (name_buffer, "__LINE__"))
e78566595089 initial import
mandel
parents:
diff changeset
260 {
e78566595089 initial import
mandel
parents:
diff changeset
261 char buf [16];
e78566595089 initial import
mandel
parents:
diff changeset
262 char *end = buf + pp_snprintf (buf, 16, "%d", env.current_line + lines);
e78566595089 initial import
mandel
parents:
diff changeset
263
e78566595089 initial import
mandel
parents:
diff changeset
264 std::copy (&buf [0], end, __result);
e78566595089 initial import
mandel
parents:
diff changeset
265 continue;
e78566595089 initial import
mandel
parents:
diff changeset
266 }
e78566595089 initial import
mandel
parents:
diff changeset
267
e78566595089 initial import
mandel
parents:
diff changeset
268 else if (! strcmp (name_buffer, "__FILE__"))
e78566595089 initial import
mandel
parents:
diff changeset
269 {
e78566595089 initial import
mandel
parents:
diff changeset
270 __result++ = '"';
e78566595089 initial import
mandel
parents:
diff changeset
271 std::copy (env.current_file.begin (), env.current_file.end (), __result); // ### quote
e78566595089 initial import
mandel
parents:
diff changeset
272 __result++ = '"';
e78566595089 initial import
mandel
parents:
diff changeset
273 continue;
e78566595089 initial import
mandel
parents:
diff changeset
274 }
e78566595089 initial import
mandel
parents:
diff changeset
275 }
e78566595089 initial import
mandel
parents:
diff changeset
276
e78566595089 initial import
mandel
parents:
diff changeset
277 std::copy (name_begin, name_end, __result);
e78566595089 initial import
mandel
parents:
diff changeset
278 continue;
e78566595089 initial import
mandel
parents:
diff changeset
279 }
e78566595089 initial import
mandel
parents:
diff changeset
280
e78566595089 initial import
mandel
parents:
diff changeset
281 if (! macro->function_like)
e78566595089 initial import
mandel
parents:
diff changeset
282 {
e78566595089 initial import
mandel
parents:
diff changeset
283 pp_macro *m = 0;
e78566595089 initial import
mandel
parents:
diff changeset
284
e78566595089 initial import
mandel
parents:
diff changeset
285 if (macro->definition)
e78566595089 initial import
mandel
parents:
diff changeset
286 {
e78566595089 initial import
mandel
parents:
diff changeset
287 macro->hidden = true;
e78566595089 initial import
mandel
parents:
diff changeset
288
e78566595089 initial import
mandel
parents:
diff changeset
289 std::string __tmp;
e78566595089 initial import
mandel
parents:
diff changeset
290 __tmp.reserve (256);
e78566595089 initial import
mandel
parents:
diff changeset
291
e78566595089 initial import
mandel
parents:
diff changeset
292 pp_macro_expander expand_macro (env);
e78566595089 initial import
mandel
parents:
diff changeset
293 expand_macro (macro->definition->begin (), macro->definition->end (), std::back_inserter (__tmp));
e78566595089 initial import
mandel
parents:
diff changeset
294 generated_lines += expand_macro.lines;
e78566595089 initial import
mandel
parents:
diff changeset
295
e78566595089 initial import
mandel
parents:
diff changeset
296 if (! __tmp.empty ())
e78566595089 initial import
mandel
parents:
diff changeset
297 {
e78566595089 initial import
mandel
parents:
diff changeset
298 std::string::iterator __begin_id = skip_whitespaces (__tmp.begin (), __tmp.end ());
e78566595089 initial import
mandel
parents:
diff changeset
299 std::string::iterator __end_id = skip_identifier (__begin_id, __tmp.end ());
e78566595089 initial import
mandel
parents:
diff changeset
300
e78566595089 initial import
mandel
parents:
diff changeset
301 if (__end_id == __tmp.end ())
e78566595089 initial import
mandel
parents:
diff changeset
302 {
e78566595089 initial import
mandel
parents:
diff changeset
303 std::string __id;
e78566595089 initial import
mandel
parents:
diff changeset
304 __id.assign (__begin_id, __end_id);
e78566595089 initial import
mandel
parents:
diff changeset
305
e78566595089 initial import
mandel
parents:
diff changeset
306 std::size_t x;
e78566595089 initial import
mandel
parents:
diff changeset
307 #if defined(__SUNPRO_CC)
e78566595089 initial import
mandel
parents:
diff changeset
308 std::distance (__begin_id, __end_id, x);
e78566595089 initial import
mandel
parents:
diff changeset
309 #else
e78566595089 initial import
mandel
parents:
diff changeset
310 x = std::distance (__begin_id, __end_id);
e78566595089 initial import
mandel
parents:
diff changeset
311 #endif
e78566595089 initial import
mandel
parents:
diff changeset
312 m = env.resolve (__id.c_str (), x);
e78566595089 initial import
mandel
parents:
diff changeset
313 }
e78566595089 initial import
mandel
parents:
diff changeset
314
e78566595089 initial import
mandel
parents:
diff changeset
315 if (! m)
e78566595089 initial import
mandel
parents:
diff changeset
316 std::copy (__tmp.begin (), __tmp.end (), __result);
e78566595089 initial import
mandel
parents:
diff changeset
317 }
e78566595089 initial import
mandel
parents:
diff changeset
318
e78566595089 initial import
mandel
parents:
diff changeset
319 macro->hidden = false;
e78566595089 initial import
mandel
parents:
diff changeset
320 }
e78566595089 initial import
mandel
parents:
diff changeset
321
e78566595089 initial import
mandel
parents:
diff changeset
322 if (! m)
e78566595089 initial import
mandel
parents:
diff changeset
323 continue;
e78566595089 initial import
mandel
parents:
diff changeset
324
e78566595089 initial import
mandel
parents:
diff changeset
325 macro = m;
e78566595089 initial import
mandel
parents:
diff changeset
326 }
e78566595089 initial import
mandel
parents:
diff changeset
327
e78566595089 initial import
mandel
parents:
diff changeset
328 // function like macro
e78566595089 initial import
mandel
parents:
diff changeset
329 _InputIterator arg_it = skip_whitespaces (__first, __last);
e78566595089 initial import
mandel
parents:
diff changeset
330
e78566595089 initial import
mandel
parents:
diff changeset
331 if (arg_it == __last || *arg_it != '(')
e78566595089 initial import
mandel
parents:
diff changeset
332 {
e78566595089 initial import
mandel
parents:
diff changeset
333 std::copy (name_begin, name_end, __result);
e78566595089 initial import
mandel
parents:
diff changeset
334 lines += skip_whitespaces.lines;
e78566595089 initial import
mandel
parents:
diff changeset
335 __first = arg_it;
e78566595089 initial import
mandel
parents:
diff changeset
336 continue;
e78566595089 initial import
mandel
parents:
diff changeset
337 }
e78566595089 initial import
mandel
parents:
diff changeset
338
e78566595089 initial import
mandel
parents:
diff changeset
339 std::vector<std::string> actuals;
e78566595089 initial import
mandel
parents:
diff changeset
340 actuals.reserve (5);
e78566595089 initial import
mandel
parents:
diff changeset
341 ++arg_it; // skip '('
e78566595089 initial import
mandel
parents:
diff changeset
342
e78566595089 initial import
mandel
parents:
diff changeset
343 pp_macro_expander expand_actual (env, frame);
e78566595089 initial import
mandel
parents:
diff changeset
344
e78566595089 initial import
mandel
parents:
diff changeset
345 _InputIterator arg_end = skip_argument_variadics (actuals, macro, arg_it, __last);
e78566595089 initial import
mandel
parents:
diff changeset
346 if (arg_it != arg_end)
e78566595089 initial import
mandel
parents:
diff changeset
347 {
e78566595089 initial import
mandel
parents:
diff changeset
348 std::string actual (arg_it, arg_end);
e78566595089 initial import
mandel
parents:
diff changeset
349 actuals.resize (actuals.size() + 1);
e78566595089 initial import
mandel
parents:
diff changeset
350 actuals.back ().reserve (255);
e78566595089 initial import
mandel
parents:
diff changeset
351 expand_actual (actual.begin (), actual.end(), std::back_inserter (actuals.back()));
e78566595089 initial import
mandel
parents:
diff changeset
352 arg_it = arg_end;
e78566595089 initial import
mandel
parents:
diff changeset
353 }
e78566595089 initial import
mandel
parents:
diff changeset
354
e78566595089 initial import
mandel
parents:
diff changeset
355 while (arg_it != __last && *arg_end == ',')
e78566595089 initial import
mandel
parents:
diff changeset
356 {
e78566595089 initial import
mandel
parents:
diff changeset
357 ++arg_it; // skip ','
e78566595089 initial import
mandel
parents:
diff changeset
358
e78566595089 initial import
mandel
parents:
diff changeset
359 arg_end = skip_argument_variadics (actuals, macro, arg_it, __last);
e78566595089 initial import
mandel
parents:
diff changeset
360 std::string actual (arg_it, arg_end);
e78566595089 initial import
mandel
parents:
diff changeset
361 actuals.resize (actuals.size() + 1);
e78566595089 initial import
mandel
parents:
diff changeset
362 actuals.back ().reserve (255);
e78566595089 initial import
mandel
parents:
diff changeset
363 expand_actual (actual.begin (), actual.end(), std::back_inserter (actuals.back()));
e78566595089 initial import
mandel
parents:
diff changeset
364 arg_it = arg_end;
e78566595089 initial import
mandel
parents:
diff changeset
365 }
e78566595089 initial import
mandel
parents:
diff changeset
366
e78566595089 initial import
mandel
parents:
diff changeset
367 assert (arg_it != __last && *arg_it == ')');
e78566595089 initial import
mandel
parents:
diff changeset
368
e78566595089 initial import
mandel
parents:
diff changeset
369 ++arg_it; // skip ')'
e78566595089 initial import
mandel
parents:
diff changeset
370 __first = arg_it;
e78566595089 initial import
mandel
parents:
diff changeset
371
e78566595089 initial import
mandel
parents:
diff changeset
372 #if 0 // ### enable me
e78566595089 initial import
mandel
parents:
diff changeset
373 assert ((macro->variadics && macro->formals.size () >= actuals.size ())
e78566595089 initial import
mandel
parents:
diff changeset
374 || macro->formals.size() == actuals.size());
e78566595089 initial import
mandel
parents:
diff changeset
375 #endif
e78566595089 initial import
mandel
parents:
diff changeset
376
e78566595089 initial import
mandel
parents:
diff changeset
377 pp_frame frame (macro, &actuals);
e78566595089 initial import
mandel
parents:
diff changeset
378 pp_macro_expander expand_macro (env, &frame);
e78566595089 initial import
mandel
parents:
diff changeset
379 macro->hidden = true;
e78566595089 initial import
mandel
parents:
diff changeset
380 expand_macro (macro->definition->begin (), macro->definition->end (), __result);
e78566595089 initial import
mandel
parents:
diff changeset
381 macro->hidden = false;
e78566595089 initial import
mandel
parents:
diff changeset
382 generated_lines += expand_macro.lines;
e78566595089 initial import
mandel
parents:
diff changeset
383 }
e78566595089 initial import
mandel
parents:
diff changeset
384 else
e78566595089 initial import
mandel
parents:
diff changeset
385 *__result++ = *__first++;
e78566595089 initial import
mandel
parents:
diff changeset
386 }
e78566595089 initial import
mandel
parents:
diff changeset
387
e78566595089 initial import
mandel
parents:
diff changeset
388 return __first;
e78566595089 initial import
mandel
parents:
diff changeset
389 }
e78566595089 initial import
mandel
parents:
diff changeset
390
e78566595089 initial import
mandel
parents:
diff changeset
391 template <typename _InputIterator>
e78566595089 initial import
mandel
parents:
diff changeset
392 _InputIterator skip_argument_variadics (std::vector<std::string> const &__actuals, pp_macro *__macro,
e78566595089 initial import
mandel
parents:
diff changeset
393 _InputIterator __first, _InputIterator __last)
e78566595089 initial import
mandel
parents:
diff changeset
394 {
e78566595089 initial import
mandel
parents:
diff changeset
395 _InputIterator arg_end = skip_argument (__first, __last);
e78566595089 initial import
mandel
parents:
diff changeset
396
e78566595089 initial import
mandel
parents:
diff changeset
397 while (__macro->variadics && __first != arg_end && arg_end != __last && *arg_end == ','
e78566595089 initial import
mandel
parents:
diff changeset
398 && (__actuals.size () + 1) == __macro->formals.size ())
e78566595089 initial import
mandel
parents:
diff changeset
399 {
e78566595089 initial import
mandel
parents:
diff changeset
400 arg_end = skip_argument (++arg_end, __last);
e78566595089 initial import
mandel
parents:
diff changeset
401 }
e78566595089 initial import
mandel
parents:
diff changeset
402
e78566595089 initial import
mandel
parents:
diff changeset
403 return arg_end;
e78566595089 initial import
mandel
parents:
diff changeset
404 }
e78566595089 initial import
mandel
parents:
diff changeset
405 };
e78566595089 initial import
mandel
parents:
diff changeset
406
e78566595089 initial import
mandel
parents:
diff changeset
407 } // namespace rpp
e78566595089 initial import
mandel
parents:
diff changeset
408
e78566595089 initial import
mandel
parents:
diff changeset
409 #endif // PP_MACRO_EXPANDER_H
e78566595089 initial import
mandel
parents:
diff changeset
410
e78566595089 initial import
mandel
parents:
diff changeset
411 // kate: space-indent on; indent-width 2; replace-tabs on;