32void ProxyOutputBackend::anchor() {}
33void OnDiskOutputBackend::anchor() {}
38 return const_cast<NullOutputBackend *
>(
this);
41 createFileImpl(
StringRef Path, std::optional<OutputConfig>)
override {
42 return std::make_unique<NullOutputFileImpl>();
55 std::optional<OutputConfig> Config)
override {
58 return std::make_unique<NullOutputFileImpl>();
63 getUnderlyingBackend().clone(),
Filter);
66 FilteringOutputBackend(
77 std::move(UnderlyingBackend), std::move(
Filter));
94 Error discard()
final {
96 return joinErrors(F1->discard(), F2->discard());
100 void write_impl(
const char *
Ptr,
size_t Size)
override {
109 uint64_t current_pos()
const override {
return F1->getOS().tell(); }
110 size_t preferred_buffer_size()
const override {
111 return PreferredBufferSize;
113 void reserveExtraSpace(
uint64_t ExtraSize)
override {
114 F1->getOS().reserveExtraSpace(ExtraSize);
115 F2->getOS().reserveExtraSpace(ExtraSize);
117 bool is_displayed()
const override {
118 return F1->getOS().is_displayed() && F2->getOS().is_displayed();
120 bool has_colors()
const override {
121 return F1->getOS().has_colors() && F2->getOS().has_colors();
123 void enable_colors(
bool enable)
override {
125 F1->getOS().enable_colors(enable);
126 F2->getOS().enable_colors(enable);
129 MirroringOutput(std::unique_ptr<OutputFileImpl> F1,
130 std::unique_ptr<OutputFileImpl> F2)
131 : PreferredBufferSize(std::max(F1->getOS().GetBufferSize(),
132 F1->getOS().GetBufferSize())),
133 F1(std::move(F1)), F2(std::move(F2)) {
135 this->F1->getOS().SetUnbuffered();
136 this->F2->getOS().SetUnbuffered();
138 size_t PreferredBufferSize;
139 std::unique_ptr<OutputFileImpl> F1;
140 std::unique_ptr<OutputFileImpl> F2;
142 struct MirroringOutputBackend :
public ProxyOutputBackend1,
143 public ProxyOutputBackend2 {
146 std::optional<OutputConfig> Config)
override {
147 std::unique_ptr<OutputFileImpl> File1;
148 std::unique_ptr<OutputFileImpl> File2;
150 ProxyOutputBackend1::createFileImpl(Path, Config).moveInto(File1))
153 ProxyOutputBackend2::createFileImpl(Path, Config).moveInto(File2))
154 return joinErrors(std::move(E), File1->discard());
159 return std::move(File2);
163 return std::move(File1);
165 return std::make_unique<MirroringOutput>(std::move(File1),
172 ProxyOutputBackend1::getUnderlyingBackend().clone(),
173 ProxyOutputBackend2::getUnderlyingBackend().clone()));
175 void Retain()
const { ProxyOutputBackend1::Retain(); }
176 void Release()
const { ProxyOutputBackend1::Release(); }
180 : ProxyOutputBackend1(std::move(Backend1)),
181 ProxyOutputBackend2(std::move(Backend2)) {}
184 assert(Backend1 &&
"Expected actual backend");
185 assert(Backend2 &&
"Expected actual backend");
188 std::move(Backend2)));
197 Config->setNoAtomicWrite();
199 Config->setNoDiscardOnSignal();
206 Error keep()
override;
207 Error discard()
override;
208 raw_pwrite_stream &getOS()
override {
209 assert(FileOS &&
"Expected valid file");
223 Error tryToCreateTemporary(std::optional<int> &FD);
225 Error initializeFile(std::optional<int> &FD);
226 Error initializeStream();
229 OnDiskOutputFile(StringRef OutputPath, std::optional<OutputConfig> Config,
230 const OnDiskOutputBackend::OutputSettings &Settings)
232 OutputPath(OutputPath.str()) {}
235 const std::string OutputPath;
236 std::optional<std::string> TempPath;
237 std::optional<raw_fd_ostream> FileOS;
238 std::optional<buffer_ostream> BufferOS;
245 return handleErrors(CreateFile(), [&](std::unique_ptr<ECError> EC) {
246 if (EC->convertToErrorCode() != std::errc::no_such_file_or_directory ||
247 Config.getNoImplyCreateDirectories())
248 return Error(std::move(EC));
257Error OnDiskOutputFile::tryToCreateTemporary(std::optional<int> &FD) {
263 SmallString<128> ModelPath =
264 StringRef(OutputPath).drop_back(OutputExtension.
size());
265 ModelPath +=
"-%%%%%%%%";
266 ModelPath += OutputExtension;
271 SmallString<128> UniquePath;
272 if (std::error_code EC =
276 if (Config.getDiscardOnSignal())
279 TempPath = UniquePath.
str().
str();
285Error OnDiskOutputFile::initializeFile(std::optional<int> &FD) {
286 assert(OutputPath !=
"-" &&
"Unexpected request for FD of stdout");
291 if (Config.getAtomicWrite()) {
292 sys::fs::file_status
Status;
296 Config.setNoAtomicWrite();
302 std::make_error_code(std::errc::operation_not_permitted));
308 if (Config.getAtomicWrite())
318 else if (Config.getText())
320 if (Config.getAppend())
327 if (Config.getDiscardOnSignal())
333Error OnDiskOutputFile::initializeStream() {
335 if (OutputPath ==
"-") {
337 FileOS.emplace(OutputPath, EC);
341 std::optional<int> FD;
342 if (
Error E = initializeFile(FD))
344 FileOS.emplace(*FD,
true);
348 if (!FileOS->supportsSeeking() && !Config.getText())
349 BufferOS.emplace(*FileOS);
356 static const int InvalidFd = -1;
367enum class FileDifference : uint8_t {
379static Expected<FileDifference>
382 return FileDifference::IdenticalFile;
384 OpenFileRAII SourceFile;
394 OpenFileRAII DestFile;
397 if (std::error_code
Error =
399 return FileDifference::DifferentContents;
403 return FileDifference::DifferentContents;
408 return FileDifference::DifferentContents;
412 return FileDifference::SameContents;
416 std::error_code SourceRegionErr;
423 std::error_code DestRegionErr;
429 return FileDifference::DifferentContents;
432 return FileDifference::DifferentContents;
434 return FileDifference::SameContents;
437Error OnDiskOutputFile::reset() {
444 std::error_code
EC = FileOS->error();
446 FileOS->clear_error();
451Error OnDiskOutputFile::keep() {
452 if (
auto E = reset())
457 if (Config.getDiscardOnSignal())
465 if (Config.getAppend() && OutputPath !=
"-") {
474 llvm::LockFileManager Lock(OutputPath);
476 if (
Error Err = Lock.tryLock().moveInto(Owned)) {
479 Lock.unsafeMaybeUnlock();
481 OutputPath, std::make_error_code(std::errc::no_lock_available));
489 Out << (*Content)->getBuffer();
491 Lock.unsafeMaybeUnlock();
499 switch (Lock.waitForUnlockFor(std::chrono::seconds(256))) {
511 Lock.unsafeMaybeUnlock();
519 if (Config.getOnlyIfDifferent()) {
522 return Result.takeError();
524 case FileDifference::IdenticalFile:
528 case FileDifference::SameContents:
533 case FileDifference::DifferentContents:
555Error OnDiskOutputFile::discard() {
557 if (
auto E = reset())
561 if (OutputPath ==
"-")
564 auto discardPath = [&](StringRef
Path) {
574 discardPath(*TempPath));
584 std::optional<OutputConfig> Config) {
593 auto File = std::make_unique<OnDiskOutputFile>(Path, Config,
Settings);
597 return std::move(
File);
assert(UImm &&(UImm !=~static_cast< T >(0)) &&"Invalid immediate!")
static GCRegistry::Add< CoreCLRGC > E("coreclr", "CoreCLR-compatible GC")
#define LLVM_FALLTHROUGH
LLVM_FALLTHROUGH - Mark fallthrough cases in switch statements.
Merge contiguous icmps into a memcmp
Provides a library for accessing information about this process and other processes on the operating ...
This file defines the make_scope_exit function, which executes user-defined cleanup logic at scope ex...
static Error createDirectoriesOnDemand(StringRef OutputPath, OutputConfig Config, llvm::function_ref< Error()> CreateFile)
static Expected< FileDifference > areFilesDifferent(const llvm::Twine &Source, const llvm::Twine &Destination)
static OutputConfig applySettings(std::optional< OutputConfig > &&Config, const OnDiskOutputBackend::OutputSettings &Settings)
This file contains the declarations of the concrete VirtualOutputBackend classes, which are the imple...
This file contains the declarations of the OutputConfig class.
This file contains the declarations of the OutputError class.
Lightweight error class with error context and mandatory checking.
static ErrorSuccess success()
Create a success value.
Tagged union holding either a T or a Error.
A smart pointer to a reference-counted object that inherits from RefCountedBase or ThreadSafeRefCount...
static ErrorOr< std::unique_ptr< MemoryBuffer > > getFile(const Twine &Filename, bool IsText=false, bool RequiresNullTerminator=true, bool IsVolatile=false, std::optional< Align > Alignment=std::nullopt)
Open the specified file as a MemoryBuffer, returning a new MemoryBuffer if successful,...
SmallString - A SmallString is just a SmallVector with methods and accessors that make it work better...
StringRef str() const
Explicit conversion to StringRef.
This class consists of common code factored out of the SmallVector class to reduce code duplication b...
StringRef - Represent a constant reference to a string, i.e.
std::string str() const
str - Get the contents as an std::string.
constexpr size_t size() const
size - Get the string size.
Twine - A lightweight data structure for efficiently representing the concatenation of temporary valu...
An efficient, type-erasing, non-owning reference to a callable.
raw_ostream & write(unsigned char C)
virtual void enable_colors(bool enable)
An abstract base class for streams implementations that also support a pwrite operation.
static LLVM_ABI std::error_code SafelyCloseFileDescriptor(int FD)
Represents the result of a call to sys::fs::status().
This class represents a memory mapped file.
@ readonly
May only access map via const_data as read only.
LLVM_ABI const char * const_data() const
Get a const view of the data.
Expected< std::unique_ptr< OutputFileImpl > > createFileImpl(StringRef Path, std::optional< OutputConfig > Config) override
Create a file for Path.
OutputSettings Settings
Settings for this backend.
Error makeAbsolute(SmallVectorImpl< char > &Path) const
Resolve an absolute path.
Interface for virtualized outputs.
A helper class for proxying another backend, with the default implementation to forward to the underl...
ProxyOutputBackend(IntrusiveRefCntPtr< OutputBackend > UnderlyingBackend)
Expected< std::unique_ptr< OutputFileImpl > > createFileImpl(StringRef Path, std::optional< OutputConfig > Config) override
Create a file for Path.
The result of a status operation.
LLVM_ABI bool is_regular_file(const basic_file_status &status)
Does status represent a regular file?
LLVM_ABI void make_absolute(const Twine ¤t_directory, SmallVectorImpl< char > &path)
Make path an absolute path.
LLVM_ABI std::error_code rename(const Twine &from, const Twine &to)
Rename from to to.
bool can_write(const Twine &Path)
Can we write this file?
LLVM_ABI bool exists(const basic_file_status &status)
Does file exist?
@ OF_Text
The file should be opened in text mode on platforms like z/OS that make this distinction.
@ OF_TextWithCRLF
The file should be opened in text mode and use a carriage linefeed '\r '.
@ OF_Append
The file should be opened in append mode.
LLVM_ABI std::error_code createUniqueFile(const Twine &Model, int &ResultFD, SmallVectorImpl< char > &ResultPath, OpenFlags Flags=OF_None, unsigned Mode=all_read|all_write)
Create a uniquely named file.
LLVM_ABI std::error_code remove(const Twine &path, bool IgnoreNonExisting=true)
Remove path.
@ CD_CreateAlways
CD_CreateAlways - When opening a file:
std::error_code openFileForWrite(const Twine &Name, int &ResultFD, CreationDisposition Disp=CD_CreateAlways, OpenFlags Flags=OF_None, unsigned Mode=0666)
Opens the file with the given name in a write-only or read-write mode, returning its open file descri...
LLVM_ABI std::error_code create_directories(const Twine &path, bool IgnoreExisting=true, perms Perms=owner_all|group_all)
Create all the non-existent directories in path.
LLVM_ABI std::error_code copy_file(const Twine &From, const Twine &To)
Copy the contents of From to To.
LLVM_ABI file_t convertFDToNativeFile(int FD)
Converts from a Posix file descriptor number to a native file handle.
LLVM_ABI std::error_code status(const Twine &path, file_status &result, bool follow=true)
Get file status as if by POSIX stat().
LLVM_ABI std::error_code openFileForRead(const Twine &Name, int &ResultFD, OpenFlags Flags=OF_None, SmallVectorImpl< char > *RealPath=nullptr)
Opens the file with the given name in a read-only mode, returning its open file descriptor.
LLVM_ABI bool equivalent(file_status A, file_status B)
Do file_status's represent the same thing?
LLVM_ABI StringRef parent_path(StringRef path LLVM_LIFETIME_BOUND, Style style=Style::native)
Get parent path.
LLVM_ABI StringRef extension(StringRef path LLVM_LIFETIME_BOUND, Style style=Style::native)
Get extension.
LLVM_ABI void DontRemoveFileOnSignal(StringRef Filename)
This function removes a file from the list of files to be removed on signal delivery.
LLVM_ABI bool RemoveFileOnSignal(StringRef Filename, std::string *ErrMsg=nullptr)
This function registers signal handlers to ensure that if a signal gets delivered that the named file...
Error convertToOutputError(const Twine &OutputPath, std::error_code EC)
Return Error::success() or use OutputPath to create an OutputError, depending on EC.
IntrusiveRefCntPtr< OutputBackend > makeNullOutputBackend()
Create a backend that ignores all output.
IntrusiveRefCntPtr< OutputBackend > makeFilteringOutputBackend(IntrusiveRefCntPtr< OutputBackend > UnderlyingBackend, std::function< bool(StringRef, std::optional< OutputConfig >)> Filter)
Make a backend where OutputBackend::createFile() forwards to UnderlyingBackend when Filter is true,...
IntrusiveRefCntPtr< OutputBackend > makeMirroringOutputBackend(IntrusiveRefCntPtr< OutputBackend > Backend1, IntrusiveRefCntPtr< OutputBackend > Backend2)
Create a backend that forwards OutputBackend::createFile() to both Backend1 and Backend2.
Error convertToTempFileOutputError(const Twine &TempPath, const Twine &OutputPath, std::error_code EC)
Return Error::success() or use TempPath and OutputPath to create a TempFileOutputError,...
This is an optimization pass for GlobalISel generic memory operations.
bool errorToBool(Error Err)
Helper for converting an Error to a bool.
detail::scope_exit< std::decay_t< Callable > > make_scope_exit(Callable &&F)
Error handleErrors(Error E, HandlerTs &&... Hs)
Pass the ErrorInfo(s) contained in E to their respective handlers.
IntrusiveRefCntPtr< T > makeIntrusiveRefCnt(Args &&...A)
Factory function for creating intrusive ref counted pointers.
Error joinErrors(Error E1, Error E2)
Concatenate errors.
bool isa(const From &Val)
isa<X> - Return true if the parameter to the template is an instance of one of the template type argu...
@ Success
The lock was released successfully.
@ OwnerDied
Owner died while holding the lock.
@ Timeout
Reached timeout while waiting for the owner to release the lock.
Error make_error(ArgTs &&... Args)
Make a Error instance representing failure using the given error info type.
OutputIt move(R &&Range, OutputIt Out)
Provide wrappers to std::move which take ranges instead of having to pass begin/end explicitly.
LLVM_ABI Error errorCodeToError(std::error_code EC)
Helper for converting an std::error_code to a Error.
void consumeError(Error Err)
Consume a Error without doing anything.
OutputConfig DefaultConfig
bool RemoveOnSignal
Register output files to be deleted if a signal is received.
bool UseTemporaries
Use temporary files.
Full configuration for an output for use by the OutputBackend.
constexpr bool getTextWithCRLF() const