mirror of
https://github.com/ClickHouse/ClickHouse.git
synced 2024-11-25 00:52:02 +00:00
commit
03deb642d5
@ -1,10 +1,8 @@
|
|||||||
#include <Columns/ColumnString.h>
|
#include <Columns/ColumnString.h>
|
||||||
#include <Functions/FunctionFactory.h>
|
#include <Functions/FunctionFactory.h>
|
||||||
#include <Functions/FunctionStringToString.h>
|
#include <Functions/FunctionStringToString.h>
|
||||||
|
#include <common/find_symbols.h>
|
||||||
|
|
||||||
#ifdef __SSE4_2__
|
|
||||||
#include <nmmintrin.h>
|
|
||||||
#endif
|
|
||||||
|
|
||||||
namespace DB
|
namespace DB
|
||||||
{
|
{
|
||||||
@ -60,7 +58,7 @@ public:
|
|||||||
execute(reinterpret_cast<const UInt8 *>(&data[prev_offset]), offsets[i] - prev_offset - 1, start, length);
|
execute(reinterpret_cast<const UInt8 *>(&data[prev_offset]), offsets[i] - prev_offset - 1, start, length);
|
||||||
|
|
||||||
res_data.resize(res_data.size() + length + 1);
|
res_data.resize(res_data.size() + length + 1);
|
||||||
memcpy(&res_data[res_offset], start, length);
|
memcpySmallAllowReadWriteOverflow15(&res_data[res_offset], start, length);
|
||||||
res_offset += length + 1;
|
res_offset += length + 1;
|
||||||
res_data[res_offset - 1] = '\0';
|
res_data[res_offset - 1] = '\0';
|
||||||
|
|
||||||
@ -77,59 +75,27 @@ public:
|
|||||||
private:
|
private:
|
||||||
static void execute(const UInt8 * data, size_t size, const UInt8 *& res_data, size_t & res_size)
|
static void execute(const UInt8 * data, size_t size, const UInt8 *& res_data, size_t & res_size)
|
||||||
{
|
{
|
||||||
size_t chars_to_trim_left = 0;
|
const char * char_data = reinterpret_cast<const char *>(data);
|
||||||
size_t chars_to_trim_right = 0;
|
const char * char_end = char_data + size;
|
||||||
char whitespace = ' ';
|
|
||||||
#ifdef __SSE4_2__
|
|
||||||
const auto bytes_sse = sizeof(__m128i);
|
|
||||||
const auto size_sse = size - (size % bytes_sse);
|
|
||||||
const auto whitespace_mask = _mm_set1_epi8(whitespace);
|
|
||||||
constexpr auto base_sse_mode = _SIDD_UBYTE_OPS | _SIDD_CMP_EQUAL_EACH | _SIDD_NEGATIVE_POLARITY;
|
|
||||||
auto mask = bytes_sse;
|
|
||||||
#endif
|
|
||||||
|
|
||||||
if constexpr (mode::trim_left)
|
if constexpr (mode::trim_left)
|
||||||
{
|
{
|
||||||
#ifdef __SSE4_2__
|
const char * found = find_first_not_symbols<' '>(char_data, char_end);
|
||||||
/// skip whitespace from left in blocks of up to 16 characters
|
size_t num_chars = found - char_data;
|
||||||
|
char_data += num_chars;
|
||||||
/// Avoid gcc bug: _mm_cmpistri: error: the third argument must be an 8-bit immediate
|
|
||||||
enum { left_sse_mode = base_sse_mode | _SIDD_LEAST_SIGNIFICANT };
|
|
||||||
while (mask == bytes_sse && chars_to_trim_left < size_sse)
|
|
||||||
{
|
|
||||||
const auto chars = _mm_loadu_si128(reinterpret_cast<const __m128i *>(data + chars_to_trim_left));
|
|
||||||
mask = _mm_cmpistri(whitespace_mask, chars, left_sse_mode);
|
|
||||||
chars_to_trim_left += mask;
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
/// skip remaining whitespace from left, character by character
|
|
||||||
while (chars_to_trim_left < size && data[chars_to_trim_left] == whitespace)
|
|
||||||
++chars_to_trim_left;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if constexpr (mode::trim_right)
|
if constexpr (mode::trim_right)
|
||||||
{
|
{
|
||||||
const auto trim_right_size = size - chars_to_trim_left;
|
const char * found = find_last_not_symbols_or_null<' '>(char_data, char_end);
|
||||||
#ifdef __SSE4_2__
|
if (found)
|
||||||
/// try to skip whitespace from right in blocks of up to 16 characters
|
char_end = found + 1;
|
||||||
|
else
|
||||||
/// Avoid gcc bug: _mm_cmpistri: error: the third argument must be an 8-bit immediate
|
char_end = char_data;
|
||||||
enum { right_sse_mode = base_sse_mode | _SIDD_MOST_SIGNIFICANT };
|
|
||||||
const auto trim_right_size_sse = trim_right_size - (trim_right_size % bytes_sse);
|
|
||||||
while (mask == bytes_sse && chars_to_trim_right < trim_right_size_sse)
|
|
||||||
{
|
|
||||||
const auto chars = _mm_loadu_si128(reinterpret_cast<const __m128i *>(data + size - chars_to_trim_right - bytes_sse));
|
|
||||||
mask = _mm_cmpistri(whitespace_mask, chars, right_sse_mode);
|
|
||||||
chars_to_trim_right += mask;
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
/// skip remaining whitespace from right, character by character
|
|
||||||
while (chars_to_trim_right < trim_right_size && data[size - chars_to_trim_right - 1] == whitespace)
|
|
||||||
++chars_to_trim_right;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
res_data = data + chars_to_trim_left;
|
res_data = reinterpret_cast<const UInt8 *>(char_data);
|
||||||
res_size = size - chars_to_trim_left - chars_to_trim_right;
|
res_size = char_end - char_data;
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
0
dbms/tests/queries/0_stateless/00997_trim.reference
Normal file
0
dbms/tests/queries/0_stateless/00997_trim.reference
Normal file
20
dbms/tests/queries/0_stateless/00997_trim.sql
Normal file
20
dbms/tests/queries/0_stateless/00997_trim.sql
Normal file
@ -0,0 +1,20 @@
|
|||||||
|
WITH
|
||||||
|
'0123456789abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ' AS x,
|
||||||
|
replaceRegexpAll(x, '.', ' ') AS spaces,
|
||||||
|
concat(substring(spaces, 1, rand(1) % 62), substring(x, 1, rand(2) % 62), substring(spaces, 1, rand(3) % 62)) AS s,
|
||||||
|
trimLeft(s) AS sl,
|
||||||
|
trimRight(s) AS sr,
|
||||||
|
trimBoth(s) AS t,
|
||||||
|
replaceRegexpOne(s, '^ +', '') AS slr,
|
||||||
|
replaceRegexpOne(s, ' +$', '') AS srr,
|
||||||
|
replaceRegexpOne(s, '^ *(.*?) *$', '\\1') AS tr
|
||||||
|
SELECT
|
||||||
|
replaceAll(s, ' ', '_'),
|
||||||
|
replaceAll(sl, ' ', '_'),
|
||||||
|
replaceAll(slr, ' ', '_'),
|
||||||
|
replaceAll(sr, ' ', '_'),
|
||||||
|
replaceAll(srr, ' ', '_'),
|
||||||
|
replaceAll(t, ' ', '_'),
|
||||||
|
replaceAll(tr, ' ', '_')
|
||||||
|
FROM numbers(100000)
|
||||||
|
WHERE NOT ((sl = slr) AND (sr = srr) AND (t = tr))
|
@ -17,7 +17,7 @@
|
|||||||
* but with the following differencies:
|
* but with the following differencies:
|
||||||
* - works with any memory ranges, including containing zero bytes;
|
* - works with any memory ranges, including containing zero bytes;
|
||||||
* - doesn't require terminating zero byte: end of memory range is passed explicitly;
|
* - doesn't require terminating zero byte: end of memory range is passed explicitly;
|
||||||
* - if not found, returns pointer to end instead of NULL;
|
* - if not found, returns pointer to end instead of nullptr;
|
||||||
* - maximum number of symbols to search is 16.
|
* - maximum number of symbols to search is 16.
|
||||||
*
|
*
|
||||||
* Uses SSE 2 in case of small number of symbols for search and SSE 4.2 in the case of large number of symbols,
|
* Uses SSE 2 in case of small number of symbols for search and SSE 4.2 in the case of large number of symbols,
|
||||||
@ -65,115 +65,154 @@ inline __m128i mm_is_in(__m128i bytes)
|
|||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
template <bool positive>
|
||||||
template <char... symbols>
|
bool maybe_negate(bool x)
|
||||||
inline const char * find_first_symbols_sse2(const char * begin, const char * end)
|
|
||||||
{
|
{
|
||||||
|
if constexpr (positive)
|
||||||
|
return x;
|
||||||
|
else
|
||||||
|
return !x;
|
||||||
|
}
|
||||||
|
|
||||||
|
template <bool positive>
|
||||||
|
uint16_t maybe_negate(uint16_t x)
|
||||||
|
{
|
||||||
|
if constexpr (positive)
|
||||||
|
return x;
|
||||||
|
else
|
||||||
|
return ~x;
|
||||||
|
}
|
||||||
|
|
||||||
|
enum class ReturnMode
|
||||||
|
{
|
||||||
|
End,
|
||||||
|
Nullptr,
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
template <bool positive, ReturnMode return_mode, char... symbols>
|
||||||
|
inline const char * find_first_symbols_sse2(const char * const begin, const char * const end)
|
||||||
|
{
|
||||||
|
const char * pos = begin;
|
||||||
|
|
||||||
#if defined(__SSE2__)
|
#if defined(__SSE2__)
|
||||||
for (; begin + 15 < end; begin += 16)
|
for (; pos + 15 < end; pos += 16)
|
||||||
{
|
{
|
||||||
__m128i bytes = _mm_loadu_si128(reinterpret_cast<const __m128i *>(begin));
|
__m128i bytes = _mm_loadu_si128(reinterpret_cast<const __m128i *>(pos));
|
||||||
|
|
||||||
__m128i eq = mm_is_in<symbols...>(bytes);
|
__m128i eq = mm_is_in<symbols...>(bytes);
|
||||||
|
|
||||||
uint16_t bit_mask = _mm_movemask_epi8(eq);
|
uint16_t bit_mask = maybe_negate<positive>(uint16_t(_mm_movemask_epi8(eq)));
|
||||||
if (bit_mask)
|
if (bit_mask)
|
||||||
return begin + __builtin_ctz(bit_mask);
|
return pos + __builtin_ctz(bit_mask);
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
for (; begin < end; ++begin)
|
for (; pos < end; ++pos)
|
||||||
if (is_in<symbols...>(*begin))
|
if (maybe_negate<positive>(is_in<symbols...>(*pos)))
|
||||||
return begin;
|
return pos;
|
||||||
return end;
|
|
||||||
|
return return_mode == ReturnMode::End ? end : nullptr;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
template <char... symbols>
|
template <bool positive, ReturnMode return_mode, char... symbols>
|
||||||
inline const char * find_last_symbols_or_null_sse2(const char * begin, const char * end)
|
inline const char * find_last_symbols_sse2(const char * const begin, const char * const end)
|
||||||
{
|
{
|
||||||
|
const char * pos = end;
|
||||||
|
|
||||||
#if defined(__SSE2__)
|
#if defined(__SSE2__)
|
||||||
for (; end - 16 >= begin; end -= 16) /// Assuming the pointer cannot overflow. Assuming we can compare these pointers.
|
for (; pos - 16 >= begin; pos -= 16) /// Assuming the pointer cannot overflow. Assuming we can compare these pointers.
|
||||||
{
|
{
|
||||||
__m128i bytes = _mm_loadu_si128(reinterpret_cast<const __m128i *>(end - 16));
|
__m128i bytes = _mm_loadu_si128(reinterpret_cast<const __m128i *>(pos - 16));
|
||||||
|
|
||||||
__m128i eq = mm_is_in<symbols...>(bytes);
|
__m128i eq = mm_is_in<symbols...>(bytes);
|
||||||
|
|
||||||
uint16_t bit_mask = _mm_movemask_epi8(eq);
|
uint16_t bit_mask = maybe_negate<positive>(uint16_t(_mm_movemask_epi8(eq)));
|
||||||
if (bit_mask)
|
if (bit_mask)
|
||||||
return end - 1 - (__builtin_clz(bit_mask) - 16); /// because __builtin_clz works with mask as uint32.
|
return pos - 1 - (__builtin_clz(bit_mask) - 16); /// because __builtin_clz works with mask as uint32.
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
--end;
|
--pos;
|
||||||
for (; end >= begin; --end)
|
for (; pos >= begin; --pos)
|
||||||
if (is_in<symbols...>(*end))
|
if (maybe_negate<positive>(is_in<symbols...>(*pos)))
|
||||||
return end;
|
return pos;
|
||||||
|
|
||||||
return nullptr;
|
return return_mode == ReturnMode::End ? end : nullptr;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
template <size_t num_chars,
|
template <bool positive, ReturnMode return_mode, size_t num_chars,
|
||||||
char c01, char c02 = 0, char c03 = 0, char c04 = 0,
|
char c01, char c02 = 0, char c03 = 0, char c04 = 0,
|
||||||
char c05 = 0, char c06 = 0, char c07 = 0, char c08 = 0,
|
char c05 = 0, char c06 = 0, char c07 = 0, char c08 = 0,
|
||||||
char c09 = 0, char c10 = 0, char c11 = 0, char c12 = 0,
|
char c09 = 0, char c10 = 0, char c11 = 0, char c12 = 0,
|
||||||
char c13 = 0, char c14 = 0, char c15 = 0, char c16 = 0>
|
char c13 = 0, char c14 = 0, char c15 = 0, char c16 = 0>
|
||||||
inline const char * find_first_symbols_sse42_impl(const char * begin, const char * end)
|
inline const char * find_first_symbols_sse42_impl(const char * const begin, const char * const end)
|
||||||
{
|
{
|
||||||
|
const char * pos = begin;
|
||||||
|
|
||||||
#if defined(__SSE4_2__)
|
#if defined(__SSE4_2__)
|
||||||
#define MODE (_SIDD_UBYTE_OPS | _SIDD_CMP_EQUAL_ANY | _SIDD_LEAST_SIGNIFICANT)
|
#define MODE (_SIDD_UBYTE_OPS | _SIDD_CMP_EQUAL_ANY | _SIDD_LEAST_SIGNIFICANT)
|
||||||
__m128i set = _mm_setr_epi8(c01, c02, c03, c04, c05, c06, c07, c08, c09, c10, c11, c12, c13, c14, c15, c16);
|
__m128i set = _mm_setr_epi8(c01, c02, c03, c04, c05, c06, c07, c08, c09, c10, c11, c12, c13, c14, c15, c16);
|
||||||
|
|
||||||
for (; begin + 15 < end; begin += 16)
|
for (; pos + 15 < end; pos += 16)
|
||||||
{
|
{
|
||||||
__m128i bytes = _mm_loadu_si128(reinterpret_cast<const __m128i *>(begin));
|
__m128i bytes = _mm_loadu_si128(reinterpret_cast<const __m128i *>(pos));
|
||||||
|
|
||||||
|
if constexpr (positive)
|
||||||
|
{
|
||||||
if (_mm_cmpestrc(set, num_chars, bytes, 16, MODE))
|
if (_mm_cmpestrc(set, num_chars, bytes, 16, MODE))
|
||||||
return begin + _mm_cmpestri(set, num_chars, bytes, 16, MODE);
|
return pos + _mm_cmpestri(set, num_chars, bytes, 16, MODE);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
if (_mm_cmpestrc(set, num_chars, bytes, 16, MODE | _SIDD_NEGATIVE_POLARITY))
|
||||||
|
return pos + _mm_cmpestri(set, num_chars, bytes, 16, MODE | _SIDD_NEGATIVE_POLARITY);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
#undef MODE
|
#undef MODE
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
for (; begin < end; ++begin)
|
for (; pos < end; ++pos)
|
||||||
if ( (num_chars >= 1 && *begin == c01)
|
if ( (num_chars >= 1 && maybe_negate<positive>(*pos == c01))
|
||||||
|| (num_chars >= 2 && *begin == c02)
|
|| (num_chars >= 2 && maybe_negate<positive>(*pos == c02))
|
||||||
|| (num_chars >= 3 && *begin == c03)
|
|| (num_chars >= 3 && maybe_negate<positive>(*pos == c03))
|
||||||
|| (num_chars >= 4 && *begin == c04)
|
|| (num_chars >= 4 && maybe_negate<positive>(*pos == c04))
|
||||||
|| (num_chars >= 5 && *begin == c05)
|
|| (num_chars >= 5 && maybe_negate<positive>(*pos == c05))
|
||||||
|| (num_chars >= 6 && *begin == c06)
|
|| (num_chars >= 6 && maybe_negate<positive>(*pos == c06))
|
||||||
|| (num_chars >= 7 && *begin == c07)
|
|| (num_chars >= 7 && maybe_negate<positive>(*pos == c07))
|
||||||
|| (num_chars >= 8 && *begin == c08)
|
|| (num_chars >= 8 && maybe_negate<positive>(*pos == c08))
|
||||||
|| (num_chars >= 9 && *begin == c09)
|
|| (num_chars >= 9 && maybe_negate<positive>(*pos == c09))
|
||||||
|| (num_chars >= 10 && *begin == c10)
|
|| (num_chars >= 10 && maybe_negate<positive>(*pos == c10))
|
||||||
|| (num_chars >= 11 && *begin == c11)
|
|| (num_chars >= 11 && maybe_negate<positive>(*pos == c11))
|
||||||
|| (num_chars >= 12 && *begin == c12)
|
|| (num_chars >= 12 && maybe_negate<positive>(*pos == c12))
|
||||||
|| (num_chars >= 13 && *begin == c13)
|
|| (num_chars >= 13 && maybe_negate<positive>(*pos == c13))
|
||||||
|| (num_chars >= 14 && *begin == c14)
|
|| (num_chars >= 14 && maybe_negate<positive>(*pos == c14))
|
||||||
|| (num_chars >= 15 && *begin == c15)
|
|| (num_chars >= 15 && maybe_negate<positive>(*pos == c15))
|
||||||
|| (num_chars >= 16 && *begin == c16))
|
|| (num_chars >= 16 && maybe_negate<positive>(*pos == c16)))
|
||||||
return begin;
|
return pos;
|
||||||
return end;
|
return return_mode == ReturnMode::End ? end : nullptr;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
template <char... symbols>
|
template <bool positive, ReturnMode return_mode, char... symbols>
|
||||||
inline const char * find_first_symbols_sse42(const char * begin, const char * end)
|
inline const char * find_first_symbols_sse42(const char * begin, const char * end)
|
||||||
{
|
{
|
||||||
return find_first_symbols_sse42_impl<sizeof...(symbols), symbols...>(begin, end);
|
return find_first_symbols_sse42_impl<positive, return_mode, sizeof...(symbols), symbols...>(begin, end);
|
||||||
}
|
}
|
||||||
|
|
||||||
/// NOTE No SSE 4.2 implementation for find_last_symbols_or_null. Not worth to do.
|
/// NOTE No SSE 4.2 implementation for find_last_symbols_or_null. Not worth to do.
|
||||||
|
|
||||||
template <char... symbols>
|
template <bool positive, ReturnMode return_mode, char... symbols>
|
||||||
inline const char * find_first_symbols_dispatch(const char * begin, const char * end)
|
inline const char * find_first_symbols_dispatch(const char * begin, const char * end)
|
||||||
{
|
{
|
||||||
#if defined(__SSE4_2__)
|
#if defined(__SSE4_2__)
|
||||||
if (sizeof...(symbols) >= 5)
|
if (sizeof...(symbols) >= 5)
|
||||||
return find_first_symbols_sse42<symbols...>(begin, end);
|
return find_first_symbols_sse42<positive, return_mode, symbols...>(begin, end);
|
||||||
else
|
else
|
||||||
#endif
|
#endif
|
||||||
return find_first_symbols_sse2<symbols...>(begin, end);
|
return find_first_symbols_sse2<positive, return_mode, symbols...>(begin, end);
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
@ -182,7 +221,7 @@ inline const char * find_first_symbols_dispatch(const char * begin, const char *
|
|||||||
template <char... symbols>
|
template <char... symbols>
|
||||||
inline const char * find_first_symbols(const char * begin, const char * end)
|
inline const char * find_first_symbols(const char * begin, const char * end)
|
||||||
{
|
{
|
||||||
return detail::find_first_symbols_dispatch<symbols...>(begin, end);
|
return detail::find_first_symbols_dispatch<true, detail::ReturnMode::End, symbols...>(begin, end);
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Returning non const result for non const arguments.
|
/// Returning non const result for non const arguments.
|
||||||
@ -190,18 +229,66 @@ inline const char * find_first_symbols(const char * begin, const char * end)
|
|||||||
template <char... symbols>
|
template <char... symbols>
|
||||||
inline char * find_first_symbols(char * begin, char * end)
|
inline char * find_first_symbols(char * begin, char * end)
|
||||||
{
|
{
|
||||||
return const_cast<char *>(detail::find_first_symbols_dispatch<symbols...>(begin, end));
|
return const_cast<char *>(detail::find_first_symbols_dispatch<true, detail::ReturnMode::End, symbols...>(begin, end));
|
||||||
|
}
|
||||||
|
|
||||||
|
template <char... symbols>
|
||||||
|
inline const char * find_first_not_symbols(const char * begin, const char * end)
|
||||||
|
{
|
||||||
|
return detail::find_first_symbols_dispatch<false, detail::ReturnMode::End, symbols...>(begin, end);
|
||||||
|
}
|
||||||
|
|
||||||
|
template <char... symbols>
|
||||||
|
inline char * find_first_not_symbols(char * begin, char * end)
|
||||||
|
{
|
||||||
|
return const_cast<char *>(detail::find_first_symbols_dispatch<false, detail::ReturnMode::End, symbols...>(begin, end));
|
||||||
|
}
|
||||||
|
|
||||||
|
template <char... symbols>
|
||||||
|
inline const char * find_first_symbols_or_null(const char * begin, const char * end)
|
||||||
|
{
|
||||||
|
return detail::find_first_symbols_dispatch<true, detail::ReturnMode::Nullptr, symbols...>(begin, end);
|
||||||
|
}
|
||||||
|
|
||||||
|
template <char... symbols>
|
||||||
|
inline char * find_first_symbols_or_null(char * begin, char * end)
|
||||||
|
{
|
||||||
|
return const_cast<char *>(detail::find_first_symbols_dispatch<true, detail::ReturnMode::Nullptr, symbols...>(begin, end));
|
||||||
|
}
|
||||||
|
|
||||||
|
template <char... symbols>
|
||||||
|
inline const char * find_first_not_symbols_or_null(const char * begin, const char * end)
|
||||||
|
{
|
||||||
|
return detail::find_first_symbols_dispatch<false, detail::ReturnMode::Nullptr, symbols...>(begin, end);
|
||||||
|
}
|
||||||
|
|
||||||
|
template <char... symbols>
|
||||||
|
inline char * find_first_not_symbols_or_null(char * begin, char * end)
|
||||||
|
{
|
||||||
|
return const_cast<char *>(detail::find_first_symbols_dispatch<false, detail::ReturnMode::Nullptr, symbols...>(begin, end));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
template <char... symbols>
|
template <char... symbols>
|
||||||
inline const char * find_last_symbols_or_null(const char * begin, const char * end)
|
inline const char * find_last_symbols_or_null(const char * begin, const char * end)
|
||||||
{
|
{
|
||||||
return detail::find_last_symbols_or_null_sse2<symbols...>(begin, end);
|
return detail::find_last_symbols_sse2<true, detail::ReturnMode::Nullptr, symbols...>(begin, end);
|
||||||
}
|
}
|
||||||
|
|
||||||
template <char... symbols>
|
template <char... symbols>
|
||||||
inline char * find_last_symbols_or_null(char * begin, char * end)
|
inline char * find_last_symbols_or_null(char * begin, char * end)
|
||||||
{
|
{
|
||||||
return const_cast<char *>(detail::find_last_symbols_or_null_sse2<symbols...>(begin, end));
|
return const_cast<char *>(detail::find_last_symbols_sse2<true, detail::ReturnMode::Nullptr, symbols...>(begin, end));
|
||||||
|
}
|
||||||
|
|
||||||
|
template <char... symbols>
|
||||||
|
inline const char * find_last_not_symbols_or_null(const char * begin, const char * end)
|
||||||
|
{
|
||||||
|
return detail::find_last_symbols_sse2<false, detail::ReturnMode::Nullptr, symbols...>(begin, end);
|
||||||
|
}
|
||||||
|
|
||||||
|
template <char... symbols>
|
||||||
|
inline char * find_last_not_symbols_or_null(char * begin, char * end)
|
||||||
|
{
|
||||||
|
return const_cast<char *>(detail::find_last_symbols_sse2<false, detail::ReturnMode::Nullptr, symbols...>(begin, end));
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user