mirror of
https://github.com/apple/swift.git
synced 2025-12-14 20:36:38 +01:00
This is an important fixit, but was lost because it is on the note instead of the error. rdar://problem/26593117
1205 lines
40 KiB
C++
1205 lines
40 KiB
C++
//===--- FrontendTool.cpp - Swift Compiler Frontend -----------------------===//
|
|
//
|
|
// This source file is part of the Swift.org open source project
|
|
//
|
|
// Copyright (c) 2014 - 2016 Apple Inc. and the Swift project authors
|
|
// Licensed under Apache License v2.0 with Runtime Library Exception
|
|
//
|
|
// See http://swift.org/LICENSE.txt for license information
|
|
// See http://swift.org/CONTRIBUTORS.txt for the list of Swift project authors
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
///
|
|
/// \file
|
|
/// \brief This is the entry point to the swift -frontend functionality, which
|
|
/// implements the core compiler functionality along with a number of additional
|
|
/// tools for demonstration and testing purposes.
|
|
///
|
|
/// This is separate from the rest of libFrontend to reduce the dependencies
|
|
/// required by that library.
|
|
///
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
#include "swift/FrontendTool/FrontendTool.h"
|
|
|
|
#include "swift/Subsystems.h"
|
|
#include "swift/AST/DiagnosticsFrontend.h"
|
|
#include "swift/AST/DiagnosticsSema.h"
|
|
#include "swift/AST/IRGenOptions.h"
|
|
#include "swift/AST/Mangle.h"
|
|
#include "swift/AST/NameLookup.h"
|
|
#include "swift/AST/ReferencedNameTracker.h"
|
|
#include "swift/AST/TypeRefinementContext.h"
|
|
#include "swift/Basic/Dwarf.h"
|
|
#include "swift/Basic/Fallthrough.h"
|
|
#include "swift/Basic/FileSystem.h"
|
|
#include "swift/Basic/SourceManager.h"
|
|
#include "swift/Basic/Timer.h"
|
|
#include "swift/Frontend/DiagnosticVerifier.h"
|
|
#include "swift/Frontend/Frontend.h"
|
|
#include "swift/Frontend/PrintingDiagnosticConsumer.h"
|
|
#include "swift/Frontend/SerializedDiagnosticConsumer.h"
|
|
#include "swift/Immediate/Immediate.h"
|
|
#include "swift/Option/Options.h"
|
|
#include "swift/PrintAsObjC/PrintAsObjC.h"
|
|
#include "swift/Serialization/SerializationOptions.h"
|
|
#include "swift/SILOptimizer/PassManager/Passes.h"
|
|
|
|
// FIXME: We're just using CompilerInstance::createOutputFile.
|
|
// This API should be sunk down to LLVM.
|
|
#include "clang/Frontend/CompilerInstance.h"
|
|
|
|
#include "llvm/ADT/Statistic.h"
|
|
#include "llvm/IR/LLVMContext.h"
|
|
#include "llvm/IR/Module.h"
|
|
#include "llvm/IRReader/IRReader.h"
|
|
#include "llvm/Option/Option.h"
|
|
#include "llvm/Option/OptTable.h"
|
|
#include "llvm/Support/FileSystem.h"
|
|
#include "llvm/Support/Path.h"
|
|
#include "llvm/Support/raw_ostream.h"
|
|
#include "llvm/Support/TargetSelect.h"
|
|
#include "llvm/Support/Timer.h"
|
|
#include "llvm/Support/YAMLParser.h"
|
|
|
|
#include <memory>
|
|
#include <unordered_set>
|
|
|
|
using namespace swift;
|
|
|
|
static std::string displayName(StringRef MainExecutablePath) {
|
|
std::string Name = llvm::sys::path::stem(MainExecutablePath);
|
|
Name += " -frontend";
|
|
return Name;
|
|
}
|
|
|
|
/// Emits a Make-style dependencies file.
|
|
static bool emitMakeDependencies(DiagnosticEngine &diags,
|
|
DependencyTracker &depTracker,
|
|
const FrontendOptions &opts) {
|
|
std::error_code EC;
|
|
llvm::raw_fd_ostream out(opts.DependenciesFilePath, EC,
|
|
llvm::sys::fs::F_None);
|
|
|
|
if (out.has_error() || EC) {
|
|
diags.diagnose(SourceLoc(), diag::error_opening_output,
|
|
opts.DependenciesFilePath, EC.message());
|
|
out.clear_error();
|
|
return true;
|
|
}
|
|
|
|
// Declare a helper for escaping file names for use in Makefiles.
|
|
llvm::SmallString<256> pathBuf;
|
|
auto escape = [&](StringRef raw) -> StringRef {
|
|
pathBuf.clear();
|
|
|
|
static const char badChars[] = " $#:\n";
|
|
size_t prev = 0;
|
|
for (auto index = raw.find_first_of(badChars); index != StringRef::npos;
|
|
index = raw.find_first_of(badChars, index+1)) {
|
|
pathBuf.append(raw.slice(prev, index));
|
|
if (raw[index] == '$')
|
|
pathBuf.push_back('$');
|
|
else
|
|
pathBuf.push_back('\\');
|
|
prev = index;
|
|
}
|
|
pathBuf.append(raw.substr(prev));
|
|
return pathBuf;
|
|
};
|
|
|
|
// FIXME: Xcode can't currently handle multiple targets in a single
|
|
// dependency line.
|
|
opts.forAllOutputPaths([&](StringRef targetName) {
|
|
out << escape(targetName) << " :";
|
|
// First include all other files in the module. Make-style dependencies
|
|
// need to be conservative!
|
|
for (StringRef path : opts.InputFilenames)
|
|
out << ' ' << escape(path);
|
|
// Then print dependencies we've picked up during compilation.
|
|
for (StringRef path : depTracker.getDependencies())
|
|
out << ' ' << escape(path);
|
|
out << '\n';
|
|
});
|
|
|
|
return false;
|
|
}
|
|
|
|
static void findNominals(llvm::MapVector<const NominalTypeDecl *, bool> &found,
|
|
DeclRange members) {
|
|
for (const Decl *D : members) {
|
|
auto nominal = dyn_cast<NominalTypeDecl>(D);
|
|
if (!nominal)
|
|
continue;
|
|
found[nominal] |= true;
|
|
findNominals(found, nominal->getMembers());
|
|
}
|
|
}
|
|
|
|
static bool declIsPrivate(const Decl *member) {
|
|
auto *VD = dyn_cast<ValueDecl>(member);
|
|
if (!VD) {
|
|
switch (member->getKind()) {
|
|
case DeclKind::Import:
|
|
case DeclKind::PatternBinding:
|
|
case DeclKind::EnumCase:
|
|
case DeclKind::TopLevelCode:
|
|
case DeclKind::IfConfig:
|
|
return true;
|
|
|
|
case DeclKind::Extension:
|
|
case DeclKind::InfixOperator:
|
|
case DeclKind::PrefixOperator:
|
|
case DeclKind::PostfixOperator:
|
|
return false;
|
|
|
|
default:
|
|
llvm_unreachable("everything else is a ValueDecl");
|
|
}
|
|
}
|
|
|
|
return VD->getFormalAccess() == Accessibility::Private;
|
|
}
|
|
|
|
static bool extendedTypeIsPrivate(TypeLoc inheritedType) {
|
|
if (!inheritedType.getType())
|
|
return true;
|
|
|
|
SmallVector<ProtocolDecl *, 2> protocols;
|
|
if (!inheritedType.getType()->isAnyExistentialType(protocols)) {
|
|
// Be conservative. We don't know how to deal with other extended types.
|
|
return false;
|
|
}
|
|
|
|
return std::all_of(protocols.begin(), protocols.end(), declIsPrivate);
|
|
}
|
|
|
|
static std::string mangleTypeAsContext(const NominalTypeDecl *type) {
|
|
Mangle::Mangler mangler(/*debug style=*/false, /*Unicode=*/true);
|
|
mangler.mangleContext(type);
|
|
return mangler.finalize();
|
|
}
|
|
|
|
/// Emits a Swift-style dependencies file.
|
|
static bool emitReferenceDependencies(DiagnosticEngine &diags,
|
|
SourceFile *SF,
|
|
DependencyTracker &depTracker,
|
|
const FrontendOptions &opts) {
|
|
if (!SF) {
|
|
diags.diagnose(SourceLoc(),
|
|
diag::emit_reference_dependencies_without_primary_file);
|
|
return true;
|
|
}
|
|
|
|
std::error_code EC;
|
|
llvm::raw_fd_ostream out(opts.ReferenceDependenciesFilePath, EC,
|
|
llvm::sys::fs::F_None);
|
|
|
|
if (out.has_error() || EC) {
|
|
diags.diagnose(SourceLoc(), diag::error_opening_output,
|
|
opts.ReferenceDependenciesFilePath, EC.message());
|
|
out.clear_error();
|
|
return true;
|
|
}
|
|
|
|
auto escape = [](Identifier name) -> std::string {
|
|
return llvm::yaml::escape(name.str());
|
|
};
|
|
|
|
out << "### Swift dependencies file v0 ###\n";
|
|
|
|
llvm::MapVector<const NominalTypeDecl *, bool> extendedNominals;
|
|
llvm::SmallVector<const ExtensionDecl *, 8> extensionsWithJustMembers;
|
|
|
|
out << "provides-top-level:\n";
|
|
for (const Decl *D : SF->Decls) {
|
|
switch (D->getKind()) {
|
|
case DeclKind::Module:
|
|
break;
|
|
|
|
case DeclKind::Import:
|
|
// FIXME: Handle re-exported decls.
|
|
break;
|
|
|
|
case DeclKind::Extension: {
|
|
auto *ED = cast<ExtensionDecl>(D);
|
|
auto *NTD = ED->getExtendedType()->getAnyNominal();
|
|
if (!NTD)
|
|
break;
|
|
if (NTD->hasAccessibility() &&
|
|
NTD->getFormalAccess() == Accessibility::Private) {
|
|
break;
|
|
}
|
|
|
|
bool justMembers = std::all_of(ED->getInherited().begin(),
|
|
ED->getInherited().end(),
|
|
extendedTypeIsPrivate);
|
|
if (justMembers) {
|
|
if (std::all_of(ED->getMembers().begin(), ED->getMembers().end(),
|
|
declIsPrivate)) {
|
|
break;
|
|
} else {
|
|
extensionsWithJustMembers.push_back(ED);
|
|
}
|
|
}
|
|
extendedNominals[NTD] |= !justMembers;
|
|
findNominals(extendedNominals, ED->getMembers());
|
|
break;
|
|
}
|
|
|
|
case DeclKind::InfixOperator:
|
|
case DeclKind::PrefixOperator:
|
|
case DeclKind::PostfixOperator:
|
|
out << "- \"" << escape(cast<OperatorDecl>(D)->getName()) << "\"\n";
|
|
break;
|
|
|
|
case DeclKind::Enum:
|
|
case DeclKind::Struct:
|
|
case DeclKind::Class:
|
|
case DeclKind::Protocol: {
|
|
auto *NTD = cast<NominalTypeDecl>(D);
|
|
if (!NTD->hasName())
|
|
break;
|
|
if (NTD->hasAccessibility() &&
|
|
NTD->getFormalAccess() == Accessibility::Private) {
|
|
break;
|
|
}
|
|
out << "- \"" << escape(NTD->getName()) << "\"\n";
|
|
extendedNominals[NTD] |= true;
|
|
findNominals(extendedNominals, NTD->getMembers());
|
|
break;
|
|
}
|
|
|
|
case DeclKind::TypeAlias:
|
|
case DeclKind::Var:
|
|
case DeclKind::Func: {
|
|
auto *VD = cast<ValueDecl>(D);
|
|
if (!VD->hasName())
|
|
break;
|
|
if (VD->hasAccessibility() &&
|
|
VD->getFormalAccess() == Accessibility::Private) {
|
|
break;
|
|
}
|
|
out << "- \"" << escape(VD->getName()) << "\"\n";
|
|
break;
|
|
}
|
|
|
|
case DeclKind::PatternBinding:
|
|
case DeclKind::TopLevelCode:
|
|
case DeclKind::IfConfig:
|
|
// No action necessary.
|
|
break;
|
|
|
|
case DeclKind::EnumCase:
|
|
case DeclKind::GenericTypeParam:
|
|
case DeclKind::AssociatedType:
|
|
case DeclKind::Param:
|
|
case DeclKind::Subscript:
|
|
case DeclKind::Constructor:
|
|
case DeclKind::Destructor:
|
|
case DeclKind::EnumElement:
|
|
llvm_unreachable("cannot appear at the top level of a file");
|
|
}
|
|
}
|
|
|
|
out << "provides-nominal:\n";
|
|
for (auto entry : extendedNominals) {
|
|
if (!entry.second)
|
|
continue;
|
|
out << "- \"";
|
|
out << mangleTypeAsContext(entry.first);
|
|
out << "\"\n";
|
|
}
|
|
|
|
out << "provides-member:\n";
|
|
for (auto entry : extendedNominals) {
|
|
out << "- [\"";
|
|
out << mangleTypeAsContext(entry.first);
|
|
out << "\", \"\"]\n";
|
|
}
|
|
|
|
// This is also part of "provides-member".
|
|
for (auto *ED : extensionsWithJustMembers) {
|
|
auto mangledName = mangleTypeAsContext(
|
|
ED->getExtendedType()->getAnyNominal());
|
|
|
|
for (auto *member : ED->getMembers()) {
|
|
auto *VD = dyn_cast<ValueDecl>(member);
|
|
if (!VD || !VD->hasName() ||
|
|
VD->getFormalAccess() == Accessibility::Private) {
|
|
continue;
|
|
}
|
|
out << "- [\"" << mangledName << "\", \""
|
|
<< escape(VD->getName()) << "\"]\n";
|
|
}
|
|
}
|
|
|
|
if (SF->getASTContext().LangOpts.EnableObjCInterop) {
|
|
// FIXME: This requires a traversal of the whole file to compute.
|
|
// We should (a) see if there's a cheaper way to keep it up to date,
|
|
// and/or (b) see if we can fast-path cases where there's no ObjC involved.
|
|
out << "provides-dynamic-lookup:\n";
|
|
class ValueDeclPrinter : public VisibleDeclConsumer {
|
|
private:
|
|
raw_ostream &out;
|
|
std::string (*escape)(Identifier);
|
|
public:
|
|
ValueDeclPrinter(raw_ostream &out, decltype(escape) escape)
|
|
: out(out), escape(escape) {}
|
|
|
|
void foundDecl(ValueDecl *VD, DeclVisibilityKind Reason) override {
|
|
out << "- \"" << escape(VD->getName()) << "\"\n";
|
|
}
|
|
};
|
|
ValueDeclPrinter printer(out, escape);
|
|
SF->lookupClassMembers({}, printer);
|
|
}
|
|
|
|
ReferencedNameTracker *tracker = SF->getReferencedNameTracker();
|
|
|
|
// FIXME: Sort these?
|
|
out << "depends-top-level:\n";
|
|
for (auto &entry : tracker->getTopLevelNames()) {
|
|
assert(!entry.first.empty());
|
|
out << "- ";
|
|
if (!entry.second)
|
|
out << "!private ";
|
|
out << "\"" << escape(entry.first) << "\"\n";
|
|
}
|
|
|
|
out << "depends-member:\n";
|
|
auto &memberLookupTable = tracker->getUsedMembers();
|
|
using TableEntryTy = std::pair<ReferencedNameTracker::MemberPair, bool>;
|
|
std::vector<TableEntryTy> sortedMembers{
|
|
memberLookupTable.begin(), memberLookupTable.end()
|
|
};
|
|
llvm::array_pod_sort(sortedMembers.begin(), sortedMembers.end(),
|
|
[](const TableEntryTy *lhs,
|
|
const TableEntryTy *rhs) -> int {
|
|
if (lhs->first.first == rhs->first.first)
|
|
return lhs->first.second.compare(rhs->first.second);
|
|
|
|
if (lhs->first.first->getName() != rhs->first.first->getName())
|
|
return lhs->first.first->getName().compare(rhs->first.first->getName());
|
|
|
|
// Break type name ties by mangled name.
|
|
auto lhsMangledName = mangleTypeAsContext(lhs->first.first);
|
|
auto rhsMangledName = mangleTypeAsContext(rhs->first.first);
|
|
return lhsMangledName.compare(rhsMangledName);
|
|
});
|
|
|
|
for (auto &entry : sortedMembers) {
|
|
assert(entry.first.first != nullptr);
|
|
if (entry.first.first->hasAccessibility() &&
|
|
entry.first.first->getFormalAccess() == Accessibility::Private)
|
|
continue;
|
|
|
|
out << "- ";
|
|
if (!entry.second)
|
|
out << "!private ";
|
|
out << "[\"";
|
|
out << mangleTypeAsContext(entry.first.first);
|
|
out << "\", \"";
|
|
if (!entry.first.second.empty())
|
|
out << escape(entry.first.second);
|
|
out << "\"]\n";
|
|
}
|
|
|
|
out << "depends-nominal:\n";
|
|
for (auto i = sortedMembers.begin(), e = sortedMembers.end(); i != e; ++i) {
|
|
bool isCascading = i->second;
|
|
while (i+1 != e && i[0].first.first == i[1].first.first) {
|
|
++i;
|
|
isCascading |= i->second;
|
|
}
|
|
|
|
if (i->first.first->hasAccessibility() &&
|
|
i->first.first->getFormalAccess() == Accessibility::Private)
|
|
continue;
|
|
|
|
out << "- ";
|
|
if (!isCascading)
|
|
out << "!private ";
|
|
out << "\"";
|
|
out << mangleTypeAsContext(i->first.first);
|
|
out << "\"\n";
|
|
}
|
|
|
|
// FIXME: Sort these?
|
|
out << "depends-dynamic-lookup:\n";
|
|
for (auto &entry : tracker->getDynamicLookupNames()) {
|
|
assert(!entry.first.empty());
|
|
out << "- ";
|
|
if (!entry.second)
|
|
out << "!private ";
|
|
out << "\"" << escape(entry.first) << "\"\n";
|
|
}
|
|
|
|
out << "depends-external:\n";
|
|
for (auto &entry : depTracker.getDependencies()) {
|
|
out << "- \"" << llvm::yaml::escape(entry) << "\"\n";
|
|
}
|
|
|
|
llvm::SmallString<32> interfaceHash;
|
|
SF->getInterfaceHash(interfaceHash);
|
|
out << "interface-hash: \"" << interfaceHash << "\"\n";
|
|
|
|
return false;
|
|
}
|
|
|
|
/// Writes SIL out to the given file.
|
|
static bool writeSIL(SILModule &SM, Module *M, bool EmitVerboseSIL,
|
|
StringRef OutputFilename, bool SortSIL) {
|
|
std::error_code EC;
|
|
llvm::raw_fd_ostream OS(OutputFilename, EC, llvm::sys::fs::F_None);
|
|
if (EC) {
|
|
M->getASTContext().Diags.diagnose(SourceLoc(), diag::error_opening_output,
|
|
OutputFilename, EC.message());
|
|
return true;
|
|
}
|
|
SM.print(OS, EmitVerboseSIL, M, SortSIL);
|
|
return false;
|
|
}
|
|
|
|
static bool printAsObjC(const std::string &outputPath, Module *M,
|
|
StringRef bridgingHeader, bool moduleIsPublic) {
|
|
using namespace llvm::sys;
|
|
|
|
clang::CompilerInstance Clang;
|
|
|
|
std::string tmpFilePath;
|
|
std::error_code EC;
|
|
std::unique_ptr<llvm::raw_pwrite_stream> out =
|
|
Clang.createOutputFile(outputPath, EC,
|
|
/*binary=*/false,
|
|
/*removeOnSignal=*/true,
|
|
/*inputPath=*/"",
|
|
path::extension(outputPath),
|
|
/*temporary=*/true,
|
|
/*createDirs=*/false,
|
|
/*finalPath=*/nullptr,
|
|
&tmpFilePath);
|
|
|
|
if (!out) {
|
|
M->getASTContext().Diags.diagnose(SourceLoc(), diag::error_opening_output,
|
|
tmpFilePath, EC.message());
|
|
return true;
|
|
}
|
|
|
|
auto requiredAccess = moduleIsPublic ? Accessibility::Public
|
|
: Accessibility::Internal;
|
|
bool hadError = printAsObjC(*out, M, bridgingHeader, requiredAccess);
|
|
out->flush();
|
|
|
|
EC = swift::moveFileIfDifferent(tmpFilePath, outputPath);
|
|
if (EC) {
|
|
M->getASTContext().Diags.diagnose(SourceLoc(), diag::error_opening_output,
|
|
outputPath, EC.message());
|
|
return true;
|
|
}
|
|
|
|
return hadError;
|
|
}
|
|
|
|
/// Returns the OutputKind for the given Action.
|
|
static IRGenOutputKind getOutputKind(FrontendOptions::ActionType Action) {
|
|
switch (Action) {
|
|
case FrontendOptions::EmitIR:
|
|
return IRGenOutputKind::LLVMAssembly;
|
|
case FrontendOptions::EmitBC:
|
|
return IRGenOutputKind::LLVMBitcode;
|
|
case FrontendOptions::EmitAssembly:
|
|
return IRGenOutputKind::NativeAssembly;
|
|
case FrontendOptions::EmitObject:
|
|
return IRGenOutputKind::ObjectFile;
|
|
case FrontendOptions::Immediate:
|
|
return IRGenOutputKind::Module;
|
|
default:
|
|
llvm_unreachable("Unknown ActionType which requires IRGen");
|
|
return IRGenOutputKind::ObjectFile;
|
|
}
|
|
}
|
|
|
|
namespace {
|
|
|
|
/// If there is an error with fixits it writes the fixits as edits in json
|
|
/// format.
|
|
class JSONFixitWriter : public DiagnosticConsumer {
|
|
std::unique_ptr<llvm::raw_ostream> OSPtr;
|
|
bool FixitAll;
|
|
|
|
public:
|
|
JSONFixitWriter(std::unique_ptr<llvm::raw_ostream> OS,
|
|
const DiagnosticOptions &DiagOpts)
|
|
: OSPtr(std::move(OS)),
|
|
FixitAll(DiagOpts.FixitCodeForAllDiagnostics) {
|
|
*OSPtr << "[\n";
|
|
}
|
|
~JSONFixitWriter() {
|
|
*OSPtr << "]\n";
|
|
}
|
|
|
|
private:
|
|
void handleDiagnostic(SourceManager &SM, SourceLoc Loc,
|
|
DiagnosticKind Kind, StringRef Text,
|
|
const DiagnosticInfo &Info) override {
|
|
if (!shouldFix(Kind, Info))
|
|
return;
|
|
for (const auto &Fix : Info.FixIts) {
|
|
writeEdit(SM, Fix.getRange(), Fix.getText(), *OSPtr);
|
|
}
|
|
}
|
|
|
|
bool shouldFix(DiagnosticKind Kind, const DiagnosticInfo &Info) {
|
|
if (FixitAll)
|
|
return true;
|
|
|
|
// Do not add a semi as it is wrong in most cases during migration
|
|
if (Info.ID == diag::statement_same_line_without_semi.ID)
|
|
return false;
|
|
// The following interact badly with the swift migrator, they are undoing
|
|
// migration of arguments to preserve the no-label for first argument.
|
|
if (Info.ID == diag::witness_argument_name_mismatch.ID ||
|
|
Info.ID == diag::missing_argument_labels.ID ||
|
|
Info.ID == diag::override_argument_name_mismatch.ID)
|
|
return false;
|
|
// This also interacts badly with the swift migrator, it unnecessary adds
|
|
// @objc(selector) attributes triggered by the mismatched label changes.
|
|
if (Info.ID == diag::objc_witness_selector_mismatch.ID ||
|
|
Info.ID == diag::witness_non_objc.ID)
|
|
return false;
|
|
|
|
if (Kind == DiagnosticKind::Error)
|
|
return true;
|
|
if (Info.ID == diag::forced_downcast_coercion.ID ||
|
|
Info.ID == diag::forced_downcast_noop.ID ||
|
|
Info.ID == diag::variable_never_mutated.ID ||
|
|
Info.ID == diag::function_type_no_parens.ID ||
|
|
Info.ID == diag::convert_let_to_var.ID ||
|
|
Info.ID == diag::parameter_extraneous_double_up.ID)
|
|
return true;
|
|
return false;
|
|
}
|
|
|
|
void writeEdit(SourceManager &SM, CharSourceRange Range, StringRef Text,
|
|
llvm::raw_ostream &OS) {
|
|
SourceLoc Loc = Range.getStart();
|
|
unsigned BufID = SM.findBufferContainingLoc(Loc);
|
|
unsigned Offset = SM.getLocOffsetInBuffer(Loc, BufID);
|
|
unsigned Length = Range.getByteLength();
|
|
SmallString<200> Path =
|
|
StringRef(SM.getIdentifierForBuffer(BufID));
|
|
|
|
OS << " {\n";
|
|
OS << " \"file\": \"";
|
|
OS.write_escaped(Path.str()) << "\",\n";
|
|
OS << " \"offset\": " << Offset << ",\n";
|
|
if (Length != 0)
|
|
OS << " \"remove\": " << Length << ",\n";
|
|
if (!Text.empty()) {
|
|
OS << " \"text\": \"";
|
|
OS.write_escaped(Text) << "\",\n";
|
|
}
|
|
OS << " },\n";
|
|
}
|
|
};
|
|
|
|
} // anonymous namespace
|
|
|
|
// This is a separate function so that it shows up in stack traces.
|
|
LLVM_ATTRIBUTE_NOINLINE
|
|
static void debugFailWithAssertion() {
|
|
// This assertion should always fail, per the user's request, and should
|
|
// not be converted to llvm_unreachable.
|
|
assert(0 && "This is an assertion!");
|
|
}
|
|
|
|
// This is a separate function so that it shows up in stack traces.
|
|
LLVM_ATTRIBUTE_NOINLINE
|
|
static void debugFailWithCrash() {
|
|
LLVM_BUILTIN_TRAP;
|
|
}
|
|
|
|
/// Performs the compile requested by the user.
|
|
/// \returns true on error
|
|
static bool performCompile(CompilerInstance &Instance,
|
|
CompilerInvocation &Invocation,
|
|
ArrayRef<const char *> Args,
|
|
int &ReturnValue,
|
|
FrontendObserver *observer) {
|
|
FrontendOptions opts = Invocation.getFrontendOptions();
|
|
FrontendOptions::ActionType Action = opts.RequestedAction;
|
|
|
|
IRGenOptions &IRGenOpts = Invocation.getIRGenOptions();
|
|
|
|
bool inputIsLLVMIr = Invocation.getInputKind() == InputFileKind::IFK_LLVM_IR;
|
|
if (inputIsLLVMIr) {
|
|
auto &LLVMContext = llvm::getGlobalContext();
|
|
|
|
// Load in bitcode file.
|
|
assert(Invocation.getInputFilenames().size() == 1 &&
|
|
"We expect a single input for bitcode input!");
|
|
llvm::ErrorOr<std::unique_ptr<llvm::MemoryBuffer>> FileBufOrErr =
|
|
llvm::MemoryBuffer::getFileOrSTDIN(Invocation.getInputFilenames()[0]);
|
|
if (!FileBufOrErr) {
|
|
Instance.getASTContext().Diags.diagnose(SourceLoc(),
|
|
diag::error_open_input_file,
|
|
Invocation.getInputFilenames()[0],
|
|
FileBufOrErr.getError().message());
|
|
return true;
|
|
}
|
|
llvm::MemoryBuffer *MainFile = FileBufOrErr.get().get();
|
|
|
|
llvm::SMDiagnostic Err;
|
|
std::unique_ptr<llvm::Module> Module = llvm::parseIR(
|
|
MainFile->getMemBufferRef(),
|
|
Err, LLVMContext);
|
|
if (!Module) {
|
|
// TODO: Translate from the diagnostic info to the SourceManager location
|
|
// if available.
|
|
Instance.getASTContext().Diags.diagnose(SourceLoc(),
|
|
diag::error_parse_input_file,
|
|
Invocation.getInputFilenames()[0],
|
|
Err.getMessage());
|
|
return true;
|
|
}
|
|
|
|
// TODO: remove once the frontend understands what action it should perform
|
|
IRGenOpts.OutputKind = getOutputKind(Action);
|
|
|
|
return performLLVM(IRGenOpts, Instance.getASTContext(), Module.get());
|
|
}
|
|
|
|
ReferencedNameTracker nameTracker;
|
|
bool shouldTrackReferences = !opts.ReferenceDependenciesFilePath.empty();
|
|
if (shouldTrackReferences)
|
|
Instance.setReferencedNameTracker(&nameTracker);
|
|
|
|
if (Action == FrontendOptions::DumpParse ||
|
|
Action == FrontendOptions::DumpInterfaceHash)
|
|
Instance.performParseOnly();
|
|
else
|
|
Instance.performSema();
|
|
|
|
if (observer) {
|
|
observer->performedSemanticAnalysis(Instance);
|
|
}
|
|
|
|
FrontendOptions::DebugCrashMode CrashMode = opts.CrashMode;
|
|
if (CrashMode == FrontendOptions::DebugCrashMode::AssertAfterParse)
|
|
debugFailWithAssertion();
|
|
else if (CrashMode == FrontendOptions::DebugCrashMode::CrashAfterParse)
|
|
debugFailWithCrash();
|
|
|
|
ASTContext &Context = Instance.getASTContext();
|
|
|
|
if (Action == FrontendOptions::REPL) {
|
|
runREPL(Instance, ProcessCmdLine(Args.begin(), Args.end()),
|
|
Invocation.getParseStdlib());
|
|
return false;
|
|
}
|
|
|
|
SourceFile *PrimarySourceFile = Instance.getPrimarySourceFile();
|
|
|
|
// We've been told to dump the AST (either after parsing or type-checking,
|
|
// which is already differentiated in CompilerInstance::performSema()),
|
|
// so dump or print the main source file and return.
|
|
if (Action == FrontendOptions::DumpParse ||
|
|
Action == FrontendOptions::DumpAST ||
|
|
Action == FrontendOptions::PrintAST ||
|
|
Action == FrontendOptions::DumpTypeRefinementContexts ||
|
|
Action == FrontendOptions::DumpInterfaceHash) {
|
|
SourceFile *SF = PrimarySourceFile;
|
|
if (!SF) {
|
|
SourceFileKind Kind = Invocation.getSourceFileKind();
|
|
SF = &Instance.getMainModule()->getMainSourceFile(Kind);
|
|
}
|
|
if (Action == FrontendOptions::PrintAST)
|
|
SF->print(llvm::outs(), PrintOptions::printEverything());
|
|
else if (Action == FrontendOptions::DumpTypeRefinementContexts)
|
|
SF->getTypeRefinementContext()->dump(llvm::errs(), Context.SourceMgr);
|
|
else if (Action == FrontendOptions::DumpInterfaceHash)
|
|
SF->dumpInterfaceHash(llvm::errs());
|
|
else
|
|
SF->dump();
|
|
return false;
|
|
}
|
|
|
|
// If we were asked to print Clang stats, do so.
|
|
if (opts.PrintClangStats && Context.getClangModuleLoader())
|
|
Context.getClangModuleLoader()->printStatistics();
|
|
|
|
if (!opts.DependenciesFilePath.empty())
|
|
(void)emitMakeDependencies(Context.Diags, *Instance.getDependencyTracker(),
|
|
opts);
|
|
|
|
if (shouldTrackReferences)
|
|
emitReferenceDependencies(Context.Diags, Instance.getPrimarySourceFile(),
|
|
*Instance.getDependencyTracker(), opts);
|
|
|
|
if (Context.hadError())
|
|
return true;
|
|
|
|
// FIXME: This is still a lousy approximation of whether the module file will
|
|
// be externally consumed.
|
|
bool moduleIsPublic =
|
|
!Instance.getMainModule()->hasEntryPoint() &&
|
|
opts.ImplicitObjCHeaderPath.empty() &&
|
|
!Context.LangOpts.EnableAppExtensionRestrictions;
|
|
|
|
// We've just been told to perform a parse, so we can return now.
|
|
if (Action == FrontendOptions::Parse) {
|
|
if (!opts.ObjCHeaderOutputPath.empty())
|
|
return printAsObjC(opts.ObjCHeaderOutputPath, Instance.getMainModule(),
|
|
opts.ImplicitObjCHeaderPath, moduleIsPublic);
|
|
return false;
|
|
}
|
|
|
|
assert(Action >= FrontendOptions::EmitSILGen &&
|
|
"All actions not requiring SILGen must have been handled!");
|
|
|
|
std::unique_ptr<SILModule> SM = Instance.takeSILModule();
|
|
if (!SM) {
|
|
if (opts.PrimaryInput.hasValue() && opts.PrimaryInput.getValue().isFilename()) {
|
|
FileUnit *PrimaryFile = PrimarySourceFile;
|
|
if (!PrimaryFile) {
|
|
auto Index = opts.PrimaryInput.getValue().Index;
|
|
PrimaryFile = Instance.getMainModule()->getFiles()[Index];
|
|
}
|
|
SM = performSILGeneration(*PrimaryFile, Invocation.getSILOptions(),
|
|
None, opts.SILSerializeAll);
|
|
} else {
|
|
SM = performSILGeneration(Instance.getMainModule(), Invocation.getSILOptions(),
|
|
opts.SILSerializeAll,
|
|
true);
|
|
}
|
|
}
|
|
|
|
if (observer) {
|
|
observer->performedSILGeneration(*SM);
|
|
}
|
|
|
|
// We've been told to emit SIL after SILGen, so write it now.
|
|
if (Action == FrontendOptions::EmitSILGen) {
|
|
// If we are asked to link all, link all.
|
|
if (Invocation.getSILOptions().LinkMode == SILOptions::LinkAll)
|
|
performSILLinking(SM.get(), true);
|
|
return writeSIL(*SM, Instance.getMainModule(), opts.EmitVerboseSIL,
|
|
opts.getSingleOutputFilename(), opts.EmitSortedSIL);
|
|
}
|
|
|
|
if (Action == FrontendOptions::EmitSIBGen) {
|
|
// If we are asked to link all, link all.
|
|
if (Invocation.getSILOptions().LinkMode == SILOptions::LinkAll)
|
|
performSILLinking(SM.get(), true);
|
|
|
|
auto DC = PrimarySourceFile ? ModuleOrSourceFile(PrimarySourceFile) :
|
|
Instance.getMainModule();
|
|
if (!opts.ModuleOutputPath.empty()) {
|
|
SerializationOptions serializationOpts;
|
|
serializationOpts.OutputPath = opts.ModuleOutputPath.c_str();
|
|
serializationOpts.SerializeAllSIL = true;
|
|
serializationOpts.IsSIB = true;
|
|
|
|
serialize(DC, serializationOpts, SM.get());
|
|
}
|
|
return false;
|
|
}
|
|
|
|
// Perform "stable" optimizations that are invariant across compiler versions.
|
|
if (!Invocation.getDiagnosticOptions().SkipDiagnosticPasses) {
|
|
if (runSILDiagnosticPasses(*SM))
|
|
return true;
|
|
|
|
if (observer) {
|
|
observer->performedSILDiagnostics(*SM);
|
|
}
|
|
}
|
|
|
|
// Now if we are asked to link all, link all.
|
|
if (Invocation.getSILOptions().LinkMode == SILOptions::LinkAll)
|
|
performSILLinking(SM.get(), true);
|
|
|
|
{
|
|
SharedTimer timer("SIL verification (pre-optimization)");
|
|
SM->verify();
|
|
}
|
|
|
|
// Perform SIL optimization passes if optimizations haven't been disabled.
|
|
// These may change across compiler versions.
|
|
{
|
|
SharedTimer timer("SIL optimization");
|
|
if (Invocation.getSILOptions().Optimization >
|
|
SILOptions::SILOptMode::None) {
|
|
StringRef CustomPipelinePath =
|
|
Invocation.getSILOptions().ExternalPassPipelineFilename;
|
|
if (!CustomPipelinePath.empty()) {
|
|
runSILOptimizationPassesWithFileSpecification(*SM, CustomPipelinePath);
|
|
} else {
|
|
runSILOptimizationPasses(*SM);
|
|
}
|
|
} else {
|
|
runSILPassesForOnone(*SM);
|
|
}
|
|
}
|
|
|
|
if (observer) {
|
|
observer->performedSILOptimization(*SM);
|
|
}
|
|
|
|
{
|
|
SharedTimer timer("SIL verification (post-optimization)");
|
|
SM->verify();
|
|
}
|
|
|
|
// Gather instruction counts if we are asked to do so.
|
|
if (SM->getOptions().PrintInstCounts) {
|
|
performSILInstCount(&*SM);
|
|
}
|
|
|
|
// Get the main source file's private discriminator and attach it to
|
|
// the compile unit's flags.
|
|
if (PrimarySourceFile) {
|
|
Identifier PD = PrimarySourceFile->getPrivateDiscriminator();
|
|
if (!PD.empty())
|
|
IRGenOpts.DWARFDebugFlags += (" -private-discriminator "+PD.str()).str();
|
|
}
|
|
|
|
if (!opts.ObjCHeaderOutputPath.empty()) {
|
|
(void)printAsObjC(opts.ObjCHeaderOutputPath, Instance.getMainModule(),
|
|
opts.ImplicitObjCHeaderPath, moduleIsPublic);
|
|
}
|
|
|
|
if (Action == FrontendOptions::EmitSIB) {
|
|
auto DC = PrimarySourceFile ? ModuleOrSourceFile(PrimarySourceFile) :
|
|
Instance.getMainModule();
|
|
if (!opts.ModuleOutputPath.empty()) {
|
|
SerializationOptions serializationOpts;
|
|
serializationOpts.OutputPath = opts.ModuleOutputPath.c_str();
|
|
serializationOpts.SerializeAllSIL = true;
|
|
serializationOpts.IsSIB = true;
|
|
|
|
serialize(DC, serializationOpts, SM.get());
|
|
}
|
|
return false;
|
|
}
|
|
|
|
if (!opts.ModuleOutputPath.empty() || !opts.ModuleDocOutputPath.empty()) {
|
|
auto DC = PrimarySourceFile ? ModuleOrSourceFile(PrimarySourceFile) :
|
|
Instance.getMainModule();
|
|
if (!opts.ModuleOutputPath.empty()) {
|
|
SerializationOptions serializationOpts;
|
|
serializationOpts.OutputPath = opts.ModuleOutputPath.c_str();
|
|
serializationOpts.DocOutputPath = opts.ModuleDocOutputPath.c_str();
|
|
serializationOpts.GroupInfoPath = opts.GroupInfoPath.c_str();
|
|
serializationOpts.SerializeAllSIL = opts.SILSerializeAll;
|
|
if (opts.SerializeBridgingHeader)
|
|
serializationOpts.ImportedHeader = opts.ImplicitObjCHeaderPath;
|
|
serializationOpts.ModuleLinkName = opts.ModuleLinkName;
|
|
serializationOpts.ExtraClangOptions =
|
|
Invocation.getClangImporterOptions().ExtraArgs;
|
|
if (!IRGenOpts.ForceLoadSymbolName.empty())
|
|
serializationOpts.AutolinkForceLoad = true;
|
|
|
|
// Options contain information about the developer's computer,
|
|
// so only serialize them if the module isn't going to be shipped to
|
|
// the public.
|
|
serializationOpts.SerializeOptionsForDebugging =
|
|
!moduleIsPublic || opts.AlwaysSerializeDebuggingOptions;
|
|
|
|
serialize(DC, serializationOpts, SM.get());
|
|
}
|
|
|
|
if (Action == FrontendOptions::EmitModuleOnly)
|
|
return false;
|
|
}
|
|
|
|
assert(Action >= FrontendOptions::EmitSIL &&
|
|
"All actions not requiring SILPasses must have been handled!");
|
|
|
|
// We've been told to write canonical SIL, so write it now.
|
|
if (Action == FrontendOptions::EmitSIL) {
|
|
return writeSIL(*SM, Instance.getMainModule(), opts.EmitVerboseSIL,
|
|
opts.getSingleOutputFilename(), opts.EmitSortedSIL);
|
|
}
|
|
|
|
assert(Action >= FrontendOptions::Immediate &&
|
|
"All actions not requiring IRGen must have been handled!");
|
|
assert(Action != FrontendOptions::REPL &&
|
|
"REPL mode must be handled immediately after Instance.performSema()");
|
|
|
|
// Check if we had any errors; if we did, don't proceed to IRGen.
|
|
if (Context.hadError())
|
|
return true;
|
|
|
|
// Cleanup instructions/builtin calls not suitable for IRGen.
|
|
performSILCleanup(SM.get());
|
|
|
|
// TODO: remove once the frontend understands what action it should perform
|
|
IRGenOpts.OutputKind = getOutputKind(Action);
|
|
if (Action == FrontendOptions::Immediate) {
|
|
assert(!PrimarySourceFile && "-i doesn't work in -primary-file mode");
|
|
IRGenOpts.UseJIT = true;
|
|
IRGenOpts.DebugInfoKind = IRGenDebugInfoKind::Normal;
|
|
const ProcessCmdLine &CmdLine = ProcessCmdLine(opts.ImmediateArgv.begin(),
|
|
opts.ImmediateArgv.end());
|
|
Instance.setSILModule(std::move(SM));
|
|
|
|
if (observer) {
|
|
observer->aboutToRunImmediately(Instance);
|
|
}
|
|
|
|
ReturnValue =
|
|
RunImmediately(Instance, CmdLine, IRGenOpts, Invocation.getSILOptions());
|
|
return false;
|
|
}
|
|
|
|
// FIXME: We shouldn't need to use the global context here, but
|
|
// something is persisting across calls to performIRGeneration.
|
|
auto &LLVMContext = llvm::getGlobalContext();
|
|
if (PrimarySourceFile) {
|
|
performIRGeneration(IRGenOpts, *PrimarySourceFile, SM.get(),
|
|
opts.getSingleOutputFilename(), LLVMContext);
|
|
} else {
|
|
performIRGeneration(IRGenOpts, Instance.getMainModule(), SM.get(),
|
|
opts.getSingleOutputFilename(), LLVMContext);
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
/// Returns true if an error occurred.
|
|
static bool dumpAPI(Module *Mod, StringRef OutDir) {
|
|
using namespace llvm::sys;
|
|
|
|
auto getOutPath = [&](SourceFile *SF) -> std::string {
|
|
SmallString<256> Path = OutDir;
|
|
StringRef Filename = SF->getFilename();
|
|
path::append(Path, path::filename(Filename));
|
|
return Path.str();
|
|
};
|
|
|
|
std::unordered_set<std::string> Filenames;
|
|
|
|
auto dumpFile = [&](SourceFile *SF) -> bool {
|
|
SmallString<512> TempBuf;
|
|
llvm::raw_svector_ostream TempOS(TempBuf);
|
|
|
|
PrintOptions PO = PrintOptions::printInterface();
|
|
PO.PrintOriginalSourceText = true;
|
|
PO.Indent = 2;
|
|
PO.PrintAccessibility = false;
|
|
PO.SkipUnderscoredStdlibProtocols = true;
|
|
SF->print(TempOS, PO);
|
|
if (TempOS.str().trim().empty())
|
|
return false; // nothing to show.
|
|
|
|
std::string OutPath = getOutPath(SF);
|
|
bool WasInserted = Filenames.insert(OutPath).second;
|
|
if (!WasInserted) {
|
|
llvm::errs() << "multiple source files ended up with the same dump API "
|
|
"filename to write to: " << OutPath << '\n';
|
|
return true;
|
|
}
|
|
|
|
std::error_code EC;
|
|
llvm::raw_fd_ostream OS(OutPath, EC, fs::OpenFlags::F_RW);
|
|
if (EC) {
|
|
llvm::errs() << "error opening file '" << OutPath << "': "
|
|
<< EC.message() << '\n';
|
|
return true;
|
|
}
|
|
|
|
OS << TempOS.str();
|
|
return false;
|
|
};
|
|
|
|
std::error_code EC = fs::create_directories(OutDir);
|
|
if (EC) {
|
|
llvm::errs() << "error creating directory '" << OutDir << "': "
|
|
<< EC.message() << '\n';
|
|
return true;
|
|
}
|
|
|
|
for (auto *FU : Mod->getFiles()) {
|
|
if (SourceFile *SF = dyn_cast<SourceFile>(FU))
|
|
if (dumpFile(SF))
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
int swift::performFrontend(ArrayRef<const char *> Args,
|
|
const char *Argv0, void *MainAddr,
|
|
FrontendObserver *observer) {
|
|
llvm::InitializeAllTargets();
|
|
llvm::InitializeAllTargetMCs();
|
|
llvm::InitializeAllAsmPrinters();
|
|
llvm::InitializeAllAsmParsers();
|
|
|
|
CompilerInstance Instance;
|
|
PrintingDiagnosticConsumer PDC;
|
|
Instance.addDiagnosticConsumer(&PDC);
|
|
|
|
if (Args.empty()) {
|
|
Instance.getDiags().diagnose(SourceLoc(), diag::error_no_frontend_args);
|
|
return 1;
|
|
}
|
|
|
|
CompilerInvocation Invocation;
|
|
std::string MainExecutablePath = llvm::sys::fs::getMainExecutable(Argv0,
|
|
MainAddr);
|
|
Invocation.setMainExecutablePath(MainExecutablePath);
|
|
|
|
SmallString<128> workingDirectory;
|
|
llvm::sys::fs::current_path(workingDirectory);
|
|
|
|
// Parse arguments.
|
|
if (Invocation.parseArgs(Args, Instance.getDiags(), workingDirectory)) {
|
|
return 1;
|
|
}
|
|
|
|
// Setting DWARF Version depend on platform
|
|
IRGenOptions &IRGenOpts = Invocation.getIRGenOptions();
|
|
IRGenOpts.DWARFVersion = swift::GenericDWARFVersion;
|
|
if (Invocation.getLangOptions().Target.isWindowsCygwinEnvironment())
|
|
IRGenOpts.DWARFVersion = swift::CygwinDWARFVersion;
|
|
|
|
// The compiler invocation is now fully configured; notify our observer.
|
|
if (observer) {
|
|
observer->parsedArgs(Invocation);
|
|
}
|
|
|
|
if (Invocation.getFrontendOptions().PrintHelp ||
|
|
Invocation.getFrontendOptions().PrintHelpHidden) {
|
|
unsigned IncludedFlagsBitmask = options::FrontendOption;
|
|
unsigned ExcludedFlagsBitmask =
|
|
Invocation.getFrontendOptions().PrintHelpHidden ? 0 :
|
|
llvm::opt::HelpHidden;
|
|
std::unique_ptr<llvm::opt::OptTable> Options(createSwiftOptTable());
|
|
Options->PrintHelp(llvm::outs(), displayName(MainExecutablePath).c_str(),
|
|
"Swift frontend", IncludedFlagsBitmask,
|
|
ExcludedFlagsBitmask);
|
|
return 0;
|
|
}
|
|
|
|
if (Invocation.getFrontendOptions().RequestedAction ==
|
|
FrontendOptions::NoneAction) {
|
|
Instance.getDiags().diagnose(SourceLoc(),
|
|
diag::error_missing_frontend_action);
|
|
return 1;
|
|
}
|
|
|
|
// TODO: reorder, if possible, so that diagnostics emitted during
|
|
// CompilerInvocation::parseArgs are included in the serialized file.
|
|
std::unique_ptr<DiagnosticConsumer> SerializedConsumer;
|
|
{
|
|
const std::string &SerializedDiagnosticsPath =
|
|
Invocation.getFrontendOptions().SerializedDiagnosticsPath;
|
|
if (!SerializedDiagnosticsPath.empty()) {
|
|
std::error_code EC;
|
|
std::unique_ptr<llvm::raw_fd_ostream> OS;
|
|
OS.reset(new llvm::raw_fd_ostream(SerializedDiagnosticsPath,
|
|
EC,
|
|
llvm::sys::fs::F_None));
|
|
|
|
if (EC) {
|
|
Instance.getDiags().diagnose(SourceLoc(),
|
|
diag::cannot_open_serialized_file,
|
|
SerializedDiagnosticsPath, EC.message());
|
|
return 1;
|
|
}
|
|
|
|
SerializedConsumer.reset(
|
|
serialized_diagnostics::createConsumer(std::move(OS)));
|
|
Instance.addDiagnosticConsumer(SerializedConsumer.get());
|
|
}
|
|
}
|
|
|
|
std::unique_ptr<DiagnosticConsumer> FixitsConsumer;
|
|
{
|
|
const std::string &FixitsOutputPath =
|
|
Invocation.getFrontendOptions().FixitsOutputPath;
|
|
if (!FixitsOutputPath.empty()) {
|
|
std::error_code EC;
|
|
std::unique_ptr<llvm::raw_fd_ostream> OS;
|
|
OS.reset(new llvm::raw_fd_ostream(FixitsOutputPath,
|
|
EC,
|
|
llvm::sys::fs::F_None));
|
|
|
|
if (EC) {
|
|
Instance.getDiags().diagnose(SourceLoc(),
|
|
diag::cannot_open_file,
|
|
FixitsOutputPath, EC.message());
|
|
return 1;
|
|
}
|
|
|
|
FixitsConsumer.reset(new JSONFixitWriter(std::move(OS),
|
|
Invocation.getDiagnosticOptions()));
|
|
Instance.addDiagnosticConsumer(FixitsConsumer.get());
|
|
}
|
|
}
|
|
|
|
if (Invocation.getDiagnosticOptions().UseColor)
|
|
PDC.forceColors();
|
|
|
|
if (Invocation.getFrontendOptions().DebugTimeCompilation)
|
|
SharedTimer::enableCompilationTimers();
|
|
|
|
if (Invocation.getFrontendOptions().PrintStats) {
|
|
llvm::EnableStatistics();
|
|
}
|
|
|
|
if (Invocation.getDiagnosticOptions().VerifyDiagnostics) {
|
|
enableDiagnosticVerifier(Instance.getSourceMgr());
|
|
}
|
|
|
|
DependencyTracker depTracker;
|
|
if (!Invocation.getFrontendOptions().DependenciesFilePath.empty() ||
|
|
!Invocation.getFrontendOptions().ReferenceDependenciesFilePath.empty()) {
|
|
Instance.setDependencyTracker(&depTracker);
|
|
}
|
|
|
|
if (Instance.setup(Invocation)) {
|
|
return 1;
|
|
}
|
|
|
|
// The compiler instance has been configured; notify our observer.
|
|
if (observer) {
|
|
observer->configuredCompiler(Instance);
|
|
}
|
|
|
|
int ReturnValue = 0;
|
|
bool HadError =
|
|
performCompile(Instance, Invocation, Args, ReturnValue, observer) ||
|
|
Instance.getASTContext().hadError();
|
|
|
|
if (!HadError && !Invocation.getFrontendOptions().DumpAPIPath.empty()) {
|
|
HadError = dumpAPI(Instance.getMainModule(),
|
|
Invocation.getFrontendOptions().DumpAPIPath);
|
|
}
|
|
|
|
if (Invocation.getDiagnosticOptions().VerifyDiagnostics) {
|
|
HadError = verifyDiagnostics(Instance.getSourceMgr(),
|
|
Instance.getInputBufferIDs());
|
|
DiagnosticEngine &diags = Instance.getDiags();
|
|
if (diags.hasFatalErrorOccurred() &&
|
|
!Invocation.getDiagnosticOptions().ShowDiagnosticsAfterFatalError) {
|
|
diags.resetHadAnyError();
|
|
diags.diagnose(SourceLoc(), diag::verify_encountered_fatal);
|
|
HadError = true;
|
|
}
|
|
}
|
|
|
|
return (HadError ? 1 : ReturnValue);
|
|
}
|
|
|
|
void FrontendObserver::parsedArgs(CompilerInvocation &invocation) {}
|
|
void FrontendObserver::configuredCompiler(CompilerInstance &instance) {}
|
|
void FrontendObserver::performedSemanticAnalysis(CompilerInstance &instance) {}
|
|
void FrontendObserver::performedSILGeneration(SILModule &module) {}
|
|
void FrontendObserver::performedSILDiagnostics(SILModule &module) {}
|
|
void FrontendObserver::performedSILOptimization(SILModule &module) {}
|
|
void FrontendObserver::aboutToRunImmediately(CompilerInstance &instance) {}
|