Blame src/thread-utils.h

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
#ifndef INCLUDE_thread_utils_h__
Packit ae9e2a
#define INCLUDE_thread_utils_h__
Packit ae9e2a
Packit ae9e2a
#if defined(__GNUC__) && defined(GIT_THREADS)
Packit ae9e2a
# if (__GNUC__ < 4 || (__GNUC__ == 4 && __GNUC_MINOR__ < 1))
Packit ae9e2a
#  error Atomic primitives do not exist on this version of gcc; configure libgit2 with -DTHREADSAFE=OFF
Packit ae9e2a
# endif
Packit ae9e2a
#endif
Packit ae9e2a
Packit ae9e2a
/* Common operations even if threading has been disabled */
Packit ae9e2a
typedef struct {
Packit ae9e2a
#if defined(GIT_WIN32)
Packit ae9e2a
	volatile long val;
Packit ae9e2a
#else
Packit ae9e2a
	volatile int val;
Packit ae9e2a
#endif
Packit ae9e2a
} git_atomic;
Packit ae9e2a
Packit ae9e2a
#ifdef GIT_ARCH_64
Packit ae9e2a
Packit ae9e2a
typedef struct {
Packit ae9e2a
#if defined(GIT_WIN32)
Packit ae9e2a
	__int64 val;
Packit ae9e2a
#else
Packit ae9e2a
	int64_t val;
Packit ae9e2a
#endif
Packit ae9e2a
} git_atomic64;
Packit ae9e2a
Packit ae9e2a
typedef git_atomic64 git_atomic_ssize;
Packit ae9e2a
Packit ae9e2a
#define git_atomic_ssize_add git_atomic64_add
Packit ae9e2a
Packit ae9e2a
#else
Packit ae9e2a
Packit ae9e2a
typedef git_atomic git_atomic_ssize;
Packit ae9e2a
Packit ae9e2a
#define git_atomic_ssize_add git_atomic_add
Packit ae9e2a
Packit ae9e2a
#endif
Packit ae9e2a
Packit ae9e2a
#ifdef GIT_THREADS
Packit ae9e2a
Packit ae9e2a
#ifdef GIT_WIN32
Packit ae9e2a
#   include "win32/thread.h"
Packit ae9e2a
#else
Packit ae9e2a
#   include "unix/pthread.h"
Packit ae9e2a
#endif
Packit ae9e2a
Packit ae9e2a
GIT_INLINE(void) git_atomic_set(git_atomic *a, int val)
Packit ae9e2a
{
Packit ae9e2a
#if defined(GIT_WIN32)
Packit ae9e2a
	InterlockedExchange(&a->val, (LONG)val);
Packit ae9e2a
#elif defined(__GNUC__)
Packit ae9e2a
	__sync_lock_test_and_set(&a->val, val);
Packit ae9e2a
#else
Packit ae9e2a
#	error "Unsupported architecture for atomic operations"
Packit ae9e2a
#endif
Packit ae9e2a
}
Packit ae9e2a
Packit ae9e2a
GIT_INLINE(int) git_atomic_inc(git_atomic *a)
Packit ae9e2a
{
Packit ae9e2a
#if defined(GIT_WIN32)
Packit ae9e2a
	return InterlockedIncrement(&a->val);
Packit ae9e2a
#elif defined(__GNUC__)
Packit ae9e2a
	return __sync_add_and_fetch(&a->val, 1);
Packit ae9e2a
#else
Packit ae9e2a
#	error "Unsupported architecture for atomic operations"
Packit ae9e2a
#endif
Packit ae9e2a
}
Packit ae9e2a
Packit ae9e2a
GIT_INLINE(int) git_atomic_add(git_atomic *a, int32_t addend)
Packit ae9e2a
{
Packit ae9e2a
#if defined(GIT_WIN32)
Packit ae9e2a
	return InterlockedExchangeAdd(&a->val, addend);
Packit ae9e2a
#elif defined(__GNUC__)
Packit ae9e2a
	return __sync_add_and_fetch(&a->val, addend);
Packit ae9e2a
#else
Packit ae9e2a
#	error "Unsupported architecture for atomic operations"
Packit ae9e2a
#endif
Packit ae9e2a
}
Packit ae9e2a
Packit ae9e2a
GIT_INLINE(int) git_atomic_dec(git_atomic *a)
Packit ae9e2a
{
Packit ae9e2a
#if defined(GIT_WIN32)
Packit ae9e2a
	return InterlockedDecrement(&a->val);
Packit ae9e2a
#elif defined(__GNUC__)
Packit ae9e2a
	return __sync_sub_and_fetch(&a->val, 1);
Packit ae9e2a
#else
Packit ae9e2a
#	error "Unsupported architecture for atomic operations"
Packit ae9e2a
#endif
Packit ae9e2a
}
Packit ae9e2a
Packit ae9e2a
GIT_INLINE(void *) git___compare_and_swap(
Packit ae9e2a
	void * volatile *ptr, void *oldval, void *newval)
