2012-11-02 01:52:58 +08:00
|
|
|
//===- unittests/Lex/PPCallbacksTest.cpp - PPCallbacks tests ------===//
|
|
|
|
//
|
|
|
|
// The LLVM Compiler Infrastructure
|
|
|
|
//
|
|
|
|
// This file is distributed under the University of Illinois Open Source
|
|
|
|
// License. See LICENSE.TXT for details.
|
|
|
|
//
|
|
|
|
//===--------------------------------------------------------------===//
|
|
|
|
|
2012-12-04 17:45:34 +08:00
|
|
|
#include "clang/Lex/Preprocessor.h"
|
2014-01-07 19:51:46 +08:00
|
|
|
#include "clang/AST/ASTConsumer.h"
|
|
|
|
#include "clang/AST/ASTContext.h"
|
2012-11-02 01:52:58 +08:00
|
|
|
#include "clang/Basic/Diagnostic.h"
|
2014-05-11 00:31:55 +08:00
|
|
|
#include "clang/Basic/DiagnosticOptions.h"
|
2012-11-02 01:52:58 +08:00
|
|
|
#include "clang/Basic/FileManager.h"
|
|
|
|
#include "clang/Basic/LangOptions.h"
|
Reapply "Modules: Cache PCMs in memory and avoid a use-after-free"
This reverts commit r298185, effectively reapplying r298165, after fixing the
new unit tests (PR32338). The memory buffer generator doesn't null-terminate
the MemoryBuffer it creates; this version of the commit informs getMemBuffer
about that to avoid the assert.
Original commit message follows:
----
Clang's internal build system for implicit modules uses lock files to
ensure that after a process writes a PCM it will read the same one back
in (without contention from other -cc1 commands). Since PCMs are read
from disk repeatedly while invalidating, building, and importing, the
lock is not released quickly. Furthermore, the LockFileManager is not
robust in every environment. Other -cc1 commands can stall until
timeout (after about eight minutes).
This commit changes the lock file from being necessary for correctness
to a (possibly dubious) performance hack. The remaining benefit is to
reduce duplicate work in competing -cc1 commands which depend on the
same module. Follow-up commits will change the internal build system to
continue after a timeout, and reduce the timeout. Perhaps we should
reconsider blocking at all.
This also fixes a use-after-free, when one part of a compilation
validates a PCM and starts using it, and another tries to swap out the
PCM for something new.
The PCMCache is a new type called MemoryBufferCache, which saves memory
buffers based on their filename. Its ownership is shared by the
CompilerInstance and ModuleManager.
- The ModuleManager stores PCMs there that it loads from disk, never
touching the disk if the cache is hot.
- When modules fail to validate, they're removed from the cache.
- When a CompilerInstance is spawned to build a new module, each
already-loaded PCM is assumed to be valid, and is frozen to avoid
the use-after-free.
- Any newly-built module is written directly to the cache to avoid the
round-trip to the filesystem, making lock files unnecessary for
correctness.
Original patch by Manman Ren; most testcases by Adrian Prantl!
llvm-svn: 298278
2017-03-21 01:58:26 +08:00
|
|
|
#include "clang/Basic/MemoryBufferCache.h"
|
2012-11-02 01:52:58 +08:00
|
|
|
#include "clang/Basic/SourceManager.h"
|
|
|
|
#include "clang/Basic/TargetInfo.h"
|
|
|
|
#include "clang/Basic/TargetOptions.h"
|
|
|
|
#include "clang/Lex/HeaderSearch.h"
|
|
|
|
#include "clang/Lex/HeaderSearchOptions.h"
|
|
|
|
#include "clang/Lex/ModuleLoader.h"
|
|
|
|
#include "clang/Lex/PreprocessorOptions.h"
|
2013-10-12 17:29:48 +08:00
|
|
|
#include "clang/Parse/Parser.h"
|
|
|
|
#include "clang/Sema/Sema.h"
|
2012-11-02 01:52:58 +08:00
|
|
|
#include "llvm/ADT/SmallString.h"
|
2013-06-12 06:15:02 +08:00
|
|
|
#include "llvm/Support/Path.h"
|
2012-11-02 01:52:58 +08:00
|
|
|
#include "gtest/gtest.h"
|
|
|
|
|
|
|
|
using namespace clang;
|
|
|
|
|
|
|
|
namespace {
|
|
|
|
|
|
|
|
// Stub to collect data from InclusionDirective callbacks.
|
|
|
|
class InclusionDirectiveCallbacks : public PPCallbacks {
|
|
|
|
public:
|
2015-04-11 10:00:23 +08:00
|
|
|
void InclusionDirective(SourceLocation HashLoc, const Token &IncludeTok,
|
|
|
|
StringRef FileName, bool IsAngled,
|
|
|
|
CharSourceRange FilenameRange, const FileEntry *File,
|
|
|
|
StringRef SearchPath, StringRef RelativePath,
|
2018-05-10 06:25:47 +08:00
|
|
|
const Module *Imported) override {
|
|
|
|
this->HashLoc = HashLoc;
|
|
|
|
this->IncludeTok = IncludeTok;
|
|
|
|
this->FileName = FileName.str();
|
|
|
|
this->IsAngled = IsAngled;
|
|
|
|
this->FilenameRange = FilenameRange;
|
|
|
|
this->File = File;
|
|
|
|
this->SearchPath = SearchPath.str();
|
|
|
|
this->RelativePath = RelativePath.str();
|
|
|
|
this->Imported = Imported;
|
2012-11-02 01:52:58 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
SourceLocation HashLoc;
|
|
|
|
Token IncludeTok;
|
|
|
|
SmallString<16> FileName;
|
|
|
|
bool IsAngled;
|
|
|
|
CharSourceRange FilenameRange;
|
|
|
|
const FileEntry* File;
|
|
|
|
SmallString<16> SearchPath;
|
|
|
|
SmallString<16> RelativePath;
|
|
|
|
const Module* Imported;
|
|
|
|
};
|
|
|
|
|
2013-10-12 17:29:48 +08:00
|
|
|
// Stub to collect data from PragmaOpenCLExtension callbacks.
|
|
|
|
class PragmaOpenCLExtensionCallbacks : public PPCallbacks {
|
|
|
|
public:
|
|
|
|
typedef struct {
|
2013-10-14 15:13:59 +08:00
|
|
|
SmallString<16> Name;
|
2013-10-12 17:29:48 +08:00
|
|
|
unsigned State;
|
|
|
|
} CallbackParameters;
|
|
|
|
|
2015-07-23 04:46:26 +08:00
|
|
|
PragmaOpenCLExtensionCallbacks() : Name("Not called."), State(99) {}
|
2013-10-12 17:29:48 +08:00
|
|
|
|
2015-04-11 10:00:23 +08:00
|
|
|
void PragmaOpenCLExtension(clang::SourceLocation NameLoc,
|
|
|
|
const clang::IdentifierInfo *Name,
|
|
|
|
clang::SourceLocation StateLoc,
|
|
|
|
unsigned State) override {
|
2013-10-12 17:29:48 +08:00
|
|
|
this->NameLoc = NameLoc;
|
2013-10-14 15:13:59 +08:00
|
|
|
this->Name = Name->getName();
|
2013-10-12 17:29:48 +08:00
|
|
|
this->StateLoc = StateLoc;
|
|
|
|
this->State = State;
|
2015-07-23 04:46:26 +08:00
|
|
|
}
|
2013-10-12 17:29:48 +08:00
|
|
|
|
|
|
|
SourceLocation NameLoc;
|
2013-10-14 15:13:59 +08:00
|
|
|
SmallString<16> Name;
|
2013-10-12 17:29:48 +08:00
|
|
|
SourceLocation StateLoc;
|
|
|
|
unsigned State;
|
|
|
|
};
|
|
|
|
|
2012-11-02 01:52:58 +08:00
|
|
|
// PPCallbacks test fixture.
|
|
|
|
class PPCallbacksTest : public ::testing::Test {
|
|
|
|
protected:
|
|
|
|
PPCallbacksTest()
|
2015-10-08 22:20:14 +08:00
|
|
|
: InMemoryFileSystem(new vfs::InMemoryFileSystem),
|
|
|
|
FileMgr(FileSystemOptions(), InMemoryFileSystem),
|
|
|
|
DiagID(new DiagnosticIDs()), DiagOpts(new DiagnosticOptions()),
|
2014-07-06 13:26:44 +08:00
|
|
|
Diags(DiagID, DiagOpts.get(), new IgnoringDiagConsumer()),
|
|
|
|
SourceMgr(Diags, FileMgr), TargetOpts(new TargetOptions()) {
|
2012-11-02 01:52:58 +08:00
|
|
|
TargetOpts->Triple = "x86_64-apple-darwin11.1.0";
|
2014-07-06 13:26:44 +08:00
|
|
|
Target = TargetInfo::CreateTargetInfo(Diags, TargetOpts);
|
2012-11-02 01:52:58 +08:00
|
|
|
}
|
|
|
|
|
2015-10-08 22:20:14 +08:00
|
|
|
IntrusiveRefCntPtr<vfs::InMemoryFileSystem> InMemoryFileSystem;
|
2012-11-02 01:52:58 +08:00
|
|
|
FileManager FileMgr;
|
|
|
|
IntrusiveRefCntPtr<DiagnosticIDs> DiagID;
|
|
|
|
IntrusiveRefCntPtr<DiagnosticOptions> DiagOpts;
|
|
|
|
DiagnosticsEngine Diags;
|
|
|
|
SourceManager SourceMgr;
|
|
|
|
LangOptions LangOpts;
|
2014-07-06 13:26:44 +08:00
|
|
|
std::shared_ptr<TargetOptions> TargetOpts;
|
2012-11-02 01:52:58 +08:00
|
|
|
IntrusiveRefCntPtr<TargetInfo> Target;
|
|
|
|
|
|
|
|
// Register a header path as a known file and add its location
|
|
|
|
// to search path.
|
|
|
|
void AddFakeHeader(HeaderSearch& HeaderInfo, const char* HeaderPath,
|
|
|
|
bool IsSystemHeader) {
|
|
|
|
// Tell FileMgr about header.
|
2015-10-08 22:20:14 +08:00
|
|
|
InMemoryFileSystem->addFile(HeaderPath, 0,
|
|
|
|
llvm::MemoryBuffer::getMemBuffer("\n"));
|
2012-11-02 01:52:58 +08:00
|
|
|
|
|
|
|
// Add header's parent path to search path.
|
2014-10-16 06:00:40 +08:00
|
|
|
StringRef SearchPath = llvm::sys::path::parent_path(HeaderPath);
|
2012-11-02 01:52:58 +08:00
|
|
|
const DirectoryEntry *DE = FileMgr.getDirectory(SearchPath);
|
2013-01-25 09:50:28 +08:00
|
|
|
DirectoryLookup DL(DE, SrcMgr::C_User, false);
|
2012-11-02 01:52:58 +08:00
|
|
|
HeaderInfo.AddSearchPath(DL, IsSystemHeader);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Get the raw source string of the range.
|
|
|
|
StringRef GetSourceString(CharSourceRange Range) {
|
|
|
|
const char* B = SourceMgr.getCharacterData(Range.getBegin());
|
|
|
|
const char* E = SourceMgr.getCharacterData(Range.getEnd());
|
|
|
|
|
|
|
|
return StringRef(B, E - B);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Run lexer over SourceText and collect FilenameRange from
|
|
|
|
// the InclusionDirective callback.
|
|
|
|
CharSourceRange InclusionDirectiveFilenameRange(const char* SourceText,
|
|
|
|
const char* HeaderPath, bool SystemHeader) {
|
2014-10-16 06:00:40 +08:00
|
|
|
std::unique_ptr<llvm::MemoryBuffer> Buf =
|
|
|
|
llvm::MemoryBuffer::getMemBuffer(SourceText);
|
2014-08-29 15:59:55 +08:00
|
|
|
SourceMgr.setMainFileID(SourceMgr.createFileID(std::move(Buf)));
|
2012-11-02 01:52:58 +08:00
|
|
|
|
2017-06-10 03:22:32 +08:00
|
|
|
TrivialModuleLoader ModLoader;
|
Reapply "Modules: Cache PCMs in memory and avoid a use-after-free"
This reverts commit r298185, effectively reapplying r298165, after fixing the
new unit tests (PR32338). The memory buffer generator doesn't null-terminate
the MemoryBuffer it creates; this version of the commit informs getMemBuffer
about that to avoid the assert.
Original commit message follows:
----
Clang's internal build system for implicit modules uses lock files to
ensure that after a process writes a PCM it will read the same one back
in (without contention from other -cc1 commands). Since PCMs are read
from disk repeatedly while invalidating, building, and importing, the
lock is not released quickly. Furthermore, the LockFileManager is not
robust in every environment. Other -cc1 commands can stall until
timeout (after about eight minutes).
This commit changes the lock file from being necessary for correctness
to a (possibly dubious) performance hack. The remaining benefit is to
reduce duplicate work in competing -cc1 commands which depend on the
same module. Follow-up commits will change the internal build system to
continue after a timeout, and reduce the timeout. Perhaps we should
reconsider blocking at all.
This also fixes a use-after-free, when one part of a compilation
validates a PCM and starts using it, and another tries to swap out the
PCM for something new.
The PCMCache is a new type called MemoryBufferCache, which saves memory
buffers based on their filename. Its ownership is shared by the
CompilerInstance and ModuleManager.
- The ModuleManager stores PCMs there that it loads from disk, never
touching the disk if the cache is hot.
- When modules fail to validate, they're removed from the cache.
- When a CompilerInstance is spawned to build a new module, each
already-loaded PCM is assumed to be valid, and is frozen to avoid
the use-after-free.
- Any newly-built module is written directly to the cache to avoid the
round-trip to the filesystem, making lock files unnecessary for
correctness.
Original patch by Manman Ren; most testcases by Adrian Prantl!
llvm-svn: 298278
2017-03-21 01:58:26 +08:00
|
|
|
MemoryBufferCache PCMCache;
|
2012-11-02 01:52:58 +08:00
|
|
|
|
2017-01-06 09:04:46 +08:00
|
|
|
HeaderSearch HeaderInfo(std::make_shared<HeaderSearchOptions>(), SourceMgr,
|
|
|
|
Diags, LangOpts, Target.get());
|
2012-11-02 01:52:58 +08:00
|
|
|
AddFakeHeader(HeaderInfo, HeaderPath, SystemHeader);
|
|
|
|
|
2017-01-06 03:11:36 +08:00
|
|
|
Preprocessor PP(std::make_shared<PreprocessorOptions>(), Diags, LangOpts,
|
Reapply "Modules: Cache PCMs in memory and avoid a use-after-free"
This reverts commit r298185, effectively reapplying r298165, after fixing the
new unit tests (PR32338). The memory buffer generator doesn't null-terminate
the MemoryBuffer it creates; this version of the commit informs getMemBuffer
about that to avoid the assert.
Original commit message follows:
----
Clang's internal build system for implicit modules uses lock files to
ensure that after a process writes a PCM it will read the same one back
in (without contention from other -cc1 commands). Since PCMs are read
from disk repeatedly while invalidating, building, and importing, the
lock is not released quickly. Furthermore, the LockFileManager is not
robust in every environment. Other -cc1 commands can stall until
timeout (after about eight minutes).
This commit changes the lock file from being necessary for correctness
to a (possibly dubious) performance hack. The remaining benefit is to
reduce duplicate work in competing -cc1 commands which depend on the
same module. Follow-up commits will change the internal build system to
continue after a timeout, and reduce the timeout. Perhaps we should
reconsider blocking at all.
This also fixes a use-after-free, when one part of a compilation
validates a PCM and starts using it, and another tries to swap out the
PCM for something new.
The PCMCache is a new type called MemoryBufferCache, which saves memory
buffers based on their filename. Its ownership is shared by the
CompilerInstance and ModuleManager.
- The ModuleManager stores PCMs there that it loads from disk, never
touching the disk if the cache is hot.
- When modules fail to validate, they're removed from the cache.
- When a CompilerInstance is spawned to build a new module, each
already-loaded PCM is assumed to be valid, and is frozen to avoid
the use-after-free.
- Any newly-built module is written directly to the cache to avoid the
round-trip to the filesystem, making lock files unnecessary for
correctness.
Original patch by Manman Ren; most testcases by Adrian Prantl!
llvm-svn: 298278
2017-03-21 01:58:26 +08:00
|
|
|
SourceMgr, PCMCache, HeaderInfo, ModLoader,
|
2014-06-08 16:38:12 +08:00
|
|
|
/*IILookup =*/nullptr,
|
2014-05-02 11:43:38 +08:00
|
|
|
/*OwnsHeaderSearch =*/false);
|
2014-05-02 11:43:30 +08:00
|
|
|
PP.Initialize(*Target);
|
2012-11-02 01:52:58 +08:00
|
|
|
InclusionDirectiveCallbacks* Callbacks = new InclusionDirectiveCallbacks;
|
2014-09-10 12:53:53 +08:00
|
|
|
PP.addPPCallbacks(std::unique_ptr<PPCallbacks>(Callbacks));
|
2012-11-02 01:52:58 +08:00
|
|
|
|
|
|
|
// Lex source text.
|
|
|
|
PP.EnterMainSourceFile();
|
|
|
|
|
|
|
|
while (true) {
|
|
|
|
Token Tok;
|
|
|
|
PP.Lex(Tok);
|
|
|
|
if (Tok.is(tok::eof))
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Callbacks have been executed at this point -- return filename range.
|
2018-05-10 06:25:47 +08:00
|
|
|
return Callbacks->FilenameRange;
|
2012-11-02 01:52:58 +08:00
|
|
|
}
|
2013-10-12 17:29:48 +08:00
|
|
|
|
|
|
|
PragmaOpenCLExtensionCallbacks::CallbackParameters
|
|
|
|
PragmaOpenCLExtensionCall(const char* SourceText) {
|
|
|
|
LangOptions OpenCLLangOpts;
|
|
|
|
OpenCLLangOpts.OpenCL = 1;
|
|
|
|
|
2014-10-16 06:00:40 +08:00
|
|
|
std::unique_ptr<llvm::MemoryBuffer> SourceBuf =
|
|
|
|
llvm::MemoryBuffer::getMemBuffer(SourceText, "test.cl");
|
2014-08-29 15:59:55 +08:00
|
|
|
SourceMgr.setMainFileID(SourceMgr.createFileID(std::move(SourceBuf)));
|
2013-10-12 17:29:48 +08:00
|
|
|
|
2017-06-10 03:22:32 +08:00
|
|
|
TrivialModuleLoader ModLoader;
|
Reapply "Modules: Cache PCMs in memory and avoid a use-after-free"
This reverts commit r298185, effectively reapplying r298165, after fixing the
new unit tests (PR32338). The memory buffer generator doesn't null-terminate
the MemoryBuffer it creates; this version of the commit informs getMemBuffer
about that to avoid the assert.
Original commit message follows:
----
Clang's internal build system for implicit modules uses lock files to
ensure that after a process writes a PCM it will read the same one back
in (without contention from other -cc1 commands). Since PCMs are read
from disk repeatedly while invalidating, building, and importing, the
lock is not released quickly. Furthermore, the LockFileManager is not
robust in every environment. Other -cc1 commands can stall until
timeout (after about eight minutes).
This commit changes the lock file from being necessary for correctness
to a (possibly dubious) performance hack. The remaining benefit is to
reduce duplicate work in competing -cc1 commands which depend on the
same module. Follow-up commits will change the internal build system to
continue after a timeout, and reduce the timeout. Perhaps we should
reconsider blocking at all.
This also fixes a use-after-free, when one part of a compilation
validates a PCM and starts using it, and another tries to swap out the
PCM for something new.
The PCMCache is a new type called MemoryBufferCache, which saves memory
buffers based on their filename. Its ownership is shared by the
CompilerInstance and ModuleManager.
- The ModuleManager stores PCMs there that it loads from disk, never
touching the disk if the cache is hot.
- When modules fail to validate, they're removed from the cache.
- When a CompilerInstance is spawned to build a new module, each
already-loaded PCM is assumed to be valid, and is frozen to avoid
the use-after-free.
- Any newly-built module is written directly to the cache to avoid the
round-trip to the filesystem, making lock files unnecessary for
correctness.
Original patch by Manman Ren; most testcases by Adrian Prantl!
llvm-svn: 298278
2017-03-21 01:58:26 +08:00
|
|
|
MemoryBufferCache PCMCache;
|
2017-01-06 09:04:46 +08:00
|
|
|
HeaderSearch HeaderInfo(std::make_shared<HeaderSearchOptions>(), SourceMgr,
|
|
|
|
Diags, OpenCLLangOpts, Target.get());
|
2013-10-12 17:29:48 +08:00
|
|
|
|
2017-01-06 03:11:36 +08:00
|
|
|
Preprocessor PP(std::make_shared<PreprocessorOptions>(), Diags,
|
Reapply "Modules: Cache PCMs in memory and avoid a use-after-free"
This reverts commit r298185, effectively reapplying r298165, after fixing the
new unit tests (PR32338). The memory buffer generator doesn't null-terminate
the MemoryBuffer it creates; this version of the commit informs getMemBuffer
about that to avoid the assert.
Original commit message follows:
----
Clang's internal build system for implicit modules uses lock files to
ensure that after a process writes a PCM it will read the same one back
in (without contention from other -cc1 commands). Since PCMs are read
from disk repeatedly while invalidating, building, and importing, the
lock is not released quickly. Furthermore, the LockFileManager is not
robust in every environment. Other -cc1 commands can stall until
timeout (after about eight minutes).
This commit changes the lock file from being necessary for correctness
to a (possibly dubious) performance hack. The remaining benefit is to
reduce duplicate work in competing -cc1 commands which depend on the
same module. Follow-up commits will change the internal build system to
continue after a timeout, and reduce the timeout. Perhaps we should
reconsider blocking at all.
This also fixes a use-after-free, when one part of a compilation
validates a PCM and starts using it, and another tries to swap out the
PCM for something new.
The PCMCache is a new type called MemoryBufferCache, which saves memory
buffers based on their filename. Its ownership is shared by the
CompilerInstance and ModuleManager.
- The ModuleManager stores PCMs there that it loads from disk, never
touching the disk if the cache is hot.
- When modules fail to validate, they're removed from the cache.
- When a CompilerInstance is spawned to build a new module, each
already-loaded PCM is assumed to be valid, and is frozen to avoid
the use-after-free.
- Any newly-built module is written directly to the cache to avoid the
round-trip to the filesystem, making lock files unnecessary for
correctness.
Original patch by Manman Ren; most testcases by Adrian Prantl!
llvm-svn: 298278
2017-03-21 01:58:26 +08:00
|
|
|
OpenCLLangOpts, SourceMgr, PCMCache, HeaderInfo, ModLoader,
|
2017-01-06 03:11:36 +08:00
|
|
|
/*IILookup =*/nullptr,
|
2014-05-02 11:43:30 +08:00
|
|
|
/*OwnsHeaderSearch =*/false);
|
|
|
|
PP.Initialize(*Target);
|
2013-10-12 17:29:48 +08:00
|
|
|
|
|
|
|
// parser actually sets correct pragma handlers for preprocessor
|
|
|
|
// according to LangOptions, so we init Parser to register opencl
|
|
|
|
// pragma handlers
|
2014-05-03 11:46:04 +08:00
|
|
|
ASTContext Context(OpenCLLangOpts, SourceMgr,
|
2013-10-12 17:29:48 +08:00
|
|
|
PP.getIdentifierTable(), PP.getSelectorTable(),
|
2014-05-03 11:46:04 +08:00
|
|
|
PP.getBuiltinInfo());
|
|
|
|
Context.InitBuiltinTypes(*Target);
|
|
|
|
|
2013-10-12 17:29:48 +08:00
|
|
|
ASTConsumer Consumer;
|
|
|
|
Sema S(PP, Context, Consumer);
|
|
|
|
Parser P(PP, S, false);
|
|
|
|
PragmaOpenCLExtensionCallbacks* Callbacks = new PragmaOpenCLExtensionCallbacks;
|
2014-09-10 12:53:53 +08:00
|
|
|
PP.addPPCallbacks(std::unique_ptr<PPCallbacks>(Callbacks));
|
2013-10-12 17:29:48 +08:00
|
|
|
|
|
|
|
// Lex source text.
|
|
|
|
PP.EnterMainSourceFile();
|
|
|
|
while (true) {
|
|
|
|
Token Tok;
|
|
|
|
PP.Lex(Tok);
|
|
|
|
if (Tok.is(tok::eof))
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
PragmaOpenCLExtensionCallbacks::CallbackParameters RetVal = {
|
2013-10-14 15:13:59 +08:00
|
|
|
Callbacks->Name,
|
2013-10-12 17:29:48 +08:00
|
|
|
Callbacks->State
|
|
|
|
};
|
|
|
|
return RetVal;
|
|
|
|
}
|
2012-11-02 01:52:58 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
TEST_F(PPCallbacksTest, QuotedFilename) {
|
|
|
|
const char* Source =
|
|
|
|
"#include \"quoted.h\"\n";
|
|
|
|
|
|
|
|
CharSourceRange Range =
|
|
|
|
InclusionDirectiveFilenameRange(Source, "/quoted.h", false);
|
|
|
|
|
|
|
|
ASSERT_EQ("\"quoted.h\"", GetSourceString(Range));
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(PPCallbacksTest, AngledFilename) {
|
|
|
|
const char* Source =
|
|
|
|
"#include <angled.h>\n";
|
|
|
|
|
|
|
|
CharSourceRange Range =
|
|
|
|
InclusionDirectiveFilenameRange(Source, "/angled.h", true);
|
|
|
|
|
|
|
|
ASSERT_EQ("<angled.h>", GetSourceString(Range));
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(PPCallbacksTest, QuotedInMacro) {
|
|
|
|
const char* Source =
|
|
|
|
"#define MACRO_QUOTED \"quoted.h\"\n"
|
|
|
|
"#include MACRO_QUOTED\n";
|
|
|
|
|
|
|
|
CharSourceRange Range =
|
|
|
|
InclusionDirectiveFilenameRange(Source, "/quoted.h", false);
|
|
|
|
|
|
|
|
ASSERT_EQ("\"quoted.h\"", GetSourceString(Range));
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(PPCallbacksTest, AngledInMacro) {
|
|
|
|
const char* Source =
|
|
|
|
"#define MACRO_ANGLED <angled.h>\n"
|
|
|
|
"#include MACRO_ANGLED\n";
|
|
|
|
|
|
|
|
CharSourceRange Range =
|
|
|
|
InclusionDirectiveFilenameRange(Source, "/angled.h", true);
|
|
|
|
|
|
|
|
ASSERT_EQ("<angled.h>", GetSourceString(Range));
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(PPCallbacksTest, StringizedMacroArgument) {
|
|
|
|
const char* Source =
|
|
|
|
"#define MACRO_STRINGIZED(x) #x\n"
|
|
|
|
"#include MACRO_STRINGIZED(quoted.h)\n";
|
|
|
|
|
|
|
|
CharSourceRange Range =
|
|
|
|
InclusionDirectiveFilenameRange(Source, "/quoted.h", false);
|
|
|
|
|
|
|
|
ASSERT_EQ("\"quoted.h\"", GetSourceString(Range));
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(PPCallbacksTest, ConcatenatedMacroArgument) {
|
|
|
|
const char* Source =
|
|
|
|
"#define MACRO_ANGLED <angled.h>\n"
|
|
|
|
"#define MACRO_CONCAT(x, y) x ## _ ## y\n"
|
|
|
|
"#include MACRO_CONCAT(MACRO, ANGLED)\n";
|
|
|
|
|
|
|
|
CharSourceRange Range =
|
|
|
|
InclusionDirectiveFilenameRange(Source, "/angled.h", false);
|
|
|
|
|
|
|
|
ASSERT_EQ("<angled.h>", GetSourceString(Range));
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(PPCallbacksTest, TrigraphFilename) {
|
|
|
|
const char* Source =
|
2012-11-04 04:58:26 +08:00
|
|
|
"#include \"tri\?\?-graph.h\"\n";
|
2012-11-02 01:52:58 +08:00
|
|
|
|
|
|
|
CharSourceRange Range =
|
|
|
|
InclusionDirectiveFilenameRange(Source, "/tri~graph.h", false);
|
|
|
|
|
2012-11-04 04:58:26 +08:00
|
|
|
ASSERT_EQ("\"tri\?\?-graph.h\"", GetSourceString(Range));
|
2012-11-02 01:52:58 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(PPCallbacksTest, TrigraphInMacro) {
|
|
|
|
const char* Source =
|
2012-11-04 04:58:26 +08:00
|
|
|
"#define MACRO_TRIGRAPH \"tri\?\?-graph.h\"\n"
|
2012-11-02 01:52:58 +08:00
|
|
|
"#include MACRO_TRIGRAPH\n";
|
|
|
|
|
|
|
|
CharSourceRange Range =
|
|
|
|
InclusionDirectiveFilenameRange(Source, "/tri~graph.h", false);
|
|
|
|
|
2012-11-04 04:58:26 +08:00
|
|
|
ASSERT_EQ("\"tri\?\?-graph.h\"", GetSourceString(Range));
|
2012-11-02 01:52:58 +08:00
|
|
|
}
|
|
|
|
|
2013-10-12 17:29:48 +08:00
|
|
|
TEST_F(PPCallbacksTest, OpenCLExtensionPragmaEnabled) {
|
|
|
|
const char* Source =
|
|
|
|
"#pragma OPENCL EXTENSION cl_khr_fp64 : enable\n";
|
|
|
|
|
|
|
|
PragmaOpenCLExtensionCallbacks::CallbackParameters Parameters =
|
|
|
|
PragmaOpenCLExtensionCall(Source);
|
|
|
|
|
|
|
|
ASSERT_EQ("cl_khr_fp64", Parameters.Name);
|
|
|
|
unsigned ExpectedState = 1;
|
|
|
|
ASSERT_EQ(ExpectedState, Parameters.State);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(PPCallbacksTest, OpenCLExtensionPragmaDisabled) {
|
|
|
|
const char* Source =
|
|
|
|
"#pragma OPENCL EXTENSION cl_khr_fp16 : disable\n";
|
|
|
|
|
|
|
|
PragmaOpenCLExtensionCallbacks::CallbackParameters Parameters =
|
|
|
|
PragmaOpenCLExtensionCall(Source);
|
|
|
|
|
|
|
|
ASSERT_EQ("cl_khr_fp16", Parameters.Name);
|
|
|
|
unsigned ExpectedState = 0;
|
|
|
|
ASSERT_EQ(ExpectedState, Parameters.State);
|
|
|
|
}
|
|
|
|
|
2015-06-23 07:07:51 +08:00
|
|
|
} // anonoymous namespace
|