summaryrefslogtreecommitdiff
path: root/src/thread/pthread_mutex_trylock.c
blob: 1e3817bbf8960fc53f36bba70c8f9c6171985e0a (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
#include "pthread_impl.h"

int pthread_mutex_trylock(pthread_mutex_t *m)
{
	if (m->__type == PTHREAD_MUTEX_RECURSIVE) {
		pthread_t self = pthread_self();
		if (m->__owner == self) {
			if ((unsigned)m->__lock >= INT_MAX) return EAGAIN;
			a_inc(&m->__lock);
			return 0;
		}
		if (a_fetch_add(&m->__lock, 1)) {
			if (a_fetch_add(&m->__lock, -1)==1 && m->__waiters)
				__wake(&m->__lock, 1, 0);
			return EBUSY;
		}
		m->__owner = self;
		return 0;
	}

	if (a_xchg(&m->__lock, 1))
		if (m->__type == PTHREAD_MUTEX_ERRORCHECK
		 && m->__owner == pthread_self()) return EDEADLK;
		else return EBUSY;
	if (m->__type == PTHREAD_MUTEX_ERRORCHECK)
		m->__owner = pthread_self();
	return 0;
}