2011-02-18 13:58:08 +00:00
|
|
|
/*
|
2009-07-18 07:11:37 +00: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,
|
2010-02-12 13:34:04 +00:00
|
|
|
* Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
2009-07-18 07:11:37 +00:00
|
|
|
*
|
|
|
|
* The Original Code is Copyright (C) 2009 Blender Foundation, Joshua Leung
|
|
|
|
* All rights reserved.
|
|
|
|
*/
|
|
|
|
|
2012-02-17 18:59:41 +00:00
|
|
|
#pragma once
|
2009-07-18 07:11:37 +00:00
|
|
|
|
2019-02-18 08:08:12 +11:00
|
|
|
/** \file
|
|
|
|
* \ingroup bli
|
2021-12-09 20:01:44 +11:00
|
|
|
*
|
|
|
|
* Double-Linked Red-Black Tree Implementation:
|
2018-06-01 18:19:39 +02:00
|
|
|
*
|
2009-07-18 07:11:37 +00:00
|
|
|
* This is simply a Red-Black Tree implementation whose nodes can later
|
|
|
|
* be arranged + retrieved as elements in a Double-Linked list (i.e. ListBase).
|
|
|
|
* The Red-Black Tree implementation is based on the methods defined by Wikipedia.
|
|
|
|
*/
|
|
|
|
|
2021-12-09 20:01:44 +11:00
|
|
|
#ifdef __cplusplus
|
|
|
|
extern "C" {
|
|
|
|
#endif
|
|
|
|
|
2009-07-18 07:11:37 +00:00
|
|
|
/* ********************************************** */
|
2021-06-24 15:56:58 +10:00
|
|
|
/* Data Types and Type Defines */
|
2009-07-18 07:11:37 +00:00
|
|
|
|
2021-12-09 20:01:44 +11:00
|
|
|
/* -------------------------------------------------------------------- */
|
|
|
|
/** \name Base Structs
|
|
|
|
* \{ */
|
2009-07-18 07:11:37 +00:00
|
|
|
|
2021-12-20 19:01:14 +11:00
|
|
|
/** Basic Layout for a Node. */
|
2009-07-18 07:11:37 +00:00
|
|
|
typedef struct DLRBT_Node {
|
|
|
|
/* ListBase capabilities */
|
2012-10-21 05:46:41 +00:00
|
|
|
struct DLRBT_Node *next, *prev;
|
2018-06-17 16:32:54 +02:00
|
|
|
|
2009-07-18 07:11:37 +00:00
|
|
|
/* Tree Associativity settings */
|
|
|
|
struct DLRBT_Node *left, *right;
|
|
|
|
struct DLRBT_Node *parent;
|
2018-06-17 16:32:54 +02:00
|
|
|
|
2009-07-18 07:11:37 +00:00
|
|
|
char tree_col;
|
|
|
|
/* ... for nice alignment, next item should usually be a char too... */
|
|
|
|
} DLRBT_Node;
|
|
|
|
|
2021-12-20 19:01:14 +11:00
|
|
|
/** Red/Black defines for tree_col. */
|
2009-11-15 11:20:44 +00:00
|
|
|
typedef enum eDLRBT_Colors {
|
2012-05-12 20:39:39 +00:00
|
|
|
DLRBT_BLACK = 0,
|
2009-07-18 07:11:37 +00:00
|
|
|
DLRBT_RED,
|
2009-11-15 11:20:44 +00:00
|
|
|
} eDLRBT_Colors;
|
2009-07-18 07:11:37 +00:00
|
|
|
|
|
|
|
/* -------- */
|
|
|
|
|
2021-12-20 19:01:14 +11:00
|
|
|
/** The Tree Data. */
|
2009-07-18 07:11:37 +00:00
|
|
|
typedef struct DLRBT_Tree {
|
|
|
|
/* ListBase capabilities */
|
2012-05-12 20:39:39 +00:00
|
|
|
void *first, *last; /* these should be based on DLRBT_Node-s */
|
2009-07-18 07:11:37 +00:00
|
|
|
|
|
|
|
/* Root Node */
|
2012-05-12 20:39:39 +00:00
|
|
|
void *root; /* this should be based on DLRBT_Node-s */
|
2009-07-18 07:11:37 +00:00
|
|
|
} DLRBT_Tree;
|
|
|
|
|
2021-12-09 20:01:44 +11:00
|
|
|
/** \} */
|
|
|
|
|
|
|
|
/* -------------------------------------------------------------------- */
|
|
|
|
/** \name Callback Types
|
|
|
|
* \{ */
|
2009-11-15 11:20:44 +00:00
|
|
|
|
2021-12-09 20:01:44 +11:00
|
|
|
/**
|
|
|
|
* Return -1, 0, 1 for whether the given data is less than,
|
2019-04-22 00:54:27 +10:00
|
|
|
* equal to, or greater than the given node.
|
2021-12-09 20:01:44 +11:00
|
|
|
* \param node: <DLRBT_Node> the node to compare to.
|
|
|
|
* \param data: pointer to the relevant data or values stored in the bit-pattern.
|
|
|
|
* Dependent on the function.
|
2009-11-15 11:20:44 +00:00
|
|
|
*/
|
|
|
|
typedef short (*DLRBT_Comparator_FP)(void *node, void *data);
|
|
|
|
|
2021-12-09 20:01:44 +11:00
|
|
|
/**
|
|
|
|
* Return a new node instance wrapping the given data
|
|
|
|
* - data: Pointer to the relevant data to create a subclass of node from.
|
2009-11-15 11:20:44 +00:00
|
|
|
*/
|
|
|
|
typedef DLRBT_Node *(*DLRBT_NAlloc_FP)(void *data);
|
|
|
|
|
2021-12-09 20:01:44 +11:00
|
|
|
/**
|
|
|
|
* Update an existing node instance accordingly to be in sync with the given data.
|
|
|
|
* \param node: <DLRBT_Node> the node to update.
|
|
|
|
* \param data: Pointer to the relevant data or values stored in the bit-pattern.
|
|
|
|
* Dependent on the function.
|
2009-11-15 11:20:44 +00:00
|
|
|
*/
|
|
|
|
typedef void (*DLRBT_NUpdate_FP)(void *node, void *data);
|
|
|
|
|
GPencil: Update-on-write using update cache
This implements the update cache described in T95401.
The cache is currently only used for drawing strokes and
sculpting (using the push brush).
**Note: Making use of the cache throughout grease pencil will
have to be done incrementally in other patches. **
The update cache stores what elements have changed in the
original data-block since the last time the eval object
was updated. Additionally, the update cache can store multiple
updates to the data and minimizes the number of elements
that need to be copied.
Elements can be tagged using `BKE_gpencil_tag_full_update` and
`BKE_gpencil_tag_light_update`. A full update means that the element
itself will be copied but also all of the content inside. E.g. when a
layer is tagged for a full update, the layer, all the frames inside the
layer and all the strokes inside the frames will be copied.
A light update means that only the properties of the element are copied
without any of the content. E.g. if a layer is tagged with a light
update, it will copy the layer name, opacity, transform, etc.
When the update cache is in use (e.g. elements have been tagged) then
the depsgraph will not trigger a copy-on-write, but an update-on-write.
This means that the update cache will be used to determine what elements
have changed and then only those elements will be copied over to the
eval object.
If the update cache is empty or the data block was tagged with a full
update, we always fall back to a copy-on-write.
Currently, the update cache is only used by the active depsgraph. This
is because we need to free the update cache after an update-on-write so
it's reset and we need to make sure it is not freed or read by other
depsgraphs.
Co-authored-by: @yann-lty
This patch was contributed by The SPA Studios.
Reviewed By: sergey, antoniov, #dependency_graph, pepeland, mendio
Maniphest Tasks: T95401
Differential Revision: https://developer.blender.org/D13984
2022-02-10 11:34:12 +01:00
|
|
|
/**
|
|
|
|
* Free a node and the wrapped data.
|
|
|
|
* \param node: <DLRBT_Node> the node to free.
|
|
|
|
*/
|
|
|
|
typedef void (*DLRBT_NFree_FP)(void *node);
|
|
|
|
|
2009-07-18 07:11:37 +00:00
|
|
|
/* ********************************************** */
|
|
|
|
/* Public API */
|
|
|
|
|
2021-12-09 20:01:44 +11:00
|
|
|
/** \} */
|
2009-11-15 11:20:44 +00:00
|
|
|
|
2021-12-09 20:01:44 +11:00
|
|
|
/* -------------------------------------------------------------------- */
|
|
|
|
/** \name ADT Management
|
|
|
|
* \{ */
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Create a new tree, and initialize as necessary.
|
|
|
|
*/
|
2009-07-18 07:11:37 +00:00
|
|
|
DLRBT_Tree *BLI_dlrbTree_new(void);
|
|
|
|
|
2021-12-09 20:01:44 +11:00
|
|
|
/**
|
|
|
|
* Initializes some given trees.
|
|
|
|
* Just zero out the pointers used.
|
|
|
|
*/
|
2009-07-18 07:11:37 +00:00
|
|
|
void BLI_dlrbTree_init(DLRBT_Tree *tree);
|
|
|
|
|
2021-12-09 20:01:44 +11:00
|
|
|
/**
|
|
|
|
* Free the given tree's data but not the tree itself.
|
|
|
|
*/
|
GPencil: Update-on-write using update cache
This implements the update cache described in T95401.
The cache is currently only used for drawing strokes and
sculpting (using the push brush).
**Note: Making use of the cache throughout grease pencil will
have to be done incrementally in other patches. **
The update cache stores what elements have changed in the
original data-block since the last time the eval object
was updated. Additionally, the update cache can store multiple
updates to the data and minimizes the number of elements
that need to be copied.
Elements can be tagged using `BKE_gpencil_tag_full_update` and
`BKE_gpencil_tag_light_update`. A full update means that the element
itself will be copied but also all of the content inside. E.g. when a
layer is tagged for a full update, the layer, all the frames inside the
layer and all the strokes inside the frames will be copied.
A light update means that only the properties of the element are copied
without any of the content. E.g. if a layer is tagged with a light
update, it will copy the layer name, opacity, transform, etc.
When the update cache is in use (e.g. elements have been tagged) then
the depsgraph will not trigger a copy-on-write, but an update-on-write.
This means that the update cache will be used to determine what elements
have changed and then only those elements will be copied over to the
eval object.
If the update cache is empty or the data block was tagged with a full
update, we always fall back to a copy-on-write.
Currently, the update cache is only used by the active depsgraph. This
is because we need to free the update cache after an update-on-write so
it's reset and we need to make sure it is not freed or read by other
depsgraphs.
Co-authored-by: @yann-lty
This patch was contributed by The SPA Studios.
Reviewed By: sergey, antoniov, #dependency_graph, pepeland, mendio
Maniphest Tasks: T95401
Differential Revision: https://developer.blender.org/D13984
2022-02-10 11:34:12 +01:00
|
|
|
void BLI_dlrbTree_free(DLRBT_Tree *tree, DLRBT_NFree_FP free_cb);
|
2009-07-18 07:11:37 +00:00
|
|
|
|
2021-12-09 20:01:44 +11:00
|
|
|
/**
|
|
|
|
* Make sure the tree's Double-Linked list representation is valid.
|
|
|
|
*/
|
2009-07-18 07:11:37 +00:00
|
|
|
void BLI_dlrbTree_linkedlist_sync(DLRBT_Tree *tree);
|
|
|
|
|
2021-12-09 20:01:44 +11:00
|
|
|
/** \} */
|
|
|
|
|
|
|
|
/* -------------------------------------------------------------------- */
|
|
|
|
/** \name Tree Searching Utilities
|
|
|
|
* \{ */
|
2009-07-18 07:11:37 +00:00
|
|
|
|
2021-12-09 20:01:44 +11:00
|
|
|
/**
|
|
|
|
* Find the node which matches or is the closest to the requested node.
|
|
|
|
*/
|
2021-07-20 13:42:49 +02:00
|
|
|
DLRBT_Node *BLI_dlrbTree_search(const DLRBT_Tree *tree,
|
|
|
|
DLRBT_Comparator_FP cmp_cb,
|
|
|
|
void *search_data);
|
2009-11-15 11:20:44 +00:00
|
|
|
|
2021-12-09 20:01:44 +11:00
|
|
|
/**
|
|
|
|
* Find the node which exactly matches the required data.
|
|
|
|
*/
|
2021-07-20 13:42:49 +02:00
|
|
|
DLRBT_Node *BLI_dlrbTree_search_exact(const DLRBT_Tree *tree,
|
2009-11-15 11:20:44 +00:00
|
|
|
DLRBT_Comparator_FP cmp_cb,
|
|
|
|
void *search_data);
|
|
|
|
|
2021-12-09 20:01:44 +11:00
|
|
|
/**
|
|
|
|
* Find the node which occurs immediately before the best matching node.
|
|
|
|
*/
|
2021-07-20 13:42:49 +02:00
|
|
|
DLRBT_Node *BLI_dlrbTree_search_prev(const DLRBT_Tree *tree,
|
2009-11-15 11:20:44 +00:00
|
|
|
DLRBT_Comparator_FP cmp_cb,
|
|
|
|
void *search_data);
|
|
|
|
|
2021-12-09 20:01:44 +11:00
|
|
|
/**
|
|
|
|
* Find the node which occurs immediately after the best matching node.
|
|
|
|
*/
|
2021-07-20 13:42:49 +02:00
|
|
|
DLRBT_Node *BLI_dlrbTree_search_next(const DLRBT_Tree *tree,
|
2009-11-15 11:20:44 +00:00
|
|
|
DLRBT_Comparator_FP cmp_cb,
|
|
|
|
void *search_data);
|
|
|
|
|
2021-12-09 20:01:44 +11:00
|
|
|
/**
|
|
|
|
* Check whether there is a node matching the requested node.
|
|
|
|
*/
|
2009-11-15 11:20:44 +00:00
|
|
|
short BLI_dlrbTree_contains(DLRBT_Tree *tree, DLRBT_Comparator_FP cmp_cb, void *search_data);
|
|
|
|
|
2021-12-09 20:01:44 +11:00
|
|
|
/** \} */
|
|
|
|
|
|
|
|
/* -------------------------------------------------------------------- */
|
|
|
|
/** \name Node Operations (Managed)
|
|
|
|
* \{ */
|
|
|
|
|
|
|
|
/**
|
|
|
|
* These methods automate the process of adding/removing nodes from the BST,
|
|
|
|
* using the supplied data and callbacks.
|
2009-07-18 07:11:37 +00:00
|
|
|
*/
|
2009-11-15 11:20:44 +00:00
|
|
|
|
2021-12-09 20:01:44 +11:00
|
|
|
/**
|
|
|
|
* Add the given data to the tree, and return the node added.
|
|
|
|
* \note for duplicates, the update_cb is called (if available),
|
|
|
|
* and the existing node is returned.
|
|
|
|
*/
|
2018-06-17 16:32:54 +02:00
|
|
|
DLRBT_Node *BLI_dlrbTree_add(DLRBT_Tree *tree,
|
|
|
|
DLRBT_Comparator_FP cmp_cb,
|
2012-05-12 20:39:39 +00:00
|
|
|
DLRBT_NAlloc_FP new_cb,
|
|
|
|
DLRBT_NUpdate_FP update_cb,
|
|
|
|
void *data);
|
2009-07-18 07:11:37 +00:00
|
|
|
|
2021-12-09 20:01:44 +11:00
|
|
|
/* FIXME: this is not implemented yet. */
|
|
|
|
/**
|
|
|
|
* Remove the given element from the tree and balance again.
|
|
|
|
*/
|
2012-02-29 15:00:37 +00:00
|
|
|
// void BLI_dlrbTree_remove(DLRBT_Tree *tree, DLRBT_Node *node);
|
2009-07-18 07:11:37 +00:00
|
|
|
|
2021-12-09 20:01:44 +11:00
|
|
|
/** \} */
|
|
|
|
|
|
|
|
/* -------------------------------------------------------------------- */
|
|
|
|
/** \name Node Operations (Manual)
|
|
|
|
*
|
|
|
|
* These methods require custom code for creating BST nodes and adding them to the
|
2009-11-15 11:20:44 +00:00
|
|
|
* tree in special ways, such that the node can then be balanced.
|
|
|
|
*
|
2021-12-09 20:01:44 +11:00
|
|
|
* It is recommended that these methods are only used where the other method is too cumbersome.
|
|
|
|
* \{ */
|
2009-11-15 11:20:44 +00:00
|
|
|
|
2021-12-09 20:01:44 +11:00
|
|
|
/**
|
|
|
|
* Balance the tree after the given node has been added to it
|
2009-11-15 11:20:44 +00:00
|
|
|
* (using custom code, in the Binary Tree way).
|
|
|
|
*/
|
|
|
|
void BLI_dlrbTree_insert(DLRBT_Tree *tree, DLRBT_Node *node);
|
|
|
|
|
2021-12-14 15:49:31 +11:00
|
|
|
/** \} */
|
|
|
|
|
2020-03-02 15:04:53 +01:00
|
|
|
#ifdef __cplusplus
|
|
|
|
}
|
|
|
|
#endif
|