pthread.c 2.69 KB
Newer Older
1
/*
schu committed
2
 * Copyright (C) 2009-2012 the libgit2 contributors
3
 *
Vicent Marti committed
4 5
 * This file is part of libgit2, distributed under the GNU GPL v2 with
 * a Linking Exception. For full terms see the included COPYING file.
6 7 8 9
 */

#include "pthread.h"

10 11 12 13 14
int pthread_create(
	pthread_t *GIT_RESTRICT thread,
	const pthread_attr_t *GIT_RESTRICT attr,
	void *(*start_routine)(void*),
	void *GIT_RESTRICT arg)
15
{
16
	GIT_UNUSED(attr);
17 18 19
	*thread = (pthread_t) CreateThread(
		NULL, 0, (LPTHREAD_START_ROUTINE)start_routine, arg, 0, NULL);
	return *thread ? 0 : -1;
20 21 22 23
}

int pthread_join(pthread_t thread, void **value_ptr)
{
Vicent Marti committed
24 25 26 27 28
	int ret;
	ret = WaitForSingleObject(thread, INFINITE);
	if (ret && value_ptr)
		GetExitCodeThread(thread, (void*) value_ptr);
	return -(!!ret);
29 30
}

Vicent Marti committed
31
int pthread_mutex_init(pthread_mutex_t *GIT_RESTRICT mutex,
32
						const pthread_mutexattr_t *GIT_RESTRICT mutexattr)
33
{
34
	GIT_UNUSED(mutexattr);
Vicent Marti committed
35 36
	InitializeCriticalSection(mutex);
	return 0;
37 38 39 40
}

int pthread_mutex_destroy(pthread_mutex_t *mutex)
{
Vicent Marti committed
41 42
	DeleteCriticalSection(mutex);
	return 0;
43 44
}

Vicent Marti committed
45
int pthread_mutex_lock(pthread_mutex_t *mutex)
46
{
Vicent Marti committed
47 48
	EnterCriticalSection(mutex);
	return 0;
49 50
}

Vicent Marti committed
51
int pthread_mutex_unlock(pthread_mutex_t *mutex)
52
{
Vicent Marti committed
53 54
	LeaveCriticalSection(mutex);
	return 0;
55 56
}

57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80
int pthread_cond_init(pthread_cond_t *cond, const pthread_condattr_t *attr)
{
	/* We don't support non-default attributes. */
	if (attr)
		return EINVAL;

	/* This is an auto-reset event. */
	*cond = CreateEventW(NULL, FALSE, FALSE, NULL);
	assert(*cond);

	/* If we can't create the event, claim that the reason was out-of-memory.
	 * The actual reason can be fetched with GetLastError(). */
	return *cond ? 0 : ENOMEM;
}

int pthread_cond_destroy(pthread_cond_t *cond)
{
	BOOL closed;

	if (!cond)
		return EINVAL;

	closed = CloseHandle(*cond);
	assert(closed);
81
	GIT_UNUSED(closed);
82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102

	*cond = NULL;
	return 0;
}

int pthread_cond_wait(pthread_cond_t *cond, pthread_mutex_t *mutex)
{
	int error;
	DWORD wait_result;

	if (!cond || !mutex)
		return EINVAL;

	/* The caller must be holding the mutex. */
	error = pthread_mutex_unlock(mutex);

	if (error)
		return error;

	wait_result = WaitForSingleObject(*cond, INFINITE);
	assert(WAIT_OBJECT_0 == wait_result);
103
	GIT_UNUSED(wait_result);
104 105 106 107 108 109 110 111 112 113 114 115 116

	return pthread_mutex_lock(mutex);
}

int pthread_cond_signal(pthread_cond_t *cond)
{
	BOOL signaled;

	if (!cond)
		return EINVAL;

	signaled = SetEvent(*cond);
	assert(signaled);
117
	GIT_UNUSED(signaled);
118 119 120 121 122 123 124

	return 0;
}

/* pthread_cond_broadcast is not implemented because doing so with just Win32 events
 * is quite complicated, and no caller in libgit2 uses it yet. */

125 126
int pthread_num_processors_np(void)
{
Vicent Marti committed
127 128
	DWORD_PTR p, s;
	int n = 0;
129

Vicent Marti committed
130 131 132
	if (GetProcessAffinityMask(GetCurrentProcess(), &p, &s))
		for (; p; p >>= 1)
			n += p&1;
133

Vicent Marti committed
134
	return n ? n : 1;
135 136
}