mirror of
				https://github.com/KevinMidboe/linguist.git
				synced 2025-10-29 17:50:22 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			100 lines
		
	
	
		
			2.2 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			100 lines
		
	
	
		
			2.2 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
/*
 | 
						|
 * Copyright (C) 2009-2012 the libgit2 contributors
 | 
						|
 *
 | 
						|
 * This file is part of libgit2, distributed under the GNU GPL v2 with
 | 
						|
 * a Linking Exception. For full terms see the included COPYING file.
 | 
						|
 */
 | 
						|
 | 
						|
#include "common.h"
 | 
						|
#include "repository.h"
 | 
						|
#include "commit.h"
 | 
						|
#include "thread-utils.h"
 | 
						|
#include "util.h"
 | 
						|
#include "cache.h"
 | 
						|
 | 
						|
int git_cache_init(git_cache *cache, size_t size, git_cached_obj_freeptr free_ptr)
 | 
						|
{
 | 
						|
	if (size < 8)
 | 
						|
		size = 8;
 | 
						|
	size = git__size_t_powerof2(size);
 | 
						|
 | 
						|
	cache->size_mask = size - 1;
 | 
						|
	cache->lru_count = 0;
 | 
						|
	cache->free_obj = free_ptr;
 | 
						|
 | 
						|
	git_mutex_init(&cache->lock);
 | 
						|
 | 
						|
	cache->nodes = git__malloc(size * sizeof(git_cached_obj *));
 | 
						|
	GITERR_CHECK_ALLOC(cache->nodes);
 | 
						|
 | 
						|
	memset(cache->nodes, 0x0, size * sizeof(git_cached_obj *));
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
void git_cache_free(git_cache *cache)
 | 
						|
{
 | 
						|
	size_t i;
 | 
						|
 | 
						|
	for (i = 0; i < (cache->size_mask + 1); ++i) {
 | 
						|
		if (cache->nodes[i] != NULL)
 | 
						|
			git_cached_obj_decref(cache->nodes[i], cache->free_obj);
 | 
						|
	}
 | 
						|
 | 
						|
	git__free(cache->nodes);
 | 
						|
}
 | 
						|
 | 
						|
void *git_cache_get(git_cache *cache, const git_oid *oid)
 | 
						|
{
 | 
						|
	uint32_t hash;
 | 
						|
	git_cached_obj *node = NULL, *result = NULL;
 | 
						|
 | 
						|
	memcpy(&hash, oid->id, sizeof(hash));
 | 
						|
 | 
						|
	git_mutex_lock(&cache->lock);
 | 
						|
	{
 | 
						|
		node = cache->nodes[hash & cache->size_mask];
 | 
						|
 | 
						|
		if (node != NULL && git_oid_cmp(&node->oid, oid) == 0) {
 | 
						|
			git_cached_obj_incref(node);
 | 
						|
			result = node;
 | 
						|
		}
 | 
						|
	}
 | 
						|
	git_mutex_unlock(&cache->lock);
 | 
						|
 | 
						|
	return result;
 | 
						|
}
 | 
						|
 | 
						|
void *git_cache_try_store(git_cache *cache, void *_entry)
 | 
						|
{
 | 
						|
	git_cached_obj *entry = _entry;
 | 
						|
	uint32_t hash;
 | 
						|
 | 
						|
	memcpy(&hash, &entry->oid, sizeof(uint32_t));
 | 
						|
 | 
						|
	/* increase the refcount on this object, because
 | 
						|
	 * the cache now owns it */
 | 
						|
	git_cached_obj_incref(entry);
 | 
						|
 | 
						|
	git_mutex_lock(&cache->lock);
 | 
						|
	{
 | 
						|
		git_cached_obj *node = cache->nodes[hash & cache->size_mask];
 | 
						|
 | 
						|
		if (node == NULL) {
 | 
						|
			cache->nodes[hash & cache->size_mask] = entry;
 | 
						|
		} else if (git_oid_cmp(&node->oid, &entry->oid) == 0) {
 | 
						|
			git_cached_obj_decref(entry, cache->free_obj);
 | 
						|
			entry = node;
 | 
						|
		} else {
 | 
						|
			git_cached_obj_decref(node, cache->free_obj);
 | 
						|
			cache->nodes[hash & cache->size_mask] = entry;
 | 
						|
		}
 | 
						|
	}
 | 
						|
	git_mutex_unlock(&cache->lock);
 | 
						|
 | 
						|
	/* increase the refcount again, because we are
 | 
						|
	 * returning it to the user */
 | 
						|
	git_cached_obj_incref(entry);
 | 
						|
 | 
						|
	return entry;
 | 
						|
}
 |