DDC 0.10.0
Loading...
Searching...
No Matches
non_uniform_point_sampling.hpp
1// Copyright (C) The DDC development team, see COPYRIGHT.md file
2//
3// SPDX-License-Identifier: MIT
4
5#pragma once
6
7#include <algorithm>
8#include <cassert>
9#include <cstddef>
10#include <initializer_list>
11#include <iterator>
12#include <ostream>
13#include <stdexcept>
14#include <tuple>
15#include <type_traits>
16#include <utility>
17#include <vector>
18
19#include <Kokkos_Core.hpp>
20
21#include "coordinate.hpp"
22#include "discrete_domain.hpp"
23#include "discrete_element.hpp"
24#include "discrete_space.hpp"
25#include "discrete_vector.hpp"
26
27namespace ddc {
28
29namespace detail {
30
31struct NonUniformPointSamplingBase
32{
33};
34
35} // namespace detail
36
37/// `NonUniformPointSampling` models a non-uniform discretization of the `CDim` segment \f$[a, b]\f$.
38template <class CDim>
39class NonUniformPointSampling : detail::NonUniformPointSamplingBase
40{
41public:
42 using continuous_dimension_type = CDim;
43
44 using discrete_dimension_type = NonUniformPointSampling;
45
46public:
47 template <class DDim, class MemorySpace>
48 class Impl
49 {
50 template <class ODDim, class OMemorySpace>
51 friend class Impl;
52
53 Kokkos::View<Coordinate<CDim>*, MemorySpace> m_points;
54
55 DiscreteElement<DDim> m_reference;
56
57 public:
58 using discrete_dimension_type = NonUniformPointSampling;
59
60 using discrete_domain_type = DiscreteDomain<DDim>;
61
62 using discrete_element_type = DiscreteElement<DDim>;
63
64 using discrete_vector_type = DiscreteVector<DDim>;
65
66 Impl() = default;
67
68 /// @brief Construct a `NonUniformPointSampling` using a brace-list, i.e. `NonUniformPointSampling mesh({0., 1.})`
69 Impl(std::initializer_list<Coordinate<CDim>> const points)
70 : Impl(points.begin(), points.end())
71 {
72 }
73
74 /// @brief Construct a `NonUniformPointSampling` using a C++20 "common range".
75 template <class InputRange>
76 explicit Impl(InputRange const& points) : Impl(points.begin(), points.end())
77 {
78 }
79
80 /// @brief Construct a `NonUniformPointSampling` using a pair of iterators.
81 template <class InputIt>
82 Impl(InputIt const points_begin, InputIt const points_end)
83 : m_reference(create_reference_discrete_element<DDim>())
84 {
85 using view_type = Kokkos::View<Coordinate<CDim>*, MemorySpace>;
86 if (!std::is_sorted(points_begin, points_end)) {
87 throw std::runtime_error("Input points must be sorted");
88 }
89 // Make a contiguous copy of [points_begin, points_end[
90 std::vector<Coordinate<CDim>> host_points(points_begin, points_end);
91 m_points = view_type("NonUniformPointSampling::points", host_points.size());
92 Kokkos::deep_copy(m_points, view_type(host_points.data(), host_points.size()));
93 }
94
95 template <class OriginMemorySpace>
96 explicit Impl(Impl<DDim, OriginMemorySpace> const& impl)
97 : m_points(Kokkos::create_mirror_view_and_copy(MemorySpace(), impl.m_points))
98 , m_reference(impl.m_reference)
99 {
100 }
101
102 Impl(Impl const& x) = delete;
103
104 Impl(Impl&& x) = default;
105
106 ~Impl() = default;
107
108 Impl& operator=(Impl const& x) = delete;
109
110 Impl& operator=(Impl&& x) = default;
111
112 KOKKOS_FUNCTION std::size_t size() const
113 {
114 return m_points.size();
115 }
116
117 /// @brief Lower bound index of the mesh
118 KOKKOS_FUNCTION discrete_element_type front() const noexcept
119 {
120 return m_reference;
121 }
122
123 /// @brief Convert a mesh index into a position in `CDim`
124 KOKKOS_FUNCTION Coordinate<CDim> coordinate(
125 discrete_element_type const& icoord) const noexcept
126 {
127 return m_points((icoord - front()).value());
128 }
129 };
130
131 /** Construct an Impl<Kokkos::HostSpace> and associated discrete_domain_type from a range
132 * containing the points coordinates along the `DDim` dimension.
133 *
134 * @param non_uniform_points a range (std::vector, std::array, ...) containing the coordinates of the points of the domain.
135 */
136 template <class DDim, class InputRange>
137 static std::tuple<typename DDim::template Impl<DDim, Kokkos::HostSpace>, DiscreteDomain<DDim>>
138 init(InputRange const& non_uniform_points)
139 {
140 assert(!non_uniform_points.empty());
141 DiscreteVector<DDim> const n(non_uniform_points.size());
142 typename DDim::template Impl<DDim, Kokkos::HostSpace> disc(non_uniform_points);
143 DiscreteDomain<DDim> domain(disc.front(), n);
144 return std::make_tuple(std::move(disc), std::move(domain));
145 }
146
147 /** Construct 4 non-uniform `DiscreteDomain` and an Impl<Kokkos::HostSpace> from 3 ranges containing the points coordinates along the `DDim` dimension.
148 *
149 * @param domain_r a range containing the coordinates of the points of the main domain along the DDim position
150 * @param pre_ghost_r a range containing the positions of the ghost points before the main domain the DDim position
151 * @param post_ghost_r a range containing the positions of the ghost points after the main domain the DDim position
152 */
153 template <class DDim, class InputRange>
154 static std::tuple<
155 typename DDim::template Impl<DDim, Kokkos::HostSpace>,
156 DiscreteDomain<DDim>,
157 DiscreteDomain<DDim>,
158 DiscreteDomain<DDim>,
159 DiscreteDomain<DDim>>
161 InputRange const& domain_r,
162 InputRange const& pre_ghost_r,
163 InputRange const& post_ghost_r)
164 {
165 assert(!domain_r.empty());
166
167 DiscreteVector<DDim> const n(domain_r.size());
168 DiscreteVector<DDim> const n_ghosts_before(pre_ghost_r.size());
169 DiscreteVector<DDim> const n_ghosts_after(post_ghost_r.size());
170
171 std::vector<typename InputRange::value_type> full_domain;
172
173 std::copy(pre_ghost_r.begin(), pre_ghost_r.end(), std::back_inserter(full_domain));
174 std::copy(domain_r.begin(), domain_r.end(), std::back_inserter(full_domain));
175 std::copy(post_ghost_r.begin(), post_ghost_r.end(), std::back_inserter(full_domain));
176
177 typename DDim::template Impl<DDim, Kokkos::HostSpace> disc(full_domain);
178
179 DiscreteDomain<DDim> ghosted_domain(disc.front(), n + n_ghosts_before + n_ghosts_after);
180 DiscreteDomain<DDim> pre_ghost = ghosted_domain.take_first(n_ghosts_before);
181 DiscreteDomain<DDim> main_domain = ghosted_domain.remove(n_ghosts_before, n_ghosts_after);
182 DiscreteDomain<DDim> post_ghost = ghosted_domain.take_last(n_ghosts_after);
183 return std::make_tuple(
184 std::move(disc),
185 std::move(main_domain),
186 std::move(ghosted_domain),
187 std::move(pre_ghost),
188 std::move(post_ghost));
189 }
190};
191
192template <class DDim>
194 : public std::is_base_of<detail::NonUniformPointSamplingBase, DDim>::type
195{
196};
197
198template <class DDim>
200
201template <
202 class DDimImpl,
203 std::enable_if_t<
204 is_non_uniform_point_sampling_v<typename DDimImpl::discrete_dimension_type>,
205 int>
206 = 0>
207std::ostream& operator<<(std::ostream& out, DDimImpl const& mesh)
208{
209 return out << "NonUniformPointSampling(" << mesh.size() << ")";
210}
211
212template <class DDim, std::enable_if_t<is_non_uniform_point_sampling_v<DDim>, int> = 0>
213KOKKOS_FUNCTION Coordinate<typename DDim::continuous_dimension_type> coordinate(
214 DiscreteElement<DDim> const& c)
215{
216 return discrete_space<DDim>().coordinate(c);
217}
218
219template <class DDim, std::enable_if_t<is_non_uniform_point_sampling_v<DDim>, int> = 0>
221 DiscreteElement<DDim> i)
222{
223 return coordinate(i) - coordinate(i - 1);
224}
225
226template <class DDim, std::enable_if_t<is_non_uniform_point_sampling_v<DDim>, int> = 0>
228 DiscreteElement<DDim> i)
229{
230 return coordinate(i + 1) - coordinate(i);
231}
232
233template <class DDim, std::enable_if_t<is_non_uniform_point_sampling_v<DDim>, int> = 0>
234KOKKOS_FUNCTION Coordinate<typename DDim::continuous_dimension_type> rmin(
235 DiscreteDomain<DDim> const& d)
236{
237 return coordinate(d.front());
238}
239
240template <class DDim, std::enable_if_t<is_non_uniform_point_sampling_v<DDim>, int> = 0>
241KOKKOS_FUNCTION Coordinate<typename DDim::continuous_dimension_type> rmax(
242 DiscreteDomain<DDim> const& d)
243{
244 return coordinate(d.back());
245}
246
247template <class DDim, std::enable_if_t<is_non_uniform_point_sampling_v<DDim>, int> = 0>
248KOKKOS_FUNCTION Coordinate<typename DDim::continuous_dimension_type> rlength(
249 DiscreteDomain<DDim> const& d)
250{
251 return rmax(d) - rmin(d);
252}
253
254} // namespace ddc
friend class DiscreteDomain
KOKKOS_FUNCTION constexpr bool operator!=(DiscreteVector< OTags... > const &rhs) const noexcept
KOKKOS_FUNCTION std::size_t size() const
KOKKOS_FUNCTION Coordinate< CDim > coordinate(discrete_element_type const &icoord) const noexcept
Convert a mesh index into a position in CDim
Impl(Impl< DDim, OriginMemorySpace > const &impl)
Impl & operator=(Impl &&x)=default
Impl & operator=(Impl const &x)=delete
Impl(InputIt const points_begin, InputIt const points_end)
Construct a NonUniformPointSampling using a pair of iterators.
KOKKOS_FUNCTION discrete_element_type front() const noexcept
Lower bound index of the mesh.
Impl(InputRange const &points)
Construct a NonUniformPointSampling using a C++20 "common range".
Impl(std::initializer_list< Coordinate< CDim > > const points)
Construct a NonUniformPointSampling using a brace-list, i.e. NonUniformPointSampling mesh({0....
NonUniformPointSampling models a non-uniform discretization of the CDim segment .
static std::tuple< typename DDim::template Impl< DDim, Kokkos::HostSpace >, DiscreteDomain< DDim > > init(InputRange const &non_uniform_points)
Construct an Impl<Kokkos::HostSpace> and associated discrete_domain_type from a range containing the ...
static std::tuple< typename DDim::template Impl< DDim, Kokkos::HostSpace >, DiscreteDomain< DDim >, DiscreteDomain< DDim >, DiscreteDomain< DDim >, DiscreteDomain< DDim > > init_ghosted(InputRange const &domain_r, InputRange const &pre_ghost_r, InputRange const &post_ghost_r)
Construct 4 non-uniform DiscreteDomain and an Impl<Kokkos::HostSpace> from 3 ranges containing the po...
The top-level namespace of DDC.
constexpr bool is_non_uniform_point_sampling_v
KOKKOS_FUNCTION Coordinate< typename DDim::continuous_dimension_type > rmax(DiscreteDomain< DDim > const &d)
KOKKOS_FUNCTION Coordinate< typename DDim::continuous_dimension_type > rlength(DiscreteDomain< DDim > const &d)
KOKKOS_FUNCTION Coordinate< typename DDim::continuous_dimension_type > distance_at_left(DiscreteElement< DDim > i)
KOKKOS_FUNCTION Coordinate< typename DDim::continuous_dimension_type > rmin(DiscreteDomain< DDim > const &d)
KOKKOS_FUNCTION Coordinate< typename DDim::continuous_dimension_type > coordinate(DiscreteElement< DDim > const &c)
KOKKOS_FUNCTION Coordinate< typename DDim::continuous_dimension_type > distance_at_right(DiscreteElement< DDim > i)