-
Notifications
You must be signed in to change notification settings - Fork 10.5k
/
Copy pathModuleDependencies.cpp
434 lines (383 loc) · 15.5 KB
/
ModuleDependencies.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
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
//===--- ModuleDependencies.h - Module Dependencies -------------*- C++ -*-===//
//
// This source file is part of the Swift.org open source project
//
// Copyright (c) 2014 - 2017 Apple Inc. and the Swift project authors
// Licensed under Apache License v2.0 with Runtime Library Exception
//
// See https://swift.org/LICENSE.txt for license information
// See https://swift.org/CONTRIBUTORS.txt for the list of Swift project authors
//
//===----------------------------------------------------------------------===//
//
// This file implements data structures for capturing module dependencies.
//
//===----------------------------------------------------------------------===//
#include "swift/AST/ModuleDependencies.h"
#include "swift/AST/Decl.h"
#include "swift/AST/SourceFile.h"
using namespace swift;
ModuleDependenciesKind& operator++(ModuleDependenciesKind& e) {
if (e == ModuleDependenciesKind::LastKind) {
llvm_unreachable("Attempting to incrementa last enum value on ModuleDependenciesKind");
}
e = ModuleDependenciesKind(static_cast<std::underlying_type<ModuleDependenciesKind>::type>(e) + 1);
return e;
}
ModuleDependenciesStorageBase::~ModuleDependenciesStorageBase() { }
bool ModuleDependencies::isSwiftModule() const {
return isSwiftTextualModule() ||
isSwiftBinaryModule() ||
isSwiftPlaceholderModule();
}
bool ModuleDependencies::isSwiftTextualModule() const {
return isa<SwiftTextualModuleDependenciesStorage>(storage.get());
}
bool ModuleDependencies::isSwiftBinaryModule() const {
return isa<SwiftBinaryModuleDependencyStorage>(storage.get());
}
bool ModuleDependencies::isSwiftPlaceholderModule() const {
return isa<SwiftPlaceholderModuleDependencyStorage>(storage.get());
}
bool ModuleDependencies::isClangModule() const {
return isa<ClangModuleDependenciesStorage>(storage.get());
}
/// Retrieve the dependencies for a Swift module.
const SwiftTextualModuleDependenciesStorage *
ModuleDependencies::getAsSwiftTextualModule() const {
return dyn_cast<SwiftTextualModuleDependenciesStorage>(storage.get());
}
/// Retrieve the dependencies for a binary Swift dependency module.
const SwiftBinaryModuleDependencyStorage *
ModuleDependencies::getAsSwiftBinaryModule() const {
return dyn_cast<SwiftBinaryModuleDependencyStorage>(storage.get());
}
/// Retrieve the dependencies for a Clang module.
const ClangModuleDependenciesStorage *
ModuleDependencies::getAsClangModule() const {
return dyn_cast<ClangModuleDependenciesStorage>(storage.get());
}
/// Retrieve the dependencies for a placeholder dependency module stub.
const SwiftPlaceholderModuleDependencyStorage *
ModuleDependencies::getAsPlaceholderDependencyModule() const {
return dyn_cast<SwiftPlaceholderModuleDependencyStorage>(storage.get());
}
void ModuleDependencies::addModuleDependency(
StringRef module, llvm::StringSet<> *alreadyAddedModules) {
if (!alreadyAddedModules || alreadyAddedModules->insert(module).second)
storage->moduleDependencies.push_back(module.str());
}
void ModuleDependencies::addModuleDependencies(
const SourceFile &sf, llvm::StringSet<> &alreadyAddedModules) {
// Add all of the module dependencies.
SmallVector<Decl *, 32> decls;
sf.getTopLevelDecls(decls);
for (auto decl : decls) {
auto importDecl = dyn_cast<ImportDecl>(decl);
if (!importDecl)
continue;
addModuleDependency(importDecl->getModulePath(), &alreadyAddedModules);
}
auto fileName = sf.getFilename();
if (fileName.empty())
return;
// If the storage is for an interface file, the only source file we
// should see is that interface file.
auto swiftStorage = cast<SwiftTextualModuleDependenciesStorage>(storage.get());
if (swiftStorage->swiftInterfaceFile) {
assert(fileName == *swiftStorage->swiftInterfaceFile);
return;
}
// Otherwise, record the source file.
swiftStorage->sourceFiles.push_back(fileName.str());
}
Optional<std::string> ModuleDependencies::getBridgingHeader() const {
auto swiftStorage = cast<SwiftTextualModuleDependenciesStorage>(storage.get());
return swiftStorage->bridgingHeaderFile;
}
void ModuleDependencies::addBridgingHeader(StringRef bridgingHeader) {
auto swiftStorage = cast<SwiftTextualModuleDependenciesStorage>(storage.get());
assert(!swiftStorage->bridgingHeaderFile);
swiftStorage->bridgingHeaderFile = bridgingHeader.str();
}
/// Add source files that the bridging header depends on.
void ModuleDependencies::addBridgingSourceFile(StringRef bridgingSourceFile) {
auto swiftStorage = cast<SwiftTextualModuleDependenciesStorage>(storage.get());
swiftStorage->bridgingSourceFiles.push_back(bridgingSourceFile.str());
}
void ModuleDependencies::addSourceFile(StringRef sourceFile) {
auto swiftStorage = cast<SwiftTextualModuleDependenciesStorage>(storage.get());
swiftStorage->sourceFiles.push_back(sourceFile.str());
}
/// Add (Clang) module on which the bridging header depends.
void ModuleDependencies::addBridgingModuleDependency(
StringRef module, llvm::StringSet<> &alreadyAddedModules) {
auto swiftStorage = cast<SwiftTextualModuleDependenciesStorage>(storage.get());
if (alreadyAddedModules.insert(module).second)
swiftStorage->bridgingModuleDependencies.push_back(module.str());
}
llvm::StringMap<ModuleDependenciesVector> &
GlobalModuleDependenciesCache::getDependenciesMap(ModuleDependenciesKind kind) {
auto it = ModuleDependenciesKindMap.find(kind);
assert(it != ModuleDependenciesKindMap.end() && "invalid dependency kind");
return it->second;
}
const llvm::StringMap<ModuleDependenciesVector> &
GlobalModuleDependenciesCache::getDependenciesMap(ModuleDependenciesKind kind) const {
auto it = ModuleDependenciesKindMap.find(kind);
assert(it != ModuleDependenciesKindMap.end() && "invalid dependency kind");
return it->second;
}
static std::string moduleBasePath(const StringRef modulePath) {
auto parent = llvm::sys::path::parent_path(modulePath);
// If the modulePath is that of a .swiftinterface contained in a .swiftmodule,
// disambiguate further to parent.
if (llvm::sys::path::extension(parent) == ".swiftmodule") {
parent = llvm::sys::path::parent_path(parent);
}
// If the module is a part of a framework, disambiguate to the framework's parent
if (llvm::sys::path::filename(parent) == "Modules") {
auto grandParent = llvm::sys::path::parent_path(parent);
if (llvm::sys::path::extension(grandParent) == ".framework") {
parent = llvm::sys::path::parent_path(grandParent);
}
}
return parent.str();
}
static bool moduleContainedInImportPathSet(const StringRef modulePath,
const llvm::StringSet<> &importPaths)
{
return importPaths.contains(moduleBasePath(modulePath));
}
static bool moduleContainedInImportPathSet(const ModuleDependencies &module,
const llvm::StringSet<> &importPaths)
{
std::string modulePath = "";
switch (module.getKind()) {
case swift::ModuleDependenciesKind::SwiftTextual: {
auto *swiftDep = module.getAsSwiftTextualModule();
if (swiftDep->swiftInterfaceFile)
modulePath = *(swiftDep->swiftInterfaceFile);
else {
// If we encountered a Swift textual dependency without an interface
// file, we are seeing the main scan module itself. This means that
// our search-path disambiguation is not necessary here.
return true;
}
break;
}
case swift::ModuleDependenciesKind::SwiftBinary: {
auto *swiftBinaryDep = module.getAsSwiftBinaryModule();
modulePath = swiftBinaryDep->compiledModulePath;
break;
}
case swift::ModuleDependenciesKind::Clang: {
auto *clangDep = module.getAsClangModule();
modulePath = clangDep->moduleMapFile;
break;
}
case swift::ModuleDependenciesKind::SwiftPlaceholder: {
// Placeholders are resolved as `true` because they are not associated with
// any specific search path.
return true;
}
default:
llvm_unreachable("Unhandled dependency kind.");
}
if (moduleContainedInImportPathSet(modulePath, importPaths)) {
return true;
}
return false;
}
GlobalModuleDependenciesCache::GlobalModuleDependenciesCache() {
for (auto kind = ModuleDependenciesKind::FirstKind;
kind != ModuleDependenciesKind::LastKind; ++kind) {
ModuleDependenciesKindMap.insert(
{kind, llvm::StringMap<ModuleDependenciesVector>()});
}
ModuleDependenciesKindMap.insert(
{ModuleDependenciesKind::SwiftBinary,
llvm::StringMap<ModuleDependenciesVector>()});
ModuleDependenciesKindMap.insert(
{ModuleDependenciesKind::SwiftPlaceholder,
llvm::StringMap<ModuleDependenciesVector>()});
ModuleDependenciesKindMap.insert(
{ModuleDependenciesKind::Clang,
llvm::StringMap<ModuleDependenciesVector>()});
}
Optional<ModuleDependencies> GlobalModuleDependenciesCache::findDependencies(
StringRef moduleName,
ModuleLookupSpecifics details) const {
if (!details.kind) {
for (auto kind = ModuleDependenciesKind::FirstKind;
kind != ModuleDependenciesKind::LastKind; ++kind) {
auto dep = findDependencies(moduleName, {kind, details.currentSearchPaths});
if (dep.hasValue())
return dep.getValue();
}
return None;
}
const auto &map = getDependenciesMap(*details.kind);
auto known = map.find(moduleName);
if (known != map.end()) {
assert(!known->second.empty());
for (auto &dep : known->second) {
if (moduleContainedInImportPathSet(dep, details.currentSearchPaths))
return dep;
}
return None;
}
return None;
}
bool GlobalModuleDependenciesCache::hasDependencies(
StringRef moduleName,
ModuleLookupSpecifics details) const {
return findDependencies(moduleName, details).hasValue();
}
Optional<ModuleDependenciesVector>
GlobalModuleDependenciesCache::findAllDependenciesIrrespectiveOfSearchPaths(
StringRef moduleName, Optional<ModuleDependenciesKind> kind) const {
if (!kind) {
for (auto kind = ModuleDependenciesKind::FirstKind;
kind != ModuleDependenciesKind::LastKind; ++kind) {
auto deps = findAllDependenciesIrrespectiveOfSearchPaths(moduleName, kind);
if (deps.hasValue())
return deps.getValue();
}
return None;
}
const auto &map = getDependenciesMap(*kind);
auto known = map.find(moduleName);
if (known != map.end()) {
assert(!known->second.empty());
return known->second;
}
return None;
}
static std::string modulePathForVerification(const ModuleDependencies &module) {
std::string existingModulePath = "";
switch (module.getKind()) {
case swift::ModuleDependenciesKind::SwiftTextual: {
auto *swiftDep = module.getAsSwiftTextualModule();
if (swiftDep->swiftInterfaceFile)
existingModulePath = *(swiftDep->swiftInterfaceFile);
break;
}
case swift::ModuleDependenciesKind::SwiftBinary: {
auto *swiftBinaryDep = module.getAsSwiftBinaryModule();
existingModulePath = swiftBinaryDep->compiledModulePath;
break;
}
case swift::ModuleDependenciesKind::Clang: {
auto *clangDep = module.getAsClangModule();
existingModulePath = clangDep->moduleMapFile;
break;
}
case swift::ModuleDependenciesKind::SwiftPlaceholder:
default:
llvm_unreachable("Unhandled dependency kind.");
}
return existingModulePath;
}
const ModuleDependencies* GlobalModuleDependenciesCache::recordDependencies(
StringRef moduleName,
ModuleDependencies dependencies) {
auto kind = dependencies.getKind();
auto &map = getDependenciesMap(kind);
// Cache may already have a dependency for this module
if (map.count(moduleName) != 0) {
// Ensure that the existing dependencies objects are at a different path.
// i.e. do not record duplicate dependencies.
auto newModulePath = modulePathForVerification(dependencies);
for (auto &existingDeps : map[moduleName]) {
if (modulePathForVerification(existingDeps) == newModulePath)
return &existingDeps;
}
map[moduleName].emplace_back(std::move(dependencies));
return map[moduleName].end()-1;
} else {
map.insert({moduleName, ModuleDependenciesVector{std::move(dependencies)}});
AllModules.push_back({moduleName.str(), kind});
return &(map[moduleName].front());
}
}
const ModuleDependencies* GlobalModuleDependenciesCache::updateDependencies(
ModuleDependencyID moduleID, ModuleDependencies dependencies) {
auto &map = getDependenciesMap(moduleID.second);
auto known = map.find(moduleID.first);
assert(known != map.end() && "Not yet added to map");
assert(known->second.size() == 1 &&
"Cannot update dependency with multiple candidates.");
known->second[0] = std::move(dependencies);
return &(known->second[0]);
}
llvm::StringMap<const ModuleDependencies*> &
ModuleDependenciesCache::getDependencyReferencesMap(ModuleDependenciesKind kind) {
auto it = ModuleDependenciesKindMap.find(kind);
assert(it != ModuleDependenciesKindMap.end() && "invalid dependency kind");
return it->second;
}
const llvm::StringMap<const ModuleDependencies*> &
ModuleDependenciesCache::getDependencyReferencesMap(ModuleDependenciesKind kind) const {
auto it = ModuleDependenciesKindMap.find(kind);
assert(it != ModuleDependenciesKindMap.end() && "invalid dependency kind");
return it->second;
}
ModuleDependenciesCache::ModuleDependenciesCache(GlobalModuleDependenciesCache &globalCache)
: globalCache(globalCache) {
for (auto kind = ModuleDependenciesKind::FirstKind;
kind != ModuleDependenciesKind::LastKind; ++kind) {
ModuleDependenciesKindMap.insert(
{kind, llvm::StringMap<const ModuleDependencies *>()});
}
}
Optional<const ModuleDependencies*> ModuleDependenciesCache::findDependencies(
StringRef moduleName, Optional<ModuleDependenciesKind> kind) const {
if (!kind) {
for (auto kind = ModuleDependenciesKind::FirstKind;
kind != ModuleDependenciesKind::LastKind; ++kind) {
auto dep = findDependencies(moduleName, kind);
if (dep.hasValue())
return dep.getValue();
}
return None;
}
const auto &map = getDependencyReferencesMap(*kind);
auto known = map.find(moduleName);
if (known != map.end())
return known->second;
return None;
}
Optional<ModuleDependencies>
ModuleDependenciesCache::findDependencies(StringRef moduleName,
ModuleLookupSpecifics details) const {
// 1. Query the local scan results
// 2. If no module found, query the global cache using the module details
// lookup
auto localResult = findDependencies(moduleName, details.kind);
if (localResult.hasValue())
return *(localResult.getValue());
else
return globalCache.findDependencies(moduleName, details);
}
bool ModuleDependenciesCache::hasDependencies(
StringRef moduleName, ModuleLookupSpecifics details) const {
return findDependencies(moduleName, details).hasValue();
}
void ModuleDependenciesCache::recordDependencies(
StringRef moduleName, ModuleDependencies dependencies) {
auto globalDepPtr = globalCache.recordDependencies(moduleName, dependencies);
auto kind = globalDepPtr->getKind();
auto &map = getDependencyReferencesMap(kind);
assert(map.count(moduleName) == 0 && "Already added to map");
map.insert({moduleName, globalDepPtr});
}
void ModuleDependenciesCache::updateDependencies(
ModuleDependencyID moduleID, ModuleDependencies dependencies) {
auto globalDepRef = globalCache.updateDependencies(moduleID, dependencies);
auto &map = getDependencyReferencesMap(moduleID.second);
auto known = map.find(moduleID.first);
if (known != map.end())
map.erase(known);
map.insert({moduleID.first, globalDepRef});
}