mirror of
https://github.com/apple/swift.git
synced 2025-12-14 20:36:38 +01:00
147 lines
6.0 KiB
C++
147 lines
6.0 KiB
C++
//===--- InputFile.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_INPUTFILE_H
|
|
#define SWIFT_FRONTEND_INPUTFILE_H
|
|
|
|
#include "swift/Basic/FileTypes.h"
|
|
#include "swift/Basic/PrimarySpecificPaths.h"
|
|
#include "swift/Basic/SupplementaryOutputPaths.h"
|
|
#include "llvm/ADT/PointerIntPair.h"
|
|
#include "llvm/Support/MemoryBuffer.h"
|
|
#include "llvm/Support/Path.h"
|
|
#include <string>
|
|
|
|
namespace swift {
|
|
|
|
/// An \c InputFile encapsulates information about an input passed to the
|
|
/// frontend.
|
|
///
|
|
/// Compiler inputs are usually passed on the command line without a leading
|
|
/// flag. However, there are clients that use the \c CompilerInvocation as
|
|
/// a library like LLDB and SourceKit that generate their own \c InputFile
|
|
/// instances programmatically. Note that an \c InputFile need not actually be
|
|
/// backed by a physical file, nor does its file name actually reflect its
|
|
/// contents. \c InputFile has a constructor that will try to figure out the file
|
|
/// type from the file name if none is provided, but many clients that
|
|
/// construct \c InputFile instances themselves may provide bogus file names
|
|
/// with pre-computed kinds. It is imperative that \c InputFile::getType be used
|
|
/// as a source of truth for this information.
|
|
///
|
|
/// \warning \c InputFile takes an unfortunately lax view of the ownership of
|
|
/// its primary data. It currently only owns the file name and a copy of any
|
|
/// assigned \c PrimarySpecificPaths outright. It is the responsibility of the
|
|
/// caller to ensure that an associated memory buffer outlives the \c InputFile.
|
|
class InputFile final {
|
|
std::string Filename;
|
|
file_types::ID FileID;
|
|
llvm::PointerIntPair<llvm::MemoryBuffer *, 1, bool> BufferAndIsPrimary;
|
|
PrimarySpecificPaths PSPs;
|
|
|
|
public:
|
|
/// Constructs an input file from the provided data.
|
|
///
|
|
/// \warning This entrypoint infers the type of the file from its extension
|
|
/// and is therefore not suitable for most clients that use files synthesized
|
|
/// from memory buffers. Use the overload of this constructor accepting a
|
|
/// memory buffer and an explicit \c file_types::ID instead.
|
|
InputFile(StringRef name, bool isPrimary,
|
|
llvm::MemoryBuffer *buffer = nullptr)
|
|
: InputFile(name, isPrimary, buffer,
|
|
file_types::lookupTypeForExtension(
|
|
llvm::sys::path::extension(name))) {}
|
|
|
|
/// Constructs an input file from the provided data.
|
|
InputFile(StringRef name, bool isPrimary, llvm::MemoryBuffer *buffer,
|
|
file_types::ID FileID)
|
|
: Filename(
|
|
convertBufferNameFromLLVM_getFileOrSTDIN_toSwiftConventions(name)),
|
|
FileID(FileID), BufferAndIsPrimary(buffer, isPrimary),
|
|
PSPs(PrimarySpecificPaths()) {
|
|
assert(!name.empty());
|
|
}
|
|
|
|
public:
|
|
/// Retrieves the type of this input file.
|
|
file_types::ID getType() const { return FileID; };
|
|
|
|
/// Retrieves whether this input file was passed as a primary to the frontend.
|
|
bool isPrimary() const { return BufferAndIsPrimary.getInt(); }
|
|
|
|
/// Retrieves the backing buffer for this input file, if any.
|
|
llvm::MemoryBuffer *getBuffer() const {
|
|
return BufferAndIsPrimary.getPointer();
|
|
}
|
|
|
|
/// The name of this \c InputFile, or `-` if this input corresponds to the
|
|
/// standard input stream.
|
|
///
|
|
/// The returned file name is guaranteed not to be the empty string.
|
|
const std::string &getFileName() const {
|
|
assert(!Filename.empty());
|
|
return Filename;
|
|
}
|
|
|
|
/// Return Swift-standard file name from a buffer name set by
|
|
/// llvm::MemoryBuffer::getFileOrSTDIN, which uses "<stdin>" instead of "-".
|
|
static StringRef convertBufferNameFromLLVM_getFileOrSTDIN_toSwiftConventions(
|
|
StringRef filename) {
|
|
return filename == "<stdin>" ? "-" : filename;
|
|
}
|
|
|
|
/// Retrieves the name of the output file corresponding to this input.
|
|
///
|
|
/// If there is no such corresponding file, the result is the empty string.
|
|
/// If there the resulting output should be directed to the standard output
|
|
/// stream, the result is "-".
|
|
std::string outputFilename() const { return PSPs.OutputFilename; }
|
|
|
|
std::string indexUnitOutputFilename() const {
|
|
if (!PSPs.IndexUnitOutputFilename.empty())
|
|
return PSPs.IndexUnitOutputFilename;
|
|
return outputFilename();
|
|
}
|
|
|
|
/// If there are explicit primary inputs (i.e. designated with -primary-input
|
|
/// or -primary-filelist), the paths specific to those inputs (other than the
|
|
/// input file path itself) are kept here. If there are no explicit primary
|
|
/// inputs (for instance for whole module optimization), the corresponding
|
|
/// paths are kept in the first input file.
|
|
const PrimarySpecificPaths &getPrimarySpecificPaths() const { return PSPs; }
|
|
|
|
void setPrimarySpecificPaths(PrimarySpecificPaths &&PSPs) {
|
|
this->PSPs = std::move(PSPs);
|
|
}
|
|
|
|
// The next set of functions provides access to those primary-specific paths
|
|
// accessed directly from an InputFile, as opposed to via
|
|
// FrontendInputsAndOutputs. They merely make the call sites
|
|
// a bit shorter. Add more forwarding methods as needed.
|
|
|
|
StringRef getDependenciesFilePath() const {
|
|
return getPrimarySpecificPaths().SupplementaryOutputs.DependenciesFilePath;
|
|
}
|
|
StringRef getLoadedModuleTracePath() const {
|
|
return getPrimarySpecificPaths().SupplementaryOutputs.LoadedModuleTracePath;
|
|
}
|
|
StringRef getSerializedDiagnosticsPath() const {
|
|
return getPrimarySpecificPaths().SupplementaryOutputs
|
|
.SerializedDiagnosticsPath;
|
|
}
|
|
StringRef getFixItsOutputPath() const {
|
|
return getPrimarySpecificPaths().SupplementaryOutputs.FixItsOutputPath;
|
|
}
|
|
};
|
|
} // namespace swift
|
|
|
|
#endif // SWIFT_FRONTEND_INPUTFILE_H
|