aboutsummaryrefslogtreecommitdiff
path: root/mutex.h
blob: 4bb23df21c64434c4eefd34a16398e747b463d46 (plain) (blame)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
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
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
// Selection of mutexes (semaphores, futexes (may get rid of), or some miserable spinlocks (for use if all else fails))
//
// Written by: Test_User <hax@runxiyu.org>
//
// This is free and unencumbered software released into the public
// domain.
//
// Anyone is free to copy, modify, publish, use, compile, sell, or
// distribute this software, either in source code form or as a compiled
// binary, for any purpose, commercial or non-commercial, and by any
// means.
//
// In jurisdictions that recognize copyright laws, the author or authors
// of this software dedicate any and all copyright interest in the
// software to the public domain. We make this dedication for the benefit
// of the public at large and to the detriment of our heirs and
// successors. We intend this dedication to be an overt act of
// relinquishment in perpetuity of all present and future rights to this
// software under copyright law.
//
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
// EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
// MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.
// IN NO EVENT SHALL THE AUTHORS BE LIABLE FOR ANY CLAIM, DAMAGES OR
// OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE,
// ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
// OTHER DEALINGS IN THE SOFTWARE.

#pragma once

#if defined(USE_FUTEX)

#include <unistd.h>
#include <sys/syscall.h>
#include <stdint.h>
#include <linux/futex.h>

#define MUTEX_TYPE uint32_t

inline void mutex_init(uint32_t *futex) {
	*futex = 0;
}

inline void mutex_lock(uint32_t *futex) {
	if ((__sync_fetch_and_or(futex, 0x1) & 0x1) == 0)
		return;

	__sync_fetch_and_add(futex, 0x2);

	uint32_t val;
	while ((val = __sync_fetch_and_or(futex, 0x1)) & 0x1)
		syscall(SYS_futex, futex, FUTEX_PRIVATE_FLAG | FUTEX_WAIT, val | 0x1, NULL, 0, 0);

	__sync_fetch_and_sub(futex, 0x2);
}

inline void mutex_unlock(uint32_t *futex) {
	if (__sync_and_and_fetch(futex, 0xFFFFFFFE))
		syscall(SYS_futex, futex, FUTEX_PRIVATE_FLAG | FUTEX_WAKE, 1, NULL, 0, 0);
}

inline void mutex_destroy(uint32_t *futex) {
	return;
}

#elif defined(USE_MISERABLE_SPINLOCKS)

#define MUTEX_TYPE char

inline void mutex_init(char *lock) {
	*lock = 0;
}

inline void mutex_lock(char *lock) {
	while (__sync_lock_test_and_set(lock, 0x1));
}

inline void mutex_unlock(char *lock) {
	__sync_lock_release(lock);
}

inline void mutex_destroy(char *lock) {
	return;
}

#else

#include <semaphore.h>
#include <errno.h>

#define MUTEX_TYPE sem_t

inline void mutex_init(sem_t *mutex) {
	sem_init(mutex, 0, 1);
}

inline void mutex_lock(sem_t *mutex) {
	while (sem_wait(mutex) == -1);
}

inline void mutex_unlock(sem_t *mutex) {
	while (sem_trywait(mutex) == -1 && errno == EINTR);
	sem_post(mutex);
}

inline void mutex_destroy(sem_t *mutex) {
	sem_destroy(mutex);
}

#endif