//===-------------- DependencyScanningTool.h - Swift Compiler -------------===// // // This source file is part of the Swift.org open source project // // Copyright (c) 2014 - 2020 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_DEPENDENCY_SCANNING_TOOL_H #define SWIFT_DEPENDENCY_SCANNING_TOOL_H #include "swift-c/DependencyScan/DependencyScan.h" #include "swift/Frontend/Frontend.h" #include "swift/AST/ModuleDependencies.h" #include "swift/DependencyScan/ScanDependencies.h" #include "swift/Frontend/PrintingDiagnosticConsumer.h" #include "llvm/Support/Error.h" #include "llvm/Support/StringSaver.h" namespace swift { namespace dependencies { class DependencyScanningTool; class DependencyScanDiagnosticCollector; struct ScanQueryInstance { std::unique_ptr ScanInstance; std::shared_ptr ScanDiagnostics; }; /// Pure virtual Diagnostic consumer intended for collecting /// emitted diagnostics in a thread-safe fashion class ThreadSafeDiagnosticCollector : public DiagnosticConsumer { private: llvm::sys::SmartMutex DiagnosticConsumerStateLock; void handleDiagnostic(SourceManager &SM, const DiagnosticInfo &Info) final; protected: virtual void addDiagnostic(SourceManager &SM, const DiagnosticInfo &Info) = 0; public: ThreadSafeDiagnosticCollector() {} bool finishProcessing() final { return false; } }; /// Diagnostic consumer that simply collects the diagnostics emitted so-far class DependencyScanDiagnosticCollector : public ThreadSafeDiagnosticCollector { private: struct ScannerDiagnosticInfo { std::string Message; llvm::SourceMgr::DiagKind Severity; std::optional ImportLocation; }; std::vector Diagnostics; protected: void addDiagnostic(SourceManager &SM, const DiagnosticInfo &Info) override; public: friend DependencyScanningTool; DependencyScanDiagnosticCollector() {} void reset() { Diagnostics.clear(); } const std::vector &getDiagnostics() const { return Diagnostics; } }; /// Given a set of arguments to a print-target-info frontend tool query, produce the /// JSON target info. llvm::ErrorOr getTargetInfo(ArrayRef Command, const char *main_executable_path); /// The high-level implementation of the dependency scanner that runs on /// an individual worker thread. class DependencyScanningTool { public: /// Construct a dependency scanning tool. DependencyScanningTool(); /// Collect the full module dependency graph for the input, ignoring any /// placeholder modules. /// /// \returns a \c StringError with the diagnostic output if errors /// occurred, \c swiftscan_dependency_result_t otherwise. llvm::ErrorOr getDependencies(ArrayRef Command, const llvm::StringSet<> &PlaceholderModules, StringRef WorkingDirectory); /// Collect the set of imports for the input module /// /// \returns a \c StringError with the diagnostic output if errors /// occurred, \c swiftscan_prescan_result_t otherwise. llvm::ErrorOr getImports(ArrayRef Command, StringRef WorkingDirectory); /// Using the specified invocation command, instantiate a CompilerInstance /// that will be used for this scan. llvm::ErrorOr initCompilerInstanceForScan(ArrayRef Command, StringRef WorkingDirectory, std::shared_ptr scannerDiagnosticsCollector); private: /// Shared cache of module dependencies, re-used by individual full-scan queries /// during the lifetime of this Tool. std::unique_ptr ScanningService; /// Shared state mutual-exclusivity lock llvm::sys::SmartMutex DependencyScanningToolStateLock; llvm::BumpPtrAllocator Alloc; llvm::StringSaver Saver; }; swiftscan_diagnostic_set_t *mapCollectedDiagnosticsForOutput(const DependencyScanDiagnosticCollector *diagnosticCollector); } // end namespace dependencies } // end namespace swift #endif // SWIFT_DEPENDENCY_SCANNING_TOOL_H