33void ProxyOutputBackend::anchor() {}
34void OnDiskOutputBackend::anchor() {}
39 return const_cast<NullOutputBackend *
>(
this);
42 createFileImpl(
StringRef Path, std::optional<OutputConfig>)
override {
43 return std::make_unique<NullOutputFileImpl>();
56 std::optional<OutputConfig> Config)
override {
59 return std::make_unique<NullOutputFileImpl>();
64 getUnderlyingBackend().clone(),
Filter);
67 FilteringOutputBackend(
78 std::move(UnderlyingBackend), std::move(
Filter));
95 Error discard()
final {
97 return joinErrors(F1->discard(), F2->discard());
101 void write_impl(
const char *Ptr,
size_t Size)
override {
110 uint64_t current_pos()
const override {
return F1->getOS().tell(); }
111 size_t preferred_buffer_size()
const override {
112 return PreferredBufferSize;
114 void reserveExtraSpace(
uint64_t ExtraSize)
override {
115 F1->getOS().reserveExtraSpace(ExtraSize);
116 F2->getOS().reserveExtraSpace(ExtraSize);
118 bool is_displayed()
const override {
119 return F1->getOS().is_displayed() && F2->getOS().is_displayed();
121 bool has_colors()
const override {
122 return F1->getOS().has_colors() && F2->getOS().has_colors();
124 void enable_colors(
bool enable)
override {
126 F1->getOS().enable_colors(enable);
127 F2->getOS().enable_colors(enable);
130 MirroringOutput(std::unique_ptr<OutputFileImpl> F1,
131 std::unique_ptr<OutputFileImpl> F2)
132 : PreferredBufferSize(std::max(F1->getOS().GetBufferSize(),
133 F1->getOS().GetBufferSize())),
134 F1(std::move(F1)), F2(std::move(F2)) {
136 this->F1->getOS().SetUnbuffered();
137 this->F2->getOS().SetUnbuffered();
139 size_t PreferredBufferSize;
140 std::unique_ptr<OutputFileImpl> F1;
141 std::unique_ptr<OutputFileImpl> F2;
143 struct MirroringOutputBackend :
public ProxyOutputBackend1,
144 public ProxyOutputBackend2 {
147 std::optional<OutputConfig> Config)
override {
148 std::unique_ptr<OutputFileImpl> File1;
149 std::unique_ptr<OutputFileImpl> File2;
151 ProxyOutputBackend1::createFileImpl(Path, Config).moveInto(File1))
154 ProxyOutputBackend2::createFileImpl(Path, Config).moveInto(File2))
155 return joinErrors(std::move(E), File1->discard());
160 return std::move(File2);
164 return std::move(File1);
166 return std::make_unique<MirroringOutput>(std::move(File1),
173 ProxyOutputBackend1::getUnderlyingBackend().clone(),
174 ProxyOutputBackend2::getUnderlyingBackend().clone()));
176 void Retain()
const { ProxyOutputBackend1::Retain(); }
177 void Release()
const { ProxyOutputBackend1::Release(); }
181 : ProxyOutputBackend1(std::move(Backend1)),
182 ProxyOutputBackend2(std::move(Backend2)) {}
185 assert(Backend1 &&
"Expected actual backend");
186 assert(Backend2 &&
"Expected actual backend");
189 std::move(Backend2)));
198 Config->setNoAtomicWrite();
200 Config->setNoDiscardOnSignal();
207 Error keep()
override;
208 Error discard()
override;
209 raw_pwrite_stream &getOS()
override {
210 assert(FileOS &&
"Expected valid file");
224 Error tryToCreateTemporary(std::optional<int> &FD);
226 Error initializeFile(std::optional<int> &FD);
227 Error initializeStream();
230 OnDiskOutputFile(StringRef OutputPath, std::optional<OutputConfig> Config,
231 const OnDiskOutputBackend::OutputSettings &Settings)
233 OutputPath(OutputPath.str()) {}
236 const std::string OutputPath;
237 std::optional<std::string> TempPath;
238 std::optional<raw_fd_ostream> FileOS;
239 std::optional<buffer_ostream> BufferOS;
246 return handleErrors(CreateFile(), [&](std::unique_ptr<ECError> EC) {
247 if (EC->convertToErrorCode() != std::errc::no_such_file_or_directory ||
248 Config.getNoImplyCreateDirectories())
249 return Error(std::move(EC));
262 else if (Config.getText())
266 if (Config.getAppend() && !Config.getAtomicWrite())
272Error OnDiskOutputFile::tryToCreateTemporary(std::optional<int> &FD) {
280 SmallString<128> ModelPath =
281 StringRef(OutputPath).drop_back(OutputExtension.
size());
282 ModelPath +=
"-%%%%%%%%";
283 ModelPath += OutputExtension;
288 SmallString<128> UniquePath;
290 if (std::error_code EC =
294 if (Config.getDiscardOnSignal())
297 TempPath = UniquePath.
str().
str();
303Error OnDiskOutputFile::initializeFile(std::optional<int> &FD) {
306 assert(OutputPath !=
"-" &&
"Unexpected request for FD of stdout");
311 if (Config.getAtomicWrite()) {
312 sys::fs::file_status
Status;
316 Config.setNoAtomicWrite();
322 std::make_error_code(std::errc::operation_not_permitted));
328 if (Config.getAtomicWrite())
341 if (Config.getDiscardOnSignal())
347Error OnDiskOutputFile::initializeStream() {
351 if (OutputPath ==
"-") {
353 FileOS.emplace(OutputPath, EC);
357 std::optional<int> FD;
358 if (
Error E = initializeFile(FD))
360 FileOS.emplace(*FD,
true);
364 if (!FileOS->supportsSeeking() && !Config.getText())
365 BufferOS.emplace(*FileOS);
372 static const int InvalidFd = -1;
383enum class FileDifference : uint8_t {
395static Expected<FileDifference>
398 return FileDifference::IdenticalFile;
400 OpenFileRAII SourceFile;
410 OpenFileRAII DestFile;
413 if (std::error_code
Error =
415 return FileDifference::DifferentContents;
419 return FileDifference::DifferentContents;
424 return FileDifference::DifferentContents;
428 return FileDifference::SameContents;
432 std::error_code SourceRegionErr;
439 std::error_code DestRegionErr;
445 return FileDifference::DifferentContents;
448 return FileDifference::DifferentContents;
450 return FileDifference::SameContents;
453Error OnDiskOutputFile::reset() {
462 std::error_code
EC = FileOS->error();
464 FileOS->clear_error();
469Error OnDiskOutputFile::keep() {
472 if (
auto E = reset())
477 if (Config.getDiscardOnSignal())
485 if (Config.getAppend() && OutputPath !=
"-") {
494 llvm::LockFileManager Lock(OutputPath);
496 if (
Error Err = Lock.tryLock().moveInto(Owned)) {
499 Lock.unsafeMaybeUnlock();
501 OutputPath, std::make_error_code(std::errc::no_lock_available));
509 Out << (*Content)->getBuffer();
511 Lock.unsafeMaybeUnlock();
519 switch (Lock.waitForUnlockFor(std::chrono::seconds(256))) {
531 Lock.unsafeMaybeUnlock();
539 if (Config.getOnlyIfDifferent()) {
542 return Result.takeError();
544 case FileDifference::IdenticalFile:
548 case FileDifference::SameContents:
553 case FileDifference::DifferentContents:
575Error OnDiskOutputFile::discard() {
579 if (
auto E = reset())
583 if (OutputPath ==
"-")
586 auto discardPath = [&](StringRef
Path) {
596 discardPath(*TempPath));
608 std::optional<OutputConfig> Config) {
619 auto File = std::make_unique<OnDiskOutputFile>(Path, Config,
Settings);
623 return std::move(
File);
assert(UImm &&(UImm !=~static_cast< T >(0)) &&"Invalid immediate!")
static GCRegistry::Add< CoreCLRGC > E("coreclr", "CoreCLR-compatible GC")
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 sys::fs::OpenFlags generateFlagsFromConfig(OutputConfig Config)
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.
LLVM_ABI 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 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 make_absolute(SmallVectorImpl< char > &path)
Make path an absolute 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.
ScopedSetting scopedDisable()
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.
LLVM_ABI IntrusiveRefCntPtr< OutputBackend > makeNullOutputBackend()
Create a backend that ignores all output.
LLVM_ABI 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,...
LLVM_ABI 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