Blame src/win32/thread.c

Packit ae9e2a
/*
Packit ae9e2a
 * Copyright (C) the libgit2 contributors. All rights reserved.
Packit ae9e2a
 *
Packit ae9e2a
 * This file is part of libgit2, distributed under the GNU GPL v2 with
Packit ae9e2a
 * a Linking Exception. For full terms see the included COPYING file.
Packit ae9e2a
 */
Packit ae9e2a
Packit ae9e2a
#include "thread.h"
Packit ae9e2a
#include "../global.h"
Packit ae9e2a
Packit ae9e2a
#define CLEAN_THREAD_EXIT 0x6F012842
Packit ae9e2a
Packit ae9e2a
typedef void (WINAPI *win32_srwlock_fn)(GIT_SRWLOCK *);
Packit ae9e2a
Packit ae9e2a
static win32_srwlock_fn win32_srwlock_initialize;
Packit ae9e2a
static win32_srwlock_fn win32_srwlock_acquire_shared;
Packit ae9e2a
static win32_srwlock_fn win32_srwlock_release_shared;
Packit ae9e2a
static win32_srwlock_fn win32_srwlock_acquire_exclusive;
Packit ae9e2a
static win32_srwlock_fn win32_srwlock_release_exclusive;
Packit ae9e2a
Packit ae9e2a
/* The thread procedure stub used to invoke the caller's procedure
Packit ae9e2a
 * and capture the return value for later collection. Windows will
Packit ae9e2a
 * only hold a DWORD, but we need to be able to store an entire
Packit ae9e2a
 * void pointer. This requires the indirection. */
Packit ae9e2a
static DWORD WINAPI git_win32__threadproc(LPVOID lpParameter)
Packit ae9e2a
{
Packit ae9e2a
	git_thread *thread = lpParameter;
Packit ae9e2a
Packit ae9e2a
	/* Set the current thread for `git_thread_exit` */
Packit ae9e2a
	GIT_GLOBAL->current_thread = thread;
Packit ae9e2a
Packit ae9e2a
	thread->result = thread->proc(thread->param);
Packit ae9e2a
Packit ae9e2a
	git__free_tls_data();
Packit ae9e2a
Packit ae9e2a
	return CLEAN_THREAD_EXIT;
Packit ae9e2a
}
Packit ae9e2a
Packit ae9e2a
int git_threads_init(void)
Packit ae9e2a
{
Packit ae9e2a
	HMODULE hModule = GetModuleHandleW(L"kernel32");
Packit ae9e2a
Packit ae9e2a
	if (hModule) {
Packit ae9e2a
		win32_srwlock_initialize = (win32_srwlock_fn)
Packit ae9e2a
			GetProcAddress(hModule, "InitializeSRWLock");
Packit ae9e2a
		win32_srwlock_acquire_shared = (win32_srwlock_fn)
Packit ae9e2a
			GetProcAddress(hModule, "AcquireSRWLockShared");
Packit ae9e2a
		win32_srwlock_release_shared = (win32_srwlock_fn)
Packit ae9e2a
			GetProcAddress(hModule, "ReleaseSRWLockShared");
Packit ae9e2a
		win32_srwlock_acquire_exclusive = (win32_srwlock_fn)
Packit ae9e2a
			GetProcAddress(hModule, "AcquireSRWLockExclusive");
Packit ae9e2a
		win32_srwlock_release_exclusive = (win32_srwlock_fn)
Packit ae9e2a
			GetProcAddress(hModule, "ReleaseSRWLockExclusive");
Packit ae9e2a
	}
Packit ae9e2a
Packit ae9e2a
	return 0;
Packit ae9e2a
}
Packit ae9e2a
Packit ae9e2a
int git_thread_create(
Packit ae9e2a
	git_thread *GIT_RESTRICT thread,
Packit ae9e2a
	void *(*start_routine)(void*),
Packit ae9e2a
	void *GIT_RESTRICT arg)
Packit ae9e2a
{
Packit ae9e2a
	thread->result = NULL;
Packit ae9e2a
	thread->param = arg;
Packit ae9e2a
	thread->proc = start_routine;
Packit ae9e2a
	thread->thread = CreateThread(
Packit ae9e2a
		NULL, 0, git_win32__threadproc, thread, 0, NULL);
Packit ae9e2a
Packit ae9e2a
	return thread->thread ? 0 : -1;
Packit ae9e2a
}
Packit ae9e2a
Packit ae9e2a
int git_thread_join(
Packit ae9e2a
	git_thread *thread,
Packit ae9e2a
	void **value_ptr)
