In previous optimization in outliner I assumed that order in treehash was not important. But testing outliner in datablocks mode revealed a problem: when user expands multiple recursive levels and then closes any element, it always closed the top level of recursion. Now it should work fine with recursive trees. Now treehash contains groups of elements indexed by (id,nr,type). Adding an element with the same (id,nr,type) results in appending it to existing group. No duplicates are possible in treehash. This commit should also make lookups a little bit faster, because searching in small arrays by "used" is faster than searching in hashtable with duplicates by "id,nr,type,used".
174 lines
5.6 KiB
C++
174 lines
5.6 KiB
C++
/*
|
|
* ***** BEGIN GPL LICENSE BLOCK *****
|
|
*
|
|
* 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.
|
|
*
|
|
* The Original Code is Copyright (C) 2001-2002 by NaN Holding BV.
|
|
* All rights reserved.
|
|
*
|
|
* The Original Code is: all of this file.
|
|
*
|
|
* Contributor(s): none yet.
|
|
*
|
|
* ***** END GPL LICENSE BLOCK *****
|
|
*/
|
|
|
|
#ifndef __BLI_GHASH_H__
|
|
#define __BLI_GHASH_H__
|
|
|
|
/** \file BLI_ghash.h
|
|
* \ingroup bli
|
|
* \brief A general (pointer -> pointer) hash table ADT
|
|
*/
|
|
|
|
#include "BLI_sys_types.h" /* for bool */
|
|
|
|
#ifdef __cplusplus
|
|
extern "C" {
|
|
#endif
|
|
|
|
typedef unsigned int (*GHashHashFP) (const void *key);
|
|
typedef int (*GHashCmpFP) (const void *a, const void *b);
|
|
typedef void (*GHashKeyFreeFP) (void *key);
|
|
typedef void (*GHashValFreeFP) (void *val);
|
|
|
|
typedef struct GHash GHash;
|
|
|
|
typedef struct GHashIterator {
|
|
GHash *gh;
|
|
struct Entry *curEntry;
|
|
unsigned int curBucket;
|
|
} GHashIterator;
|
|
|
|
enum {
|
|
GHASH_FLAG_ALLOW_DUPES = (1 << 0), /* only checked for in debug mode */
|
|
};
|
|
|
|
/* *** */
|
|
|
|
GHash *BLI_ghash_new(GHashHashFP hashfp, GHashCmpFP cmpfp, const char *info);
|
|
void BLI_ghash_free(GHash *gh, GHashKeyFreeFP keyfreefp, GHashValFreeFP valfreefp);
|
|
void BLI_ghash_insert(GHash *gh, void *key, void *val);
|
|
void BLI_ghash_reinsert(GHash *gh, void *key, void *val, GHashKeyFreeFP keyfreefp, GHashValFreeFP valfreefp);
|
|
void *BLI_ghash_lookup(GHash *gh, const void *key);
|
|
void **BLI_ghash_lookup_p(GHash *gh, const void *key);
|
|
bool BLI_ghash_remove(GHash *gh, void *key, GHashKeyFreeFP keyfreefp, GHashValFreeFP valfreefp);
|
|
void BLI_ghash_clear(GHash *gh, GHashKeyFreeFP keyfreefp, GHashValFreeFP valfreefp);
|
|
void *BLI_ghash_pop(GHash *gh, void *key, GHashKeyFreeFP keyfreefp);
|
|
bool BLI_ghash_haskey(GHash *gh, const void *key);
|
|
int BLI_ghash_size(GHash *gh);
|
|
void BLI_ghash_flag_set(GHash *gh, unsigned short flag);
|
|
void BLI_ghash_flag_clear(GHash *gh, unsigned short flag);
|
|
|
|
/* *** */
|
|
|
|
/**
|
|
* Create a new GHashIterator. The hash table must not be mutated
|
|
* while the iterator is in use, and the iterator will step exactly
|
|
* BLI_ghash_size(gh) times before becoming done.
|
|
*
|
|
* \param gh The GHash to iterate over.
|
|
* \return Pointer to a new DynStr.
|
|
*/
|
|
GHashIterator *BLI_ghashIterator_new(GHash *gh);
|
|
/**
|
|
* Init an already allocated GHashIterator. The hash table must not
|
|
* be mutated while the iterator is in use, and the iterator will
|
|
* step exactly BLI_ghash_size(gh) times before becoming done.
|
|
*
|
|
* \param ghi The GHashIterator to initialize.
|
|
* \param gh The GHash to iterate over.
|
|
*/
|
|
void BLI_ghashIterator_init(GHashIterator *ghi, GHash *gh);
|
|
/**
|
|
* Free a GHashIterator.
|
|
*
|
|
* \param ghi The iterator to free.
|
|
*/
|
|
void BLI_ghashIterator_free(GHashIterator *ghi);
|
|
|
|
/**
|
|
* Retrieve the key from an iterator.
|
|
*
|
|
* \param ghi The iterator.
|
|
* \return The key at the current index, or NULL if the
|
|
* iterator is done.
|
|
*/
|
|
void *BLI_ghashIterator_getKey(GHashIterator *ghi);
|
|
/**
|
|
* Retrieve the value from an iterator.
|
|
*
|
|
* \param ghi The iterator.
|
|
* \return The value at the current index, or NULL if the
|
|
* iterator is done.
|
|
*/
|
|
void *BLI_ghashIterator_getValue(GHashIterator *ghi);
|
|
/**
|
|
* Steps the iterator to the next index.
|
|
*
|
|
* \param ghi The iterator.
|
|
*/
|
|
void BLI_ghashIterator_step(GHashIterator *ghi);
|
|
/**
|
|
* Determine if an iterator is done (has reached the end of
|
|
* the hash table).
|
|
*
|
|
* \param ghi The iterator.
|
|
* \return True if done, False otherwise.
|
|
*/
|
|
bool BLI_ghashIterator_done(GHashIterator *ghi);
|
|
|
|
#define GHASH_ITER(gh_iter_, ghash_) \
|
|
for (BLI_ghashIterator_init(&gh_iter_, ghash_); \
|
|
BLI_ghashIterator_done(&gh_iter_) == false; \
|
|
BLI_ghashIterator_step(&gh_iter_))
|
|
|
|
#define GHASH_ITER_INDEX(gh_iter_, ghash_, i_) \
|
|
for (BLI_ghashIterator_init(&gh_iter_, ghash_), i_ = 0; \
|
|
BLI_ghashIterator_done(&gh_iter_) == false; \
|
|
BLI_ghashIterator_step(&gh_iter_), i_++)
|
|
|
|
/* *** */
|
|
|
|
unsigned int BLI_ghashutil_ptrhash(const void *key);
|
|
int BLI_ghashutil_ptrcmp(const void *a, const void *b);
|
|
|
|
unsigned int BLI_ghashutil_strhash(const void *key);
|
|
int BLI_ghashutil_strcmp(const void *a, const void *b);
|
|
|
|
unsigned int BLI_ghashutil_inthash(const void *ptr);
|
|
int BLI_ghashutil_intcmp(const void *a, const void *b);
|
|
|
|
GHash *BLI_ghash_ptr_new(const char *info);
|
|
GHash *BLI_ghash_str_new(const char *info);
|
|
GHash *BLI_ghash_int_new(const char *info);
|
|
GHash *BLI_ghash_pair_new(const char *info);
|
|
|
|
typedef struct GHashPair {
|
|
const void *first;
|
|
const void *second;
|
|
} GHashPair;
|
|
|
|
GHashPair *BLI_ghashutil_pairalloc(const void *first, const void *second);
|
|
unsigned int BLI_ghashutil_pairhash(const void *ptr);
|
|
int BLI_ghashutil_paircmp(const void *a, const void *b);
|
|
void BLI_ghashutil_pairfree(void *ptr);
|
|
|
|
#ifdef __cplusplus
|
|
}
|
|
#endif
|
|
|
|
#endif /* __BLI_GHASH_H__ */
|