2023-08-16 00:20:26 +10:00
|
|
|
/* SPDX-FileCopyrightText: 2023 Blender Authors
|
2023-05-31 16:19:06 +02:00
|
|
|
*
|
|
|
|
|
* SPDX-License-Identifier: GPL-2.0-or-later */
|
2016-05-30 15:25:36 +10:00
|
|
|
|
|
|
|
|
#pragma once
|
|
|
|
|
|
2019-02-18 08:08:12 +11:00
|
|
|
/** \file
|
|
|
|
|
* \ingroup bli
|
|
|
|
|
* \brief Efficient in-memory storage of multiple similar arrays.
|
2016-05-30 15:25:36 +10:00
|
|
|
*/
|
|
|
|
|
|
2025-01-07 12:39:13 +01:00
|
|
|
#include "BLI_sys_types.h"
|
|
|
|
|
|
2016-05-30 15:25:36 +10:00
|
|
|
typedef struct BArrayState BArrayState;
|
2019-01-28 21:08:24 +11:00
|
|
|
typedef struct BArrayStore BArrayStore;
|
2016-05-30 15:25:36 +10:00
|
|
|
|
2021-12-09 20:01:44 +11:00
|
|
|
/**
|
|
|
|
|
* Create a new array store, which can store any number of arrays
|
|
|
|
|
* as long as their stride matches.
|
|
|
|
|
*
|
|
|
|
|
* \param stride: `sizeof()` each element,
|
|
|
|
|
*
|
|
|
|
|
* \note while a stride of `1` will always work,
|
|
|
|
|
* its less efficient since duplicate chunks of memory will be searched
|
|
|
|
|
* at positions unaligned with the array data.
|
|
|
|
|
*
|
|
|
|
|
* \param chunk_count: Number of elements to split each chunk into.
|
|
|
|
|
* - A small value increases the ability to de-duplicate chunks,
|
|
|
|
|
* but adds overhead by increasing the number of chunks to look up when searching for duplicates,
|
|
|
|
|
* as well as some overhead constructing the original array again, with more calls to `memcpy`.
|
|
|
|
|
* - Larger values reduce the *book keeping* overhead,
|
|
|
|
|
* but increase the chance a small,
|
|
|
|
|
* isolated change will cause a larger amount of data to be duplicated.
|
|
|
|
|
*
|
|
|
|
|
* \return A new array store, to be freed with #BLI_array_store_destroy.
|
|
|
|
|
*/
|
2016-05-30 15:25:36 +10:00
|
|
|
BArrayStore *BLI_array_store_create(unsigned int stride, unsigned int chunk_count);
|
2021-12-09 20:01:44 +11:00
|
|
|
/**
|
|
|
|
|
* Free the #BArrayStore, including all states and chunks.
|
|
|
|
|
*/
|
2016-05-30 15:25:36 +10:00
|
|
|
void BLI_array_store_destroy(BArrayStore *bs);
|
2021-12-09 20:01:44 +11:00
|
|
|
/**
|
|
|
|
|
* Clear all contents, allowing reuse of \a bs.
|
|
|
|
|
*/
|
2016-05-30 15:25:36 +10:00
|
|
|
void BLI_array_store_clear(BArrayStore *bs);
|
|
|
|
|
|
2021-12-09 20:01:44 +11:00
|
|
|
/**
|
|
|
|
|
* Find the memory used by all states (expanded & real).
|
|
|
|
|
*
|
|
|
|
|
* \return the total amount of memory that would be used by getting the arrays for all states.
|
|
|
|
|
*/
|
2016-05-30 15:25:36 +10:00
|
|
|
size_t BLI_array_store_calc_size_expanded_get(const BArrayStore *bs);
|
2021-12-09 20:01:44 +11:00
|
|
|
/**
|
|
|
|
|
* \return the amount of memory used by all #BChunk.data
|
|
|
|
|
* (duplicate chunks are only counted once).
|
|
|
|
|
*/
|
2016-05-30 15:25:36 +10:00
|
|
|
size_t BLI_array_store_calc_size_compacted_get(const BArrayStore *bs);
|
2019-04-17 06:17:24 +02:00
|
|
|
|
2021-12-09 20:01:44 +11:00
|
|
|
/**
|
|
|
|
|
* \param data: Data used to create
|
|
|
|
|
* \param state_reference: The state to use as a reference when adding the new state,
|
|
|
|
|
* typically this is the previous state,
|
|
|
|
|
* however it can be any previously created state from this \a bs.
|
|
|
|
|
*
|
|
|
|
|
* \return The new state,
|
|
|
|
|
* which is used by the caller as a handle to get back the contents of \a data.
|
|
|
|
|
* This may be removed using #BLI_array_store_state_remove,
|
|
|
|
|
* otherwise it will be removed with #BLI_array_store_destroy.
|
|
|
|
|
*/
|
2016-05-30 15:25:36 +10:00
|
|
|
BArrayState *BLI_array_store_state_add(BArrayStore *bs,
|
|
|
|
|
const void *data,
|
2022-01-12 12:49:36 +01:00
|
|
|
size_t data_len,
|
2016-05-30 15:25:36 +10:00
|
|
|
const BArrayState *state_reference);
|
2021-12-09 20:01:44 +11:00
|
|
|
/**
|
|
|
|
|
* Remove a state and free any unused #BChunk data.
|
|
|
|
|
*
|
|
|
|
|
* The states can be freed in any order.
|
|
|
|
|
*/
|
2016-05-30 15:25:36 +10:00
|
|
|
void BLI_array_store_state_remove(BArrayStore *bs, BArrayState *state);
|
2019-04-17 06:17:24 +02:00
|
|
|
|
2021-12-09 20:01:44 +11:00
|
|
|
/**
|
|
|
|
|
* \return the expanded size of the array,
|
|
|
|
|
* use this to know how much memory to allocate #BLI_array_store_state_data_get's argument.
|
|
|
|
|
*/
|
2025-05-22 17:16:36 +10:00
|
|
|
size_t BLI_array_store_state_size_get(const BArrayState *state);
|
2021-12-09 20:01:44 +11:00
|
|
|
/**
|
|
|
|
|
* Fill in existing allocated memory with the contents of \a state.
|
|
|
|
|
*/
|
2024-04-17 11:36:38 +10:00
|
|
|
void BLI_array_store_state_data_get(const BArrayState *state, void *data);
|
2021-12-09 20:01:44 +11:00
|
|
|
/**
|
|
|
|
|
* Allocate an array for \a state and return it.
|
|
|
|
|
*/
|
2025-05-22 17:16:36 +10:00
|
|
|
void *BLI_array_store_state_data_get_alloc(const BArrayState *state, size_t *r_data_len);
|
2016-05-30 15:25:36 +10:00
|
|
|
|
2021-12-09 20:01:44 +11:00
|
|
|
/**
|
|
|
|
|
* \note Only for tests.
|
|
|
|
|
*/
|
2016-05-30 15:25:36 +10:00
|
|
|
bool BLI_array_store_is_valid(BArrayStore *bs);
|
2025-05-27 05:48:48 +00:00
|
|
|
|
|
|
|
|
/* `array_store_rle.cc` */
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Return a run-length encoded copy of `data_dec`.
|
|
|
|
|
*
|
|
|
|
|
* \param data_dec: The data to encode.
|
|
|
|
|
* \param data_dec_len: The size of the data to encode.
|
|
|
|
|
* \param data_enc_extra_size: Allocate extra memory at the beginning of the array.
|
|
|
|
|
* - This doesn't impact the value of `r_data_enc_len`.
|
|
|
|
|
* - This must be skipped when decoding.
|
|
|
|
|
* \param r_data_enc_len: The size of the resulting RLE encoded data.
|
|
|
|
|
*/
|
|
|
|
|
uint8_t *BLI_array_store_rle_encode(const uint8_t *data_dec,
|
|
|
|
|
size_t data_dec_len,
|
|
|
|
|
size_t data_enc_extra_size,
|
|
|
|
|
size_t *r_data_enc_len);
|
|
|
|
|
/**
|
|
|
|
|
* Decode a run-length encoded array, writing the result into `data_dec_v`.
|
|
|
|
|
*
|
|
|
|
|
* \param data_enc: The data to encode (returned by #BLI_array_store_rle_encode).
|
|
|
|
|
* \param data_enc_len: The size of `data_enc`.
|
|
|
|
|
* \param data_dec: The destination for the decoded data to be written to.
|
|
|
|
|
* \param data_dec_len: The size of the destination (as passed to #BLI_array_store_rle_encode).
|
|
|
|
|
*/
|
|
|
|
|
void BLI_array_store_rle_decode(const uint8_t *data_enc,
|
|
|
|
|
const size_t data_enc_len,
|
|
|
|
|
void *data_dec_v,
|
|
|
|
|
const size_t data_dec_len);
|