Add userspace spinlock implementation
[akaros.git] / user / parlib / spinlock.c
1 /*
2  * Copyright (c) 2011 The Regents of the University of California
3  * Barret Rhoden <brho@cs.berkeley.edu>
4  * Kevin Klues <klueska@cs.berkeley.edu>
5  *
6  * This file is part of Parlib.
7  * 
8  * Parlib is free software: you can redistribute it and/or modify
9  * it under the terms of the Lesser GNU General Public License as published by
10  * the Free Software Foundation, either version 3 of the License, or
11  * (at your option) any later version.
12  * 
13  * Parlib is distributed in the hope that it will be useful,
14  * but WITHOUT ANY WARRANTY; without even the implied warranty of
15  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
16  * Lesser GNU General Public License for more details.
17  * 
18  * See COPYING.LESSER for details on the GNU Lesser General Public License.
19  * See COPYING for details on the GNU General Public License.
20  */
21
22 #include <stdlib.h>
23 #include <errno.h>
24 #include <assert.h>
25 #include "spinlock.h"
26
27 void spinlock_init(spinlock_t *lock)
28 {
29   assert(lock);
30   *lock = SPINLOCK_UNLOCKED;
31 }
32
33
34 int spinlock_trylock(spinlock_t *lock) 
35 {
36   assert(lock);
37   if (*lock == SPINLOCK_LOCKED)
38     return EBUSY;
39
40   return (int)atomic_cas(lock, (long)SPINLOCK_LOCKED, (long)SPINLOCK_UNLOCKED);
41 }
42
43
44 void spinlock_lock(spinlock_t *lock) 
45 {
46   assert(lock);
47   while (spinlock_trylock(lock) != (int)SPINLOCK_UNLOCKED)
48     cpu_relax();
49 }
50
51
52 void spinlock_unlock(spinlock_t *lock) 
53 {
54   assert(lock);
55   *lock = SPINLOCK_UNLOCKED;
56 }