/*
 * Copyright (C) the libgit2 contributors. All rights reserved.
 *
 * 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 "git2_util.h"

#if !defined(GIT_THREADS)

#define TLSDATA_MAX 16

typedef struct {
	void *value;
	void (GIT_SYSTEM_CALL *destroy_fn)(void *);
} tlsdata_value;

static tlsdata_value tlsdata_values[TLSDATA_MAX];
static int tlsdata_cnt = 0;

int git_tlsdata_init(git_tlsdata_key *key, void (GIT_SYSTEM_CALL *destroy_fn)(void *))
{
	if (tlsdata_cnt >= TLSDATA_MAX)
		return -1;

	tlsdata_values[tlsdata_cnt].value = NULL;
	tlsdata_values[tlsdata_cnt].destroy_fn = destroy_fn;

	*key = tlsdata_cnt;
	tlsdata_cnt++;

	return 0;
}

int git_tlsdata_set(git_tlsdata_key key, void *value)
{
	if (key < 0 || key > tlsdata_cnt)
		return -1;

	tlsdata_values[key].value = value;
	return 0;
}

void *git_tlsdata_get(git_tlsdata_key key)
{
	if (key < 0 || key > tlsdata_cnt)
		return NULL;

	return tlsdata_values[key].value;
}

int git_tlsdata_dispose(git_tlsdata_key key)
{
	void *value;
	void (*destroy_fn)(void *) = NULL;

	if (key < 0 || key > tlsdata_cnt)
		return -1;

	value = tlsdata_values[key].value;
	destroy_fn = tlsdata_values[key].destroy_fn;

	tlsdata_values[key].value = NULL;
	tlsdata_values[key].destroy_fn = NULL;

	if (value && destroy_fn)
		destroy_fn(value);

	return 0;
}

#elif defined(GIT_WIN32)

int git_tlsdata_init(git_tlsdata_key *key, void (GIT_SYSTEM_CALL *destroy_fn)(void *))
{
	DWORD fls_index = FlsAlloc(destroy_fn);

	if (fls_index == FLS_OUT_OF_INDEXES)
		return -1;

	*key = fls_index;
	return 0;
}

int git_tlsdata_set(git_tlsdata_key key, void *value)
{
	if (!FlsSetValue(key, value))
		return -1;

	return 0;
}

void *git_tlsdata_get(git_tlsdata_key key)
{
	return FlsGetValue(key);
}

int git_tlsdata_dispose(git_tlsdata_key key)
{
	if (!FlsFree(key))
		return -1;

	return 0;
}

#elif defined(_POSIX_THREADS)

int git_tlsdata_init(git_tlsdata_key *key, void (GIT_SYSTEM_CALL *destroy_fn)(void *))
{
	if (pthread_key_create(key, destroy_fn) != 0)
		return -1;

	return 0;
}

int git_tlsdata_set(git_tlsdata_key key, void *value)
{
	if (pthread_setspecific(key, value) != 0)
		return -1;

	return 0;
}

void *git_tlsdata_get(git_tlsdata_key key)
{
	return pthread_getspecific(key);
}

int git_tlsdata_dispose(git_tlsdata_key key)
{
	if (pthread_key_delete(key) != 0)
		return -1;

	return 0;
}

#else
# error unknown threading model
#endif