blob: 4267dd546c4dc9fdfe2df79e493ae26b44df9df1 (
plain)
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
|
//===-- Implementation of puts --------------------------------------------===//
//
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
// See https://llvm.org/LICENSE.txt for license information.
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
//
//===----------------------------------------------------------------------===//
#include "src/stdio/puts.h"
#include "src/__support/CPP/string_view.h"
#include "src/__support/File/file.h"
#include "hdr/types/FILE.h"
#include "src/__support/libc_errno.h"
#include "src/__support/macros/config.h"
#include <stddef.h>
namespace LIBC_NAMESPACE_DECL {
namespace {
// Simple helper to unlock the file once destroyed.
struct ScopedLock {
ScopedLock(LIBC_NAMESPACE::File *stream) : stream(stream) { stream->lock(); }
~ScopedLock() { stream->unlock(); }
private:
LIBC_NAMESPACE::File *stream;
};
} // namespace
LLVM_LIBC_FUNCTION(int, puts, (const char *__restrict str)) {
cpp::string_view str_view(str);
// We need to lock the stream to ensure the newline is always appended.
ScopedLock lock(LIBC_NAMESPACE::stdout);
auto result = LIBC_NAMESPACE::stdout->write_unlocked(str, str_view.size());
if (result.has_error())
libc_errno = result.error;
size_t written = result.value;
if (str_view.size() != written) {
// The stream should be in an error state in this case.
return EOF;
}
result = LIBC_NAMESPACE::stdout->write_unlocked("\n", 1);
if (result.has_error())
libc_errno = result.error;
written = result.value;
if (1 != written) {
// The stream should be in an error state in this case.
return EOF;
}
return 0;
}
} // namespace LIBC_NAMESPACE_DECL
|