SeqAn3 3.4.1-rc.1
The Modern C++ library for sequence analysis.
Loading...
Searching...
No Matches
utility/simd/concept.hpp
Go to the documentation of this file.
1// SPDX-FileCopyrightText: 2006-2025 Knut Reinert & Freie Universität Berlin
2// SPDX-FileCopyrightText: 2016-2025 Knut Reinert & MPI für molekulare Genetik
3// SPDX-License-Identifier: BSD-3-Clause
4
10#pragma once
11
12#include <concepts>
13#include <type_traits>
14
16
17namespace seqan3::detail
18{
20template <template <typename> typename rebind>
21struct simd_traits_has_rebind : std::true_type
22{};
23// NOTE: this definition should be used for seqan3::simd, but gcc has a bug that it will not fail silently if
24// simd_t is a pointer to a incomplete type. Furthermore the is_pointer_v should prevent those cases by checking the type
25// beforehand, but for some reasons the short-circuit semantic of `&&` does not work in this case and gcc still evaluates
26// the requires clause which in turn triggers the error.
27//
28// If this concept is used directly on incomplete types it will produces this compiler error:
29// error: invalid use of incomplete type ‘struct incomplete::template_type<int>’
30// requires std::same_as<decltype(a - b), simd_t>;
31template <typename simd_t>
32concept simd_concept = requires (simd_t a, simd_t b) {
33 typename simd_traits<std::remove_reference_t<simd_t>>::scalar_type;
34 typename simd_traits<std::remove_reference_t<simd_t>>::mask_type;
35 typename simd_traits<std::remove_reference_t<simd_t>>::swizzle_type;
36 requires simd_traits_has_rebind<simd_traits<std::remove_reference_t<simd_t>>::template rebind>::value;
37
38 // require that static member variables are defined
39 requires std::integral<decltype(simd_traits<std::remove_reference_t<simd_t>>::length)>;
40 requires std::integral<decltype(simd_traits<std::remove_reference_t<simd_t>>::max_length)>;
41
42 // assume array access that returns a scalar_type type
43 { a[0] } -> std::convertible_to<typename simd_traits<std::remove_reference_t<simd_t>>::scalar_type>;
44
45 // require comparison operators
46 { a == b } -> std::same_as<typename simd_traits<std::remove_reference_t<simd_t>>::mask_type>;
47 { a != b } -> std::same_as<typename simd_traits<std::remove_reference_t<simd_t>>::mask_type>;
48 { a < b } -> std::same_as<typename simd_traits<std::remove_reference_t<simd_t>>::mask_type>;
49 { a > b } -> std::same_as<typename simd_traits<std::remove_reference_t<simd_t>>::mask_type>;
50 { a <= b } -> std::same_as<typename simd_traits<std::remove_reference_t<simd_t>>::mask_type>;
51 { a >= b } -> std::same_as<typename simd_traits<std::remove_reference_t<simd_t>>::mask_type>;
52
53 // require arithmetic operators
54 { a + b } -> std::same_as<std::remove_reference_t<simd_t>>;
55 { a - b } -> std::same_as<std::remove_reference_t<simd_t>>;
56 { a * b } -> std::same_as<std::remove_reference_t<simd_t>>;
57 { a / b } -> std::same_as<std::remove_reference_t<simd_t>>;
58 { a += b } -> std::same_as<std::remove_reference_t<simd_t> &>;
59 { a -= b } -> std::same_as<std::remove_reference_t<simd_t> &>;
60 { a *= b } -> std::same_as<std::remove_reference_t<simd_t> &>;
61 { a /= b } -> std::same_as<std::remove_reference_t<simd_t> &>;
62};
64
65} // namespace seqan3::detail
66
67namespace seqan3
68{
69
70inline namespace simd
71{
72
87template <typename simd_t>
88concept simd_concept = !std::is_pointer_v<std::decay_t<simd_t>> && detail::simd_concept<simd_t>;
90
100template <typename t>
101concept simd_index = simd::simd_concept<t> &&
102 requires ()
103{
104 requires std::integral<typename simd_traits<std::remove_reference_t<t>>::scalar_type>;
105};
107
108} // namespace simd
109
110} // namespace seqan3
The generic simd concept.
Refines the seqan3::simd::simd_concept requiring the underlying scalar type to model std::integral.
The internal SeqAn3 namespace.
Definition aligned_sequence_concept.hpp:26
The main SeqAn3 namespace.
Definition aligned_sequence_concept.hpp:26
Provides seqan3::simd::simd_traits.
Hide me