spinlock.h
上传用户:jlfgdled
上传日期:2013-04-10
资源大小:33168k
文件大小:4k
- #ifndef __ASM_SPINLOCK_H
- #define __ASM_SPINLOCK_H
- /*
- * Simple spin lock operations.
- *
- * Copyright (C) 2001 Paul Mackerras <paulus@au.ibm.com>, IBM
- * Copyright (C) 2001 Anton Blanchard <anton@au.ibm.com>, IBM
- *
- * Type of int is used as a full 64b word is not necessary.
- *
- * This program is free software; you can redistribute it and/or
- * modify it under the terms of the GNU General Public License
- * as published by the Free Software Foundation; either version
- * 2 of the License, or (at your option) any later version.
- */
- typedef struct {
- volatile unsigned int lock;
- } spinlock_t;
- #ifdef __KERNEL__
- #define SPIN_LOCK_UNLOCKED (spinlock_t) { 0 }
- #define spin_is_locked(x) ((x)->lock != 0)
- static __inline__ int spin_trylock(spinlock_t *lock)
- {
- unsigned int tmp;
- __asm__ __volatile__(
- "1: lwarx %0,0,%1 # spin_trylockn
- cmpwi 0,%0,0n
- li %0,0n
- bne- 2fn
- li %0,1n
- stwcx. %0,0,%1n
- bne- 1bn
- isyncn
- 2:" : "=&r"(tmp)
- : "r"(&lock->lock)
- : "cr0", "memory");
- return tmp;
- }
- static __inline__ void spin_lock(spinlock_t *lock)
- {
- unsigned int tmp;
- __asm__ __volatile__(
- "b 2f # spin_lockn
- 1: or 1,1,1 # spin at low priorityn
- lwzx %0,0,%1n
- cmpwi 0,%0,0n
- bne+ 1bn
- or 2,2,2 # back to medium priorityn
- 2: lwarx %0,0,%1n
- cmpwi 0,%0,0n
- bne- 1bn
- stwcx. %2,0,%1n
- bne- 2bn
- isync"
- : "=&r"(tmp)
- : "r"(&lock->lock), "r"(1)
- : "cr0", "memory");
- }
- static __inline__ void spin_unlock(spinlock_t *lock)
- {
- __asm__ __volatile__("lwsync # spin_unlock": : :"memory");
- lock->lock = 0;
- }
- /*
- * Read-write spinlocks, allowing multiple readers
- * but only one writer.
- *
- * NOTE! it is quite common to have readers in interrupts
- * but no interrupt writers. For those circumstances we
- * can "mix" irq-safe locks - any writer needs to get a
- * irq-safe write-lock, but readers can get non-irqsafe
- * read-locks.
- */
- typedef struct {
- volatile signed int lock;
- } rwlock_t;
- #define RW_LOCK_UNLOCKED (rwlock_t) { 0 }
- static __inline__ int read_trylock(rwlock_t *rw)
- {
- unsigned int tmp;
- unsigned int ret;
- __asm__ __volatile__(
- "1: lwarx %0,0,%2 # read_trylockn
- li %1,0n
- extsw %0,%0n
- addic. %0,%0,1n
- ble- 2fn
- stwcx. %0,0,%2n
- bne- 1bn
- li %1,1n
- isyncn
- 2:" : "=&r"(tmp), "=&r"(ret)
- : "r"(&rw->lock)
- : "cr0", "memory");
- return ret;
- }
- static __inline__ void read_lock(rwlock_t *rw)
- {
- unsigned int tmp;
- __asm__ __volatile__(
- "b 2f # read_lockn
- 1: or 1,1,1 # spin at low priorityn
- lwax %0,0,%1n
- cmpwi 0,%0,0n
- blt+ 1bn
- or 2,2,2 # back to medium priorityn
- 2: lwarx %0,0,%1n
- extsw %0,%0n
- addic. %0,%0,1n
- ble- 1bn
- stwcx. %0,0,%1n
- bne- 2bn
- isync"
- : "=&r"(tmp)
- : "r"(&rw->lock)
- : "cr0", "memory");
- }
- static __inline__ void read_unlock(rwlock_t *rw)
- {
- unsigned int tmp;
- __asm__ __volatile__(
- "lwsync # read_unlockn
- 1: lwarx %0,0,%1n
- addic %0,%0,-1n
- stwcx. %0,0,%1n
- bne- 1b"
- : "=&r"(tmp)
- : "r"(&rw->lock)
- : "cr0", "memory");
- }
- static __inline__ int write_trylock(rwlock_t *rw)
- {
- unsigned int tmp;
- unsigned int ret;
- __asm__ __volatile__(
- "1: lwarx %0,0,%2 # write_trylockn
- cmpwi 0,%0,0n
- li %1,0n
- bne- 2fn
- stwcx. %3,0,%2n
- bne- 1bn
- li %1,1n
- isyncn
- 2:" : "=&r"(tmp), "=&r"(ret)
- : "r"(&rw->lock), "r"(-1)
- : "cr0", "memory");
- return ret;
- }
- static __inline__ void write_lock(rwlock_t *rw)
- {
- unsigned int tmp;
- __asm__ __volatile__(
- "b 2f # write_lockn
- 1: or 1,1,1 # spin at low priorityn
- lwax %0,0,%1n
- cmpwi 0,%0,0n
- bne+ 1bn
- or 2,2,2 # back to medium priorityn
- 2: lwarx %0,0,%1n
- cmpwi 0,%0,0n
- bne- 1bn
- stwcx. %2,0,%1n
- bne- 2bn
- isync"
- : "=&r"(tmp)
- : "r"(&rw->lock), "r"(-1)
- : "cr0", "memory");
- }
- static __inline__ void write_unlock(rwlock_t *rw)
- {
- __asm__ __volatile__("lwsync # write_unlock": : :"memory");
- rw->lock = 0;
- }
- static __inline__ int is_read_locked(rwlock_t *rw)
- {
- return rw->lock > 0;
- }
- static __inline__ int is_write_locked(rwlock_t *rw)
- {
- return rw->lock < 0;
- }
- #define spin_lock_init(x) do { *(x) = SPIN_LOCK_UNLOCKED; } while(0)
- #define spin_unlock_wait(x) do { barrier(); } while(spin_is_locked(x))
- #define rwlock_init(x) do { *(x) = RW_LOCK_UNLOCKED; } while(0)
- #endif /* __KERNEL__ */
- #endif /* __ASM_SPINLOCK_H */