blob: fda884c4da4c32b0955813df695b04f9e5c0c72e [file] [log] [blame]
Reid Spencer5f016e22007-07-11 17:01:13 +00001//===--- MacroInfo.cpp - Information about #defined identifiers -----------===//
2//
3// The LLVM Compiler Infrastructure
4//
Chris Lattner0bc735f2007-12-29 19:59:25 +00005// This file is distributed under the University of Illinois Open Source
6// License. See LICENSE.TXT for details.
Reid Spencer5f016e22007-07-11 17:01:13 +00007//
8//===----------------------------------------------------------------------===//
9//
10// This file implements the MacroInfo interface.
11//
12//===----------------------------------------------------------------------===//
13
14#include "clang/Lex/MacroInfo.h"
15#include "clang/Lex/Preprocessor.h"
16using namespace clang;
17
18MacroInfo::MacroInfo(SourceLocation DefLoc) : Location(DefLoc) {
19 IsFunctionLike = false;
20 IsC99Varargs = false;
21 IsGNUVarargs = false;
22 IsBuiltinMacro = false;
Reid Spencer5f016e22007-07-11 17:01:13 +000023 IsDisabled = false;
24 IsUsed = true;
Mike Stump1eb44332009-09-09 15:08:12 +000025
Chris Lattner25c96482007-07-14 22:46:43 +000026 ArgumentList = 0;
27 NumArguments = 0;
Reid Spencer5f016e22007-07-11 17:01:13 +000028}
29
30/// isIdenticalTo - Return true if the specified macro definition is equal to
31/// this macro in spelling, arguments, and whitespace. This is used to emit
32/// duplicate definition warnings. This implements the rules in C99 6.10.3.
33///
Reid Spencer5f016e22007-07-11 17:01:13 +000034bool MacroInfo::isIdenticalTo(const MacroInfo &Other, Preprocessor &PP) const {
35 // Check # tokens in replacement, number of args, and various flags all match.
36 if (ReplacementTokens.size() != Other.ReplacementTokens.size() ||
Chris Lattner25c96482007-07-14 22:46:43 +000037 getNumArgs() != Other.getNumArgs() ||
Reid Spencer5f016e22007-07-11 17:01:13 +000038 isFunctionLike() != Other.isFunctionLike() ||
39 isC99Varargs() != Other.isC99Varargs() ||
40 isGNUVarargs() != Other.isGNUVarargs())
41 return false;
42
43 // Check arguments.
44 for (arg_iterator I = arg_begin(), OI = Other.arg_begin(), E = arg_end();
45 I != E; ++I, ++OI)
46 if (*I != *OI) return false;
Mike Stump1eb44332009-09-09 15:08:12 +000047
Reid Spencer5f016e22007-07-11 17:01:13 +000048 // Check all the tokens.
49 for (unsigned i = 0, e = ReplacementTokens.size(); i != e; ++i) {
Chris Lattnerd2177732007-07-20 16:59:19 +000050 const Token &A = ReplacementTokens[i];
51 const Token &B = Other.ReplacementTokens[i];
Chris Lattner688a2482009-03-09 20:33:32 +000052 if (A.getKind() != B.getKind())
53 return false;
Mike Stump1eb44332009-09-09 15:08:12 +000054
Chris Lattner688a2482009-03-09 20:33:32 +000055 // If this isn't the first first token, check that the whitespace and
56 // start-of-line characteristics match.
57 if (i != 0 &&
58 (A.isAtStartOfLine() != B.isAtStartOfLine() ||
59 A.hasLeadingSpace() != B.hasLeadingSpace()))
Reid Spencer5f016e22007-07-11 17:01:13 +000060 return false;
Mike Stump1eb44332009-09-09 15:08:12 +000061
Reid Spencer5f016e22007-07-11 17:01:13 +000062 // If this is an identifier, it is easy.
63 if (A.getIdentifierInfo() || B.getIdentifierInfo()) {
64 if (A.getIdentifierInfo() != B.getIdentifierInfo())
65 return false;
66 continue;
67 }
Mike Stump1eb44332009-09-09 15:08:12 +000068
Reid Spencer5f016e22007-07-11 17:01:13 +000069 // Otherwise, check the spelling.
70 if (PP.getSpelling(A) != PP.getSpelling(B))
71 return false;
72 }
Mike Stump1eb44332009-09-09 15:08:12 +000073
Reid Spencer5f016e22007-07-11 17:01:13 +000074 return true;
75}