Wildmeshing Toolkit
Loading...
Searching...
No Matches
CachingAttribute.hxx
Go to the documentation of this file.
1#pragma once
2#include <wmtk/Types.hpp>
5
6#if defined(WMTK_ENABLED_DEV_MODE)
7#include <fmt/ranges.h>
8#include <spdlog/spdlog.h>
9#include <ranges>
10#include <span>
11#define WMTK_CACHING_ATTRIBUTE_INLINE
12
13
14#else
15#define WMTK_CACHING_ATTRIBUTE_INLINE inline
16#endif
17
18namespace wmtk::attribute {
19#if defined(WMTK_ENABLED_DEV_MODE)
20template <typename T>
21void CachingAttribute<T>::print_state(std::string_view prefix) const
22{
23 if constexpr (std::is_same_v<T, Rational>) {
24 } else if constexpr (std::is_same_v<T, char>) {
25 auto toint = [](auto&& v) noexcept -> int64_t { return v; };
26 spdlog::warn(
27 "Attribute {}: [{}] on transaction {} of {}",
28 BaseType::m_name,
29 prefix,
30 m_current_transaction_index,
31 m_transaction_starts.size());
32 spdlog::info("Data: {}", fmt::join(std::views::transform(BaseType::m_data, toint), ","));
33 for (size_t j = 0; j < m_transaction_starts.size(); ++j) {
34 size_t start = m_transaction_starts[j];
35 size_t end;
36 if (j == m_transaction_starts.size() - 1) {
37 end = m_indices_end;
38 } else {
39 end = m_transaction_starts[j + 1];
40 }
41 spdlog::info("Detailing transaction {} with value indices {}->{}", j, start, end);
42
43 for (size_t k = start; k < end; ++k) {
44 const auto& [attr_index, table_index] = m_indices[k];
45 spdlog::info(
46 "attr index {} has table index {} and value {}",
47 attr_index,
48 table_index,
49 fmt::join(
50 std::views::transform(
51 std::span(m_buffer.data() + table_index, BaseType::dimension()),
52 toint),
53 ","));
54 }
55 }
56 }
57}
58#endif
59
60template <typename T>
62
63 -> std::vector<std::pair<size_t, size_t>>::const_iterator
64{
65 return m_indices.begin() + m_transaction_starts[scope_index];
66}
67template <typename T>
69 -> std::vector<std::pair<size_t, size_t>>::const_iterator
70
71{
72 return m_indices.begin() + m_indices_end;
73}
74
75template <typename T>
76auto CachingAttribute<T>::transaction_start_rend(size_t scope_index) const
77 -> std::vector<std::pair<size_t, size_t>>::const_reverse_iterator
78{
79 return std::reverse_iterator(transaction_start_begin(scope_index));
80}
81template <typename T>
83 -> std::vector<std::pair<size_t, size_t>>::const_reverse_iterator
84{
85 return std::reverse_iterator(final_transaction_end());
86}
87
88
89template <typename T>
91{
92 assert(has_transactions());
93 assert(at_current_scope());
94 apply_last_scope();
95}
96
97template <typename T>
98template <int D>
100 -> MapResult<D>
101{
102 assert(at_current_scope());
103
104 auto data = BaseType::template vector_attribute<D>(index);
105 assert(data.cols() == 1);
106 if constexpr (D != Eigen::Dynamic) {
107 assert(data.size() == D);
108 }
109 // we are typically only going to write when caching is enabled so better to optimize for this
110 if (has_transactions()) {
111 cache(index, data);
112 }
113 return data;
114}
115
116
117template <typename T>
118template <int D>
121{
122 if (!at_current_scope()) {
123 assert(m_current_transaction_index < m_transaction_starts.size());
124
125 const T* ptr = get_value(index);
126 if (ptr != nullptr) {
127 const int dim = BaseType::dimension();
128 auto dat = ConstMapResult<D>(ptr, dim);
129 return dat;
130 }
131 }
132 return BaseType::template const_vector_attribute<D>(index);
133}
134
135template <typename T>
137{
138 assert(at_current_scope());
139 T& value = BaseType::scalar_attribute(index);
140 if (has_transactions()) {
141 cache(index, value);
142 }
143 return value;
144}
145
146template <typename T>
148 -> const T&
149{
150 if (!at_current_scope()) {
151 assert(m_current_transaction_index < m_transaction_starts.size());
152
153 const T* ptr = get_value(index);
154 if (ptr != nullptr) {
155 return *ptr;
156 }
157 }
158
159 return BaseType::const_scalar_attribute(index);
160}
161template <typename T>
162template <int D>
164 int64_t index,
165 int8_t vector_index) const -> const T&
166{
167 if (!at_current_scope()) {
168 assert(m_current_transaction_index < m_transaction_starts.size());
169
170 const T* ptr = get_value(index);
171 if (ptr != nullptr) {
172 const int dim = BaseType::dimension();
173 assert(vector_index < dim);
174 return ptr[vector_index];
175 }
176 }
177 return BaseType::const_vector_single_value(index, vector_index);
178}
179
180//=======================================================
181// Scope members
182//=======================================================
183template <typename T>
188template <typename T>
190{
191 pop(preserve_changes);
192}
193
194template <typename T>
196{
197 m_transaction_starts.clear();
198 change_to_current_scope();
199 clear();
200}
201
202template <typename T>
204{
205 // clearing a scope means nothing?
206 if (!has_transactions()) {
207 m_indices_end = 0;
208 m_buffer_end = 0;
209 } else {
210 m_indices_end = m_transaction_starts.back();
211 m_buffer_end = m_indices[m_indices_end].second;
212 }
213}
214
215template <typename T>
217 size_t data_size)
218{
219 // check sizes
220 if (m_indices_end + 1 >= m_indices.size()) {
221 assert(m_indices.size() > 2);
222 m_indices.resize(m_indices.size() * 1.75);
223 }
224
225 if (m_buffer_end + data_size >= m_buffer.size()) {
226 assert(m_buffer.size() > 2);
227 m_buffer.resize(m_buffer.size() * 1.75);
228 }
229}
230
231template <typename T>
232template <typename Derived>
234 int64_t index,
235 const Eigen::MatrixBase<Derived>& value)
236{
237 // basically try_emplace but optimizes to avoid accessing the pointed-to value
238
239 size_t dim = value.size();
240
241
242 update_buffer_sizes_for_add(dim);
243
244 // assert(old_size % dim == 0);
245 //assert(old_size / value.size() == m_indices.size());
246 // m_indices.emplace_back(index, m_indices.size());
247 m_indices[m_indices_end] = {index, m_buffer_end};
248
249
250 assert(m_buffer.size() >= m_buffer_end + dim);
251 // m_buffer.resize(m_buffer_end + dim);
252 std::copy(value.begin(), value.end(), m_buffer.begin() + m_buffer_end);
253 m_buffer_end += dim;
254 m_indices_end++;
255 assert(m_buffer_end == m_indices_end * dim);
256}
257
258template <typename T>
260{
261 update_buffer_sizes_for_add(1);
262 // assert(m_buffer.size() == m_indices.size());
263 m_indices[m_indices_end] = {index, m_buffer_end};
264 m_buffer[m_buffer_end] = value;
265
266 m_buffer_end++;
267 m_indices_end++;
268 assert(m_buffer_end == m_indices_end);
269}
270
271
272template <typename T>
274{
275 apply_cache(BaseType::m_data);
276}
277
278template <typename T>
280{
281 assert(at_current_scope());
282 for (auto it = final_transaction_rbegin();
283 it != transaction_start_rend(current_transaction_index() - 1);
284 ++it) {
285 const auto& [global, local] = *it;
286 auto a = BaseType::vector_attribute(global, other);
287 auto b = BaseType::const_vector_attribute_without_stride(local, m_buffer);
288 a = b;
289 }
290}
291
292template <typename T>
294{
295 for (auto it = transaction_start_begin(current_transaction_index());
296 it != final_transaction_end();
297 ++it) {
298 const auto& [global_index, local_index] = *it;
299 if (global_index == index) {
300 const T* ptr = m_buffer.data() + local_index;
301 return ptr;
302 }
303 }
304 return nullptr;
305}
306
307template <typename T>
309{
310 assert(at_current_scope()); // must only be called on leaf node
311
312 m_transaction_starts.emplace_back(m_indices_end);
313 change_to_current_scope();
314}
315template <typename T>
317{
318 assert(at_current_scope()); // must only be called on leaf node
319 // TODO consider re-enabling
320 if (!preserve_changes) {
321 apply_cache();
322 }
323
324
325 m_transaction_starts.pop_back();
326
327 change_to_current_scope();
328 if (!has_transactions()) {
329 m_indices_end = 0;
330 m_buffer_end = 0;
331 }
332}
333
334
335template <typename T>
337{
338 assert(at_current_scope());
339 assert(has_transactions());
340 apply_cache();
341}
342
343template <typename T>
345{
346 // if the previous is a nullptr it's fine
347 assert(!at_current_scope());
348 m_current_transaction_index++;
349
350}
351
352template <typename T>
354{
355 if (at_current_scope()) {
356 assert(has_transactions());
357 }
358 m_current_transaction_index--;
359}
360template <typename T>
362{
363 m_current_transaction_index = transaction_depth();
364}
365template <typename T>
367{
368 assert(m_current_transaction_index <= m_transaction_starts.size());
369 return m_current_transaction_index == transaction_depth();
370}
371
372template <typename T>
374{
375 return m_transaction_starts.size();
376}
377
378template <typename T>
380{
381 return !m_transaction_starts.empty();
382}
383
384} // namespace wmtk::attribute
385#undef WMTK_CACHING_ATTRIBUTE_INLINE
#define WMTK_CACHING_ATTRIBUTE_INLINE
ConstMapResult< D > const_vector_attribute(int64_t index) const
default immutable vector access
const T & const_scalar_attribute(int64_t index) const
default immutable scalar access
std::vector< std::pair< size_t, size_t > >::const_iterator final_transaction_end() const
void pop(bool preserve_changes)
MapResult< D > vector_attribute(int64_t index)
default mutable vector access
bool at_current_scope() const
checks that we are viewing the active state of the attribute
const T * get_value(int64_t index) const
typename BaseType::template MapResult< D > MapResult
std::vector< std::pair< size_t, size_t > >::const_reverse_iterator transaction_start_rend(size_t scope_index) const
std::vector< std::pair< size_t, size_t > >::const_reverse_iterator final_transaction_rbegin() const
void pop_scope(bool preserve_changes)
void cache(int64_t index, const Eigen::MatrixBase< Derived > &value)
std::vector< std::pair< size_t, size_t > >::const_iterator transaction_start_begin(size_t scope_index) const
typename BaseType::template ConstMapResult< D > ConstMapResult
T & scalar_attribute(int64_t index)
default mutable scalar access
const T & const_vector_single_value(int64_t index, int8_t vector_index) const
specialized immutable scalar access useful for topological operations
void update_buffer_sizes_for_add(size_t data_size)