xref: /llvm-project/llvm/lib/Support/FileOutputBuffer.cpp (revision 0d7a38a81d1fdb8c9c522dce83fb63a41d30ebc3)
1 //===- FileOutputBuffer.cpp - File Output Buffer ----------------*- C++ -*-===//
2 //
3 //                     The LLVM Compiler Infrastructure
4 //
5 // This file is distributed under the University of Illinois Open Source
6 // License. See LICENSE.TXT for details.
7 //
8 //===----------------------------------------------------------------------===//
9 //
10 // Utility for creating a in-memory buffer that will be written to a file.
11 //
12 //===----------------------------------------------------------------------===//
13 
14 #include "llvm/Support/FileOutputBuffer.h"
15 #include "llvm/ADT/STLExtras.h"
16 #include "llvm/ADT/SmallString.h"
17 #include "llvm/Support/Errc.h"
18 #include "llvm/Support/Memory.h"
19 #include "llvm/Support/Path.h"
20 #include "llvm/Support/Signals.h"
21 #include <system_error>
22 
23 #if !defined(_MSC_VER) && !defined(__MINGW32__)
24 #include <unistd.h>
25 #else
26 #include <io.h>
27 #endif
28 
29 using namespace llvm;
30 using namespace llvm::sys;
31 
32 // A FileOutputBuffer which creates a temporary file in the same directory
33 // as the final output file. The final output file is atomically replaced
34 // with the temporary file on commit().
35 class OnDiskBuffer : public FileOutputBuffer {
36 public:
37   OnDiskBuffer(StringRef Path, StringRef TempPath,
38                std::unique_ptr<fs::mapped_file_region> Buf)
39       : FileOutputBuffer(Path), Buffer(std::move(Buf)), TempPath(TempPath) {}
40 
41   static Expected<std::unique_ptr<OnDiskBuffer>>
42   create(StringRef Path, size_t Size, unsigned Mode);
43 
44   uint8_t *getBufferStart() const override { return (uint8_t *)Buffer->data(); }
45 
46   uint8_t *getBufferEnd() const override {
47     return (uint8_t *)Buffer->data() + Buffer->size();
48   }
49 
50   size_t getBufferSize() const override { return Buffer->size(); }
51 
52   Error commit() override {
53     // Unmap buffer, letting OS flush dirty pages to file on disk.
54     Buffer.reset();
55 
56     // Atomically replace the existing file with the new one.
57     auto EC = fs::rename(TempPath, FinalPath);
58     sys::DontRemoveFileOnSignal(TempPath);
59     return errorCodeToError(EC);
60   }
61 
62   ~OnDiskBuffer() override {
63     // Close the mapping before deleting the temp file, so that the removal
64     // succeeds.
65     Buffer.reset();
66     fs::remove(TempPath);
67   }
68 
69 private:
70   std::unique_ptr<fs::mapped_file_region> Buffer;
71   std::string TempPath;
72 };
73 
74 // A FileOutputBuffer which keeps data in memory and writes to the final
75 // output file on commit(). This is used only when we cannot use OnDiskBuffer.
76 class InMemoryBuffer : public FileOutputBuffer {
77 public:
78   InMemoryBuffer(StringRef Path, MemoryBlock Buf, unsigned Mode)
79       : FileOutputBuffer(Path), Buffer(Buf), Mode(Mode) {}
80 
81   static Expected<std::unique_ptr<InMemoryBuffer>>
82   create(StringRef Path, size_t Size, unsigned Mode) {
83     std::error_code EC;
84     MemoryBlock MB = Memory::allocateMappedMemory(
85         Size, nullptr, sys::Memory::MF_READ | sys::Memory::MF_WRITE, EC);
86     if (EC)
87       return errorCodeToError(EC);
88     return llvm::make_unique<InMemoryBuffer>(Path, MB, Mode);
89   }
90 
91   uint8_t *getBufferStart() const override { return (uint8_t *)Buffer.base(); }
92 
93   uint8_t *getBufferEnd() const override {
94     return (uint8_t *)Buffer.base() + Buffer.size();
95   }
96 
97   size_t getBufferSize() const override { return Buffer.size(); }
98 
99   Error commit() override {
100     int FD;
101     std::error_code EC;
102     if (auto EC = openFileForWrite(FinalPath, FD, fs::F_None, Mode))
103       return errorCodeToError(EC);
104     raw_fd_ostream OS(FD, /*shouldClose=*/true, /*unbuffered=*/true);
105     OS << StringRef((const char *)Buffer.base(), Buffer.size());
106     return Error::success();
107   }
108 
109 private:
110   OwningMemoryBlock Buffer;
111   unsigned Mode;
112 };
113 
114 Expected<std::unique_ptr<OnDiskBuffer>>
115 OnDiskBuffer::create(StringRef Path, size_t Size, unsigned Mode) {
116   // Create new file in same directory but with random name.
117   SmallString<128> TempPath;
118   int FD;
119   if (auto EC = fs::createUniqueFile(Path + ".tmp%%%%%%%", FD, TempPath, Mode))
120     return errorCodeToError(EC);
121 
122   sys::RemoveFileOnSignal(TempPath);
123 
124 #ifndef LLVM_ON_WIN32
125   // On Windows, CreateFileMapping (the mmap function on Windows)
126   // automatically extends the underlying file. We don't need to
127   // extend the file beforehand. _chsize (ftruncate on Windows) is
128   // pretty slow just like it writes specified amount of bytes,
129   // so we should avoid calling that function.
130   if (auto EC = fs::resize_file(FD, Size))
131     return errorCodeToError(EC);
132 #endif
133 
134   // Mmap it.
135   std::error_code EC;
136   auto MappedFile = llvm::make_unique<fs::mapped_file_region>(
137       FD, fs::mapped_file_region::readwrite, Size, 0, EC);
138   close(FD);
139   if (EC)
140     return errorCodeToError(EC);
141   return llvm::make_unique<OnDiskBuffer>(Path, TempPath, std::move(MappedFile));
142 }
143 
144 // Create an instance of FileOutputBuffer.
145 Expected<std::unique_ptr<FileOutputBuffer>>
146 FileOutputBuffer::create(StringRef Path, size_t Size, unsigned Flags) {
147   unsigned Mode = fs::all_read | fs::all_write;
148   if (Flags & F_executable)
149     Mode |= fs::all_exe;
150 
151   fs::file_status Stat;
152   fs::status(Path, Stat);
153 
154   // Usually, we want to create OnDiskBuffer to create a temporary file in
155   // the same directory as the destination file and atomically replaces it
156   // by rename(2).
157   //
158   // However, if the destination file is a special file, we don't want to
159   // use rename (e.g. we don't want to replace /dev/null with a regular
160   // file.) If that's the case, we create an in-memory buffer, open the
161   // destination file and write to it on commit().
162   switch (Stat.type()) {
163   case fs::file_type::directory_file:
164     return errorCodeToError(errc::is_a_directory);
165   case fs::file_type::regular_file:
166   case fs::file_type::file_not_found:
167   case fs::file_type::status_error:
168     return OnDiskBuffer::create(Path, Size, Mode);
169   default:
170     return InMemoryBuffer::create(Path, Size, Mode);
171   }
172 }
173