opnsense-src/lib/Lex/Preprocessor.cpp

544 lines
20 KiB
C++
Raw Normal View History

2009-06-02 13:58:47 -04:00
//===--- Preprocess.cpp - C Language Family Preprocessor Implementation ---===//
//
// The LLVM Compiler Infrastructure
//
// This file is distributed under the University of Illinois Open Source
// License. See LICENSE.TXT for details.
//
//===----------------------------------------------------------------------===//
//
// This file implements the Preprocessor interface.
//
//===----------------------------------------------------------------------===//
//
// Options to support:
// -H - Print the name of each header file used.
// -d[DNI] - Dump various things.
// -fworking-directory - #line's with preprocessor's working dir.
// -fpreprocessed
// -dependency-file,-M,-MM,-MF,-MG,-MP,-MT,-MQ,-MD,-MMD
// -W*
// -w
//
// Messages to emit:
// "Multiple include guards may be useful for:\n"
//
//===----------------------------------------------------------------------===//
#include "clang/Lex/Preprocessor.h"
2009-12-15 13:49:47 -05:00
#include "MacroArgs.h"
2010-01-15 10:39:40 -05:00
#include "clang/Lex/ExternalPreprocessorSource.h"
2009-06-02 13:58:47 -04:00
#include "clang/Lex/HeaderSearch.h"
#include "clang/Lex/MacroInfo.h"
#include "clang/Lex/Pragma.h"
2010-03-21 06:50:08 -04:00
#include "clang/Lex/PreprocessingRecord.h"
2009-06-02 13:58:47 -04:00
#include "clang/Lex/ScratchBuffer.h"
#include "clang/Lex/LexDiagnostic.h"
#include "clang/Lex/CodeCompletionHandler.h"
2009-06-02 13:58:47 -04:00
#include "clang/Basic/SourceManager.h"
#include "clang/Basic/FileManager.h"
#include "clang/Basic/TargetInfo.h"
#include "llvm/ADT/APFloat.h"
#include "llvm/ADT/SmallVector.h"
#include "llvm/Support/MemoryBuffer.h"
2009-10-14 14:03:49 -04:00
#include "llvm/Support/raw_ostream.h"
2009-06-02 13:58:47 -04:00
using namespace clang;
//===----------------------------------------------------------------------===//
2010-01-15 10:39:40 -05:00
ExternalPreprocessorSource::~ExternalPreprocessorSource() { }
2009-06-02 13:58:47 -04:00
Preprocessor::Preprocessor(Diagnostic &diags, const LangOptions &opts,
2009-11-18 09:59:57 -05:00
const TargetInfo &target, SourceManager &SM,
2009-06-02 13:58:47 -04:00
HeaderSearch &Headers,
2009-11-18 09:59:57 -05:00
IdentifierInfoLookup* IILookup,
bool OwnsHeaders)
2009-06-02 13:58:47 -04:00
: Diags(&diags), Features(opts), Target(target),FileMgr(Headers.getFileMgr()),
SourceMgr(SM),
HeaderInfo(Headers), ExternalSource(0),
Identifiers(opts, IILookup), BuiltinInfo(Target), CodeComplete(0),
CodeCompletionFile(0), SkipMainFilePreamble(0, true), CurPPLexer(0),
CurDirLookup(0), Callbacks(0), MacroArgCache(0), Record(0), MIChainHead(0),
MICache(0) {
2009-06-02 13:58:47 -04:00
ScratchBuf = new ScratchBuffer(SourceMgr);
CounterValue = 0; // __COUNTER__ starts at 0.
2009-11-18 09:59:57 -05:00
OwnsHeaderSearch = OwnsHeaders;
2009-10-14 14:03:49 -04:00
2009-06-02 13:58:47 -04:00
// Clear stats.
NumDirectives = NumDefined = NumUndefined = NumPragma = 0;
NumIf = NumElse = NumEndif = 0;
NumEnteredSourceFiles = 0;
NumMacroExpanded = NumFnMacroExpanded = NumBuiltinMacroExpanded = 0;
NumFastMacroExpanded = NumTokenPaste = NumFastTokenPaste = 0;
2009-10-14 14:03:49 -04:00
MaxIncludeStackDepth = 0;
2009-06-02 13:58:47 -04:00
NumSkipped = 0;
// Default to discarding comments.
KeepComments = false;
KeepMacroComments = false;
2009-10-14 14:03:49 -04:00
2009-06-02 13:58:47 -04:00
// Macro expansion is enabled.
DisableMacroExpansion = false;
InMacroArgs = false;
NumCachedTokenLexers = 0;
CachedLexPos = 0;
2010-01-15 10:39:40 -05:00
// We haven't read anything from the external source.
ReadMacrosFromExternalSource = false;
2010-03-16 12:52:15 -04:00
2009-06-02 13:58:47 -04:00
// "Poison" __VA_ARGS__, which can only appear in the expansion of a macro.
// This gets unpoisoned where it is allowed.
(Ident__VA_ARGS__ = getIdentifierInfo("__VA_ARGS__"))->setIsPoisoned();
SetPoisonReason(Ident__VA_ARGS__,diag::ext_pp_bad_vaargs_use);
2009-10-14 14:03:49 -04:00
2009-06-02 13:58:47 -04:00
// Initialize the pragma handlers.
2010-07-13 13:21:42 -04:00
PragmaHandlers = new PragmaNamespace(llvm::StringRef());
2009-06-02 13:58:47 -04:00
RegisterBuiltinPragmas();
2009-10-14 14:03:49 -04:00
2009-06-02 13:58:47 -04:00
// Initialize builtin macros like __LINE__ and friends.
RegisterBuiltinMacros();
if(Features.Borland) {
Ident__exception_info = getIdentifierInfo("_exception_info");
Ident___exception_info = getIdentifierInfo("__exception_info");
Ident_GetExceptionInfo = getIdentifierInfo("GetExceptionInformation");
Ident__exception_code = getIdentifierInfo("_exception_code");
Ident___exception_code = getIdentifierInfo("__exception_code");
Ident_GetExceptionCode = getIdentifierInfo("GetExceptionCode");
Ident__abnormal_termination = getIdentifierInfo("_abnormal_termination");
Ident___abnormal_termination = getIdentifierInfo("__abnormal_termination");
Ident_AbnormalTermination = getIdentifierInfo("AbnormalTermination");
} else {
Ident__exception_info = Ident__exception_code = Ident__abnormal_termination = 0;
Ident___exception_info = Ident___exception_code = Ident___abnormal_termination = 0;
Ident_GetExceptionInfo = Ident_GetExceptionCode = Ident_AbnormalTermination = 0;
}
2009-06-02 13:58:47 -04:00
}
Preprocessor::~Preprocessor() {
assert(BacktrackPositions.empty() && "EnableBacktrack/Backtrack imbalance!");
assert(MacroExpandingLexersStack.empty() && MacroExpandedTokens.empty() &&
"Preprocessor::HandleEndOfTokenLexer should have cleared those");
2009-06-02 13:58:47 -04:00
while (!IncludeMacroStack.empty()) {
delete IncludeMacroStack.back().TheLexer;
delete IncludeMacroStack.back().TheTokenLexer;
IncludeMacroStack.pop_back();
}
// Free any macro definitions.
for (MacroInfoChain *I = MIChainHead ; I ; I = I->Next)
I->MI.Destroy();
2009-10-14 14:03:49 -04:00
2009-06-02 13:58:47 -04:00
// Free any cached macro expanders.
for (unsigned i = 0, e = NumCachedTokenLexers; i != e; ++i)
delete TokenLexerCache[i];
2010-03-16 12:52:15 -04:00
2009-12-15 13:49:47 -05:00
// Free any cached MacroArgs.
for (MacroArgs *ArgList = MacroArgCache; ArgList; )
ArgList = ArgList->deallocate();
2009-10-14 14:03:49 -04:00
2009-06-02 13:58:47 -04:00
// Release pragma information.
delete PragmaHandlers;
// Delete the scratch buffer info.
delete ScratchBuf;
2009-11-18 09:59:57 -05:00
// Delete the header search info, if we own it.
if (OwnsHeaderSearch)
delete &HeaderInfo;
2009-06-02 13:58:47 -04:00
delete Callbacks;
}
void Preprocessor::setPTHManager(PTHManager* pm) {
PTH.reset(pm);
2009-10-23 10:22:18 -04:00
FileMgr.addStatCache(PTH->createStatCache());
2009-06-02 13:58:47 -04:00
}
void Preprocessor::DumpToken(const Token &Tok, bool DumpFlags) const {
2009-10-14 14:03:49 -04:00
llvm::errs() << tok::getTokenName(Tok.getKind()) << " '"
<< getSpelling(Tok) << "'";
2009-06-02 13:58:47 -04:00
if (!DumpFlags) return;
2009-10-14 14:03:49 -04:00
llvm::errs() << "\t";
2009-06-02 13:58:47 -04:00
if (Tok.isAtStartOfLine())
2009-10-14 14:03:49 -04:00
llvm::errs() << " [StartOfLine]";
2009-06-02 13:58:47 -04:00
if (Tok.hasLeadingSpace())
2009-10-14 14:03:49 -04:00
llvm::errs() << " [LeadingSpace]";
2009-06-02 13:58:47 -04:00
if (Tok.isExpandDisabled())
2009-10-14 14:03:49 -04:00
llvm::errs() << " [ExpandDisabled]";
2009-06-02 13:58:47 -04:00
if (Tok.needsCleaning()) {
const char *Start = SourceMgr.getCharacterData(Tok.getLocation());
llvm::errs() << " [UnClean='" << llvm::StringRef(Start, Tok.getLength())
2009-10-14 14:03:49 -04:00
<< "']";
2009-06-02 13:58:47 -04:00
}
2009-10-14 14:03:49 -04:00
llvm::errs() << "\tLoc=<";
2009-06-02 13:58:47 -04:00
DumpLocation(Tok.getLocation());
2009-10-14 14:03:49 -04:00
llvm::errs() << ">";
2009-06-02 13:58:47 -04:00
}
void Preprocessor::DumpLocation(SourceLocation Loc) const {
Loc.dump(SourceMgr);
}
void Preprocessor::DumpMacro(const MacroInfo &MI) const {
2009-10-14 14:03:49 -04:00
llvm::errs() << "MACRO: ";
2009-06-02 13:58:47 -04:00
for (unsigned i = 0, e = MI.getNumTokens(); i != e; ++i) {
DumpToken(MI.getReplacementToken(i));
2009-10-14 14:03:49 -04:00
llvm::errs() << " ";
2009-06-02 13:58:47 -04:00
}
2009-10-14 14:03:49 -04:00
llvm::errs() << "\n";
2009-06-02 13:58:47 -04:00
}
void Preprocessor::PrintStats() {
2009-10-14 14:03:49 -04:00
llvm::errs() << "\n*** Preprocessor Stats:\n";
llvm::errs() << NumDirectives << " directives found:\n";
llvm::errs() << " " << NumDefined << " #define.\n";
llvm::errs() << " " << NumUndefined << " #undef.\n";
llvm::errs() << " #include/#include_next/#import:\n";
llvm::errs() << " " << NumEnteredSourceFiles << " source files entered.\n";
llvm::errs() << " " << MaxIncludeStackDepth << " max include stack depth\n";
llvm::errs() << " " << NumIf << " #if/#ifndef/#ifdef.\n";
llvm::errs() << " " << NumElse << " #else/#elif.\n";
llvm::errs() << " " << NumEndif << " #endif.\n";
llvm::errs() << " " << NumPragma << " #pragma.\n";
llvm::errs() << NumSkipped << " #if/#ifndef#ifdef regions skipped\n";
llvm::errs() << NumMacroExpanded << "/" << NumFnMacroExpanded << "/"
2009-06-02 13:58:47 -04:00
<< NumBuiltinMacroExpanded << " obj/fn/builtin macros expanded, "
<< NumFastMacroExpanded << " on the fast path.\n";
2009-10-14 14:03:49 -04:00
llvm::errs() << (NumFastTokenPaste+NumTokenPaste)
2009-06-02 13:58:47 -04:00
<< " token paste (##) operations performed, "
<< NumFastTokenPaste << " on the fast path.\n";
}
2010-03-16 12:52:15 -04:00
Preprocessor::macro_iterator
Preprocessor::macro_begin(bool IncludeExternalMacros) const {
if (IncludeExternalMacros && ExternalSource &&
2010-01-15 10:39:40 -05:00
!ReadMacrosFromExternalSource) {
ReadMacrosFromExternalSource = true;
ExternalSource->ReadDefinedMacros();
}
2010-03-16 12:52:15 -04:00
return Macros.begin();
2010-01-15 10:39:40 -05:00
}
size_t Preprocessor::getTotalMemory() const {
return BP.getTotalMemory() + MacroExpandedTokens.capacity()*sizeof(Token);
}
2010-03-16 12:52:15 -04:00
Preprocessor::macro_iterator
Preprocessor::macro_end(bool IncludeExternalMacros) const {
if (IncludeExternalMacros && ExternalSource &&
2010-01-15 10:39:40 -05:00
!ReadMacrosFromExternalSource) {
ReadMacrosFromExternalSource = true;
ExternalSource->ReadDefinedMacros();
}
2010-03-16 12:52:15 -04:00
return Macros.end();
2010-01-15 10:39:40 -05:00
}
2010-03-16 12:52:15 -04:00
bool Preprocessor::SetCodeCompletionPoint(const FileEntry *File,
unsigned TruncateAtLine,
2009-12-15 13:49:47 -05:00
unsigned TruncateAtColumn) {
using llvm::MemoryBuffer;
CodeCompletionFile = File;
// Okay to clear out the code-completion point by passing NULL.
if (!CodeCompletionFile)
return false;
// Load the actual file's contents.
2010-03-21 06:50:08 -04:00
bool Invalid = false;
const MemoryBuffer *Buffer = SourceMgr.getMemoryBufferForFile(File, &Invalid);
if (Invalid)
2009-12-15 13:49:47 -05:00
return true;
// Find the byte position of the truncation point.
const char *Position = Buffer->getBufferStart();
for (unsigned Line = 1; Line < TruncateAtLine; ++Line) {
for (; *Position; ++Position) {
if (*Position != '\r' && *Position != '\n')
continue;
2010-03-16 12:52:15 -04:00
2009-12-15 13:49:47 -05:00
// Eat \r\n or \n\r as a single line.
if ((Position[1] == '\r' || Position[1] == '\n') &&
Position[0] != Position[1])
++Position;
++Position;
break;
}
}
2010-03-16 12:52:15 -04:00
2009-12-15 13:49:47 -05:00
Position += TruncateAtColumn - 1;
2010-03-16 12:52:15 -04:00
2009-12-15 13:49:47 -05:00
// Truncate the buffer.
if (Position < Buffer->getBufferEnd()) {
2010-04-06 11:53:59 -04:00
llvm::StringRef Data(Buffer->getBufferStart(),
Position-Buffer->getBufferStart());
2010-03-16 12:52:15 -04:00
MemoryBuffer *TruncatedBuffer
2010-04-06 11:53:59 -04:00
= MemoryBuffer::getMemBufferCopy(Data, Buffer->getBufferIdentifier());
2009-12-15 13:49:47 -05:00
SourceMgr.overrideFileContents(File, TruncatedBuffer);
}
return false;
}
bool Preprocessor::isCodeCompletionFile(SourceLocation FileLoc) const {
return CodeCompletionFile && FileLoc.isFileID() &&
SourceMgr.getFileEntryForID(SourceMgr.getFileID(FileLoc))
== CodeCompletionFile;
}
void Preprocessor::CodeCompleteNaturalLanguage() {
SetCodeCompletionPoint(0, 0, 0);
getDiagnostics().setSuppressAllDiagnostics(true);
if (CodeComplete)
CodeComplete->CodeCompleteNaturalLanguage();
}
2010-03-03 12:28:16 -05:00
/// getSpelling - This method is used to get the spelling of a token into a
/// SmallVector. Note that the returned StringRef may not point to the
/// supplied buffer if a copy can be avoided.
llvm::StringRef Preprocessor::getSpelling(const Token &Tok,
2010-03-16 12:52:15 -04:00
llvm::SmallVectorImpl<char> &Buffer,
bool *Invalid) const {
// NOTE: this has to be checked *before* testing for an IdentifierInfo.
if (Tok.isNot(tok::raw_identifier)) {
// Try the fast path.
if (const IdentifierInfo *II = Tok.getIdentifierInfo())
return II->getName();
}
2010-03-03 12:28:16 -05:00
// Resize the buffer if we need to copy into it.
if (Tok.needsCleaning())
Buffer.resize(Tok.getLength());
const char *Ptr = Buffer.data();
2010-03-16 12:52:15 -04:00
unsigned Len = getSpelling(Tok, Ptr, Invalid);
2010-03-03 12:28:16 -05:00
return llvm::StringRef(Ptr, Len);
}
2009-06-02 13:58:47 -04:00
/// CreateString - Plop the specified string into a scratch buffer and return a
/// location for it. If specified, the source location provides a source
/// location for the token.
void Preprocessor::CreateString(const char *Buf, unsigned Len, Token &Tok,
SourceLocation ExpansionLoc) {
2009-06-02 13:58:47 -04:00
Tok.setLength(Len);
2009-10-14 14:03:49 -04:00
2009-06-02 13:58:47 -04:00
const char *DestPtr;
SourceLocation Loc = ScratchBuf->getToken(Buf, Len, DestPtr);
2009-10-14 14:03:49 -04:00
if (ExpansionLoc.isValid())
Loc = SourceMgr.createInstantiationLoc(Loc, ExpansionLoc,
ExpansionLoc, Len);
2009-06-02 13:58:47 -04:00
Tok.setLocation(Loc);
2009-10-14 14:03:49 -04:00
// If this is a raw identifier or a literal token, set the pointer data.
if (Tok.is(tok::raw_identifier))
Tok.setRawIdentifierData(DestPtr);
else if (Tok.isLiteral())
2009-06-02 13:58:47 -04:00
Tok.setLiteralData(DestPtr);
}
//===----------------------------------------------------------------------===//
// Preprocessor Initialization Methods
//===----------------------------------------------------------------------===//
/// EnterMainSourceFile - Enter the specified FileID as the main source file,
/// which implicitly adds the builtin defines etc.
2010-05-04 12:12:48 -04:00
void Preprocessor::EnterMainSourceFile() {
2009-06-02 13:58:47 -04:00
// We do not allow the preprocessor to reenter the main file. Doing so will
// cause FileID's to accumulate information from both runs (e.g. #line
// information) and predefined macros aren't guaranteed to be set properly.
assert(NumEnteredSourceFiles == 0 && "Cannot reenter the main file!");
FileID MainFileID = SourceMgr.getMainFileID();
2009-10-14 14:03:49 -04:00
2009-06-02 13:58:47 -04:00
// Enter the main file source buffer.
2010-05-04 12:12:48 -04:00
EnterSourceFile(MainFileID, 0, SourceLocation());
2009-10-14 14:03:49 -04:00
// If we've been asked to skip bytes in the main file (e.g., as part of a
// precompiled preamble), do so now.
if (SkipMainFilePreamble.first > 0)
CurLexer->SkipBytes(SkipMainFilePreamble.first,
SkipMainFilePreamble.second);
2009-06-02 13:58:47 -04:00
// Tell the header info that the main file was entered. If the file is later
// #imported, it won't be re-entered.
if (const FileEntry *FE = SourceMgr.getFileEntryForID(MainFileID))
HeaderInfo.IncrementIncludeCount(FE);
2009-10-14 14:03:49 -04:00
2010-01-01 05:34:51 -05:00
// Preprocess Predefines to populate the initial preprocessor state.
2009-10-14 14:03:49 -04:00
llvm::MemoryBuffer *SB =
2010-04-06 11:53:59 -04:00
llvm::MemoryBuffer::getMemBufferCopy(Predefines, "<built-in>");
assert(SB && "Cannot create predefined source buffer");
2009-06-02 13:58:47 -04:00
FileID FID = SourceMgr.createFileIDForMemBuffer(SB);
assert(!FID.isInvalid() && "Could not create FileID for predefines?");
2009-10-14 14:03:49 -04:00
2009-06-02 13:58:47 -04:00
// Start parsing the predefines.
2010-05-04 12:12:48 -04:00
EnterSourceFile(FID, 0, SourceLocation());
2009-06-02 13:58:47 -04:00
}
2010-04-02 04:55:10 -04:00
void Preprocessor::EndSourceFile() {
// Notify the client that we reached the end of the source file.
if (Callbacks)
Callbacks->EndOfMainFile();
}
2009-06-02 13:58:47 -04:00
//===----------------------------------------------------------------------===//
// Lexer Event Handling.
//===----------------------------------------------------------------------===//
/// LookUpIdentifierInfo - Given a tok::raw_identifier token, look up the
/// identifier information for the token and install it into the token,
/// updating the token kind accordingly.
IdentifierInfo *Preprocessor::LookUpIdentifierInfo(Token &Identifier) const {
assert(Identifier.getRawIdentifierData() != 0 && "No raw identifier data!");
2009-10-14 14:03:49 -04:00
2009-06-02 13:58:47 -04:00
// Look up this token, see if it is a macro, or if it is a language keyword.
IdentifierInfo *II;
if (!Identifier.needsCleaning()) {
2009-06-02 13:58:47 -04:00
// No cleaning needed, just use the characters from the lexed buffer.
II = getIdentifierInfo(llvm::StringRef(Identifier.getRawIdentifierData(),
Identifier.getLength()));
2009-06-02 13:58:47 -04:00
} else {
// Cleaning needed, alloca a buffer, clean into it, then use the buffer.
2010-03-16 12:52:15 -04:00
llvm::SmallString<64> IdentifierBuffer;
2010-03-03 12:28:16 -05:00
llvm::StringRef CleanedStr = getSpelling(Identifier, IdentifierBuffer);
II = getIdentifierInfo(CleanedStr);
2009-06-02 13:58:47 -04:00
}
// Update the token info (identifier info and appropriate token kind).
2009-06-02 13:58:47 -04:00
Identifier.setIdentifierInfo(II);
Identifier.setKind(II->getTokenID());
2009-06-02 13:58:47 -04:00
return II;
}
void Preprocessor::SetPoisonReason(IdentifierInfo *II, unsigned DiagID) {
PoisonReasons[II] = DiagID;
}
void Preprocessor::PoisonSEHIdentifiers(bool Poison) {
assert(Ident__exception_code && Ident__exception_info);
assert(Ident___exception_code && Ident___exception_info);
Ident__exception_code->setIsPoisoned(Poison);
Ident___exception_code->setIsPoisoned(Poison);
Ident_GetExceptionCode->setIsPoisoned(Poison);
Ident__exception_info->setIsPoisoned(Poison);
Ident___exception_info->setIsPoisoned(Poison);
Ident_GetExceptionInfo->setIsPoisoned(Poison);
Ident__abnormal_termination->setIsPoisoned(Poison);
Ident___abnormal_termination->setIsPoisoned(Poison);
Ident_AbnormalTermination->setIsPoisoned(Poison);
}
void Preprocessor::HandlePoisonedIdentifier(Token & Identifier) {
assert(Identifier.getIdentifierInfo() &&
"Can't handle identifiers without identifier info!");
llvm::DenseMap<IdentifierInfo*,unsigned>::const_iterator it =
PoisonReasons.find(Identifier.getIdentifierInfo());
if(it == PoisonReasons.end())
Diag(Identifier, diag::err_pp_used_poisoned_id);
else
Diag(Identifier,it->second) << Identifier.getIdentifierInfo();
}
2009-06-02 13:58:47 -04:00
/// HandleIdentifier - This callback is invoked when the lexer reads an
/// identifier. This callback looks up the identifier in the map and/or
/// potentially macro expands it or turns it into a named token (like 'for').
///
/// Note that callers of this method are guarded by checking the
/// IdentifierInfo's 'isHandleIdentifierCase' bit. If this method changes, the
/// IdentifierInfo methods that compute these properties will need to change to
/// match.
void Preprocessor::HandleIdentifier(Token &Identifier) {
assert(Identifier.getIdentifierInfo() &&
"Can't handle identifiers without identifier info!");
2009-10-14 14:03:49 -04:00
2009-06-02 13:58:47 -04:00
IdentifierInfo &II = *Identifier.getIdentifierInfo();
// If this identifier was poisoned, and if it was not produced from a macro
// expansion, emit an error.
if (II.isPoisoned() && CurPPLexer) {
HandlePoisonedIdentifier(Identifier);
2009-06-02 13:58:47 -04:00
}
2009-10-14 14:03:49 -04:00
2009-06-02 13:58:47 -04:00
// If this is a macro to be expanded, do it.
if (MacroInfo *MI = getMacroInfo(&II)) {
if (!DisableMacroExpansion && !Identifier.isExpandDisabled()) {
if (MI->isEnabled()) {
if (!HandleMacroExpandedIdentifier(Identifier, MI))
return;
} else {
// C99 6.10.3.4p2 says that a disabled macro may never again be
// expanded, even if it's in a context where it could be expanded in the
// future.
Identifier.setFlag(Token::DisableExpand);
}
}
}
// C++ 2.11p2: If this is an alternative representation of a C++ operator,
// then we act as if it is the actual operator and not the textual
// representation of it.
if (II.isCPlusPlusOperatorKeyword())
Identifier.setIdentifierInfo(0);
// If this is an extension token, diagnose its use.
// We avoid diagnosing tokens that originate from macro definitions.
// FIXME: This warning is disabled in cases where it shouldn't be,
// like "#define TY typeof", "TY(1) x".
if (II.isExtensionToken() && !DisableMacroExpansion)
Diag(Identifier, diag::ext_token_used);
}
2009-07-04 09:58:54 -04:00
void Preprocessor::AddCommentHandler(CommentHandler *Handler) {
assert(Handler && "NULL comment handler");
assert(std::find(CommentHandlers.begin(), CommentHandlers.end(), Handler) ==
CommentHandlers.end() && "Comment handler already registered");
CommentHandlers.push_back(Handler);
}
void Preprocessor::RemoveCommentHandler(CommentHandler *Handler) {
std::vector<CommentHandler *>::iterator Pos
= std::find(CommentHandlers.begin(), CommentHandlers.end(), Handler);
assert(Pos != CommentHandlers.end() && "Comment handler not registered");
CommentHandlers.erase(Pos);
}
2010-01-23 06:10:26 -05:00
bool Preprocessor::HandleComment(Token &result, SourceRange Comment) {
bool AnyPendingTokens = false;
2009-07-04 09:58:54 -04:00
for (std::vector<CommentHandler *>::iterator H = CommentHandlers.begin(),
HEnd = CommentHandlers.end();
2010-01-23 06:10:26 -05:00
H != HEnd; ++H) {
if ((*H)->HandleComment(*this, Comment))
AnyPendingTokens = true;
}
if (!AnyPendingTokens || getCommentRetentionState())
return false;
Lex(result);
return true;
2009-07-04 09:58:54 -04:00
}
CommentHandler::~CommentHandler() { }
2010-03-21 06:50:08 -04:00
CodeCompletionHandler::~CodeCompletionHandler() { }
void Preprocessor::createPreprocessingRecord(
bool IncludeNestedMacroExpansions) {
2010-03-21 06:50:08 -04:00
if (Record)
return;
Record = new PreprocessingRecord(IncludeNestedMacroExpansions);
2010-03-21 06:50:08 -04:00
addPPCallbacks(Record);
}