//===-- Implementation header for libc_errno --------------------*- C++ -*-===// // // 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 // //===----------------------------------------------------------------------===// #ifndef LLVM_LIBC_SRC___SUPPORT_LIBC_ERRNO_H #define LLVM_LIBC_SRC___SUPPORT_LIBC_ERRNO_H // This header is to be consumed by internal implementations, in which all of // them should refer to `libc_errno` instead of using `errno` directly from // header. // Unit and hermetic tests should: // - #include "src/__support/libc_errno.h" // - NOT #include // - Only use `libc_errno` in the code // - Depend on libc.src.errno.errno // Integration tests should: // - NOT #include "src/__support/libc_errno.h" // - #include // - Use regular `errno` in the code // - Still depend on libc.src.errno.errno // libc uses a fallback default value, either system or thread local. #define LIBC_ERRNO_MODE_DEFAULT 0 // libc never stores a value; `errno` macro uses get link-time failure. #define LIBC_ERRNO_MODE_UNDEFINED 1 // libc maintains per-thread state (requires C++ `thread_local` support). #define LIBC_ERRNO_MODE_THREAD_LOCAL 2 // libc maintains shared state used by all threads, contrary to standard C // semantics unless always single-threaded; nothing prevents data races. #define LIBC_ERRNO_MODE_SHARED 3 // libc doesn't maintain any internal state, instead the embedder must define // `int *__llvm_libc_errno(void);` C function. #define LIBC_ERRNO_MODE_EXTERNAL 4 // DEPRECATED: #define LIBC_ERRNO_MODE_SYSTEM 5 // In this mode, the libc_errno is simply a macro resolved to `errno` from the // system header . There is no need to link against the // `libc.src.errno.errno` object, and public C++ symbol // `LIBC_NAMESPACE::libc_errno` doesn't exist. #define LIBC_ERRNO_MODE_SYSTEM_INLINE 6 #if !defined(LIBC_ERRNO_MODE) || LIBC_ERRNO_MODE == LIBC_ERRNO_MODE_DEFAULT #undef LIBC_ERRNO_MODE #if defined(LIBC_FULL_BUILD) || !defined(LIBC_COPT_PUBLIC_PACKAGING) #define LIBC_ERRNO_MODE LIBC_ERRNO_MODE_THREAD_LOCAL #else #define LIBC_ERRNO_MODE LIBC_ERRNO_MODE_SYSTEM_INLINE #endif #endif // LIBC_ERRNO_MODE #if LIBC_ERRNO_MODE != LIBC_ERRNO_MODE_DEFAULT && \ LIBC_ERRNO_MODE != LIBC_ERRNO_MODE_UNDEFINED && \ LIBC_ERRNO_MODE != LIBC_ERRNO_MODE_THREAD_LOCAL && \ LIBC_ERRNO_MODE != LIBC_ERRNO_MODE_SHARED && \ LIBC_ERRNO_MODE != LIBC_ERRNO_MODE_EXTERNAL && \ LIBC_ERRNO_MODE != LIBC_ERRNO_MODE_SYSTEM_INLINE #error LIBC_ERRNO_MODE must be one of the following values: \ LIBC_ERRNO_MODE_DEFAULT, \ LIBC_ERRNO_MODE_UNDEFINED, \ LIBC_ERRNO_MODE_THREAD_LOCAL, \ LIBC_ERRNO_MODE_SHARED, \ LIBC_ERRNO_MODE_EXTERNAL, \ LIBC_ERRNO_MODE_SYSTEM_INLINE. #endif #if LIBC_ERRNO_MODE == LIBC_ERRNO_MODE_SYSTEM_INLINE #include #define libc_errno errno #else // !LIBC_ERRNO_MODE_SYSTEM_INLINE #include "hdr/errno_macros.h" #include "src/__support/macros/config.h" namespace LIBC_NAMESPACE_DECL { extern "C" int *__llvm_libc_errno() noexcept; struct Errno { void operator=(int); operator int(); }; extern Errno libc_errno; } // namespace LIBC_NAMESPACE_DECL using LIBC_NAMESPACE::libc_errno; #endif // LIBC_ERRNO_MODE_SYSTEM_INLINE #endif // LLVM_LIBC_SRC___SUPPORT_LIBC_ERRNO_H