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>
81 constexpr void copy_init(T
const *
values)
noexcept {
82 for (std::size_t i{0}; i !=
N; ++i) {
92 template <
class Range>
94 not std::is_same_v<std::remove_cvref_t<Range>, T[
N]>)
98#if __cpp_lib_containers_ranges
99 template <std::ranges::input_range Range>
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>
125 throw std::length_error(
126 "Construction of Vector from Container of wrong length.");
134 for (std::size_t i = 0
u; 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 = 0
u; 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 = 0
u; 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 = 0
u; 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{};
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
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
cudaStream_t stream[1]
CUDA streams for parallel computing on CPU and GPU.
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)