1 | //== unittests/Sema/SemaNoloadLookupTest.cpp -------------------------========// |
2 | // |
3 | // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. |
4 | // See https://llvm.org/LICENSE.txt for license information. |
5 | // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception |
6 | // |
7 | //===----------------------------------------------------------------------===// |
8 | |
9 | #include "clang/AST/DeclLookups.h" |
10 | #include "clang/AST/DeclarationName.h" |
11 | #include "clang/ASTMatchers/ASTMatchFinder.h" |
12 | #include "clang/ASTMatchers/ASTMatchers.h" |
13 | #include "clang/Frontend/CompilerInstance.h" |
14 | #include "clang/Frontend/FrontendAction.h" |
15 | #include "clang/Frontend/FrontendActions.h" |
16 | #include "clang/Parse/ParseAST.h" |
17 | #include "clang/Sema/Lookup.h" |
18 | #include "clang/Sema/Sema.h" |
19 | #include "clang/Sema/SemaConsumer.h" |
20 | #include "clang/Tooling/Tooling.h" |
21 | #include "gtest/gtest.h" |
22 | |
23 | using namespace llvm; |
24 | using namespace clang; |
25 | using namespace clang::tooling; |
26 | |
27 | namespace { |
28 | |
29 | class NoloadLookupTest : public ::testing::Test { |
30 | void SetUp() override { |
31 | ASSERT_FALSE( |
32 | sys::fs::createUniqueDirectory("modules-no-comments-test" , TestDir)); |
33 | } |
34 | |
35 | void TearDown() override { sys::fs::remove_directories(path: TestDir); } |
36 | |
37 | public: |
38 | SmallString<256> TestDir; |
39 | |
40 | void addFile(StringRef Path, StringRef Contents) { |
41 | ASSERT_FALSE(sys::path::is_absolute(Path)); |
42 | |
43 | SmallString<256> AbsPath(TestDir); |
44 | sys::path::append(path&: AbsPath, a: Path); |
45 | |
46 | ASSERT_FALSE( |
47 | sys::fs::create_directories(llvm::sys::path::parent_path(AbsPath))); |
48 | |
49 | std::error_code EC; |
50 | llvm::raw_fd_ostream OS(AbsPath, EC); |
51 | ASSERT_FALSE(EC); |
52 | OS << Contents; |
53 | } |
54 | |
55 | std::string GenerateModuleInterface(StringRef ModuleName, |
56 | StringRef Contents) { |
57 | std::string FileName = llvm::Twine(ModuleName + ".cppm" ).str(); |
58 | addFile(Path: FileName, Contents); |
59 | |
60 | IntrusiveRefCntPtr<DiagnosticsEngine> Diags = |
61 | CompilerInstance::createDiagnostics(Opts: new DiagnosticOptions()); |
62 | CreateInvocationOptions CIOpts; |
63 | CIOpts.Diags = Diags; |
64 | CIOpts.VFS = llvm::vfs::createPhysicalFileSystem(); |
65 | |
66 | std::string CacheBMIPath = |
67 | llvm::Twine(TestDir + "/" + ModuleName + ".pcm" ).str(); |
68 | std::string PrebuiltModulePath = |
69 | "-fprebuilt-module-path=" + TestDir.str().str(); |
70 | const char *Args[] = {"clang++" , |
71 | "-std=c++20" , |
72 | "--precompile" , |
73 | PrebuiltModulePath.c_str(), |
74 | "-working-directory" , |
75 | TestDir.c_str(), |
76 | "-I" , |
77 | TestDir.c_str(), |
78 | FileName.c_str()}; |
79 | std::shared_ptr<CompilerInvocation> Invocation = |
80 | createInvocation(Args, Opts: CIOpts); |
81 | EXPECT_TRUE(Invocation); |
82 | |
83 | CompilerInstance Instance; |
84 | Instance.setDiagnostics(Diags.get()); |
85 | Instance.setInvocation(Invocation); |
86 | Instance.getFrontendOpts().OutputFile = CacheBMIPath; |
87 | GenerateReducedModuleInterfaceAction Action; |
88 | EXPECT_TRUE(Instance.ExecuteAction(Action)); |
89 | EXPECT_FALSE(Diags->hasErrorOccurred()); |
90 | |
91 | return CacheBMIPath; |
92 | } |
93 | }; |
94 | |
95 | struct TrivialVisibleDeclConsumer : public VisibleDeclConsumer { |
96 | TrivialVisibleDeclConsumer() {} |
97 | void EnteredContext(DeclContext *Ctx) override {} |
98 | void FoundDecl(NamedDecl *ND, NamedDecl *Hiding, DeclContext *Ctx, |
99 | bool InBaseClass) override { |
100 | FoundNum++; |
101 | } |
102 | |
103 | int FoundNum = 0; |
104 | }; |
105 | |
106 | class NoloadLookupConsumer : public SemaConsumer { |
107 | public: |
108 | void InitializeSema(Sema &S) override { SemaPtr = &S; } |
109 | |
110 | bool HandleTopLevelDecl(DeclGroupRef D) override { |
111 | if (!D.isSingleDecl()) |
112 | return true; |
113 | |
114 | Decl *TD = D.getSingleDecl(); |
115 | |
116 | auto *ID = dyn_cast<ImportDecl>(Val: TD); |
117 | if (!ID) |
118 | return true; |
119 | |
120 | clang::Module *M = ID->getImportedModule(); |
121 | assert(M); |
122 | if (M->Name != "R" ) |
123 | return true; |
124 | |
125 | auto *Std = SemaPtr->getStdNamespace(); |
126 | EXPECT_TRUE(Std); |
127 | TrivialVisibleDeclConsumer Consumer; |
128 | SemaPtr->LookupVisibleDecls(Std, Sema::LookupNameKind::LookupOrdinaryName, |
129 | Consumer, |
130 | /*IncludeGlobalScope=*/true, |
131 | /*IncludeDependentBases=*/false, |
132 | /*LoadExternal=*/false); |
133 | EXPECT_EQ(Consumer.FoundNum, 1); |
134 | return true; |
135 | } |
136 | |
137 | private: |
138 | Sema *SemaPtr = nullptr; |
139 | }; |
140 | |
141 | class NoloadLookupAction : public ASTFrontendAction { |
142 | std::unique_ptr<ASTConsumer> |
143 | CreateASTConsumer(CompilerInstance &CI, StringRef /*Unused*/) override { |
144 | return std::make_unique<NoloadLookupConsumer>(); |
145 | } |
146 | }; |
147 | |
148 | TEST_F(NoloadLookupTest, NonModulesTest) { |
149 | GenerateModuleInterface(ModuleName: "M" , Contents: R"cpp( |
150 | module; |
151 | namespace std { |
152 | int What(); |
153 | |
154 | void bar(int x = What()) { |
155 | } |
156 | } |
157 | export module M; |
158 | export using std::bar; |
159 | )cpp" ); |
160 | |
161 | GenerateModuleInterface(ModuleName: "R" , Contents: R"cpp( |
162 | module; |
163 | namespace std { |
164 | class Another; |
165 | int What(Another); |
166 | int What(); |
167 | } |
168 | export module R; |
169 | )cpp" ); |
170 | |
171 | const char *test_file_contents = R"cpp( |
172 | import M; |
173 | namespace std { |
174 | void use() { |
175 | bar(); |
176 | } |
177 | } |
178 | import R; |
179 | )cpp" ; |
180 | std::string DepArg = "-fprebuilt-module-path=" + TestDir.str().str(); |
181 | EXPECT_TRUE(runToolOnCodeWithArgs(std::make_unique<NoloadLookupAction>(), |
182 | test_file_contents, |
183 | { |
184 | "-std=c++20" , |
185 | DepArg.c_str(), |
186 | "-I" , |
187 | TestDir.c_str(), |
188 | }, |
189 | "test.cpp" )); |
190 | } |
191 | |
192 | } // namespace |
193 | |