2020-09-11 17:17:31 +08:00
|
|
|
//===--- CompilerInstance.cpp ---------------------------------------------===//
|
|
|
|
//
|
|
|
|
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
|
|
|
|
// See https://llvm.org/LICENSE.txt for license information.
|
|
|
|
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
#include "flang/Frontend/CompilerInstance.h"
|
2021-02-10 17:24:45 +08:00
|
|
|
#include "flang/Common/Fortran-features.h"
|
2020-09-11 17:17:31 +08:00
|
|
|
#include "flang/Frontend/CompilerInvocation.h"
|
2020-10-06 00:42:00 +08:00
|
|
|
#include "flang/Frontend/TextDiagnosticPrinter.h"
|
2020-10-27 20:26:47 +08:00
|
|
|
#include "flang/Parser/parsing.h"
|
2020-10-24 19:33:19 +08:00
|
|
|
#include "flang/Parser/provenance.h"
|
2020-12-09 00:27:46 +08:00
|
|
|
#include "flang/Semantics/semantics.h"
|
2020-10-24 19:33:19 +08:00
|
|
|
#include "llvm/Support/Errc.h"
|
|
|
|
#include "llvm/Support/Error.h"
|
|
|
|
#include "llvm/Support/FileSystem.h"
|
|
|
|
#include "llvm/Support/Path.h"
|
2020-09-11 17:17:31 +08:00
|
|
|
#include "llvm/Support/raw_ostream.h"
|
|
|
|
|
|
|
|
using namespace Fortran::frontend;
|
|
|
|
|
2020-10-24 19:33:19 +08:00
|
|
|
CompilerInstance::CompilerInstance()
|
|
|
|
: invocation_(new CompilerInvocation()),
|
2020-10-27 20:26:47 +08:00
|
|
|
allSources_(new Fortran::parser::AllSources()),
|
|
|
|
allCookedSources_(new Fortran::parser::AllCookedSources(*allSources_)),
|
2021-02-10 17:24:45 +08:00
|
|
|
parsing_(new Fortran::parser::Parsing(*allCookedSources_)) {
|
2020-10-27 20:26:47 +08:00
|
|
|
// TODO: This is a good default during development, but ultimately we should
|
|
|
|
// give the user the opportunity to specify this.
|
|
|
|
allSources_->set_encoding(Fortran::parser::Encoding::UTF_8);
|
|
|
|
}
|
2020-09-11 17:17:31 +08:00
|
|
|
|
2020-10-24 19:33:19 +08:00
|
|
|
CompilerInstance::~CompilerInstance() {
|
|
|
|
assert(outputFiles_.empty() && "Still output files in flight?");
|
|
|
|
}
|
|
|
|
|
2020-10-28 18:47:48 +08:00
|
|
|
void CompilerInstance::set_invocation(
|
2020-10-24 19:33:19 +08:00
|
|
|
std::shared_ptr<CompilerInvocation> value) {
|
|
|
|
invocation_ = std::move(value);
|
|
|
|
}
|
|
|
|
|
2020-12-09 00:27:46 +08:00
|
|
|
void CompilerInstance::set_semaOutputStream(raw_ostream &Value) {
|
|
|
|
ownedSemaOutputStream_.release();
|
|
|
|
semaOutputStream_ = &Value;
|
|
|
|
}
|
|
|
|
|
|
|
|
void CompilerInstance::set_semaOutputStream(
|
|
|
|
std::unique_ptr<raw_ostream> Value) {
|
|
|
|
ownedSemaOutputStream_.swap(Value);
|
|
|
|
semaOutputStream_ = ownedSemaOutputStream_.get();
|
|
|
|
}
|
|
|
|
|
2020-10-24 19:33:19 +08:00
|
|
|
// Helper method to generate the path of the output file. The following logic
|
|
|
|
// applies:
|
|
|
|
// 1. If the user specifies the output file via `-o`, then use that (i.e.
|
|
|
|
// the outputFilename parameter).
|
|
|
|
// 2. If the user does not specify the name of the output file, derive it from
|
|
|
|
// the input file (i.e. inputFilename + extension)
|
|
|
|
// 3. If the output file is not specified and the input file is `-`, then set
|
|
|
|
// the output file to `-` as well.
|
|
|
|
static std::string GetOutputFilePath(llvm::StringRef outputFilename,
|
|
|
|
llvm::StringRef inputFilename, llvm::StringRef extension) {
|
|
|
|
|
|
|
|
// Output filename _is_ specified. Just use that.
|
|
|
|
if (!outputFilename.empty())
|
|
|
|
return std::string(outputFilename);
|
|
|
|
|
|
|
|
// Output filename _is not_ specified. Derive it from the input file name.
|
|
|
|
std::string outFile = "-";
|
|
|
|
if (!extension.empty() && (inputFilename != "-")) {
|
|
|
|
llvm::SmallString<128> path(inputFilename);
|
|
|
|
llvm::sys::path::replace_extension(path, extension);
|
|
|
|
outFile = std::string(path.str());
|
|
|
|
}
|
|
|
|
|
|
|
|
return outFile;
|
|
|
|
}
|
|
|
|
|
|
|
|
std::unique_ptr<llvm::raw_pwrite_stream>
|
|
|
|
CompilerInstance::CreateDefaultOutputFile(
|
|
|
|
bool binary, llvm::StringRef baseName, llvm::StringRef extension) {
|
|
|
|
|
|
|
|
// Get the path of the output file
|
|
|
|
std::string outputFilePath =
|
2021-07-29 20:39:10 +08:00
|
|
|
GetOutputFilePath(frontendOpts().outputFile, baseName, extension);
|
2020-10-24 19:33:19 +08:00
|
|
|
|
|
|
|
// Create the output file
|
2021-08-20 18:25:11 +08:00
|
|
|
llvm::Expected<std::unique_ptr<llvm::raw_pwrite_stream>> os =
|
|
|
|
CreateOutputFileImpl(outputFilePath, binary);
|
|
|
|
|
|
|
|
// If successful, add the file to the list of tracked output files and
|
|
|
|
// return.
|
|
|
|
if (os) {
|
|
|
|
outputFiles_.emplace_back(OutputFile(outputFilePath));
|
|
|
|
return std::move(*os);
|
|
|
|
}
|
2021-08-20 17:11:19 +08:00
|
|
|
|
2021-08-20 18:25:11 +08:00
|
|
|
// If unsuccessful, issue an error and return Null
|
|
|
|
unsigned DiagID = diagnostics().getCustomDiagID(
|
|
|
|
clang::DiagnosticsEngine::Error, "unable to open output file '%0': '%1'");
|
|
|
|
diagnostics().Report(DiagID)
|
|
|
|
<< outputFilePath << llvm::errorToErrorCode(os.takeError()).message();
|
|
|
|
return nullptr;
|
2020-10-24 19:33:19 +08:00
|
|
|
}
|
|
|
|
|
2021-08-20 18:25:11 +08:00
|
|
|
llvm::Expected<std::unique_ptr<llvm::raw_pwrite_stream>>
|
|
|
|
CompilerInstance::CreateOutputFileImpl(
|
|
|
|
llvm::StringRef outputFilePath, bool binary) {
|
2020-10-24 19:33:19 +08:00
|
|
|
|
|
|
|
// Creates the file descriptor for the output file
|
|
|
|
std::unique_ptr<llvm::raw_fd_ostream> os;
|
2021-08-20 18:25:11 +08:00
|
|
|
|
|
|
|
std::error_code error;
|
|
|
|
os.reset(new llvm::raw_fd_ostream(outputFilePath, error,
|
|
|
|
(binary ? llvm::sys::fs::OF_None : llvm::sys::fs::OF_TextWithCRLF)));
|
|
|
|
if (error) {
|
|
|
|
return llvm::errorCodeToError(error);
|
2021-08-20 17:11:19 +08:00
|
|
|
}
|
2020-10-24 19:33:19 +08:00
|
|
|
|
2021-08-20 18:25:11 +08:00
|
|
|
// For seekable streams, just return the stream corresponding to the output
|
|
|
|
// file.
|
2020-10-24 19:33:19 +08:00
|
|
|
if (!binary || os->supportsSeeking())
|
|
|
|
return std::move(os);
|
|
|
|
|
2021-08-20 18:25:11 +08:00
|
|
|
// For non-seekable streams, we need to wrap the output stream into something
|
|
|
|
// that supports 'pwrite' and takes care of the ownership for us.
|
|
|
|
return std::make_unique<llvm::buffer_unique_ostream>(std::move(os));
|
2020-10-24 19:33:19 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void CompilerInstance::ClearOutputFiles(bool eraseFiles) {
|
|
|
|
for (OutputFile &of : outputFiles_)
|
|
|
|
if (!of.filename_.empty() && eraseFiles)
|
|
|
|
llvm::sys::fs::remove(of.filename_);
|
|
|
|
|
|
|
|
outputFiles_.clear();
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CompilerInstance::ExecuteAction(FrontendAction &act) {
|
2021-01-21 23:42:56 +08:00
|
|
|
auto &invoc = this->invocation();
|
|
|
|
|
2020-10-27 20:26:47 +08:00
|
|
|
// Set some sane defaults for the frontend.
|
2021-01-21 23:42:56 +08:00
|
|
|
invoc.SetDefaultFortranOpts();
|
|
|
|
// Update the fortran options based on user-based input.
|
|
|
|
invoc.setFortranOpts();
|
2021-02-05 05:11:22 +08:00
|
|
|
// Set the encoding to read all input files in based on user input.
|
|
|
|
allSources_->set_encoding(invoc.fortranOpts().encoding);
|
2021-02-10 17:24:45 +08:00
|
|
|
// Create the semantics context and set semantic options.
|
|
|
|
invoc.setSemanticsOpts(*this->allCookedSources_);
|
2020-10-24 19:33:19 +08:00
|
|
|
|
2021-01-21 23:42:56 +08:00
|
|
|
// Run the frontend action `act` for every input file.
|
2021-07-29 20:39:10 +08:00
|
|
|
for (const FrontendInputFile &fif : frontendOpts().inputs) {
|
2020-10-24 19:33:19 +08:00
|
|
|
if (act.BeginSourceFile(*this, fif)) {
|
|
|
|
if (llvm::Error err = act.Execute()) {
|
|
|
|
consumeError(std::move(err));
|
|
|
|
}
|
|
|
|
act.EndSourceFile();
|
|
|
|
}
|
|
|
|
}
|
2020-12-09 00:27:46 +08:00
|
|
|
return !diagnostics().getClient()->getNumErrors();
|
2020-10-24 19:33:19 +08:00
|
|
|
}
|
2020-09-11 17:17:31 +08:00
|
|
|
|
|
|
|
void CompilerInstance::CreateDiagnostics(
|
|
|
|
clang::DiagnosticConsumer *client, bool shouldOwnClient) {
|
|
|
|
diagnostics_ =
|
|
|
|
CreateDiagnostics(&GetDiagnosticOpts(), client, shouldOwnClient);
|
|
|
|
}
|
|
|
|
|
|
|
|
clang::IntrusiveRefCntPtr<clang::DiagnosticsEngine>
|
|
|
|
CompilerInstance::CreateDiagnostics(clang::DiagnosticOptions *opts,
|
|
|
|
clang::DiagnosticConsumer *client, bool shouldOwnClient) {
|
|
|
|
clang::IntrusiveRefCntPtr<clang::DiagnosticIDs> diagID(
|
|
|
|
new clang::DiagnosticIDs());
|
|
|
|
clang::IntrusiveRefCntPtr<clang::DiagnosticsEngine> diags(
|
|
|
|
new clang::DiagnosticsEngine(diagID, opts));
|
|
|
|
|
|
|
|
// Create the diagnostic client for reporting errors or for
|
|
|
|
// implementing -verify.
|
|
|
|
if (client) {
|
|
|
|
diags->setClient(client, shouldOwnClient);
|
|
|
|
} else {
|
2020-10-06 00:42:00 +08:00
|
|
|
diags->setClient(new TextDiagnosticPrinter(llvm::errs(), opts));
|
2020-09-11 17:17:31 +08:00
|
|
|
}
|
|
|
|
return diags;
|
|
|
|
}
|