| 
									
										
										
										
											2015-02-14 07:28:15 +11:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  * This program is free software; you can redistribute it and/or | 
					
						
							|  |  |  |  * modify it under the terms of the GNU General Public License | 
					
						
							|  |  |  |  * as published by the Free Software Foundation; either version 2 | 
					
						
							|  |  |  |  * of the License, or (at your option) any later version. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * This program is distributed in the hope that it will be useful, | 
					
						
							|  |  |  |  * but WITHOUT ANY WARRANTY; without even the implied warranty of | 
					
						
							|  |  |  |  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | 
					
						
							|  |  |  |  * GNU General Public License for more details. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * You should have received a copy of the GNU General Public License | 
					
						
							|  |  |  |  * along with this program; if not, write to the Free Software Foundation, | 
					
						
							|  |  |  |  * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-07 09:50:34 +02:00
										 |  |  | #pragma once
 | 
					
						
							| 
									
										
										
										
											2015-02-14 07:28:15 +11:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-18 08:08:12 +11:00
										 |  |  | /** \file
 | 
					
						
							|  |  |  |  * \ingroup bli | 
					
						
							|  |  |  |  * \brief Generic array manipulation API. | 
					
						
							| 
									
										
										
										
											2015-02-14 07:28:15 +11:00
										 |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-01-19 07:00:55 +11:00
										 |  |  | #include "BLI_compiler_typecheck.h"
 | 
					
						
							| 
									
										
										
										
											2020-03-06 16:45:06 +01:00
										 |  |  | #include "BLI_sys_types.h"
 | 
					
						
							| 
									
										
										
										
											2016-01-19 07:00:55 +11:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-02 15:04:53 +01:00
										 |  |  | #ifdef __cplusplus
 | 
					
						
							|  |  |  | extern "C" { | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-12-09 20:01:44 +11:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * In-place array reverse. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Access via #BLI_array_reverse | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2021-07-09 13:37:52 +10:00
										 |  |  | void _bli_array_reverse(void *arr, uint arr_len, size_t arr_stride); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | #define BLI_array_reverse(arr, arr_len) _bli_array_reverse(arr, arr_len, sizeof(*(arr)))
 | 
					
						
							| 
									
										
										
										
											2015-02-14 07:28:15 +11:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-12-09 20:01:44 +11:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * In-place array wrap. | 
					
						
							|  |  |  |  * (rotate the array one step forward or backwards). | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Access via #BLI_array_wrap | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2021-07-09 13:37:52 +10:00
										 |  |  | void _bli_array_wrap(void *arr, uint arr_len, size_t arr_stride, int dir); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | #define BLI_array_wrap(arr, arr_len, dir) _bli_array_wrap(arr, arr_len, sizeof(*(arr)), dir)
 | 
					
						
							| 
									
										
										
										
											2015-02-14 07:28:15 +11:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-12-09 20:01:44 +11:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  *In-place array permute. | 
					
						
							|  |  |  |  * (re-arrange elements based on an array of indices). | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Access via #BLI_array_wrap | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2021-07-09 13:37:52 +10:00
										 |  |  | void _bli_array_permute( | 
					
						
							| 
									
										
										
										
											2022-01-12 12:49:36 +01:00
										 |  |  |     void *arr, uint arr_len, size_t arr_stride, const uint *order, void *arr_temp); | 
					
						
							| 
									
										
										
										
											2015-04-28 04:10:31 +10:00
										 |  |  | #define BLI_array_permute(arr, arr_len, order) \
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   _bli_array_permute(arr, arr_len, sizeof(*(arr)), order, NULL) | 
					
						
							| 
									
										
										
										
											2018-08-01 16:12:36 +02:00
										 |  |  | #define BLI_array_permute_ex(arr, arr_len, order, arr_temp) \
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   _bli_array_permute(arr, arr_len, sizeof(*(arr)), order, arr_temp) | 
					
						
							| 
									
										
										
										
											2015-04-28 04:10:31 +10:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-12-09 20:01:44 +11:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * In-place array de-duplication of an ordered array. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * \return The new length of the array. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Access via #BLI_array_deduplicate_ordered | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2021-07-09 13:37:52 +10:00
										 |  |  | uint _bli_array_deduplicate_ordered(void *arr, uint arr_len, size_t arr_stride); | 
					
						
							| 
									
										
										
										
											2021-07-09 13:33:36 +10:00
										 |  |  | #define BLI_array_deduplicate_ordered(arr, arr_len) \
 | 
					
						
							|  |  |  |   _bli_array_deduplicate_ordered(arr, arr_len, sizeof(*(arr))) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-12-09 20:01:44 +11:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * Find the first index of an item in an array. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Access via #BLI_array_findindex | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * \note Not efficient, use for error checks/asserts. | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2021-07-09 13:37:52 +10:00
										 |  |  | int _bli_array_findindex(const void *arr, uint arr_len, size_t arr_stride, const void *p); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | #define BLI_array_findindex(arr, arr_len, p) _bli_array_findindex(arr, arr_len, sizeof(*(arr)), p)
 | 
					
						
							| 
									
										
										
										
											2015-02-14 07:28:15 +11:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-12-09 20:01:44 +11:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * A version of #BLI_array_findindex that searches from the end of the list. | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2021-07-09 13:37:52 +10:00
										 |  |  | int _bli_array_rfindindex(const void *arr, uint arr_len, size_t arr_stride, const void *p); | 
					
						
							| 
									
										
										
										
											2016-06-23 11:23:31 +10:00
										 |  |  | #define BLI_array_rfindindex(arr, arr_len, p) \
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   _bli_array_rfindindex(arr, arr_len, sizeof(*(arr)), p) | 
					
						
							| 
									
										
										
										
											2016-06-23 11:23:31 +10:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-01-19 07:00:55 +11:00
										 |  |  | void _bli_array_binary_and( | 
					
						
							| 
									
										
										
										
											2021-07-09 13:37:52 +10:00
										 |  |  |     void *arr, const void *arr_a, const void *arr_b, uint arr_len, size_t arr_stride); | 
					
						
							| 
									
										
										
										
											2016-01-19 07:00:55 +11:00
										 |  |  | #define BLI_array_binary_and(arr, arr_a, arr_b, arr_len) \
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   (CHECK_TYPE_PAIR_INLINE(*(arr), *(arr_a)), \ | 
					
						
							|  |  |  |    CHECK_TYPE_PAIR_INLINE(*(arr), *(arr_b)), \ | 
					
						
							|  |  |  |    _bli_array_binary_and(arr, arr_a, arr_b, arr_len, sizeof(*(arr)))) | 
					
						
							| 
									
										
										
										
											2016-01-19 07:00:55 +11:00
										 |  |  | 
 | 
					
						
							|  |  |  | void _bli_array_binary_or( | 
					
						
							| 
									
										
										
										
											2021-07-09 13:37:52 +10:00
										 |  |  |     void *arr, const void *arr_a, const void *arr_b, uint arr_len, size_t arr_stride); | 
					
						
							| 
									
										
										
										
											2016-01-19 07:00:55 +11:00
										 |  |  | #define BLI_array_binary_or(arr, arr_a, arr_b, arr_len) \
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |   (CHECK_TYPE_PAIR_INLINE(*(arr), *(arr_a)), \ | 
					
						
							|  |  |  |    CHECK_TYPE_PAIR_INLINE(*(arr), *(arr_b)), \ | 
					
						
							|  |  |  |    _bli_array_binary_or(arr, arr_a, arr_b, arr_len, sizeof(*(arr)))) | 
					
						
							| 
									
										
										
										
											2016-01-19 07:00:55 +11:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-12-09 20:01:44 +11:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * Utility function to iterate over contiguous items in an array. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * \param use_wrap: Detect contiguous ranges across the first/last points. | 
					
						
							|  |  |  |  * In this case the second index of \a span_step may be lower than the first, | 
					
						
							|  |  |  |  * which indicates the values are wrapped. | 
					
						
							|  |  |  |  * \param use_delimit_bounds: When false, | 
					
						
							|  |  |  |  * ranges that defined by the start/end indices are excluded. | 
					
						
							|  |  |  |  * This option has no effect when \a use_wrap is enabled. | 
					
						
							|  |  |  |  * \param test_fn: Function to test if the item should be included in the range. | 
					
						
							|  |  |  |  * \param user_data: User data for \a test_fn. | 
					
						
							|  |  |  |  * \param span_step: Indices to iterate over, | 
					
						
							|  |  |  |  * initialize both values to the array length to initialize iteration. | 
					
						
							|  |  |  |  * \param r_span_len: The length of the span, useful when \a use_wrap is enabled, | 
					
						
							|  |  |  |  * where calculating the length isn't a simple subtraction. | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | bool _bli_array_iter_span(const void *arr, | 
					
						
							| 
									
										
										
										
											2021-07-09 13:37:52 +10:00
										 |  |  |                           uint arr_len, | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |                           size_t arr_stride, | 
					
						
							|  |  |  |                           bool use_wrap, | 
					
						
							|  |  |  |                           bool use_delimit_bounds, | 
					
						
							|  |  |  |                           bool (*test_fn)(const void *arr_item, void *user_data), | 
					
						
							|  |  |  |                           void *user_data, | 
					
						
							| 
									
										
										
										
											2021-07-09 13:37:52 +10:00
										 |  |  |                           uint span_step[2], | 
					
						
							|  |  |  |                           uint *r_span_len); | 
					
						
							| 
									
										
										
										
											2018-03-28 08:59:26 +02:00
										 |  |  | #define BLI_array_iter_span( \
 | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  |     arr, arr_len, use_wrap, use_delimit_bounds, test_fn, user_data, span_step, r_span_len) \ | 
					
						
							|  |  |  |   _bli_array_iter_span(arr, \ | 
					
						
							|  |  |  |                        arr_len, \ | 
					
						
							|  |  |  |                        sizeof(*(arr)), \ | 
					
						
							|  |  |  |                        use_wrap, \ | 
					
						
							|  |  |  |                        use_delimit_bounds, \ | 
					
						
							|  |  |  |                        test_fn, \ | 
					
						
							|  |  |  |                        user_data, \ | 
					
						
							|  |  |  |                        span_step, \ | 
					
						
							|  |  |  |                        r_span_len) | 
					
						
							| 
									
										
										
										
											2016-05-02 18:01:00 +10:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-12-09 20:01:44 +11:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * Simple utility to check memory is zeroed. | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2021-07-09 13:37:52 +10:00
										 |  |  | bool _bli_array_is_zeroed(const void *arr, uint arr_len, size_t arr_stride); | 
					
						
							| 
									
										
										
										
											2019-04-17 06:17:24 +02:00
										 |  |  | #define BLI_array_is_zeroed(arr, arr_len) _bli_array_is_zeroed(arr, arr_len, sizeof(*(arr)))
 | 
					
						
							| 
									
										
										
										
											2018-03-28 08:59:26 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-12-09 20:01:44 +11:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * Smart function to sample a rectangle spiraling outside. | 
					
						
							|  |  |  |  * Nice for selection ID. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * \param arr_shape: dimensions [w, h]. | 
					
						
							|  |  |  |  * \param center: coordinates [x, y] indicating where to start traversing. | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2021-03-09 16:02:40 -03:00
										 |  |  | bool _bli_array_iter_spiral_square(const void *arr_v, | 
					
						
							|  |  |  |                                    const int arr_shape[2], | 
					
						
							| 
									
										
										
										
											2022-01-12 12:49:36 +01:00
										 |  |  |                                    size_t elem_size, | 
					
						
							| 
									
										
										
										
											2021-03-09 16:02:40 -03:00
										 |  |  |                                    const int center[2], | 
					
						
							| 
									
										
										
										
											2021-03-10 15:35:43 +11:00
										 |  |  |                                    bool (*test_fn)(const void *arr_item, void *user_data), | 
					
						
							| 
									
										
										
										
											2021-03-09 16:02:40 -03:00
										 |  |  |                                    void *user_data); | 
					
						
							|  |  |  | #define BLI_array_iter_spiral_square(arr, arr_shape, center, test_fn, user_data) \
 | 
					
						
							|  |  |  |   _bli_array_iter_spiral_square(arr, arr_shape, sizeof(*(arr)), center, test_fn, user_data) | 
					
						
							| 
									
										
										
										
											2020-03-02 15:04:53 +01:00
										 |  |  | #ifdef __cplusplus
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | #endif
 |