Packit ae9e2a
{
Packit ae9e2a
	DWORD exit;
Packit ae9e2a
Packit ae9e2a
	if (WaitForSingleObject(thread->thread, INFINITE) != WAIT_OBJECT_0)
Packit ae9e2a
		return -1;
Packit ae9e2a
Packit ae9e2a
	if (!GetExitCodeThread(thread->thread, &exit)) {
Packit ae9e2a
		CloseHandle(thread->thread);
Packit ae9e2a
		return -1;
Packit ae9e2a
	}
Packit ae9e2a
Packit ae9e2a
	/* Check for the thread having exited uncleanly. If exit was unclean,
Packit ae9e2a
	 * then we don't have a return value to give back to the caller. */
Packit ae9e2a
	if (exit != CLEAN_THREAD_EXIT) {
Packit ae9e2a
		assert(false);
Packit ae9e2a
		thread->result = NULL;
Packit ae9e2a
	}
Packit ae9e2a
Packit ae9e2a
	if (value_ptr)
Packit ae9e2a
		*value_ptr = thread->result;
Packit ae9e2a
Packit ae9e2a
	CloseHandle(thread->thread);
Packit ae9e2a
	return 0;
Packit ae9e2a
}
Packit ae9e2a
Packit ae9e2a
void git_thread_exit(void *value)
Packit ae9e2a
{
Packit ae9e2a
	assert(GIT_GLOBAL->current_thread);
Packit ae9e2a
	GIT_GLOBAL->current_thread->result = value;
Packit ae9e2a
Packit ae9e2a
	git__free_tls_data();
Packit ae9e2a
Packit ae9e2a
	ExitThread(CLEAN_THREAD_EXIT);
Packit ae9e2a
}
Packit ae9e2a
Packit ae9e2a
size_t git_thread_currentid(void)
Packit ae9e2a
{
Packit ae9e2a
	return GetCurrentThreadId();
Packit ae9e2a
}
Packit ae9e2a
Packit ae9e2a
int git_mutex_init(git_mutex *GIT_RESTRICT mutex)
Packit ae9e2a
{
Packit ae9e2a
	InitializeCriticalSection(mutex);
Packit ae9e2a
	return 0;
Packit ae9e2a
}
Packit ae9e2a
Packit ae9e2a
int git_mutex_free(git_mutex *mutex)
Packit ae9e2a
{
Packit ae9e2a
	DeleteCriticalSection(mutex);
Packit ae9e2a
	return 0;
Packit ae9e2a
}
Packit ae9e2a
Packit ae9e2a
int git_mutex_lock(git_mutex *mutex)
Packit ae9e2a
{
Packit ae9e2a
	EnterCriticalSection(mutex);
Packit ae9e2a
	return 0;
Packit ae9e2a
}
Packit ae9e2a
Packit ae9e2a
int git_mutex_unlock(git_mutex *mutex)
Packit ae9e2a
{
Packit ae9e2a
	LeaveCriticalSection(mutex);
Packit ae9e2a
	return 0;
Packit ae9e2a
}
Packit ae9e2a
Packit ae9e2a
int git_cond_init(git_cond *cond)
Packit ae9e2a
{
Packit ae9e2a
	/* This is an auto-reset event. */
Packit ae9e2a
	*cond = CreateEventW(NULL, FALSE, FALSE, NULL);
Packit ae9e2a
	assert(*cond);
Packit ae9e2a
Packit ae9e2a
	/* If we can't create the event, claim that the reason was out-of-memory.
Packit ae9e2a
	 * The actual reason can be fetched with GetLastError(). */
Packit ae9e2a
	return *cond ? 0 : ENOMEM;
Packit ae9e2a
}
Packit ae9e2a
Packit ae9e2a
int git_cond_free(git_cond *cond)
Packit ae9e2a
{
Packit ae9e2a
	BOOL closed;
Packit ae9e2a
Packit ae9e2a
	if (!cond)
Packit ae9e2a
		return EINVAL;
Packit ae9e2a
Packit ae9e2a
	closed = CloseHandle(*cond);
Packit ae9e2a
	assert(closed);
Packit ae9e2a
	GIT_UNUSED(closed);
Packit ae9e2a
Packit ae9e2a
	*cond = NULL;
Packit ae9e2a
	return 0;
Packit ae9e2a
}
Packit ae9e2a
Packit ae9e2a
int git_cond_wait(git_cond *cond, git_mutex *mutex)
Packit ae9e2a
{
Packit ae9e2a
	int error;
Packit ae9e2a
	DWORD wait_result;
Packit ae9e2a
Packit ae9e2a
	if (!cond || !mutex)
Packit ae9e2a
		return EINVAL;
Packit ae9e2a
Packit ae9e2a
	/* The caller must be holding the mutex. */
Packit ae9e2a
	error = git_mutex_unlock(mutex);
Packit ae9e2a
Packit ae9e2a
	if (error)
Packit ae9e2a
		return error;
Packit ae9e2a
Packit ae9e2a
	wait_result = WaitForSingleObject(*cond, INFINITE);
Packit ae9e2a
	assert(WAIT_OBJECT_0 == wait_result);
Packit ae9e2a
	GIT_UNUSED(wait_result);
Packit ae9e2a
Packit ae9e2a
	return git_mutex_lock(mutex);
Packit ae9e2a
}
Packit ae9e2a
Packit ae9e2a
int git_cond_signal(git_cond *cond)
Packit ae9e2a
{
Packit ae9e2a
	BOOL signaled;
Packit ae9e2a
Packit ae9e2a
	if (!cond)
Packit ae9e2a
		return EINVAL;
Packit ae9e2a
Packit ae9e2a
	signaled = SetEvent(*cond);
Packit ae9e2a
	assert(signaled);
Packit ae9e2a
	GIT_UNUSED(signaled);
Packit ae9e2a
Packit ae9e2a
	return 0;
Packit ae9e2a
}
Packit ae9e2a
Packit ae9e2a
int git_rwlock_init(git_rwlock *GIT_RESTRICT lock)
Packit ae9e2a
{
Packit ae9e2a
	if (win32_srwlock_initialize)
Packit ae9e2a
		win32_srwlock_initialize(&lock->native.srwl);
Packit ae9e2a
	else
Packit ae9e2a
		InitializeCriticalSection(&lock->native.csec);
Packit ae9e2a
Packit ae9e2a
	return 0;
Packit ae9e2a
}
Packit ae9e2a
Packit ae9e2a
int git_rwlock_rdlock(git_rwlock *lock)
Packit ae9e2a
{
Packit ae9e2a
	if (win32_srwlock_acquire_shared)
Packit ae9e2a
		win32_srwlock_acquire_shared(&lock->native.srwl);
Packit ae9e2a
	else
Packit ae9e2a
		EnterCriticalSection(&lock->native.csec);
Packit ae9e2a
Packit ae9e2a
	return 0;
Packit ae9e2a
}
Packit ae9e2a
Packit ae9e2a
int git_rwlock_rdunlock(git_rwlock *lock)
Packit ae9e2a
{
Packit ae9e2a
	if (win32_srwlock_release_shared)
Packit ae9e2a
		win32_srwlock_release_shared(&lock->native.srwl);
Packit ae9e2a
	else
Packit ae9e2a
		LeaveCriticalSection(&lock->native.csec);
Packit ae9e2a
Packit ae9e2a
	return 0;
Packit ae9e2a
}
Packit ae9e2a
Packit ae9e2a
int git_rwlock_wrlock(git_rwlock *lock)
Packit ae9e2a
{
Packit ae9e2a
	if (win32_srwlock_acquire_exclusive)
Packit ae9e2a
		win32_srwlock_acquire_exclusive(&lock->native.srwl);
Packit ae9e2a
	else
Packit ae9e2a
		EnterCriticalSection(&lock->native.csec);
Packit ae9e2a
Packit ae9e2a
	return 0;
Packit ae9e2a
}
Packit ae9e2a
Packit ae9e2a
int git_rwlock_wrunlock(git_rwlock *lock)
Packit ae9e2a
{
Packit ae9e2a
	if (win32_srwlock_release_exclusive)
Packit ae9e2a
		win32_srwlock_release_exclusive(&lock->native.srwl);
Packit ae9e2a
	else
Packit ae9e2a
		LeaveCriticalSection(&lock->native.csec);
Packit ae9e2a
Packit ae9e2a
	return 0;
Packit ae9e2a
}
Packit ae9e2a
Packit ae9e2a
int git_rwlock_free(git_rwlock *lock)
Packit ae9e2a
{
Packit ae9e2a
	if (!win32_srwlock_initialize)
Packit ae9e2a
		DeleteCriticalSection(&lock->native.csec);
Packit ae9e2a
	git__memzero(lock, sizeof(*lock));
Packit ae9e2a
	return 0;
Packit ae9e2a
}