blob: 6d4f0065061c028fb4c37bbb49d8c13f07fd811a [file] [log] [blame]
//===--- FrontendInputs.h ---------------------------------------*- C++ -*-===//
//
// This source file is part of the Swift.org open source project
//
// Copyright (c) 2014 - 2018 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_FRONTENDINPUTS_H
#define SWIFT_FRONTEND_FRONTENDINPUTS_H
#include "swift/AST/Module.h"
#include "swift/Frontend/InputFile.h"
#include "llvm/ADT/Hashing.h"
#include "llvm/ADT/MapVector.h"
#include <string>
#include <vector>
namespace llvm {
class MemoryBuffer;
}
namespace swift {
/// Information about all the inputs and outputs to the frontend.
class FrontendInputsAndOutputs {
friend class ArgsToFrontendInputsConverter;
std::vector<InputFile> AllInputs;
llvm::StringMap<unsigned> PrimaryInputs;
/// In Single-threaded WMO mode, all inputs are used
/// both for importing and compiling.
bool IsSingleThreadedWMO = false;
/// Punt where needed to enable batch mode experiments.
bool AreBatchModeChecksBypassed = false;
public:
bool areBatchModeChecksBypassed() const { return AreBatchModeChecksBypassed; }
void setBypassBatchModeChecks(bool bbc) { AreBatchModeChecksBypassed = bbc; }
FrontendInputsAndOutputs() = default;
FrontendInputsAndOutputs(const FrontendInputsAndOutputs &other);
FrontendInputsAndOutputs &operator=(const FrontendInputsAndOutputs &other);
// Whole-module-optimization (WMO) routines:
// SingleThreadedWMO produces only main output file. In contrast,
// multi-threaded WMO produces one main output per input, as single-file and
// batch-mode do for each primary. Both WMO modes produce only one set of
// supplementary outputs.
bool isSingleThreadedWMO() const { return IsSingleThreadedWMO; }
void setIsSingleThreadedWMO(bool istw) { IsSingleThreadedWMO = istw; }
bool isWholeModule() const { return !hasPrimaryInputs(); }
// Readers:
// All inputs:
ArrayRef<InputFile> getAllInputs() const { return AllInputs; }
std::vector<std::string> getInputFilenames() const;
unsigned inputCount() const { return AllInputs.size(); }
bool hasInputs() const { return !AllInputs.empty(); }
bool hasSingleInput() const { return inputCount() == 1; }
const InputFile &firstInput() const { return AllInputs[0]; }
InputFile &firstInput() { return AllInputs[0]; }
const InputFile &lastInput() const { return AllInputs.back(); }
StringRef getFilenameOfFirstInput() const;
bool isReadingFromStdin() const;
void forEachInput(llvm::function_ref<void(const InputFile &)> fn) const;
// Primaries:
const InputFile &firstPrimaryInput() const;
const InputFile &lastPrimaryInput() const;
void
forEachPrimaryInput(llvm::function_ref<void(const InputFile &)> fn) const;
unsigned primaryInputCount() const { return PrimaryInputs.size(); }
// Primary count readers:
bool hasUniquePrimaryInput() const { return primaryInputCount() == 1; }
bool hasPrimaryInputs() const { return primaryInputCount() > 0; }
bool hasMultiplePrimaryInputs() const { return primaryInputCount() > 1; }
/// Fails an assertion if there is more than one primary input.
/// Used in situations where only one primary input can be handled
/// and where batch mode has not been implemented yet.
void assertMustNotBeMoreThanOnePrimaryInput() const;
/// Fails an assertion when there is more than one primary input unless
/// the experimental -bypass-batch-mode-checks argument was passed to
/// the front end.
/// FIXME: When batch mode is complete, this function should be obsolete.
void
assertMustNotBeMoreThanOnePrimaryInputUnlessBatchModeChecksHaveBeenBypassed()
const;
// Count-dependend readers:
/// \return the unique primary input, if one exists.
const InputFile *getUniquePrimaryInput() const;
const InputFile &getRequiredUniquePrimaryInput() const;
/// \return the name of the unique primary input, or an empty StrinRef if
/// there isn't one.
StringRef getNameOfUniquePrimaryInputFile() const;
/// Combines all primaries for stats reporter
std::string getStatsFileMangledInputName() const;
bool isInputPrimary(StringRef file) const;
unsigned numberOfPrimaryInputsEndingWith(const char *extension) const;
// Multi-facet readers
// If we have exactly one input filename, and its extension is "bc" or "ll",
// treat the input as LLVM_IR.
bool shouldTreatAsLLVM() const;
bool shouldTreatAsSIL() const;
bool areAllNonPrimariesSIB() const;
/// \return true for error
bool verifyInputs(DiagnosticEngine &diags, bool treatAsSIL,
bool isREPLRequested, bool isNoneRequested) const;
// Changing inputs
public:
void clearInputs();
void addInput(const InputFile &input);
void addInputFile(StringRef file, llvm::MemoryBuffer *buffer = nullptr);
void addPrimaryInputFile(StringRef file,
llvm::MemoryBuffer *buffer = nullptr);
// Outputs
private:
friend class ArgsToFrontendOptionsConverter;
void setMainOutputs(ArrayRef<std::string> outputFiles);
public:
unsigned countOfInputsProducingMainOutputs() const;
const InputFile &firstInputProducingOutput() const;
const InputFile &lastInputProducingOutput() const;
/// Under single-threaded WMO, we pretend that the first input
/// generates the main output, even though it will include code
/// generated from all of them.
void forEachInputProducingAMainOutputFile(
llvm::function_ref<void(const InputFile &)> fn) const;
std::vector<std::string> copyOutputFilenames() const;
void
forEachOutputFilename(llvm::function_ref<void(const std::string &)> fn) const;
/// Gets the name of the specified output filename.
/// If multiple files are specified, the last one is returned.
StringRef getSingleOutputFilename() const;
bool isOutputFilenameStdout() const;
bool isOutputFileDirectory() const;
bool hasNamedOutputFile() const;
// Supplementary outputs
void forEachInputProducingSupplementaryOutput(
llvm::function_ref<void(const InputFile &)> fn) const;
};
} // namespace swift
#endif /* SWIFT_FRONTEND_FRONTENDINPUTS_H */