-
Notifications
You must be signed in to change notification settings - Fork 10.4k
/
Copy pathClangImporterTests.cpp
94 lines (83 loc) · 3.6 KB
/
ClangImporterTests.cpp
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
#include "swift/AST/ASTContext.h"
#include "swift/AST/DiagnosticEngine.h"
#include "swift/AST/SearchPathOptions.h"
#include "swift/Basic/Defer.h"
#include "swift/Basic/LLVMInitialize.h"
#include "swift/Basic/LangOptions.h"
#include "swift/Basic/SourceManager.h"
#include "swift/ClangImporter/ClangImporter.h"
#include "swift/SymbolGraphGen/SymbolGraphOptions.h"
#include "llvm/Support/FileSystem.h"
#include "llvm/Support/Path.h"
#include "llvm/Support/raw_ostream.h"
#include "gtest/gtest.h"
using namespace swift;
static std::string createFilename(StringRef base, StringRef name) {
SmallString<256> path = base;
llvm::sys::path::append(path, name);
return llvm::Twine(path).str();
}
static bool emitFileWithContents(StringRef path, StringRef contents,
std::string *pathOut = nullptr) {
int FD;
if (llvm::sys::fs::openFileForWrite(path, FD))
return true;
if (pathOut)
*pathOut = path.str();
llvm::raw_fd_ostream file(FD, /*shouldClose=*/true);
file << contents;
return false;
}
static bool emitFileWithContents(StringRef base, StringRef name,
StringRef contents,
std::string *pathOut = nullptr) {
return emitFileWithContents(createFilename(base, name), contents, pathOut);
}
TEST(ClangImporterTest, emitPCHInMemory) {
// Create a temporary cache on disk and clean it up at the end.
ClangImporterOptions options;
SmallString<256> temp;
ASSERT_FALSE(llvm::sys::fs::createUniqueDirectory(
"ClangImporterTest.emitPCHInMemory", temp));
SWIFT_DEFER { llvm::sys::fs::remove_directories(temp); };
// Create a cache subdirectory for the modules and PCH.
std::string cache = createFilename(temp, "cache");
ASSERT_FALSE(llvm::sys::fs::create_directory(cache));
options.ModuleCachePath = cache;
options.PrecompiledHeaderOutputDir = cache;
// Create the includes.
std::string include = createFilename(temp, "include");
ASSERT_FALSE(llvm::sys::fs::create_directory(include));
options.ExtraArgs.emplace_back("-nosysteminc");
options.ExtraArgs.emplace_back((llvm::Twine("-I") + include).str());
ASSERT_FALSE(emitFileWithContents(include, "module.modulemap",
"module A {\n"
" header \"A.h\"\n"
"}\n"));
ASSERT_FALSE(emitFileWithContents(include, "A.h", "int foo(void);\n"));
// Create a bridging header.
ASSERT_FALSE(emitFileWithContents(temp, "bridging.h", "#import <A.h>\n",
&options.BridgingHeader));
// Set up the importer and emit a bridging PCH.
swift::LangOptions langOpts;
langOpts.Target = llvm::Triple("x86_64", "apple", "darwin");
swift::SILOptions silOpts;
swift::TypeCheckerOptions typecheckOpts;
INITIALIZE_LLVM();
swift::SearchPathOptions searchPathOpts;
swift::symbolgraphgen::SymbolGraphOptions symbolGraphOpts;
swift::SourceManager sourceMgr;
swift::DiagnosticEngine diags(sourceMgr);
std::unique_ptr<ASTContext> context(
ASTContext::get(langOpts, typecheckOpts, silOpts, searchPathOpts, options,
symbolGraphOpts, sourceMgr, diags));
auto importer = ClangImporter::create(*context);
std::string PCH = createFilename(cache, "bridging.h.pch");
ASSERT_FALSE(importer->canReadPCH(PCH));
ASSERT_FALSE(importer->emitBridgingPCH(options.BridgingHeader, PCH));
ASSERT_TRUE(importer->canReadPCH(PCH));
// Overwrite the PCH with garbage. We should still be able to read it from
// the in-memory cache.
ASSERT_FALSE(emitFileWithContents(PCH, "garbage"));
ASSERT_TRUE(importer->canReadPCH(PCH));
}