-
Notifications
You must be signed in to change notification settings - Fork 0
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
yaml --- r: 360126 b: refs/heads/master c: 6e35fa2 h: refs/heads/master v: v3
- Loading branch information
Vineet Gupta
committed
Feb 11, 2013
1 parent
4c61d23
commit 8fdf5a8
Showing
4 changed files
with
189 additions
and
1 deletion.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,2 +1,2 @@ | ||
--- | ||
refs/heads/master: 3be80aaef861a60b85a9323462ebb5f623774f7a | ||
refs/heads/master: 6e35fa2d430538cd0609e499c6f789beea9e9798 |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,9 @@ | ||
/* | ||
* Copyright (C) 2004, 2007-2010, 2011-2012 Synopsys, Inc. (www.synopsys.com) | ||
* | ||
* This program is free software; you can redistribute it and/or modify | ||
* it under the terms of the GNU General Public License version 2 as | ||
* published by the Free Software Foundation. | ||
*/ | ||
|
||
#include <asm-generic/mutex-xchg.h> |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,144 @@ | ||
/* | ||
* Copyright (C) 2004, 2007-2010, 2011-2012 Synopsys, Inc. (www.synopsys.com) | ||
* | ||
* This program is free software; you can redistribute it and/or modify | ||
* it under the terms of the GNU General Public License version 2 as | ||
* published by the Free Software Foundation. | ||
*/ | ||
|
||
#ifndef __ASM_SPINLOCK_H | ||
#define __ASM_SPINLOCK_H | ||
|
||
#include <asm/spinlock_types.h> | ||
#include <asm/processor.h> | ||
#include <asm/barrier.h> | ||
|
||
#define arch_spin_is_locked(x) ((x)->slock != __ARCH_SPIN_LOCK_UNLOCKED__) | ||
#define arch_spin_lock_flags(lock, flags) arch_spin_lock(lock) | ||
#define arch_spin_unlock_wait(x) \ | ||
do { while (arch_spin_is_locked(x)) cpu_relax(); } while (0) | ||
|
||
static inline void arch_spin_lock(arch_spinlock_t *lock) | ||
{ | ||
unsigned int tmp = __ARCH_SPIN_LOCK_LOCKED__; | ||
|
||
__asm__ __volatile__( | ||
"1: ex %0, [%1] \n" | ||
" breq %0, %2, 1b \n" | ||
: "+&r" (tmp) | ||
: "r"(&(lock->slock)), "ir"(__ARCH_SPIN_LOCK_LOCKED__) | ||
: "memory"); | ||
} | ||
|
||
static inline int arch_spin_trylock(arch_spinlock_t *lock) | ||
{ | ||
unsigned int tmp = __ARCH_SPIN_LOCK_LOCKED__; | ||
|
||
__asm__ __volatile__( | ||
"1: ex %0, [%1] \n" | ||
: "+r" (tmp) | ||
: "r"(&(lock->slock)) | ||
: "memory"); | ||
|
||
return (tmp == __ARCH_SPIN_LOCK_UNLOCKED__); | ||
} | ||
|
||
static inline void arch_spin_unlock(arch_spinlock_t *lock) | ||
{ | ||
lock->slock = __ARCH_SPIN_LOCK_UNLOCKED__; | ||
smp_mb(); | ||
} | ||
|
||
/* | ||
* Read-write spinlocks, allowing multiple readers but only one writer. | ||
* | ||
* The spinlock itself is contained in @counter and access to it is | ||
* serialized with @lock_mutex. | ||
* | ||
* Unfair locking as Writers could be starved indefinitely by Reader(s) | ||
*/ | ||
|
||
/* Would read_trylock() succeed? */ | ||
#define arch_read_can_lock(x) ((x)->counter > 0) | ||
|
||
/* Would write_trylock() succeed? */ | ||
#define arch_write_can_lock(x) ((x)->counter == __ARCH_RW_LOCK_UNLOCKED__) | ||
|
||
/* 1 - lock taken successfully */ | ||
static inline int arch_read_trylock(arch_rwlock_t *rw) | ||
{ | ||
int ret = 0; | ||
|
||
arch_spin_lock(&(rw->lock_mutex)); | ||
|
||
/* | ||
* zero means writer holds the lock exclusively, deny Reader. | ||
* Otherwise grant lock to first/subseq reader | ||
*/ | ||
if (rw->counter > 0) { | ||
rw->counter--; | ||
ret = 1; | ||
} | ||
|
||
arch_spin_unlock(&(rw->lock_mutex)); | ||
|
||
smp_mb(); | ||
return ret; | ||
} | ||
|
||
/* 1 - lock taken successfully */ | ||
static inline int arch_write_trylock(arch_rwlock_t *rw) | ||
{ | ||
int ret = 0; | ||
|
||
arch_spin_lock(&(rw->lock_mutex)); | ||
|
||
/* | ||
* If reader(s) hold lock (lock < __ARCH_RW_LOCK_UNLOCKED__), | ||
* deny writer. Otherwise if unlocked grant to writer | ||
* Hence the claim that Linux rwlocks are unfair to writers. | ||
* (can be starved for an indefinite time by readers). | ||
*/ | ||
if (rw->counter == __ARCH_RW_LOCK_UNLOCKED__) { | ||
rw->counter = 0; | ||
ret = 1; | ||
} | ||
arch_spin_unlock(&(rw->lock_mutex)); | ||
|
||
return ret; | ||
} | ||
|
||
static inline void arch_read_lock(arch_rwlock_t *rw) | ||
{ | ||
while (!arch_read_trylock(rw)) | ||
cpu_relax(); | ||
} | ||
|
||
static inline void arch_write_lock(arch_rwlock_t *rw) | ||
{ | ||
while (!arch_write_trylock(rw)) | ||
cpu_relax(); | ||
} | ||
|
||
static inline void arch_read_unlock(arch_rwlock_t *rw) | ||
{ | ||
arch_spin_lock(&(rw->lock_mutex)); | ||
rw->counter++; | ||
arch_spin_unlock(&(rw->lock_mutex)); | ||
} | ||
|
||
static inline void arch_write_unlock(arch_rwlock_t *rw) | ||
{ | ||
arch_spin_lock(&(rw->lock_mutex)); | ||
rw->counter = __ARCH_RW_LOCK_UNLOCKED__; | ||
arch_spin_unlock(&(rw->lock_mutex)); | ||
} | ||
|
||
#define arch_read_lock_flags(lock, flags) arch_read_lock(lock) | ||
#define arch_write_lock_flags(lock, flags) arch_write_lock(lock) | ||
|
||
#define arch_spin_relax(lock) cpu_relax() | ||
#define arch_read_relax(lock) cpu_relax() | ||
#define arch_write_relax(lock) cpu_relax() | ||
|
||
#endif /* __ASM_SPINLOCK_H */ |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,35 @@ | ||
/* | ||
* Copyright (C) 2004, 2007-2010, 2011-2012 Synopsys, Inc. (www.synopsys.com) | ||
* | ||
* This program is free software; you can redistribute it and/or modify | ||
* it under the terms of the GNU General Public License version 2 as | ||
* published by the Free Software Foundation. | ||
*/ | ||
|
||
#ifndef __ASM_SPINLOCK_TYPES_H | ||
#define __ASM_SPINLOCK_TYPES_H | ||
|
||
typedef struct { | ||
volatile unsigned int slock; | ||
} arch_spinlock_t; | ||
|
||
#define __ARCH_SPIN_LOCK_UNLOCKED__ 0 | ||
#define __ARCH_SPIN_LOCK_LOCKED__ 1 | ||
|
||
#define __ARCH_SPIN_LOCK_UNLOCKED { __ARCH_SPIN_LOCK_UNLOCKED__ } | ||
#define __ARCH_SPIN_LOCK_LOCKED { __ARCH_SPIN_LOCK_LOCKED__ } | ||
|
||
/* | ||
* Unlocked: 0x01_00_00_00 | ||
* Read lock(s): 0x00_FF_00_00 to say 0x01 | ||
* Write lock: 0x0, but only possible if prior value "unlocked" 0x0100_0000 | ||
*/ | ||
typedef struct { | ||
volatile unsigned int counter; | ||
arch_spinlock_t lock_mutex; | ||
} arch_rwlock_t; | ||
|
||
#define __ARCH_RW_LOCK_UNLOCKED__ 0x01000000 | ||
#define __ARCH_RW_LOCK_UNLOCKED { .counter = __ARCH_RW_LOCK_UNLOCKED__ } | ||
|
||
#endif |