mirror of
https://github.com/apple/swift.git
synced 2025-12-14 20:36:38 +01:00
198 lines
7.7 KiB
C++
198 lines
7.7 KiB
C++
//===----- ModuleInterfaceBuilder.h - Compiles .swiftinterface files ------===//
|
|
//
|
|
// This source file is part of the Swift.org open source project
|
|
//
|
|
// Copyright (c) 2019 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
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
#ifndef SWIFT_FRONTEND_MODULEINTERFACEBUILDER_H
|
|
#define SWIFT_FRONTEND_MODULEINTERFACEBUILDER_H
|
|
|
|
#include "swift/AST/ModuleLoader.h"
|
|
#include "swift/Basic/LLVM.h"
|
|
#include "swift/Basic/SourceLoc.h"
|
|
#include "swift/Frontend/Frontend.h"
|
|
#include "swift/Serialization/SerializationOptions.h"
|
|
#include "llvm/Support/StringSaver.h"
|
|
|
|
namespace llvm {
|
|
namespace vfs {
|
|
class FileSystem;
|
|
}
|
|
} // namespace llvm
|
|
|
|
namespace swift {
|
|
|
|
class DiagnosticEngine;
|
|
class LangOptions;
|
|
class SearchPathOptions;
|
|
class DependencyTracker;
|
|
|
|
/// A utility class to build a Swift interface file into a module
|
|
/// using a `CompilerInstance` constructed from the flags specified in the
|
|
/// interface file.
|
|
class ImplicitModuleInterfaceBuilder {
|
|
SourceManager &sourceMgr;
|
|
DiagnosticEngine *diags;
|
|
InterfaceSubContextDelegate &subASTDelegate;
|
|
const StringRef interfacePath;
|
|
const StringRef sdkPath;
|
|
const std::optional<StringRef> sysroot;
|
|
const StringRef moduleName;
|
|
const StringRef moduleCachePath;
|
|
const StringRef prebuiltCachePath;
|
|
const StringRef backupInterfaceDir;
|
|
const StringRef ABIDescriptorPath;
|
|
const bool disableInterfaceFileLock;
|
|
const bool silenceInterfaceDiagnostics;
|
|
const SourceLoc diagnosticLoc;
|
|
DependencyTracker *const dependencyTracker;
|
|
SmallVector<StringRef, 3> extraDependencies;
|
|
|
|
public:
|
|
/// Emit a diagnostic tied to this declaration.
|
|
template <typename... ArgTypes>
|
|
static InFlightDiagnostic
|
|
diagnose(DiagnosticEngine *Diags, SourceManager &SM, StringRef InterfacePath,
|
|
SourceLoc Loc, Diag<ArgTypes...> ID,
|
|
typename detail::PassArgument<ArgTypes>::type... Args) {
|
|
if (Loc.isInvalid()) {
|
|
// Diagnose this inside the interface file, if possible.
|
|
Loc = SM.getLocFromExternalSource(InterfacePath, 1, 1);
|
|
}
|
|
return Diags->diagnose(Loc, ID, std::move(Args)...);
|
|
}
|
|
|
|
private:
|
|
/// Emit a diagnostic tied to this declaration.
|
|
template <typename... ArgTypes>
|
|
InFlightDiagnostic
|
|
diagnose(Diag<ArgTypes...> ID,
|
|
typename detail::PassArgument<ArgTypes>::type... Args) const {
|
|
return diagnose(diags, sourceMgr, interfacePath, diagnosticLoc, ID,
|
|
std::move(Args)...);
|
|
}
|
|
|
|
bool
|
|
buildSwiftModuleInternal(StringRef OutPath, bool ShouldSerializeDeps,
|
|
std::unique_ptr<llvm::MemoryBuffer> *ModuleBuffer,
|
|
ArrayRef<std::string> CandidateModules);
|
|
|
|
public:
|
|
ImplicitModuleInterfaceBuilder(
|
|
SourceManager &sourceMgr, DiagnosticEngine *diags,
|
|
InterfaceSubContextDelegate &subASTDelegate,
|
|
StringRef interfacePath, StringRef sdkPath,
|
|
std::optional<StringRef> sysroot, StringRef moduleName,
|
|
StringRef moduleCachePath, StringRef backupInterfaceDir,
|
|
StringRef prebuiltCachePath, StringRef ABIDescriptorPath,
|
|
bool disableInterfaceFileLock = false,
|
|
bool silenceInterfaceDiagnostics = false,
|
|
SourceLoc diagnosticLoc = SourceLoc(),
|
|
DependencyTracker *tracker = nullptr)
|
|
: sourceMgr(sourceMgr), diags(diags), subASTDelegate(subASTDelegate),
|
|
interfacePath(interfacePath), sdkPath(sdkPath), sysroot(sysroot),
|
|
moduleName(moduleName), moduleCachePath(moduleCachePath),
|
|
prebuiltCachePath(prebuiltCachePath),
|
|
backupInterfaceDir(backupInterfaceDir),
|
|
ABIDescriptorPath(ABIDescriptorPath),
|
|
disableInterfaceFileLock(disableInterfaceFileLock),
|
|
silenceInterfaceDiagnostics(silenceInterfaceDiagnostics),
|
|
diagnosticLoc(diagnosticLoc), dependencyTracker(tracker) {}
|
|
|
|
/// Ensures the requested file name is added as a dependency of the resulting
|
|
/// module.
|
|
void addExtraDependency(StringRef path) { extraDependencies.push_back(path); }
|
|
|
|
bool buildSwiftModule(StringRef OutPath, bool ShouldSerializeDeps,
|
|
std::unique_ptr<llvm::MemoryBuffer> *ModuleBuffer,
|
|
llvm::function_ref<void()> RemarkRebuild = nullptr,
|
|
ArrayRef<std::string> CandidateModules = {});
|
|
};
|
|
|
|
/// Use the provided `CompilerInstance` to build a swift interface into a module
|
|
class ExplicitModuleInterfaceBuilder {
|
|
public:
|
|
ExplicitModuleInterfaceBuilder(CompilerInstance &Instance,
|
|
DiagnosticEngine *diags,
|
|
SourceManager &sourceMgr,
|
|
const StringRef moduleCachePath,
|
|
const StringRef backupInterfaceDir,
|
|
const StringRef prebuiltCachePath,
|
|
const StringRef ABIDescriptorPath,
|
|
const SmallVector<StringRef, 3> &extraDependencies,
|
|
SourceLoc diagnosticLoc = SourceLoc(),
|
|
DependencyTracker *tracker = nullptr)
|
|
: Instance(Instance),
|
|
diags(diags),
|
|
sourceMgr(sourceMgr),
|
|
moduleCachePath(moduleCachePath),
|
|
prebuiltCachePath(prebuiltCachePath),
|
|
backupInterfaceDir(backupInterfaceDir),
|
|
ABIDescriptorPath(ABIDescriptorPath),
|
|
extraDependencies(extraDependencies),
|
|
diagnosticLoc(diagnosticLoc),
|
|
dependencyTracker(tracker) {}
|
|
|
|
std::error_code buildSwiftModuleFromInterface(
|
|
StringRef interfacePath, StringRef outputPath, bool ShouldSerializeDeps,
|
|
std::unique_ptr<llvm::MemoryBuffer> *ModuleBuffer,
|
|
ArrayRef<std::string> CompiledCandidates,
|
|
StringRef CompilerVersion);
|
|
|
|
/// Populate the provided \p Deps with \c FileDependency entries for all
|
|
/// dependencies \p SubInstance's DependencyTracker recorded while compiling
|
|
/// the module, excepting .swiftmodules in \p moduleCachePath or
|
|
/// \p prebuiltCachePath. Those have _their_ dependencies added instead, both
|
|
/// to avoid having to do recursive scanning when rechecking this dependency
|
|
/// in future and to make the module caches relocatable.
|
|
bool collectDepsForSerialization(
|
|
SmallVectorImpl<SerializationOptions::FileDependency> &Deps,
|
|
StringRef interfacePath, bool IsHashBased);
|
|
|
|
/// Emit a diagnostic tied to this declaration.
|
|
template <typename... ArgTypes>
|
|
static InFlightDiagnostic
|
|
diagnose(DiagnosticEngine *Diags, SourceManager &SM, StringRef InterfacePath,
|
|
SourceLoc Loc, Diag<ArgTypes...> ID,
|
|
typename detail::PassArgument<ArgTypes>::type... Args) {
|
|
if (Loc.isInvalid()) {
|
|
// Diagnose this inside the interface file, if possible.
|
|
Loc = SM.getLocFromExternalSource(InterfacePath, 1, 1);
|
|
}
|
|
return Diags->diagnose(Loc, ID, std::move(Args)...);
|
|
}
|
|
|
|
private:
|
|
/// Emit a diagnostic tied to this declaration.
|
|
template<typename ...ArgTypes>
|
|
InFlightDiagnostic diagnose(
|
|
Diag<ArgTypes...> ID,
|
|
StringRef InterfacePath,
|
|
typename detail::PassArgument<ArgTypes>::type... Args) const {
|
|
return diagnose(diags, sourceMgr, InterfacePath, diagnosticLoc,
|
|
ID, std::move(Args)...);
|
|
}
|
|
|
|
private:
|
|
CompilerInstance &Instance;
|
|
DiagnosticEngine *diags;
|
|
SourceManager &sourceMgr;
|
|
StringRef moduleCachePath;
|
|
StringRef prebuiltCachePath;
|
|
StringRef backupInterfaceDir;
|
|
StringRef ABIDescriptorPath;
|
|
const SmallVector<StringRef, 3> extraDependencies;
|
|
const SourceLoc diagnosticLoc;
|
|
DependencyTracker *const dependencyTracker;
|
|
};
|
|
|
|
} // end namespace swift
|
|
|
|
#endif // defined(SWIFT_FRONTEND_MODULEINTERFACEBUILDER_H)
|