// -*- 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 _LIBCPP_CSTRING #define _LIBCPP_CSTRING /* cstring synopsis Macros: NULL namespace std { Types: size_t void* memcpy(void* restrict s1, const void* restrict s2, size_t n); void* memmove(void* s1, const void* s2, size_t n); char* strcpy (char* restrict s1, const char* restrict s2); char* strncpy(char* restrict s1, const char* restrict s2, size_t n); char* strcat (char* restrict s1, const char* restrict s2); char* strncat(char* restrict s1, const char* restrict s2, size_t n); int memcmp(const void* s1, const void* s2, size_t n); int strcmp (const char* s1, const char* s2); int strncmp(const char* s1, const char* s2, size_t n); int strcoll(const char* s1, const char* s2); size_t strxfrm(char* restrict s1, const char* restrict s2, size_t n); const void* memchr(const void* s, int c, size_t n); void* memchr( void* s, int c, size_t n); const char* strchr(const char* s, int c); char* strchr( char* s, int c); size_t strcspn(const char* s1, const char* s2); const char* strpbrk(const char* s1, const char* s2); char* strpbrk( char* s1, const char* s2); const char* strrchr(const char* s, int c); char* strrchr( char* s, int c); size_t strspn(const char* s1, const char* s2); const char* strstr(const char* s1, const char* s2); char* strstr( char* s1, const char* s2); char* strtok(char* restrict s1, const char* restrict s2); void* memset(void* s, int c, size_t n); char* strerror(int errnum); size_t strlen(const char* s); } // std */ #include <__assert> // all public C++ headers provide the assertion handler #include <__config> #include <__type_traits/is_constant_evaluated.h> #include #ifndef _LIBCPP_STRING_H # error tried including but didn't find libc++'s header. \ This usually means that your header search paths are not configured properly. \ The header search paths should contain the C++ Standard Library headers before \ any C Standard Library, and you are probably using compiler flags that make that \ not be the case. #endif #if !defined(_LIBCPP_HAS_NO_PRAGMA_SYSTEM_HEADER) # pragma GCC system_header #endif _LIBCPP_BEGIN_NAMESPACE_STD using ::size_t _LIBCPP_USING_IF_EXISTS; using ::memcpy _LIBCPP_USING_IF_EXISTS; using ::memmove _LIBCPP_USING_IF_EXISTS; using ::strcpy _LIBCPP_USING_IF_EXISTS; using ::strncpy _LIBCPP_USING_IF_EXISTS; using ::strcat _LIBCPP_USING_IF_EXISTS; using ::strncat _LIBCPP_USING_IF_EXISTS; using ::memcmp _LIBCPP_USING_IF_EXISTS; using ::strcmp _LIBCPP_USING_IF_EXISTS; using ::strncmp _LIBCPP_USING_IF_EXISTS; using ::strcoll _LIBCPP_USING_IF_EXISTS; using ::strxfrm _LIBCPP_USING_IF_EXISTS; using ::memchr _LIBCPP_USING_IF_EXISTS; using ::strchr _LIBCPP_USING_IF_EXISTS; using ::strcspn _LIBCPP_USING_IF_EXISTS; using ::strpbrk _LIBCPP_USING_IF_EXISTS; using ::strrchr _LIBCPP_USING_IF_EXISTS; using ::strspn _LIBCPP_USING_IF_EXISTS; using ::strstr _LIBCPP_USING_IF_EXISTS; using ::strtok _LIBCPP_USING_IF_EXISTS; using ::memset _LIBCPP_USING_IF_EXISTS; using ::strerror _LIBCPP_USING_IF_EXISTS; using ::strlen _LIBCPP_USING_IF_EXISTS; inline _LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR_SINCE_CXX14 size_t __constexpr_strlen(const char* __str) { // GCC currently doesn't support __builtin_strlen for heap-allocated memory during constant evaluation. // https://gcc.gnu.org/bugzilla/show_bug.cgi?id=70816 #ifdef _LIBCPP_COMPILER_GCC if (__libcpp_is_constant_evaluated()) { size_t __i = 0; for (; __str[__i] != '\0'; ++__i) ; return __i; } #endif return __builtin_strlen(__str); } template _LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR_SINCE_CXX14 int __constexpr_memcmp(const _Tp* __lhs, const _Tp* __rhs, size_t __count) { #ifdef _LIBCPP_COMPILER_GCC if (__libcpp_is_constant_evaluated()) { for (; __count; --__count, ++__lhs, ++__rhs) { if (*__lhs < *__rhs) return -1; if (*__rhs < *__lhs) return 1; } return 0; } #endif return __builtin_memcmp(__lhs, __rhs, __count); } inline _LIBCPP_HIDE_FROM_ABI _LIBCPP_CONSTEXPR_SINCE_CXX14 const char* __constexpr_char_memchr(const char* __str, int __char, size_t __count) { #if __has_builtin(__builtin_char_memchr) return __builtin_char_memchr(__str, __char, __count); #else if (!__libcpp_is_constant_evaluated()) return static_cast(std::memchr(__str, __char, __count)); for (; __count; --__count) { if (*__str == __char) return __str; ++__str; } return nullptr; #endif } _LIBCPP_END_NAMESPACE_STD #endif // _LIBCPP_CSTRING