Packit ae9e2a
{
Packit ae9e2a
	volatile void *foundval;
Packit ae9e2a
#if defined(GIT_WIN32)
Packit ae9e2a
	foundval = InterlockedCompareExchangePointer((volatile PVOID *)ptr, newval, oldval);
Packit ae9e2a
#elif defined(__GNUC__)
Packit ae9e2a
	foundval = __sync_val_compare_and_swap(ptr, oldval, newval);
Packit ae9e2a
#else
Packit ae9e2a
#	error "Unsupported architecture for atomic operations"
Packit ae9e2a
#endif
Packit ae9e2a
	return (foundval == oldval) ? oldval : newval;
Packit ae9e2a
}
Packit ae9e2a
Packit ae9e2a
GIT_INLINE(volatile void *) git___swap(
Packit ae9e2a
	void * volatile *ptr, void *newval)
Packit ae9e2a
{
Packit ae9e2a
#if defined(GIT_WIN32)
Packit ae9e2a
	return InterlockedExchangePointer(ptr, newval);
Packit ae9e2a
#else
Packit ae9e2a
	return __sync_lock_test_and_set(ptr, newval);
Packit ae9e2a
#endif
Packit ae9e2a
}
Packit ae9e2a
Packit ae9e2a
#ifdef GIT_ARCH_64
Packit ae9e2a
Packit ae9e2a
GIT_INLINE(int64_t) git_atomic64_add(git_atomic64 *a, int64_t addend)
Packit ae9e2a
{
Packit ae9e2a
#if defined(GIT_WIN32)
Packit ae9e2a
	return InterlockedExchangeAdd64(&a->val, addend);
Packit ae9e2a
#elif defined(__GNUC__)
Packit ae9e2a
	return __sync_add_and_fetch(&a->val, addend);
Packit ae9e2a
#else
Packit ae9e2a
#	error "Unsupported architecture for atomic operations"
Packit ae9e2a
#endif
Packit ae9e2a
}
Packit ae9e2a
Packit ae9e2a
#endif
Packit ae9e2a
Packit ae9e2a
#else
Packit ae9e2a
Packit ae9e2a
#define git_thread unsigned int
Packit ae9e2a
#define git_thread_create(thread, start_routine, arg) 0
Packit ae9e2a
#define git_thread_join(id, status) (void)0
Packit ae9e2a
Packit ae9e2a
/* Pthreads Mutex */
Packit ae9e2a
#define git_mutex unsigned int
Packit ae9e2a
GIT_INLINE(int) git_mutex_init(git_mutex *mutex) \
Packit ae9e2a
	{ GIT_UNUSED(mutex); return 0; }
Packit ae9e2a
GIT_INLINE(int) git_mutex_lock(git_mutex *mutex) \
Packit ae9e2a
	{ GIT_UNUSED(mutex); return 0; }
