#ifndef LLVM_LIBC_SRC_STRING_STRING_UTILS_H
#define LLVM_LIBC_SRC_STRING_STRING_UTILS_H
#include "src/__support/CPP/bitset.h"
#include "src/__support/macros/config.h"
#include "src/__support/macros/optimization.h"
#include "src/string/memory_utils/inline_bzero.h"
#include "src/string/memory_utils/inline_memcpy.h"
#include <stddef.h>
namespace LIBC_NAMESPACE_DECL {
namespace internal {
template <typename Word> LIBC_INLINE constexpr Word repeat_byte(Word byte) {
constexpr size_t BITS_IN_BYTE = 8;
constexpr size_t BYTE_MASK = 0xff;
Word result = 0;
byte = byte & BYTE_MASK;
for (size_t i = 0; i < sizeof(Word); ++i)
result = (result << BITS_IN_BYTE) | byte;
return result;
}
template <typename Word> LIBC_INLINE constexpr bool has_zeroes(Word block) {
constexpr Word LOW_BITS = repeat_byte<Word>(0x01);
constexpr Word HIGH_BITS = repeat_byte<Word>(0x80);
Word subtracted = block - LOW_BITS;
Word inverted = ~block;
return (subtracted & inverted & HIGH_BITS) != 0;
}
template <typename Word>
LIBC_INLINE size_t string_length_wide_read(const char *src) {
const char *char_ptr = src;
for (; reinterpret_cast<uintptr_t>(char_ptr) % sizeof(Word) != 0;
++char_ptr) {
if (*char_ptr == '\0')
return char_ptr - src;
}
for (const Word *block_ptr = reinterpret_cast<const Word *>(char_ptr);
!has_zeroes<Word>(*block_ptr); ++block_ptr) {
char_ptr = reinterpret_cast<const char *>(block_ptr);
}
for (; *char_ptr != '\0'; ++char_ptr) {
;
}
return char_ptr - src;
}
LIBC_INLINE size_t string_length_byte_read(const char *src) {
size_t length;
for (length = 0; *src; ++src, ++length)
;
return length;
}
LIBC_INLINE size_t string_length(const char *src) {
#ifdef LIBC_COPT_STRING_UNSAFE_WIDE_READ
return string_length_wide_read<unsigned int>(src);
#else
return string_length_byte_read(src);
#endif
}
template <typename Word>
LIBC_INLINE void *find_first_character_wide_read(const unsigned char *src,
unsigned char ch, size_t n) {
const unsigned char *char_ptr = src;
size_t cur = 0;
for (; reinterpret_cast<uintptr_t>(char_ptr) % sizeof(Word) != 0 && cur < n;
++char_ptr, ++cur) {
if (*char_ptr == ch)
return const_cast<unsigned char *>(char_ptr);
}
const Word ch_mask = repeat_byte<Word>(ch);
for (const Word *block_ptr = reinterpret_cast<const Word *>(char_ptr);
!has_zeroes<Word>((*block_ptr) ^ ch_mask) && cur < n;
++block_ptr, cur += sizeof(Word)) {
char_ptr = reinterpret_cast<const unsigned char *>(block_ptr);
}
for (; *char_ptr != ch && cur < n; ++char_ptr, ++cur) {
;
}
if (*char_ptr != ch || cur >= n)
return static_cast<void *>(nullptr);
return const_cast<unsigned char *>(char_ptr);
}
LIBC_INLINE void *find_first_character_byte_read(const unsigned char *src,
unsigned char ch, size_t n) {
for (; n && *src != ch; --n, ++src)
;
return n ? const_cast<unsigned char *>(src) : nullptr;
}
LIBC_INLINE void *find_first_character(const unsigned char *src,
unsigned char ch, size_t max_strlen) {
#ifdef LIBC_COPT_STRING_UNSAFE_WIDE_READ
using BlockType = unsigned int;
if (max_strlen > (sizeof(BlockType) * 4)) {
return find_first_character_wide_read<BlockType>(src, ch, max_strlen);
}
#endif
return find_first_character_byte_read(src, ch, max_strlen);
}
LIBC_INLINE size_t complementary_span(const char *src, const char *segment) {
const char *initial = src;
cpp::bitset<256> bitset;
for (; *segment; ++segment)
bitset.set(*reinterpret_cast<const unsigned char *>(segment));
for (; *src && !bitset.test(*reinterpret_cast<const unsigned char *>(src));
++src)
;
return src - initial;
}
template <bool SkipDelim = true>
LIBC_INLINE char *string_token(char *__restrict src,
const char *__restrict delimiter_string,
char **__restrict saveptr) {
if (LIBC_UNLIKELY(src == nullptr && ((src = *saveptr) == nullptr)))
return nullptr;
cpp::bitset<256> delimiter_set;
for (; *delimiter_string != '\0'; ++delimiter_string)
delimiter_set.set(*delimiter_string);
if constexpr (SkipDelim)
for (; *src != '\0' && delimiter_set.test(*src); ++src)
;
if (*src == '\0') {
*saveptr = src;
return nullptr;
}
char *token = src;
for (; *src != '\0'; ++src) {
if (delimiter_set.test(*src)) {
*src = '\0';
++src;
break;
}
}
*saveptr = src;
return token;
}
LIBC_INLINE size_t strlcpy(char *__restrict dst, const char *__restrict src,
size_t size) {
size_t len = internal::string_length(src);
if (!size)
return len;
size_t n = len < size - 1 ? len : size - 1;
inline_memcpy(dst, src, n);
inline_bzero(dst + n, size - n);
return len;
}
template <bool ReturnNull = true>
LIBC_INLINE constexpr static char *strchr_implementation(const char *src,
int c) {
char ch = static_cast<char>(c);
for (; *src && *src != ch; ++src)
;
char *ret = ReturnNull ? nullptr : const_cast<char *>(src);
return *src == ch ? const_cast<char *>(src) : ret;
}
LIBC_INLINE constexpr static char *strrchr_implementation(const char *src,
int c) {
char ch = static_cast<char>(c);
char *last_occurrence = nullptr;
for (; *src; ++src) {
if (*src == ch)
last_occurrence = const_cast<char *>(src);
}
return last_occurrence;
}
}
}
#endif