File: memory_alignment.pass.cpp

package info (click to toggle)
llvm-toolchain-19 1%3A19.1.7-3~deb12u1
  • links: PTS, VCS
  • area: main
  • in suites: bookworm-proposed-updates
  • size: 1,998,492 kB
  • sloc: cpp: 6,951,680; ansic: 1,486,157; asm: 913,598; python: 232,024; f90: 80,126; objc: 75,281; lisp: 37,276; pascal: 16,990; sh: 10,009; ml: 5,058; perl: 4,724; awk: 3,523; makefile: 3,167; javascript: 2,504; xml: 892; fortran: 664; cs: 573
file content (119 lines) | stat: -rw-r--r-- 4,998 bytes parent folder | download | duplicates (9)
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
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
//===----------------------------------------------------------------------===//
//
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
// See https://llvm.org/LICENSE.txt for license information.
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
//
//===----------------------------------------------------------------------===//

// UNSUPPORTED: c++03, c++11, c++14

// <experimental/simd>
//
// [simd.traits]
// template <class T, class U = typename T::value_type> struct memory_alignment;
// template <class T, class U = typename T::value_type>
//   inline constexpr std::size_t memory_alignment_v = memory_alignment<T, U>::value;

#include "../test_utils.h"

namespace ex = std::experimental::parallelism_v2;

template <class T, std::size_t N>
struct CheckMemoryAlignmentMask {
  template <class SimdAbi>
  void operator()() {
    LIBCPP_STATIC_ASSERT(
        ex::memory_alignment<ex::simd_mask<T, SimdAbi>>::value == bit_ceil(sizeof(bool) * ex::simd_size_v<T, SimdAbi>));
    LIBCPP_STATIC_ASSERT(
        ex::memory_alignment_v<ex::simd_mask<T, SimdAbi>> == bit_ceil(sizeof(bool) * ex::simd_size_v<T, SimdAbi>));
  }
};

template <class T, std::size_t N>
struct CheckMemoryAlignmentLongDouble {
  template <class SimdAbi>
  void operator()() {
    if constexpr (std::is_same_v<T, long double>) {
      // on i686-w64-mingw32-clang++, the size of long double is 12 bytes. Disambiguation is needed.
      static_assert(
          ex::memory_alignment<ex::simd<T, SimdAbi>>::value == bit_ceil(sizeof(T) * ex::simd_size_v<T, SimdAbi>));
      static_assert(ex::memory_alignment_v<ex::simd<T, SimdAbi>> == bit_ceil(sizeof(T) * ex::simd_size_v<T, SimdAbi>));
    }
  }
};

struct CheckMemAlignmentFixedDeduce {
  template <class T, std::size_t N>
  void check() {
    if constexpr (!std::is_same_v<T, long double>) {
      static_assert(ex::memory_alignment_v<ex::simd<T, ex::simd_abi::fixed_size<N>>> == sizeof(T) * bit_ceil(N),
                    "Memory Alignment mismatch with abi fixed_size");
      static_assert(ex::memory_alignment<ex::simd<T, ex::simd_abi::fixed_size<N>>>::value == sizeof(T) * bit_ceil(N),
                    "Memory Alignment mismatch with abi fixed_size");

      static_assert(ex::memory_alignment_v<ex::simd<T, ex::simd_abi::deduce_t<T, N>>> == sizeof(T) * bit_ceil(N),
                    "Memory Alignment mismatch with abi deduce");
      static_assert(ex::memory_alignment<ex::simd<T, ex::simd_abi::deduce_t<T, N>>>::value == sizeof(T) * bit_ceil(N),
                    "Memory Alignment mismatch with abi deduce");
    }
  }

  template <class T, std::size_t... N>
  void performChecks(std::index_sequence<N...>) {
    (check<T, N + 1>(), ...);
  }

  template <class T>
  void operator()() {
    performChecks<T>(std::make_index_sequence<max_simd_size>{});
  }
};

struct CheckMemAlignmentScalarNativeCompatible {
  template <class T>
  void operator()() {
    if constexpr (!std::is_same_v<T, long double>) {
      static_assert(ex::memory_alignment<ex::simd<T, ex::simd_abi::scalar>>::value == sizeof(T));
      static_assert(ex::memory_alignment_v<ex::simd<T, ex::simd_abi::scalar>> == sizeof(T));

      LIBCPP_STATIC_ASSERT(ex::memory_alignment<ex::simd<T, ex::simd_abi::compatible<T>>>::value == 16);
      LIBCPP_STATIC_ASSERT(ex::memory_alignment_v<ex::simd<T, ex::simd_abi::compatible<T>>> == 16);

      LIBCPP_STATIC_ASSERT(
          ex::memory_alignment<ex::simd<T, ex::simd_abi::native<T>>>::value == _LIBCPP_NATIVE_SIMD_WIDTH_IN_BYTES);
      LIBCPP_STATIC_ASSERT(
          ex::memory_alignment_v<ex::simd<T, ex::simd_abi::native<T>>> == _LIBCPP_NATIVE_SIMD_WIDTH_IN_BYTES);
    }
  }
};

template <class T, class U = typename T::value_type, class = void>
struct has_memory_alignment : std::false_type {};

template <class T, class U>
struct has_memory_alignment<T, U, std::void_t<decltype(ex::memory_alignment<T, U>::value)>> : std::true_type {};

struct CheckMemoryAlignmentTraits {
  template <class T>
  void operator()() {
    static_assert(has_memory_alignment<ex::native_simd<T>>::value);
    static_assert(has_memory_alignment<ex::fixed_size_simd_mask<T, 4>>::value);
    static_assert(has_memory_alignment<ex::native_simd<T>, T>::value);
    static_assert(has_memory_alignment<ex::fixed_size_simd_mask<T, 4>, bool>::value);

    static_assert(!has_memory_alignment<T, T>::value);
    static_assert(!has_memory_alignment<T, bool>::value);
    static_assert(!has_memory_alignment<ex::native_simd<T>, bool>::value);
    static_assert(!has_memory_alignment<ex::fixed_size_simd_mask<T, 4>, T>::value);
  }
};

int main(int, char**) {
  types::for_each(arithmetic_no_bool_types(), CheckMemAlignmentFixedDeduce());
  types::for_each(arithmetic_no_bool_types(), CheckMemAlignmentScalarNativeCompatible());
  test_all_simd_abi<CheckMemoryAlignmentMask>();
  test_all_simd_abi<CheckMemoryAlignmentLongDouble>();
  types::for_each(arithmetic_no_bool_types(), CheckMemoryAlignmentTraits());
  return 0;
}