-
Notifications
You must be signed in to change notification settings - Fork 311
Expand file tree
/
Copy pathstr_split.h
More file actions
49 lines (41 loc) · 1.23 KB
/
str_split.h
File metadata and controls
49 lines (41 loc) · 1.23 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
#pragma once
#include <immintrin.h>
#include <string>
#include <string_view>
#include <vector>
namespace ylt {
namespace sse {
template <typename T>
concept StringLike =
std::same_as<T, std::string> || std::same_as<T, std::string_view>;
template <StringLike StringLike>
__attribute__((__target__("sse4.2"))) inline std::vector<StringLike>
simd_str_split(std::string_view string, const char delim) {
auto* pstr = string.data();
size_t size = string.size();
size_t start = 0;
std::vector<StringLike> output;
size_t aligned16_size = size & 0xFFFFFFFFFFFFFFF0UL;
for (size_t i = 0; i < aligned16_size; i += 16) {
__m128i data = _mm_lddqu_si128(reinterpret_cast<const __m128i*>(&pstr[i]));
const __m128i match = _mm_set1_epi8(delim);
uint32_t mask = _mm_movemask_epi8(_mm_cmpeq_epi8(data, match));
while (mask != 0) {
auto j = __builtin_ctzl(mask);
output.emplace_back(&pstr[start], i + j - start);
start = i + j + 1;
mask &= mask - 1;
}
}
size_t i = aligned16_size;
do {
while (pstr[i] != delim && i != size) {
++i;
}
output.emplace_back(&pstr[start], i - start);
start = i = i + 1;
} while (i <= size);
return output;
}
} // namespace sse
} // namespace ylt