Gamp v0.0.7-36-g24b1eb6
Gamp: Graphics, Audio, Multimedia and Processing
Loading...
Searching...
No Matches
basic_algos.hpp
Go to the documentation of this file.
1/*
2 * Author: Sven Gothel <sgothel@jausoft.com>
3 * Copyright (c) 2020 Gothel Software e.K.
4 * Copyright (c) 2020 ZAFENA AB
5 *
6 * Permission is hereby granted, free of charge, to any person obtaining
7 * a copy of this software and associated documentation files (the
8 * "Software"), to deal in the Software without restriction, including
9 * without limitation the rights to use, copy, modify, merge, publish,
10 * distribute, sublicense, and/or sell copies of the Software, and to
11 * permit persons to whom the Software is furnished to do so, subject to
12 * the following conditions:
13 *
14 * The above copyright notice and this permission notice shall be
15 * included in all copies or substantial portions of the Software.
16 *
17 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
18 * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
19 * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
20 * NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
21 * LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
22 * OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
23 * WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
24 */
25
26#ifndef JAU_BASIC_ALGOS_HPP_
27#define JAU_BASIC_ALGOS_HPP_
28
29#include <mutex>
30#include <type_traits>
31
32#include <jau/cow_iterator.hpp>
33
34namespace jau {
35 /** @defgroup Algorithms Basic Algorithms
36 * Basic algorithms.
37 *
38 * @{
39 */
40
41 /**
42 * Call on release allows the user to pass a function
43 * to be called at destruction of this instance.
44 * <p>
45 * One goal was to provide a thread exit cleanup facility,
46 * setting a 'is_running' flag to false when the thread exists
47 * normally or abnormally.
48 * <pre>
49 * jau::relaxed_atomic_bool is_running = true;
50 *
51 * void some_thread_func() {
52 * thread_local jau::call_on_release thread_cleanup([&]() {
53 * is_running = false;
54 * });
55 * ...
56 * do some work here, which might get cancelled
57 * ..
58 * thread_cleanup.set_released(); // mark orderly release
59 * }
60 * </pre>
61 * </p>
62 * @tparam UnaryFunction user provided function to be called @ dtor
63 * @see jau::service_runner
64 */
65 template <class UnaryFunction> class call_on_release {
66 private:
67 UnaryFunction f;
68 jau::sc_atomic_bool released;
69
70 public:
71 call_on_release(UnaryFunction release_func) noexcept
72 : f(release_func), released(false) {}
73 ~call_on_release() noexcept {
74 if( !released ) { f(); }
75 }
78 call_on_release& operator=(const call_on_release&) volatile = delete;
79
80 /** Mark the resource being orderly released, `release_func()` will not be called and *use after free* avoided. */
81 void set_released() noexcept { released = true; }
82 /** Query whethr the resource has been orderly released. */
83 bool is_released() const noexcept { return released; }
84 };
85
86 /****************************************************************************************
87 ****************************************************************************************/
88
89 /**
90 * Like std::find() of 'algorithm'
91 * <p>
92 * Only exists here as performance analysis over O(n*n) complexity
93 * exposes std::find() to be approximately 3x slower.<br>
94 * See test/test_cow_darray_perf01.cpp
95 * </p>
96 * @tparam InputIt the iterator type
97 * @tparam T the data type
98 * @param first range start of elements to examine
99 * @param last range end of elements to examine, exclusive
100 * @param value reference value for comparison
101 * @return Iterator to the first element satisfying the condition or last if no such element is found.
102 */
103 template<class InputIt, class T>
104 constexpr InputIt find(InputIt first, InputIt last, const T& value) noexcept
105 {
106 for (; first != last; ++first) {
107 if (*first == value) {
108 return first;
109 }
110 }
111 return last; // implicit move since C++11
112 }
113
114 /**
115 * Return true if `value` is contained in `array`.
116 * @tparam InputArray array type
117 * @tparam T value_type of value
118 * @param array the array to search
119 * @param value the value to search for
120 * @return true if contained, otherwise false
121 */
122 template<class InputArray, class T>
123 constexpr bool contains(InputArray &array, const T& value) noexcept
124 {
125 const typename InputArray::size_type size = array.size();
126 for (typename InputArray::size_type i = 0; i < size; ++i) {
127 if( value == array[i] ) {
128 return true;
129 }
130 }
131 return false;
132 }
133
134 template<class InputArray, class T>
135 constexpr bool eraseFirst(InputArray &array, const T& value)
136 {
137 const typename InputArray::size_type size = array.size();
138 for (typename InputArray::size_type i = 0; i < size; ++i) {
139 if( value == array[i] ) {
140 array.erase(i);
141 return true;
142 }
143 }
144 return false;
145 }
146
147 /**
148 * Like std::find_if() of 'algorithm'
149 * <p>
150 * Only exists here as performance analysis over O(n*n) complexity
151 * exposes std::find_if() to be approximately 3x slower for 1000 x 1000.<br>
152 * See test/test_cow_darray_perf01.cpp
153 * </p>
154 * @tparam InputIt the iterator type
155 * @tparam UnaryPredicate
156 * @param first range start of elements to examine
157 * @param last range end of elements to examine, exclusive
158 * @param p unary predicate which returns ​true for the desired element.
159 * @return Iterator to the first element satisfying the condition or last if no such element is found.
160 */
161 template<class InputIt, class UnaryPredicate>
162 constexpr InputIt find_if(InputIt first, InputIt last, UnaryPredicate p) noexcept
163 {
164 for (; first != last; ++first) {
165 if (p(*first)) {
166 return first;
167 }
168 }
169 return last; // implicit move since C++11
170 }
171
172 /**
173 * Like std::find_if_not() of 'algorithm'
174 * <p>
175 * Only exists here as performance analysis over O(n*n) complexity
176 * exposes std::find_if_not() to be approximately 3x slower for 1000 x 1000.<br>
177 * See test/test_cow_darray_perf01.cpp
178 * </p>
179 * @tparam InputIt the iterator type
180 * @tparam UnaryPredicate
181 * @param first range start of elements to examine
182 * @param last range end of elements to examine, exclusive
183 * @param q unary predicate which returns ​false for the desired element.
184 * @return Iterator to the first element satisfying the condition or last if no such element is found.
185 */
186 template<class InputIt, class UnaryPredicate>
187 constexpr InputIt find_if_not(InputIt first, InputIt last, UnaryPredicate q) noexcept
188 {
189 for (; first != last; ++first) {
190 if (!q(*first)) {
191 return first;
192 }
193 }
194 return last; // implicit move since C++11
195 }
196
197 /**
198 * Identical to C++20 std::remove() of `algorithm`
199 *
200 * @tparam ForwardIt the iterator type
201 * @tparam UnaryPredicate
202 * @param first range start of elements to examine
203 * @param last range end of elements to examine, exclusive
204 * @param value the value to remove
205 * @return past-the end iterator for the new range of values.
206 */
207 template<class ForwardIt, class T>
208 ForwardIt remove(ForwardIt first, ForwardIt last, const T& value)
209 {
210 first = jau::find(first, last, value);
211 if (first != last) {
212 for(ForwardIt i = first; ++i != last; ) {
213 if ( *i != value ) {
214 *first++ = std::move(*i);
215 }
216 }
217 }
218 return first; // implicit move since C++11
219 }
220
221 /**
222 * Identical to C++20 std::remove_if() of `algorithm`
223 *
224 * @tparam ForwardIt the iterator type
225 * @tparam UnaryPredicate
226 * @param first range start of elements to examine
227 * @param last range end of elements to examine, exclusive
228 * @param p unary predicate which returns true for the desired element to be removed
229 * @return past-the end iterator for the new range of values.
230 */
231 template<class ForwardIt, class UnaryPredicate>
232 ForwardIt remove_if(ForwardIt first, ForwardIt last, UnaryPredicate p)
233 {
234 first = jau::find_if(first, last, p);
235 if (first != last) {
236 for(ForwardIt i = first; ++i != last; ) {
237 if (!p(*i)) {
238 *first++ = std::move(*i);
239 }
240 }
241 }
242 return first; // implicit move since C++11
243 }
244
245 /**
246 * Like std::for_each() of 'algorithm'
247 * <p>
248 * Only exists here as performance analysis over O(n*n) complexity
249 * exposes std::for_each() to be 'a little' slower for 1000 x 1000.<br>
250 * See test/test_cow_darray_perf01.cpp
251 * </p>
252 * @tparam InputIt the iterator type
253 * @tparam UnaryFunction
254 * @param first range start of elements to apply the function
255 * @param last range end of elements to apply the function
256 * @param f the function object, like <code>void fun(const Type &a)</code>
257 * @return the function
258 */
259 template<class InputIt, class UnaryFunction>
260 constexpr UnaryFunction for_each(InputIt first, InputIt last, UnaryFunction f)
261 {
262 for (; first != last; ++first) {
263 f(*first);
264 }
265 return f; // implicit move since C++11
266 }
267
268 /****************************************************************************************
269 ****************************************************************************************/
270
271 /**
272 * Like jau::for_each(), see above.
273 * <p>
274 * Additionally this template function removes
275 * the <code>const</code> qualifier
276 * of the <code>UnaryFunction</code> sole argument.<br>
277 * The latter is retrieved by dereferencing the iterator,
278 * which might expose the <code>const</code> qualifier if
279 * the iterator is a <code>const_iterator</code>.
280 * </p>
281 * <p>
282 * Implementation casts argument in the following fashion
283 * <code>const_cast<value_type*>(&arg)</code>,
284 * allowing to use <code>const_iterator</code> and subsequent
285 * non-const features of the argument, see below.
286 * </p>
287 * <p>
288 * Such situations may occur when preferring to use
289 * the <code>const_iterator</code> over non-const.<br>
290 * jau::cow_darray is such a scenario, where one might
291 * not mutate the elements of the container itself
292 * but needs to invoke non-const functions <i>in good faith</i>.<br>
293 * Here we can avoid costly side-effects of copying the CoW storage for later replacement.<br>
294 * See jau::cow_ro_iterator and jau::cow_rw_iterator
295 * in conjunction with jau::cow_darray.
296 * </p>
297 * <p>
298 * Requirements for the given IteratorIt type are to
299 * have typename <code>InputIt::value_type</code> available.
300 * </p>
301 * @tparam InputIt the iterator type, which might be a 'const_iterator' for non const types.
302 * @tparam UnaryFunction
303 * @param first range start of elements to apply the function
304 * @param last range end of elements to apply the function
305 * @param f the function object, like <code>void fun(const Type &a)</code>
306 * @return the function
307 * @see jau::cow_darray
308 * @see jau::cow_ro_iterator
309 * @see jau::cow_rw_iterator
310 */
311 template<class InputIt, class UnaryFunction>
312 constexpr UnaryFunction for_each_fidelity(InputIt first, InputIt last, UnaryFunction f)
313 {
314 typedef typename InputIt::value_type value_type;
315
316 for (; first != last; ++first) {
317 f( *const_cast<value_type*>( & (*first) ) );
318 }
319 return f; // implicit move since C++11
320 }
321
322 /****************************************************************************************
323 ****************************************************************************************/
324
325 /**
326 * Custom for_each template, same as jau::for_each but using a mutex.
327 * <p>
328 * Method performs UnaryFunction on all elements [first..last).
329 * </p>
330 * <p>
331 * This method also utilizes a given mutex to ensure thread-safety,
332 * by operating within an RAII-style std::lock_guard block.
333 * </p>
334 */
335 template<class Mutex, class InputIt, class UnaryFunction>
336 constexpr UnaryFunction for_each_mtx(Mutex &mtx, InputIt first, InputIt last, UnaryFunction f)
337 {
338 const std::lock_guard<Mutex> lock(mtx); // RAII-style acquire and relinquish via destructor
339
340 for (; first != last; ++first) {
341 f(*first);
342 }
343 return f; // implicit move since C++11
344 }
345
346 /**
347 * Custom for_each template, using indices instead of iterators,
348 * allowing container to be modified within lambda 'callback'.
349 * <p>
350 * Method performs UnaryFunction on all elements [0..n-1],
351 * where n is being retrieved once before the loop!
352 * </p>
353 */
354 template<class InputArray, class UnaryFunction>
355 constexpr UnaryFunction for_each_idx(InputArray &array, UnaryFunction f)
356 {
357 const typename InputArray::size_type size = array.size();
358 for (typename InputArray::size_type i = 0; i < size; ++i) {
359 f(array[i]);
360 }
361 return f; // implicit move since C++11
362 }
363
364 /**
365 * Custom for_each template,
366 * same as jau::for_each but using indices instead of iterators and a mutex.
367 * <p>
368 * Method performs UnaryFunction on all elements [0..n-1],
369 * where n is being retrieved once before the loop!
370 * </p>
371 * <p>
372 * This method also utilizes a given mutex to ensure thread-safety,
373 * by operating within an RAII-style std::lock_guard block.
374 * </p>
375 */
376 template<class Mutex, class InputArray, class UnaryFunction>
377 constexpr UnaryFunction for_each_idx_mtx(Mutex &mtx, InputArray &array, UnaryFunction f)
378 {
379 const std::lock_guard<Mutex> lock(mtx); // RAII-style acquire and relinquish via destructor
380
381 const typename InputArray::size_type size = array.size();
382 for (typename InputArray::size_type i = 0; i < size; ++i) {
383 f(array[i]);
384 }
385 return f; // implicit move since C++11
386 }
387
388 /****************************************************************************************
389 ****************************************************************************************/
390
391 template<class T>
392 const typename T::value_type * find_const(T& data, typename T::value_type const & elem,
393 std::enable_if_t< is_cow_type<T>::value, bool> = true ) noexcept
394 {
395 for (typename T::const_iterator first = data.cbegin(); !first.is_end(); ++first) {
396 if (*first == elem) {
397 return &(*first);
398 }
399 }
400 return nullptr;
401 }
402 template<class T>
403 const typename T::value_type * find_const(T& data, typename T::value_type const & elem,
404 std::enable_if_t< !is_cow_type<T>::value, bool> = true ) noexcept
405 {
406 typename T::const_iterator first = data.cbegin();
407 typename T::const_iterator last = data.cend();
408 for (; first != last; ++first) {
409 if (*first == elem) {
410 return &(*first);
411 }
412 }
413 return nullptr;
414 }
415
416 /****************************************************************************************
417 ****************************************************************************************/
418
419 template<class T, class UnaryFunction>
420 constexpr UnaryFunction for_each_const(T& data, UnaryFunction f,
421 std::enable_if_t< is_cow_type<T>::value, bool> = true ) noexcept
422 {
423 for (typename T::const_iterator first = data.cbegin(); !first.is_end(); ++first) {
424 f(*first);
425 }
426 return f; // implicit move since C++11
427 }
428 template<class T, class UnaryFunction>
429 constexpr UnaryFunction for_each_const(T& data, UnaryFunction f,
430 std::enable_if_t< !is_cow_type<T>::value, bool> = true ) noexcept
431 {
432 typename T::const_iterator first = data.cbegin();
433 typename T::const_iterator last = data.cend();
434 for (; first != last; ++first) {
435 f(*first);
436 }
437 return f; // implicit move since C++11
438 }
439
440 /****************************************************************************************
441 ****************************************************************************************/
442
443 /**
444 * See jau::for_each_fidelity()
445 */
446 template<class T, class UnaryFunction>
447 constexpr UnaryFunction for_each_fidelity(T& data, UnaryFunction f,
448 std::enable_if_t< is_cow_type<T>::value, bool> = true ) noexcept
449 {
450 for (typename T::const_iterator first = data.cbegin(); !first.is_end(); ++first) {
451 f( *const_cast<typename T::value_type*>( & (*first) ) );
452 }
453 return f; // implicit move since C++11
454 }
455 /**
456 * See jau::for_each_fidelity()
457 */
458 template<class T, class UnaryFunction>
459 constexpr UnaryFunction for_each_fidelity(T& data, UnaryFunction f,
460 std::enable_if_t< !is_cow_type<T>::value, bool> = true ) noexcept
461 {
462 typename T::const_iterator first = data.cbegin();
463 typename T::const_iterator last = data.cend();
464 for (; first != last; ++first) {
465 f( *const_cast<typename T::value_type*>( & (*first) ) );
466 }
467 return f; // implicit move since C++11
468 }
469
470 /****************************************************************************************
471 ****************************************************************************************/
472
473 template<typename T>
475 {
476 private:
477 bool m_owning;
478 public:
479 constexpr OptDeleter() noexcept : m_owning(true) {}
480 constexpr OptDeleter(bool owner) noexcept : m_owning(owner) {}
481 constexpr OptDeleter(const OptDeleter&) noexcept = default;
482 constexpr OptDeleter(OptDeleter&&) noexcept = default;
483 void operator()(T* p) const {
484 if( m_owning ) {
485 delete p;
486 }
487 }
488 };
489
490 /**@}*/
491
492} // namespace jau
493
494#endif /* JAU_BASIC_ALGOS_HPP_ */
constexpr OptDeleter(bool owner) noexcept
constexpr OptDeleter(const OptDeleter &) noexcept=default
constexpr OptDeleter() noexcept
constexpr OptDeleter(OptDeleter &&) noexcept=default
~call_on_release() noexcept
call_on_release(UnaryFunction release_func) noexcept
call_on_release & operator=(const call_on_release &)=delete
call_on_release(const call_on_release &)=delete
void set_released() noexcept
Mark the resource being orderly released, release_func() will not be called and use after free avoide...
call_on_release & operator=(const call_on_release &) volatile =delete
bool is_released() const noexcept
Query whethr the resource has been orderly released.
ForwardIt remove_if(ForwardIt first, ForwardIt last, UnaryPredicate p)
Identical to C++20 std::remove_if() of algorithm
const T::value_type * find_const(T &data, typename T::value_type const &elem, std::enable_if_t< is_cow_type< T >::value, bool >=true) noexcept
constexpr bool contains(InputArray &array, const T &value) noexcept
Return true if value is contained in array.
constexpr UnaryFunction for_each_idx_mtx(Mutex &mtx, InputArray &array, UnaryFunction f)
Custom for_each template, same as jau::for_each but using indices instead of iterators and a mutex.
constexpr InputIt find_if_not(InputIt first, InputIt last, UnaryPredicate q) noexcept
Like std::find_if_not() of 'algorithm'.
ForwardIt remove(ForwardIt first, ForwardIt last, const T &value)
Identical to C++20 std::remove() of algorithm
constexpr InputIt find(InputIt first, InputIt last, const T &value) noexcept
Like std::find() of 'algorithm'.
constexpr UnaryFunction for_each(InputIt first, InputIt last, UnaryFunction f)
Like std::for_each() of 'algorithm'.
constexpr UnaryFunction for_each_mtx(Mutex &mtx, InputIt first, InputIt last, UnaryFunction f)
Custom for_each template, same as jau::for_each but using a mutex.
constexpr UnaryFunction for_each_fidelity(InputIt first, InputIt last, UnaryFunction f)
Like jau::for_each(), see above.
constexpr bool eraseFirst(InputArray &array, const T &value)
constexpr UnaryFunction for_each_const(T &data, UnaryFunction f, std::enable_if_t< is_cow_type< T >::value, bool >=true) noexcept
constexpr UnaryFunction for_each_idx(InputArray &array, UnaryFunction f)
Custom for_each template, using indices instead of iterators, allowing container to be modified withi...
constexpr InputIt find_if(InputIt first, InputIt last, UnaryPredicate p) noexcept
Like std::find_if() of 'algorithm'.
ordered_atomic< bool, std::memory_order_seq_cst > sc_atomic_bool
SC atomic integral scalar boolean.
constexpr bool value(const Bool rhs) noexcept
__pack(...): Produces MSVC, clang and gcc compatible lead-in and -out macros.
Definition backtrace.hpp:32
template< class T > is_cow_type<T>::value compile-time Type Trait, determining whether the given temp...