1 //===-- Internal implementation header of vfprintf --------------*- C++ -*-===//
3 // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4 // See https://llvm.org/LICENSE.txt for license information.
5 // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
7 //===----------------------------------------------------------------------===//
9 #ifndef LLVM_LIBC_SRC_STDIO_PRINTF_CORE_VFPRINTF_INTERNAL_H
10 #define LLVM_LIBC_SRC_STDIO_PRINTF_CORE_VFPRINTF_INTERNAL_H
12 #include "src/__support/File/file.h"
13 #include "src/__support/arg_list.h"
14 #include "src/__support/macros/attributes.h" // For LIBC_INLINE
15 #include "src/__support/macros/config.h"
16 #include "src/stdio/printf_core/core_structs.h"
17 #include "src/stdio/printf_core/printf_main.h"
18 #include "src/stdio/printf_core/writer.h"
20 #include "hdr/types/FILE.h"
22 namespace LIBC_NAMESPACE_DECL
{
25 #ifndef LIBC_COPT_STDIO_USE_SYSTEM_FILE
26 LIBC_INLINE
int ferror_unlocked(FILE *f
) {
27 return reinterpret_cast<LIBC_NAMESPACE::File
*>(f
)->error_unlocked();
30 LIBC_INLINE
void flockfile(FILE *f
) {
31 reinterpret_cast<LIBC_NAMESPACE::File
*>(f
)->lock();
34 LIBC_INLINE
void funlockfile(FILE *f
) {
35 reinterpret_cast<LIBC_NAMESPACE::File
*>(f
)->unlock();
38 LIBC_INLINE
size_t fwrite_unlocked(const void *ptr
, size_t size
, size_t nmemb
,
40 return reinterpret_cast<LIBC_NAMESPACE::File
*>(f
)->write_unlocked(
43 #else // defined(LIBC_COPT_STDIO_USE_SYSTEM_FILE)
44 LIBC_INLINE
int ferror_unlocked(::FILE *f
) { return ::ferror_unlocked(f
); }
46 LIBC_INLINE
void flockfile(::FILE *f
) { ::flockfile(f
); }
48 LIBC_INLINE
void funlockfile(::FILE *f
) { ::funlockfile(f
); }
50 LIBC_INLINE
size_t fwrite_unlocked(const void *ptr
, size_t size
, size_t nmemb
,
52 return ::fwrite_unlocked(ptr
, size
, nmemb
, f
);
54 #endif // LIBC_COPT_STDIO_USE_SYSTEM_FILE
55 } // namespace internal
57 namespace printf_core
{
59 LIBC_INLINE
int file_write_hook(cpp::string_view new_str
, void *fp
) {
60 ::FILE *target_file
= reinterpret_cast<::FILE *>(fp
);
61 // Write new_str to the target file. The logic preventing a zero-length write
62 // is in the writer, so we don't check here.
63 size_t written
= internal::fwrite_unlocked(new_str
.data(), sizeof(char),
64 new_str
.size(), target_file
);
65 if (written
!= new_str
.size() || internal::ferror_unlocked(target_file
))
66 return FILE_WRITE_ERROR
;
70 LIBC_INLINE
int vfprintf_internal(::FILE *__restrict stream
,
71 const char *__restrict format
,
72 internal::ArgList
&args
) {
73 constexpr size_t BUFF_SIZE
= 1024;
74 char buffer
[BUFF_SIZE
];
75 printf_core::WriteBuffer
wb(buffer
, BUFF_SIZE
, &file_write_hook
,
76 reinterpret_cast<void *>(stream
));
78 internal::flockfile(stream
);
79 int retval
= printf_main(&writer
, format
, args
);
80 int flushval
= wb
.overflow_write("");
81 if (flushval
!= WRITE_OK
)
83 internal::funlockfile(stream
);
87 } // namespace printf_core
88 } // namespace LIBC_NAMESPACE_DECL
90 #endif // LLVM_LIBC_SRC_STDIO_PRINTF_CORE_VFPRINTF_INTERNAL_H