Packit ae9e2a
#define git_mutex_unlock(a) (void)0
Packit ae9e2a
#define git_mutex_free(a) (void)0
Packit ae9e2a
Packit ae9e2a
/* Pthreads condition vars */
Packit ae9e2a
#define git_cond unsigned int
Packit ae9e2a
#define git_cond_init(c, a)	(void)0
Packit ae9e2a
#define git_cond_free(c) (void)0
Packit ae9e2a
#define git_cond_wait(c, l)	(void)0
Packit ae9e2a
#define git_cond_signal(c) (void)0
Packit ae9e2a
#define git_cond_broadcast(c) (void)0
Packit ae9e2a
Packit ae9e2a
/* Pthreads rwlock */
Packit ae9e2a
#define git_rwlock unsigned int
Packit ae9e2a
#define git_rwlock_init(a)		0
Packit ae9e2a
#define git_rwlock_rdlock(a)	0
Packit ae9e2a
#define git_rwlock_rdunlock(a)	(void)0
Packit ae9e2a
#define git_rwlock_wrlock(a)	0
Packit ae9e2a
#define git_rwlock_wrunlock(a)	(void)0
Packit ae9e2a
#define git_rwlock_free(a)		(void)0
Packit ae9e2a
#define GIT_RWLOCK_STATIC_INIT	0
Packit ae9e2a
Packit ae9e2a
Packit ae9e2a
GIT_INLINE(void) git_atomic_set(git_atomic *a, int val)
Packit ae9e2a
{
Packit ae9e2a
	a->val = val;
Packit ae9e2a
}
Packit ae9e2a
Packit ae9e2a
GIT_INLINE(int) git_atomic_inc(git_atomic *a)
Packit ae9e2a
{
Packit ae9e2a
	return ++a->val;
Packit ae9e2a
}
Packit ae9e2a
Packit ae9e2a
GIT_INLINE(int) git_atomic_add(git_atomic *a, int32_t addend)
Packit ae9e2a
{
Packit ae9e2a
	a->val += addend;
Packit ae9e2a
	return a->val;
Packit ae9e2a
}
Packit ae9e2a
Packit ae9e2a
GIT_INLINE(int) git_atomic_dec(git_atomic *a)
Packit ae9e2a
{
Packit ae9e2a
	return --a->val;
Packit ae9e2a
}
Packit ae9e2a
Packit ae9e2a
GIT_INLINE(void *) git___compare_and_swap(
Packit ae9e2a
	void * volatile *ptr, void *oldval, void *newval)
Packit ae9e2a
{
Packit ae9e2a
	if (*ptr == oldval)
Packit ae9e2a
		*ptr = newval;
Packit ae9e2a
	else
Packit ae9e2a
		oldval = newval;
Packit ae9e2a
	return oldval;
Packit ae9e2a
}
Packit ae9e2a
Packit ae9e2a
GIT_INLINE(volatile void *) git___swap(
Packit ae9e2a
	void * volatile *ptr, void *newval)
Packit ae9e2a
{
Packit ae9e2a
	volatile void *old = *ptr;
Packit ae9e2a
	*ptr = newval;
Packit ae9e2a
	return old;
Packit ae9e2a
}
Packit ae9e2a
Packit ae9e2a
#ifdef GIT_ARCH_64
Packit ae9e2a
Packit ae9e2a
GIT_INLINE(int64_t) git_atomic64_add(git_atomic64 *a, int64_t addend)
Packit ae9e2a
{
Packit ae9e2a
	a->val += addend;
Packit ae9e2a
	return a->val;
Packit ae9e2a
}
Packit ae9e2a
Packit ae9e2a
#endif
Packit ae9e2a
Packit ae9e2a
#endif
Packit ae9e2a
Packit ae9e2a
GIT_INLINE(int) git_atomic_get(git_atomic *a)
Packit ae9e2a
{
Packit ae9e2a
	return (int)a->val;
Packit ae9e2a
}
Packit ae9e2a
Packit ae9e2a
/* Atomically replace oldval with newval
Packit ae9e2a
 * @return oldval if it was replaced or newval if it was not
Packit ae9e2a
 */
Packit ae9e2a
#define git__compare_and_swap(P,O,N) \
Packit ae9e2a
	git___compare_and_swap((void * volatile *)P, O, N)
Packit ae9e2a
Packit ae9e2a
#define git__swap(ptr, val) (void *)git___swap((void * volatile *)&ptr, val)
Packit ae9e2a
Packit ae9e2a
extern int git_online_cpus(void);
Packit ae9e2a
Packit ae9e2a
#if defined(GIT_THREADS) && defined(_MSC_VER)
Packit ae9e2a
# define GIT_MEMORY_BARRIER MemoryBarrier()
Packit ae9e2a
#elif defined(GIT_THREADS)
Packit ae9e2a
# define GIT_MEMORY_BARRIER __sync_synchronize()
Packit ae9e2a
#else
Packit ae9e2a
# define GIT_MEMORY_BARRIER /* noop */
Packit ae9e2a
#endif
Packit ae9e2a
Packit ae9e2a
#endif /* INCLUDE_thread_utils_h__ */