|
7 | 7 | namespace cp_algo {
|
8 | 8 | template<typename T, size_t len>
|
9 | 9 | using simd [[gnu::vector_size(len * sizeof(T))]] = T;
|
10 |
| - using u32x8 = simd<uint32_t, 8>; |
11 | 10 | using i64x4 = simd<int64_t, 4>;
|
12 | 11 | using u64x4 = simd<uint64_t, 4>;
|
| 12 | + using u32x8 = simd<uint32_t, 8>; |
13 | 13 | using i32x4 = simd<int32_t, 4>;
|
14 | 14 | using u32x4 = simd<uint32_t, 4>;
|
15 | 15 | using i16x4 = simd<int16_t, 4>;
|
| 16 | + using u8x32 = simd<uint8_t, 32>; |
16 | 17 | using dx4 = simd<double, 4>;
|
17 | 18 |
|
18 | 19 | [[gnu::always_inline]] inline dx4 abs(dx4 a) {
|
@@ -44,23 +45,14 @@ namespace cp_algo {
|
44 | 45 | [[gnu::always_inline]] inline auto swap_bytes(auto x) {
|
45 | 46 | return decltype(x)(__builtin_shufflevector(u32x8(x), u32x8(x), 1, 0, 3, 2, 5, 4, 7, 6));
|
46 | 47 | }
|
47 |
| - [[gnu::always_inline]] inline u64x4 montgomery_reduce(u64x4 x, uint32_t mod, uint32_t imod) { |
48 |
| -#ifdef __AVX2__ |
| 48 | + [[gnu::target("avx2"), gnu::always_inline]] inline u64x4 montgomery_reduce(u64x4 x, uint32_t mod, uint32_t imod) { |
49 | 49 | auto x_ninv = u64x4(_mm256_mul_epu32(__m256i(x), __m256i() + imod));
|
50 | 50 | x += u64x4(_mm256_mul_epu32(__m256i(x_ninv), __m256i() + mod));
|
51 |
| -#else |
52 |
| - auto x_ninv = x * imod; |
53 |
| - x += low32(x_ninv) * mod; |
54 |
| -#endif |
55 | 51 | return swap_bytes(x);
|
56 | 52 | }
|
57 | 53 |
|
58 |
| - [[gnu::always_inline]] inline u64x4 montgomery_mul(u64x4 x, u64x4 y, uint32_t mod, uint32_t imod) { |
59 |
| -#ifdef __AVX2__ |
| 54 | + [[gnu::target("avx2"), gnu::always_inline]] inline u64x4 montgomery_mul(u64x4 x, u64x4 y, uint32_t mod, uint32_t imod) { |
60 | 55 | return montgomery_reduce(u64x4(_mm256_mul_epu32(__m256i(x), __m256i(y))), mod, imod);
|
61 |
| -#else |
62 |
| - return montgomery_reduce(low32(x) * low32(y), mod, imod); |
63 |
| -#endif |
64 | 56 | }
|
65 | 57 | [[gnu::always_inline]] inline u32x8 montgomery_mul(u32x8 x, u32x8 y, uint32_t mod, uint32_t imod) {
|
66 | 58 | return u32x8(montgomery_mul(u64x4(x), u64x4(y), mod, imod)) |
|
|
0 commit comments