blob: 9605c64c2f2f1a561c20acea94b93c1053139452 [file] [log] [blame]
alokp@chromium.org04d7d222012-05-16 19:24:07 +00001//
2// Copyright (c) 2011 The ANGLE Project Authors. All rights reserved.
3// Use of this source code is governed by a BSD-style license that can be
4// found in the LICENSE file.
5//
6
Corentin Wallez054f7ed2016-09-20 17:15:59 -04007#include "compiler/preprocessor/MacroExpander.h"
alokp@chromium.org04d7d222012-05-16 19:24:07 +00008
alokp@chromium.org7fc38dd2012-06-14 18:23:23 +00009#include <algorithm>
alokp@chromium.org7fc38dd2012-06-14 18:23:23 +000010
Corentin Wallez054f7ed2016-09-20 17:15:59 -040011#include "common/debug.h"
12#include "compiler/preprocessor/DiagnosticsBase.h"
13#include "compiler/preprocessor/Token.h"
alokp@chromium.org7fc38dd2012-06-14 18:23:23 +000014
alokp@chromium.org04d7d222012-05-16 19:24:07 +000015namespace pp
16{
17
Corentin Wallez2bd9c442016-09-20 16:39:18 -040018namespace
19{
20
21const size_t kMaxContextTokens = 10000;
22
alokp@chromium.org7fc38dd2012-06-14 18:23:23 +000023class TokenLexer : public Lexer
24{
25 public:
26 typedef std::vector<Token> TokenVector;
27
Zhenyao Mod526f982014-05-13 14:51:19 -070028 TokenLexer(TokenVector *tokens)
alokp@chromium.org7fc38dd2012-06-14 18:23:23 +000029 {
30 tokens->swap(mTokens);
31 mIter = mTokens.begin();
32 }
33
Corentin Walleze5a1f272015-08-21 02:58:25 +020034 void lex(Token *token) override
alokp@chromium.org7fc38dd2012-06-14 18:23:23 +000035 {
36 if (mIter == mTokens.end())
37 {
38 token->reset();
39 token->type = Token::LAST;
40 }
41 else
42 {
43 *token = *mIter++;
44 }
45 }
46
47 private:
alokp@chromium.org7fc38dd2012-06-14 18:23:23 +000048 TokenVector mTokens;
49 TokenVector::const_iterator mIter;
50};
51
Corentin Wallez2bd9c442016-09-20 16:39:18 -040052} // anonymous namespace
53
Olli Etuaho78b0c912016-11-21 14:23:06 +000054class MacroExpander::ScopedMacroReenabler final : angle::NonCopyable
55{
56 public:
57 ScopedMacroReenabler(MacroExpander *expander);
58 ~ScopedMacroReenabler();
59
60 private:
61 MacroExpander *mExpander;
62};
63
64MacroExpander::ScopedMacroReenabler::ScopedMacroReenabler(MacroExpander *expander)
65 : mExpander(expander)
66{
67 mExpander->mDeferReenablingMacros = true;
68}
69
70MacroExpander::ScopedMacroReenabler::~ScopedMacroReenabler()
71{
72 mExpander->mDeferReenablingMacros = false;
73 for (auto *macro : mExpander->mMacrosToReenable)
74 {
75 macro->disabled = false;
76 }
77 mExpander->mMacrosToReenable.clear();
78}
79
Olli Etuahof1cf5e62016-11-22 17:36:49 +000080MacroExpander::MacroExpander(Lexer *lexer,
81 MacroSet *macroSet,
82 Diagnostics *diagnostics,
83 int allowedMacroExpansionDepth)
Olli Etuaho78b0c912016-11-21 14:23:06 +000084 : mLexer(lexer),
85 mMacroSet(macroSet),
86 mDiagnostics(diagnostics),
87 mTotalTokensInContexts(0),
Olli Etuahof1cf5e62016-11-22 17:36:49 +000088 mAllowedMacroExpansionDepth(allowedMacroExpansionDepth),
Olli Etuaho78b0c912016-11-21 14:23:06 +000089 mDeferReenablingMacros(false)
alokp@chromium.org04d7d222012-05-16 19:24:07 +000090{
91}
92
alokp@chromium.org7fc38dd2012-06-14 18:23:23 +000093MacroExpander::~MacroExpander()
94{
Olli Etuaho78b0c912016-11-21 14:23:06 +000095 ASSERT(mMacrosToReenable.empty());
Corentin Wallez054f7ed2016-09-20 17:15:59 -040096 for (MacroContext *context : mContextStack)
alokp@chromium.org6c0c2d82012-07-18 15:49:56 +000097 {
Corentin Wallez054f7ed2016-09-20 17:15:59 -040098 delete context;
alokp@chromium.org6c0c2d82012-07-18 15:49:56 +000099 }
alokp@chromium.org7fc38dd2012-06-14 18:23:23 +0000100}
101
Zhenyao Mod526f982014-05-13 14:51:19 -0700102void MacroExpander::lex(Token *token)
alokp@chromium.org04d7d222012-05-16 19:24:07 +0000103{
alokp@chromium.org7fc38dd2012-06-14 18:23:23 +0000104 while (true)
105 {
106 getToken(token);
107
108 if (token->type != Token::IDENTIFIER)
109 break;
110
111 if (token->expansionDisabled())
112 break;
113
alokp@chromium.org5b6a68e2012-06-28 20:29:13 +0000114 MacroSet::const_iterator iter = mMacroSet->find(token->text);
alokp@chromium.org7fc38dd2012-06-14 18:23:23 +0000115 if (iter == mMacroSet->end())
116 break;
117
118 const Macro& macro = iter->second;
119 if (macro.disabled)
120 {
121 // If a particular token is not expanded, it is never expanded.
122 token->setExpansionDisabled(true);
123 break;
124 }
Corentin Wallez449a8032016-10-26 08:05:54 -0400125
126 // Bump the expansion count before peeking if the next token is a '('
127 // otherwise there could be a #undef of the macro before the next token.
128 macro.expansionCount++;
alokp@chromium.org7fc38dd2012-06-14 18:23:23 +0000129 if ((macro.type == Macro::kTypeFunc) && !isNextTokenLeftParen())
130 {
131 // If the token immediately after the macro name is not a '(',
132 // this macro should not be expanded.
Corentin Wallez449a8032016-10-26 08:05:54 -0400133 macro.expansionCount--;
alokp@chromium.org7fc38dd2012-06-14 18:23:23 +0000134 break;
135 }
136
137 pushMacro(macro, *token);
138 }
139}
140
Zhenyao Mod526f982014-05-13 14:51:19 -0700141void MacroExpander::getToken(Token *token)
alokp@chromium.org7fc38dd2012-06-14 18:23:23 +0000142{
143 if (mReserveToken.get())
144 {
145 *token = *mReserveToken;
146 mReserveToken.reset();
147 return;
148 }
149
150 // First pop all empty macro contexts.
151 while (!mContextStack.empty() && mContextStack.back()->empty())
152 {
153 popMacro();
154 }
155
156 if (!mContextStack.empty())
157 {
158 *token = mContextStack.back()->get();
159 }
160 else
161 {
Corentin Wallez054f7ed2016-09-20 17:15:59 -0400162 ASSERT(mTotalTokensInContexts == 0);
alokp@chromium.org7fc38dd2012-06-14 18:23:23 +0000163 mLexer->lex(token);
164 }
165}
166
Zhenyao Mod526f982014-05-13 14:51:19 -0700167void MacroExpander::ungetToken(const Token &token)
alokp@chromium.org7fc38dd2012-06-14 18:23:23 +0000168{
169 if (!mContextStack.empty())
170 {
Zhenyao Mod526f982014-05-13 14:51:19 -0700171 MacroContext *context = mContextStack.back();
alokp@chromium.org7fc38dd2012-06-14 18:23:23 +0000172 context->unget();
Corentin Wallez054f7ed2016-09-20 17:15:59 -0400173 ASSERT(context->replacements[context->index] == token);
alokp@chromium.org7fc38dd2012-06-14 18:23:23 +0000174 }
175 else
176 {
Corentin Wallez054f7ed2016-09-20 17:15:59 -0400177 ASSERT(!mReserveToken.get());
alokp@chromium.org7fc38dd2012-06-14 18:23:23 +0000178 mReserveToken.reset(new Token(token));
179 }
180}
181
182bool MacroExpander::isNextTokenLeftParen()
183{
184 Token token;
185 getToken(&token);
186
187 bool lparen = token.type == '(';
188 ungetToken(token);
189
190 return lparen;
191}
192
Zhenyao Mod526f982014-05-13 14:51:19 -0700193bool MacroExpander::pushMacro(const Macro &macro, const Token &identifier)
alokp@chromium.org7fc38dd2012-06-14 18:23:23 +0000194{
Corentin Wallez054f7ed2016-09-20 17:15:59 -0400195 ASSERT(!macro.disabled);
196 ASSERT(!identifier.expansionDisabled());
197 ASSERT(identifier.type == Token::IDENTIFIER);
198 ASSERT(identifier.text == macro.name);
alokp@chromium.org7fc38dd2012-06-14 18:23:23 +0000199
200 std::vector<Token> replacements;
201 if (!expandMacro(macro, identifier, &replacements))
202 return false;
203
204 // Macro is disabled for expansion until it is popped off the stack.
205 macro.disabled = true;
206
Zhenyao Mod526f982014-05-13 14:51:19 -0700207 MacroContext *context = new MacroContext;
alokp@chromium.org7fc38dd2012-06-14 18:23:23 +0000208 context->macro = &macro;
209 context->replacements.swap(replacements);
210 mContextStack.push_back(context);
Corentin Wallez2bd9c442016-09-20 16:39:18 -0400211 mTotalTokensInContexts += context->replacements.size();
alokp@chromium.org7fc38dd2012-06-14 18:23:23 +0000212 return true;
213}
214
215void MacroExpander::popMacro()
216{
Corentin Wallez054f7ed2016-09-20 17:15:59 -0400217 ASSERT(!mContextStack.empty());
alokp@chromium.org7fc38dd2012-06-14 18:23:23 +0000218
Zhenyao Mod526f982014-05-13 14:51:19 -0700219 MacroContext *context = mContextStack.back();
alokp@chromium.org7fc38dd2012-06-14 18:23:23 +0000220 mContextStack.pop_back();
221
Corentin Wallez054f7ed2016-09-20 17:15:59 -0400222 ASSERT(context->empty());
223 ASSERT(context->macro->disabled);
224 ASSERT(context->macro->expansionCount > 0);
Olli Etuaho78b0c912016-11-21 14:23:06 +0000225 if (mDeferReenablingMacros)
226 {
227 mMacrosToReenable.push_back(context->macro);
228 }
229 else
230 {
231 context->macro->disabled = false;
232 }
Corentin Wallezd2f195b2016-09-19 15:53:33 -0400233 context->macro->expansionCount--;
Corentin Wallez2bd9c442016-09-20 16:39:18 -0400234 mTotalTokensInContexts -= context->replacements.size();
alokp@chromium.org7fc38dd2012-06-14 18:23:23 +0000235 delete context;
236}
237
Zhenyao Mod526f982014-05-13 14:51:19 -0700238bool MacroExpander::expandMacro(const Macro &macro,
239 const Token &identifier,
240 std::vector<Token> *replacements)
alokp@chromium.org7fc38dd2012-06-14 18:23:23 +0000241{
alokp@chromium.orgf3cdb462012-06-19 18:39:48 +0000242 replacements->clear();
Olli Etuahoe6432c82015-09-08 14:21:38 +0300243
244 // In the case of an object-like macro, the replacement list gets its location
245 // from the identifier, but in the case of a function-like macro, the replacement
246 // list gets its location from the closing parenthesis of the macro invocation.
247 // This is tested by dEQP-GLES3.functional.shaders.preprocessor.predefined_macros.*
248 SourceLocation replacementLocation = identifier.location;
alokp@chromium.org7fc38dd2012-06-14 18:23:23 +0000249 if (macro.type == Macro::kTypeObj)
250 {
251 replacements->assign(macro.replacements.begin(),
252 macro.replacements.end());
alokp@chromium.orgf3cdb462012-06-19 18:39:48 +0000253
254 if (macro.predefined)
255 {
Zhenyao Mob5e17752014-10-22 10:57:10 -0700256 const char kLine[] = "__LINE__";
257 const char kFile[] = "__FILE__";
alokp@chromium.orgf3cdb462012-06-19 18:39:48 +0000258
Corentin Wallez054f7ed2016-09-20 17:15:59 -0400259 ASSERT(replacements->size() == 1);
alokp@chromium.orgf3cdb462012-06-19 18:39:48 +0000260 Token& repl = replacements->front();
261 if (macro.name == kLine)
262 {
Corentin Wallez054f7ed2016-09-20 17:15:59 -0400263 repl.text = ToString(identifier.location.line);
alokp@chromium.orgf3cdb462012-06-19 18:39:48 +0000264 }
265 else if (macro.name == kFile)
266 {
Corentin Wallez054f7ed2016-09-20 17:15:59 -0400267 repl.text = ToString(identifier.location.file);
alokp@chromium.orgf3cdb462012-06-19 18:39:48 +0000268 }
269 }
alokp@chromium.org7fc38dd2012-06-14 18:23:23 +0000270 }
271 else
272 {
Corentin Wallez054f7ed2016-09-20 17:15:59 -0400273 ASSERT(macro.type == Macro::kTypeFunc);
alokp@chromium.org7fc38dd2012-06-14 18:23:23 +0000274 std::vector<MacroArg> args;
275 args.reserve(macro.parameters.size());
Olli Etuahoe6432c82015-09-08 14:21:38 +0300276 if (!collectMacroArgs(macro, identifier, &args, &replacementLocation))
alokp@chromium.org7fc38dd2012-06-14 18:23:23 +0000277 return false;
278
279 replaceMacroParams(macro, args, replacements);
280 }
281
daniel@transgaming.coma16a55f2012-12-20 20:51:54 +0000282 for (std::size_t i = 0; i < replacements->size(); ++i)
alokp@chromium.org7fc38dd2012-06-14 18:23:23 +0000283 {
284 Token& repl = replacements->at(i);
285 if (i == 0)
286 {
287 // The first token in the replacement list inherits the padding
288 // properties of the identifier token.
289 repl.setAtStartOfLine(identifier.atStartOfLine());
290 repl.setHasLeadingSpace(identifier.hasLeadingSpace());
291 }
Olli Etuahoe6432c82015-09-08 14:21:38 +0300292 repl.location = replacementLocation;
alokp@chromium.org7fc38dd2012-06-14 18:23:23 +0000293 }
294 return true;
295}
296
Zhenyao Mod526f982014-05-13 14:51:19 -0700297bool MacroExpander::collectMacroArgs(const Macro &macro,
298 const Token &identifier,
Olli Etuahoe6432c82015-09-08 14:21:38 +0300299 std::vector<MacroArg> *args,
300 SourceLocation *closingParenthesisLocation)
alokp@chromium.org7fc38dd2012-06-14 18:23:23 +0000301{
302 Token token;
303 getToken(&token);
Corentin Wallez054f7ed2016-09-20 17:15:59 -0400304 ASSERT(token.type == '(');
alokp@chromium.org7fc38dd2012-06-14 18:23:23 +0000305
306 args->push_back(MacroArg());
Corentin Wallez054f7ed2016-09-20 17:15:59 -0400307
Olli Etuaho78b0c912016-11-21 14:23:06 +0000308 // Defer reenabling macros until args collection is finished to avoid the possibility of
309 // infinite recursion. Otherwise infinite recursion might happen when expanding the args after
310 // macros have been popped from the context stack when parsing the args.
311 ScopedMacroReenabler deferReenablingMacros(this);
312
Corentin Wallez054f7ed2016-09-20 17:15:59 -0400313 int openParens = 1;
314 while (openParens != 0)
alokp@chromium.org7fc38dd2012-06-14 18:23:23 +0000315 {
316 getToken(&token);
317
318 if (token.type == Token::LAST)
319 {
Shannon Woods7f2d7942013-11-19 15:07:58 -0500320 mDiagnostics->report(Diagnostics::PP_MACRO_UNTERMINATED_INVOCATION,
alokp@chromium.org5b6a68e2012-06-28 20:29:13 +0000321 identifier.location, identifier.text);
alokp@chromium.org7fc38dd2012-06-14 18:23:23 +0000322 // Do not lose EOF token.
323 ungetToken(token);
324 return false;
325 }
326
327 bool isArg = false; // True if token is part of the current argument.
328 switch (token.type)
329 {
330 case '(':
331 ++openParens;
332 isArg = true;
333 break;
334 case ')':
335 --openParens;
336 isArg = openParens != 0;
Olli Etuahoe6432c82015-09-08 14:21:38 +0300337 *closingParenthesisLocation = token.location;
alokp@chromium.org7fc38dd2012-06-14 18:23:23 +0000338 break;
339 case ',':
340 // The individual arguments are separated by comma tokens, but
341 // the comma tokens between matching inner parentheses do not
342 // seperate arguments.
Zhenyao Mod526f982014-05-13 14:51:19 -0700343 if (openParens == 1)
344 args->push_back(MacroArg());
alokp@chromium.org7fc38dd2012-06-14 18:23:23 +0000345 isArg = openParens != 1;
346 break;
347 default:
348 isArg = true;
349 break;
350 }
351 if (isArg)
352 {
Zhenyao Mod526f982014-05-13 14:51:19 -0700353 MacroArg &arg = args->back();
alokp@chromium.org7fc38dd2012-06-14 18:23:23 +0000354 // Initial whitespace is not part of the argument.
Zhenyao Mod526f982014-05-13 14:51:19 -0700355 if (arg.empty())
356 token.setHasLeadingSpace(false);
alokp@chromium.org7fc38dd2012-06-14 18:23:23 +0000357 arg.push_back(token);
358 }
359 }
360
Zhenyao Mod526f982014-05-13 14:51:19 -0700361 const Macro::Parameters &params = macro.parameters;
alokp@chromium.org7fc38dd2012-06-14 18:23:23 +0000362 // If there is only one empty argument, it is equivalent to no argument.
363 if (params.empty() && (args->size() == 1) && args->front().empty())
364 {
365 args->clear();
366 }
367 // Validate the number of arguments.
368 if (args->size() != params.size())
369 {
370 Diagnostics::ID id = args->size() < macro.parameters.size() ?
Shannon Woods7f2d7942013-11-19 15:07:58 -0500371 Diagnostics::PP_MACRO_TOO_FEW_ARGS :
372 Diagnostics::PP_MACRO_TOO_MANY_ARGS;
alokp@chromium.org5b6a68e2012-06-28 20:29:13 +0000373 mDiagnostics->report(id, identifier.location, identifier.text);
alokp@chromium.org7fc38dd2012-06-14 18:23:23 +0000374 return false;
375 }
376
377 // Pre-expand each argument before substitution.
378 // This step expands each argument individually before they are
379 // inserted into the macro body.
Corentin Wallez2bd9c442016-09-20 16:39:18 -0400380 size_t numTokens = 0;
Corentin Wallez054f7ed2016-09-20 17:15:59 -0400381 for (auto &arg : *args)
alokp@chromium.org7fc38dd2012-06-14 18:23:23 +0000382 {
alokp@chromium.org7fc38dd2012-06-14 18:23:23 +0000383 TokenLexer lexer(&arg);
Olli Etuahof1cf5e62016-11-22 17:36:49 +0000384 if (mAllowedMacroExpansionDepth < 1)
385 {
386 mDiagnostics->report(Diagnostics::PP_MACRO_INVOCATION_CHAIN_TOO_DEEP, token.location,
387 token.text);
388 return false;
389 }
390 MacroExpander expander(&lexer, mMacroSet, mDiagnostics, mAllowedMacroExpansionDepth - 1);
alokp@chromium.org7fc38dd2012-06-14 18:23:23 +0000391
392 arg.clear();
393 expander.lex(&token);
394 while (token.type != Token::LAST)
395 {
396 arg.push_back(token);
397 expander.lex(&token);
Corentin Wallez2bd9c442016-09-20 16:39:18 -0400398 numTokens++;
399 if (numTokens + mTotalTokensInContexts > kMaxContextTokens)
400 {
401 mDiagnostics->report(Diagnostics::PP_OUT_OF_MEMORY, token.location, token.text);
402 return false;
403 }
alokp@chromium.org7fc38dd2012-06-14 18:23:23 +0000404 }
405 }
406 return true;
407}
408
Zhenyao Mod526f982014-05-13 14:51:19 -0700409void MacroExpander::replaceMacroParams(const Macro &macro,
410 const std::vector<MacroArg> &args,
411 std::vector<Token> *replacements)
alokp@chromium.org7fc38dd2012-06-14 18:23:23 +0000412{
daniel@transgaming.coma16a55f2012-12-20 20:51:54 +0000413 for (std::size_t i = 0; i < macro.replacements.size(); ++i)
alokp@chromium.org7fc38dd2012-06-14 18:23:23 +0000414 {
Corentin Wallez2bd9c442016-09-20 16:39:18 -0400415 if (!replacements->empty() &&
416 replacements->size() + mTotalTokensInContexts > kMaxContextTokens)
417 {
418 const Token &token = replacements->back();
419 mDiagnostics->report(Diagnostics::PP_OUT_OF_MEMORY, token.location, token.text);
420 return;
421 }
422
Zhenyao Mod526f982014-05-13 14:51:19 -0700423 const Token &repl = macro.replacements[i];
alokp@chromium.org7fc38dd2012-06-14 18:23:23 +0000424 if (repl.type != Token::IDENTIFIER)
425 {
426 replacements->push_back(repl);
427 continue;
428 }
429
430 // TODO(alokp): Optimize this.
431 // There is no need to search for macro params every time.
432 // The param index can be cached with the replacement token.
433 Macro::Parameters::const_iterator iter = std::find(
alokp@chromium.org5b6a68e2012-06-28 20:29:13 +0000434 macro.parameters.begin(), macro.parameters.end(), repl.text);
alokp@chromium.org7fc38dd2012-06-14 18:23:23 +0000435 if (iter == macro.parameters.end())
436 {
437 replacements->push_back(repl);
438 continue;
439 }
440
daniel@transgaming.coma16a55f2012-12-20 20:51:54 +0000441 std::size_t iArg = std::distance(macro.parameters.begin(), iter);
Zhenyao Mod526f982014-05-13 14:51:19 -0700442 const MacroArg &arg = args[iArg];
alokp@chromium.org7fc38dd2012-06-14 18:23:23 +0000443 if (arg.empty())
444 {
445 continue;
446 }
daniel@transgaming.coma16a55f2012-12-20 20:51:54 +0000447 std::size_t iRepl = replacements->size();
alokp@chromium.org7fc38dd2012-06-14 18:23:23 +0000448 replacements->insert(replacements->end(), arg.begin(), arg.end());
449 // The replacement token inherits padding properties from
450 // macro replacement token.
451 replacements->at(iRepl).setHasLeadingSpace(repl.hasLeadingSpace());
452 }
alokp@chromium.org04d7d222012-05-16 19:24:07 +0000453}
454
Corentin Wallez054f7ed2016-09-20 17:15:59 -0400455MacroExpander::MacroContext::MacroContext() : macro(0), index(0)
456{
457}
458
459bool MacroExpander::MacroContext::empty() const
460{
461 return index == replacements.size();
462}
463
464const Token &MacroExpander::MacroContext::get()
465{
466 return replacements[index++];
467}
468
469void MacroExpander::MacroContext::unget()
470{
471 ASSERT(index > 0);
472 --index;
473}
474
alokp@chromium.org04d7d222012-05-16 19:24:07 +0000475} // namespace pp
476