1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
| static auto CheckContentSize(int64_t size) -> llvm::Error {
if (size < std::numeric_limits<int32_t>::max()) {
return llvm::Error::success();
}
return llvm::createStringError(llvm::inconvertibleErrorCode(),
"Input too large!");
}
auto SourceBuffer::CreateFromText(llvm::Twine text, llvm::StringRef filename)
-> llvm::Expected<SourceBuffer> {
std::string buffer = text.str();
auto size_check = CheckContentSize(buffer.size());
if (size_check) {
return std::move(size_check);
}
return SourceBuffer(filename.str(), std::move(buffer));
}
static auto ErrnoToError(int errno_value) -> llvm::Error {
return llvm::errorCodeToError(
std::error_code(errno_value, std::generic_category()));
}
auto SourceBuffer::CreateFromFile(llvm::StringRef filename)
-> llvm::Expected<SourceBuffer> {
std::string filename_str = filename.str();
errno = 0;
int file_descriptor = open(filename_str.c_str(), O_RDONLY);
if (file_descriptor == -1) {
return ErrnoToError(errno);
}
auto closer =
llvm::make_scope_exit([file_descriptor] { close(file_descriptor); });
struct stat stat_buffer = {};
errno = 0;
if (fstat(file_descriptor, &stat_buffer) == -1) {
return ErrnoToError(errno);
}
int64_t size = stat_buffer.st_size;
if (size == 0) {
return SourceBuffer(std::move(filename_str), std::string());
}
auto size_check = CheckContentSize(size);
if (size_check) {
return std::move(size_check);
}
errno = 0;
void* mapped_text = mmap(nullptr, size, PROT_READ, MAP_PRIVATE,
file_descriptor, /*offset=*/0);
if (mapped_text == MAP_FAILED) {
return ErrnoToError(errno);
}
errno = 0;
closer.release();
if (close(file_descriptor) == -1) {
munmap(mapped_text, size);
return ErrnoToError(errno);
}
return SourceBuffer(
std::move(filename_str),
llvm::StringRef(static_cast<const char*>(mapped_text), size));
}
SourceBuffer::SourceBuffer(SourceBuffer&& arg) noexcept
: content_mode_(
std::exchange(arg.content_mode_, ContentMode::Uninitialized)),
filename_(std::move(arg.filename_)),
text_storage_(std::move(arg.text_storage_)),
text_(content_mode_ == ContentMode::Owned ? text_storage_ : arg.text_) {}
SourceBuffer::SourceBuffer(std::string filename, std::string text)
: content_mode_(ContentMode::Owned),
filename_(std::move(filename)),
text_storage_(std::move(text)),
text_(text_storage_) {}
SourceBuffer::SourceBuffer(std::string filename, llvm::StringRef text)
: content_mode_(ContentMode::MMapped),
filename_(std::move(filename)),
text_(text) {
COCKTAIL_CHECK(!text.empty())
<< "Must not have an empty text when we have mapped data from a file!";
}
SourceBuffer::~SourceBuffer() {
if (content_mode_ == ContentMode::MMapped) {
errno = 0;
int result =
munmap(const_cast<void*>(static_cast<const void*>(text_.data())),
text_.size());
COCKTAIL_CHECK(result != -1) << "Unmapping text failed!";
}
}
|