mirror of
https://git.proxmox.com/git/libgit2
synced 2025-06-26 17:44:59 +00:00

A new method 'git_repository_object_free' allows to manually force the freeing of a repository object, even though they are still automatically managed by the repository and don't need to be freed by the user. Signed-off-by: Vicent Marti <tanoku@gmail.com>
250 lines
5.4 KiB
C
250 lines
5.4 KiB
C
/*
|
|
* This file is free software; you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License, version 2,
|
|
* as published by the Free Software Foundation.
|
|
*
|
|
* In addition to the permissions in the GNU General Public License,
|
|
* the authors give you unlimited permission to link the compiled
|
|
* version of this file into combinations with other programs,
|
|
* and to distribute those combinations without any restriction
|
|
* coming from the use of this file. (The General Public License
|
|
* restrictions do apply in other respects; for example, they cover
|
|
* modification of the file, and distribution when not linked into
|
|
* a combined executable.)
|
|
*
|
|
* This file 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; see the file COPYING. If not, write to
|
|
* the Free Software Foundation, 51 Franklin Street, Fifth Floor,
|
|
* Boston, MA 02110-1301, USA.
|
|
*/
|
|
|
|
#include "common.h"
|
|
#include "repository.h"
|
|
#include "commit.h"
|
|
|
|
static const int default_table_size = 32;
|
|
static const double max_load_factor = 0.65;
|
|
|
|
static void hashtable_resize(git_hashtable *table)
|
|
{
|
|
git_hashtable_node **new_nodes;
|
|
unsigned int new_size, i;
|
|
|
|
assert(table);
|
|
|
|
new_size = (table->size_mask + 1) * 2;
|
|
|
|
new_nodes = git__malloc(new_size * sizeof(git_hashtable_node *));
|
|
if (new_nodes == NULL)
|
|
return;
|
|
|
|
memset(new_nodes, 0x0, new_size * sizeof(git_hashtable_node *));
|
|
|
|
for (i = 0; i <= table->size_mask; ++i) {
|
|
git_hashtable_node *n;
|
|
unsigned int index;
|
|
|
|
while ((n = table->nodes[i]) != NULL) {
|
|
table->nodes[i] = n->next;
|
|
index = n->hash & (new_size - 1);
|
|
n->next = new_nodes[index];
|
|
new_nodes[index] = n;
|
|
}
|
|
}
|
|
|
|
free(table->nodes);
|
|
table->nodes = new_nodes;
|
|
table->size_mask = (new_size - 1);
|
|
table->max_count = (unsigned int)(new_size * max_load_factor);
|
|
}
|
|
|
|
git_hashtable *git_hashtable_alloc(unsigned int min_size,
|
|
git_hash_ptr hash,
|
|
git_hash_keyeq_ptr key_eq)
|
|
{
|
|
unsigned int i;
|
|
git_hashtable *table;
|
|
|
|
assert(hash && key_eq);
|
|
|
|
if ((table = git__malloc(sizeof(git_hashtable))) == NULL)
|
|
return NULL;
|
|
|
|
/* round up size to closest power of 2 */
|
|
min_size--;
|
|
min_size |= min_size >> 1;
|
|
min_size |= min_size >> 2;
|
|
min_size |= min_size >> 4;
|
|
min_size |= min_size >> 8;
|
|
min_size |= min_size >> 16;
|
|
|
|
table->size_mask = min_size;
|
|
table->count = 0;
|
|
table->max_count = (unsigned int)((min_size + 1) * max_load_factor);
|
|
|
|
table->hash = hash;
|
|
table->key_equal = key_eq;
|
|
|
|
table->nodes = git__malloc((min_size + 1) * sizeof(git_hashtable_node *));
|
|
|
|
if (table->nodes == NULL) {
|
|
free(table);
|
|
return NULL;
|
|
}
|
|
|
|
for (i = 0; i <= min_size; ++i)
|
|
table->nodes[i] = NULL;
|
|
|
|
return table;
|
|
}
|
|
|
|
void git_hashtable_clear(git_hashtable *table)
|
|
{
|
|
unsigned int index;
|
|
|
|
assert(table);
|
|
|
|
for (index = 0; index <= table->size_mask; ++index) {
|
|
git_hashtable_node *node, *next_node;
|
|
|
|
node = table->nodes[index];
|
|
while (node != NULL) {
|
|
next_node = node->next;
|
|
free(node);
|
|
node = next_node;
|
|
}
|
|
|
|
table->nodes[index] = NULL;
|
|
}
|
|
|
|
table->count = 0;
|
|
}
|
|
|
|
void git_hashtable_free(git_hashtable *table)
|
|
{
|
|
assert(table);
|
|
|
|
git_hashtable_clear(table);
|
|
free(table->nodes);
|
|
free(table);
|
|
}
|
|
|
|
|
|
int git_hashtable_insert(git_hashtable *table, const void *key, void *value)
|
|
{
|
|
git_hashtable_node *node;
|
|
uint32_t index, hash;
|
|
|
|
assert(table);
|
|
|
|
if (table->count + 1 > table->max_count)
|
|
hashtable_resize(table);
|
|
|
|
node = git__malloc(sizeof(git_hashtable_node));
|
|
if (node == NULL)
|
|
return GIT_ENOMEM;
|
|
|
|
hash = table->hash(key);
|
|
index = (hash & table->size_mask);
|
|
|
|
node->object = value;
|
|
node->hash = hash;
|
|
node->next = table->nodes[index];
|
|
|
|
table->nodes[index] = node;
|
|
table->count++;
|
|
|
|
return GIT_SUCCESS;
|
|
}
|
|
|
|
void *git_hashtable_lookup(git_hashtable *table, const void *key)
|
|
{
|
|
git_hashtable_node *node;
|
|
uint32_t index, hash;
|
|
|
|
assert(table);
|
|
|
|
hash = table->hash(key);
|
|
index = (hash & table->size_mask);
|
|
node = table->nodes[index];
|
|
|
|
while (node != NULL) {
|
|
if (node->hash == hash && table->key_equal(node->object, key))
|
|
return node->object;
|
|
|
|
node = node->next;
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
int git_hashtable_remove(git_hashtable *table, const void *key)
|
|
{
|
|
git_hashtable_node *node, *prev_node;
|
|
uint32_t index, hash;
|
|
|
|
assert(table);
|
|
|
|
hash = table->hash(key);
|
|
index = (hash & table->size_mask);
|
|
node = table->nodes[index];
|
|
|
|
prev_node = NULL;
|
|
|
|
while (node != NULL) {
|
|
if (node->hash == hash && table->key_equal(node->object, key)) {
|
|
if (prev_node == NULL)
|
|
table->nodes[index] = node->next;
|
|
else
|
|
prev_node->next = node->next;
|
|
|
|
free(node);
|
|
return GIT_SUCCESS;
|
|
}
|
|
|
|
prev_node = node;
|
|
node = node->next;
|
|
}
|
|
|
|
return GIT_ENOTFOUND;
|
|
}
|
|
|
|
|
|
|
|
void git_hashtable_iterator_init(git_hashtable *table, git_hashtable_iterator *it)
|
|
{
|
|
assert(table && it);
|
|
|
|
memset(it, 0x0, sizeof(git_hashtable_iterator));
|
|
|
|
it->nodes = table->nodes;
|
|
it->current_node = NULL;
|
|
it->current_pos = 0;
|
|
it->size = table->size_mask + 1;
|
|
}
|
|
|
|
void *git_hashtable_iterator_next(git_hashtable_iterator *it)
|
|
{
|
|
git_hashtable_node *next = NULL;
|
|
|
|
assert(it);
|
|
|
|
while (it->current_node == NULL) {
|
|
if (it->current_pos >= it->size)
|
|
return NULL;
|
|
|
|
it->current_node = it->nodes[it->current_pos++];
|
|
}
|
|
|
|
next = it->current_node;
|
|
it->current_node = it->current_node->next;
|
|
|
|
return next->object;
|
|
}
|
|
|