29#include <boost/qvm/deduce_vec.hpp>
30#include <boost/qvm/vec_traits.hpp>
40#include <initializer_list>
50template <
typename T, std::
size_t N>
class Vector :
public Array<T, N> {
56 using Base::operator[];
70 template <
class U>
struct is_vector : std::false_type {};
71 template <
class U, std::
size_t Np>
78 void swap(
Vector &rhs) { std::ranges::swap_ranges(*
this, rhs); }
81 constexpr void copy_init(T
const *values)
noexcept {
82 for (std::size_t i{0}; i != N; ++i) {
83 (*this)[i] = values[i];
92 template <
class Range>
93 requires(not is_vector<std::remove_cvref_t<Range>>::value and
94 not std::is_same_v<std::remove_cvref_t<Range>, T[N]>)
95 explicit constexpr Vector(Range &&rng)
96 :
Vector(std::begin(rng), std::end(rng)) {}
98#if __cpp_lib_containers_ranges
99 template <std::ranges::input_range Range>
100 Vector(std::from_range_t, Range &&rng)
104 explicit constexpr Vector(T
const (&v)[N]) noexcept :
Base() {
105 if constexpr (N != 0) {
106 copy_init(std::cbegin(v));
112 throw std::length_error(
113 "Construction of Vector from Container of wrong length.");
115 if constexpr (N != 0) {
116 copy_init(v.begin());
120 template <
typename InputIterator>
122 if (std::distance(first, last) == N) {
123 std::copy_n(first, N,
begin());
125 throw std::length_error(
126 "Construction of Vector from Container of wrong length.");
134 for (std::size_t i = 0u; i != N; ++i) {
145 return std::span<T, N>(
const_cast<T *
>(
begin()),
size());
148 constexpr operator std::span<T, N>()
const {
return as_span(); }
153 std::ranges::transform(*
this, ret.
begin(),
154 [](T
const &e) { return static_cast<U>(e); });
159 constexpr T
norm2()
const {
return (*
this) * (*this); }
169 auto const l =
norm();
171 for (std::size_t i = 0u; i < N; ++i)
197template <std::
size_t N,
typename T,
typename U,
typename Op>
202 using R =
decltype(std::declval<T>() + std::declval<U>());
206 std::transform(std::begin(a), std::end(a), std::begin(b), std::begin(ret),
212template <std::
size_t N,
typename T,
typename Op>
214 for (std::size_t i = 0u; i < N; ++i) {
215 if (not op(a[i], b[i])) {
223template <std::
size_t N,
typename T>
225 return detail::all_of(a, b, std::less<T>());
228template <std::
size_t N,
typename T>
230 return detail::all_of(a, b, std::greater<T>());
233template <std::
size_t N,
typename T>
235 return detail::all_of(a, b, std::less_equal<T>());
238template <std::
size_t N,
typename T>
240 return detail::all_of(a, b, std::greater_equal<T>());
243template <std::
size_t N,
typename T>
245 return detail::all_of(a, b, std::equal_to<T>());
248template <std::
size_t N,
typename T>
253template <std::
size_t N,
typename T,
typename U>
255 return detail::binary_op(a, b, std::plus<>());
258template <std::
size_t N,
typename T>
260 std::ranges::transform(a, b, std::begin(a), std::plus<T>());
264template <std::
size_t N,
typename T,
typename U>
266 return detail::binary_op(a, b, std::minus<>());
269template <std::
size_t N,
typename T>
272 std::ranges::transform(a, std::begin(ret), std::negate<T>());
276template <std::
size_t N,
typename T>
278 std::ranges::transform(a, b, std::begin(a), std::minus<T>());
283template <std::
size_t N,
typename T,
class U>
284 requires(std::is_arithmetic_v<U>)
286 using R =
decltype(a * std::declval<T>());
288 std::ranges::transform(b, std::begin(ret), [a](T
const &v) {
return a * v; });
292template <std::
size_t N,
typename T,
class U>
293 requires(std::is_arithmetic_v<U>)
295 using R =
decltype(std::declval<T>() * b);
297 std::ranges::transform(a, std::begin(ret), [b](T
const &v) {
return b * v; });
301template <std::
size_t N,
typename T>
303 std::ranges::transform(b, std::begin(b), [a](T
const &v) {
return a * v; });
308template <std::
size_t N,
typename T,
class U>
310 using R =
decltype(std::declval<T>() / b);
312 std::ranges::transform(a, std::begin(ret), [b](T
const &v) {
return v / b; });
316template <std::
size_t N,
typename T,
class U>
318 using R =
decltype(a / std::declval<T>());
320 std::ranges::transform(b, std::begin(ret), [a](T
const &v) {
return a / v; });
324template <std::
size_t N,
typename T>
326 std::ranges::transform(a, std::begin(a), [b](T
const &v) {
return v / b; });
331template <
class T>
using is_vector = Vector<int, 1>::is_vector<T>;
335template <std::
size_t N,
typename T,
class U>
336 requires(not(detail::is_vector<T>::value or detail::is_vector<U>::value))
338 using R =
decltype(std::declval<T>() * std::declval<U>());
342 for (std::size_t i = 0u; i < N; ++i) {
348template <std::
size_t N,
typename T,
class U>
349 requires(std::is_integral_v<T> and std::is_integral_v<U>)
351 using R =
decltype(std::declval<T>() % std::declval<U>());
353 std::ranges::transform(a, b, std::begin(ret), std::modulus<>());
360 using R =
decltype(
sqrt(std::declval<T>()));
362 std::ranges::transform(a, ret.
begin(), [](T
const &v) { return sqrt(v); });
368 return {a[1] * b[2] - a[2] * b[1], a[2] * b[0] - a[0] * b[2],
369 a[0] * b[1] - a[1] * b[0]};
374 return std::accumulate(v.
cbegin(), v.
cend(), T{1}, std::multiplies<T>());
377template <
class T,
class U, std::
size_t N>
379 using R =
decltype(std::declval<T>() * std::declval<U>());
380 auto constexpr proj = std::identity{};
382 std::ranges::transform(a, b, ret.
begin(), std::multiplies<>(), proj, proj);
388template <
typename T,
typename U>
389 requires(not(detail::is_vector<T>::value and detail::is_vector<U>::value))
394template <
class T,
class U, std::
size_t N>
396 using R =
decltype(std::declval<T>() / std::declval<U>());
397 auto constexpr proj = std::identity{};
399 std::ranges::transform(a, b, std::begin(ret), std::divides<>(), proj, proj);
405template <
typename T,
typename U>
406 requires(not(detail::is_vector<T>::value and detail::is_vector<U>::value))
413 return {T{1}, T{0}, T{0}};
415 return {T{0}, T{1}, T{0}};
417 return {T{0}, T{0}, T{1}};
418 throw std::domain_error(
"coordinate out of range");
433template <std::
size_t I,
class T, std::
size_t N>
438template <std::
size_t I,
class T, std::
size_t N>
445template <std::
size_t I,
class T, std::
size_t N>
446struct std::tuple_element<I,
Utils::Vector<T, N>> {
447 static_assert(I < N,
"Utils::Vector index must be in range");
451template <
class T, std::
size_t N>
452struct std::tuple_size<
Utils::Vector<T, N>>
453 : std::integral_constant<std::size_t, N> {};
459 static constexpr std::size_t dim = N;
462 template <std::
size_t I>
467 template <std::
size_t I>
468 static constexpr inline scalar_type
483template <
typename T>
struct deduce_vec<
Utils::Vector<T, 3>, 3> {
Array implementation with CUDA support.
#define UTILS_ARRAY_BOOST_MPI_T(Container, N)
Mark array types as MPI data types.
#define UTILS_ARRAY_BOOST_BIT_S(Container, N)
Mark array types as MPI bitwise serializable.
#define UTILS_ARRAY_BOOST_CLASS(Container, N, ImplementationLevel)
Redefinition of BOOST_CLASS_IMPLEMENTATION for array types.
#define UTILS_ARRAY_BOOST_TRACK(Container, N, TrackingLevel)
Redefinition of BOOST_CLASS_TRACKING for array types.
Vector() noexcept=default
constexpr Vector(T const (&v)[N]) noexcept
DEVICE_QUALIFIER constexpr iterator begin() noexcept
Vector normalized() const
DEVICE_QUALIFIER constexpr const_iterator cbegin() const noexcept
std::vector< T > as_vector() const
Vector(InputIterator first, InputIterator last)
constexpr Vector(std::initializer_list< T > v)
DEVICE_QUALIFIER constexpr const_iterator cend() const noexcept
DEVICE_QUALIFIER constexpr size_type size() const noexcept
constexpr std::span< T, N > as_span() const
DEVICE_QUALIFIER constexpr iterator end() noexcept
static DEVICE_QUALIFIER constexpr Vector< T, N > broadcast(typename Base::value_type const &value) noexcept
Create a vector that has all entries set to the same value.
constexpr Vector(Range &&rng)
constexpr T norm2() const
auto operator+(Vector< T, N > const &a, Vector< U, N > const &b)
Vector< T, 3 > unit_vector(unsigned int i)
T product(Vector< T, N > const &v)
Vector< T, N > & operator-=(Vector< T, N > &a, Vector< T, N > const &b)
constexpr bool operator<(Vector< T, N > const &a, Vector< T, N > const &b)
constexpr bool operator>=(Vector< T, N > const &a, Vector< T, N > const &b)
constexpr bool operator>(Vector< T, N > const &a, Vector< T, N > const &b)
Vector< T, 3 > vector_product(Vector< T, 3 > const &a, Vector< T, 3 > const &b)
auto & operator+=(Vector< T, N > &a, Vector< T, N > const &b)
T & get(Array< T, N > &a) noexcept
auto operator/(Vector< T, N > const &a, U const &b)
constexpr bool operator==(Vector< T, N > const &a, Vector< T, N > const &b)
constexpr bool operator!=(Vector< T, N > const &a, Vector< T, N > const &b)
constexpr bool operator<=(Vector< T, N > const &a, Vector< T, N > const &b)
auto hadamard_division(Vector< T, N > const &a, Vector< U, N > const &b)
auto & operator*=(Vector< T, N > &b, T const &a)
auto hadamard_product(Vector< T, N > const &a, Vector< U, N > const &b)
auto operator-(Vector< T, N > const &a, Vector< U, N > const &b)
auto sqrt(Vector< T, N > const &a)
auto & operator/=(Vector< T, N > &a, T const &b)
DEVICE_QUALIFIER constexpr reference at(size_type i)
DEVICE_QUALIFIER constexpr bool empty() const noexcept
DEVICE_QUALIFIER constexpr reference back()
DEVICE_QUALIFIER constexpr pointer data() noexcept
DEVICE_QUALIFIER constexpr size_type max_size() const noexcept
DEVICE_QUALIFIER constexpr iterator begin() noexcept
DEVICE_QUALIFIER constexpr const_iterator cbegin() const noexcept
DEVICE_QUALIFIER constexpr const_iterator cend() const noexcept
static DEVICE_QUALIFIER constexpr Array< T, N > broadcast(const value_type &value)
DEVICE_QUALIFIER constexpr size_type size() const noexcept
DEVICE_QUALIFIER constexpr reference front()
DEVICE_QUALIFIER constexpr iterator end() noexcept
DEVICE_QUALIFIER void fill(const value_type &value)
Meta function to turn a Vector<T, 1> into T.
static constexpr scalar_type read_element(::Utils::Vector< T, N > const &v)
static scalar_type & write_element_idx(std::size_t i, ::Utils::Vector< T, N > &v)
static constexpr scalar_type & write_element(::Utils::Vector< T, N > &v)
static scalar_type read_element_idx(std::size_t i, ::Utils::Vector< T, N > const &v)