1 // Copyright 2017 The Abseil Authors.
2 //
3 // Licensed under the Apache License, Version 2.0 (the "License");
4 // you may not use this file except in compliance with the License.
5 // You may obtain a copy of the License at
6 //
7 // https://www.apache.org/licenses/LICENSE-2.0
8 //
9 // Unless required by applicable law or agreed to in writing, software
10 // distributed under the License is distributed on an "AS IS" BASIS,
11 // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12 // See the License for the specific language governing permissions and
13 // limitations under the License.
14
15 #include "absl/strings/internal/str_format/output.h"
16
17 #include <errno.h>
18 #include <cstring>
19
20 namespace absl {
21 ABSL_NAMESPACE_BEGIN
22 namespace str_format_internal {
23
24 namespace {
25 struct ClearErrnoGuard {
ClearErrnoGuardabsl::str_format_internal::__anon9365ecbb0111::ClearErrnoGuard26 ClearErrnoGuard() : old_value(errno) { errno = 0; }
~ClearErrnoGuardabsl::str_format_internal::__anon9365ecbb0111::ClearErrnoGuard27 ~ClearErrnoGuard() {
28 if (!errno) errno = old_value;
29 }
30 int old_value;
31 };
32 } // namespace
33
Write(string_view v)34 void BufferRawSink::Write(string_view v) {
35 size_t to_write = std::min(v.size(), size_);
36 std::memcpy(buffer_, v.data(), to_write);
37 buffer_ += to_write;
38 size_ -= to_write;
39 total_written_ += v.size();
40 }
41
Write(string_view v)42 void FILERawSink::Write(string_view v) {
43 while (!v.empty() && !error_) {
44 // Reset errno to zero in case the libc implementation doesn't set errno
45 // when a failure occurs.
46 ClearErrnoGuard guard;
47
48 if (size_t result = std::fwrite(v.data(), 1, v.size(), output_)) {
49 // Some progress was made.
50 count_ += result;
51 v.remove_prefix(result);
52 } else {
53 if (errno == EINTR) {
54 continue;
55 } else if (errno) {
56 error_ = errno;
57 } else if (std::ferror(output_)) {
58 // Non-POSIX compliant libc implementations may not set errno, so we
59 // have check the streams error indicator.
60 error_ = EBADF;
61 } else {
62 // We're likely on a non-POSIX system that encountered EINTR but had no
63 // way of reporting it.
64 continue;
65 }
66 }
67 }
68 }
69
70 } // namespace str_format_internal
71 ABSL_NAMESPACE_END
72 } // namespace absl
73