-
Notifications
You must be signed in to change notification settings - Fork 49
/
popcnt-avx2-harley-seal.cpp
84 lines (67 loc) · 2.83 KB
/
popcnt-avx2-harley-seal.cpp
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
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
namespace AVX2_harley_seal {
__m256i popcount(const __m256i v)
{
const __m256i m1 = _mm256_set1_epi8(0x55);
const __m256i m2 = _mm256_set1_epi8(0x33);
const __m256i m4 = _mm256_set1_epi8(0x0F);
const __m256i t1 = _mm256_sub_epi8(v, (_mm256_srli_epi16(v, 1) & m1));
const __m256i t2 = _mm256_add_epi8(t1 & m2, (_mm256_srli_epi16(t1, 2) & m2));
const __m256i t3 = _mm256_add_epi8(t2, _mm256_srli_epi16(t2, 4)) & m4;
return _mm256_sad_epu8(t3, _mm256_setzero_si256());
}
void CSA(__m256i& h, __m256i& l, __m256i a, __m256i b, __m256i c)
{
const __m256i u = a ^ b;
h = (a & b) | (u & c);
l = u ^ c;
}
uint64_t popcnt(const __m256i* data, const uint64_t size)
{
__m256i total = _mm256_setzero_si256();
__m256i ones = _mm256_setzero_si256();
__m256i twos = _mm256_setzero_si256();
__m256i fours = _mm256_setzero_si256();
__m256i eights = _mm256_setzero_si256();
__m256i sixteens = _mm256_setzero_si256();
__m256i twosA, twosB, foursA, foursB, eightsA, eightsB;
const uint64_t limit = size - size % 16;
uint64_t i = 0;
for(; i < limit; i += 16)
{
CSA(twosA, ones, ones, data[i+0], data[i+1]);
CSA(twosB, ones, ones, data[i+2], data[i+3]);
CSA(foursA, twos, twos, twosA, twosB);
CSA(twosA, ones, ones, data[i+4], data[i+5]);
CSA(twosB, ones, ones, data[i+6], data[i+7]);
CSA(foursB, twos, twos, twosA, twosB);
CSA(eightsA,fours, fours, foursA, foursB);
CSA(twosA, ones, ones, data[i+8], data[i+9]);
CSA(twosB, ones, ones, data[i+10], data[i+11]);
CSA(foursA, twos, twos, twosA, twosB);
CSA(twosA, ones, ones, data[i+12], data[i+13]);
CSA(twosB, ones, ones, data[i+14], data[i+15]);
CSA(foursB, twos, twos, twosA, twosB);
CSA(eightsB, fours, fours, foursA, foursB);
CSA(sixteens, eights, eights, eightsA, eightsB);
total = _mm256_add_epi64(total, popcount(sixteens));
}
total = _mm256_slli_epi64(total, 4); // * 16
total = _mm256_add_epi64(total, _mm256_slli_epi64(popcount(eights), 3)); // += 8 * ...
total = _mm256_add_epi64(total, _mm256_slli_epi64(popcount(fours), 2)); // += 4 * ...
total = _mm256_add_epi64(total, _mm256_slli_epi64(popcount(twos), 1)); // += 2 * ...
total = _mm256_add_epi64(total, popcount(ones));
for(; i < size; i++)
total = _mm256_add_epi64(total, popcount(data[i]));
return static_cast<uint64_t>(_mm256_extract_epi64(total, 0))
+ static_cast<uint64_t>(_mm256_extract_epi64(total, 1))
+ static_cast<uint64_t>(_mm256_extract_epi64(total, 2))
+ static_cast<uint64_t>(_mm256_extract_epi64(total, 3));
}
} // AVX2_harley_seal
uint64_t popcnt_AVX2_harley_seal(const uint8_t* data, const size_t size)
{
uint64_t total = AVX2_harley_seal::popcnt((const __m256i*) data, size / 32);
for (size_t i = size - size % 32; i < size; i++)
total += lookup8bit[data[i]];
return total;
}