| /* |
| * |
| * linux/lib/brlock.c |
| * |
| * 'Big Reader' read-write spinlocks. See linux/brlock.h for details. |
| * |
| * Copyright 2000, Ingo Molnar <mingo@redhat.com> |
| * Copyright 2000, David S. Miller <davem@redhat.com> |
| */ |
| |
| #include <linux/config.h> |
| |
| #ifdef CONFIG_SMP |
| |
| #include <linux/sched.h> |
| #include <linux/brlock.h> |
| |
| #ifdef __BRLOCK_USE_ATOMICS |
| |
| brlock_read_lock_t __brlock_array[NR_CPUS][__BR_IDX_MAX] = |
| { [0 ... NR_CPUS-1] = { [0 ... __BR_IDX_MAX-1] = RW_LOCK_UNLOCKED } }; |
| |
| void __br_write_lock (enum brlock_indices idx) |
| { |
| int i; |
| |
| preempt_disable(); |
| for (i = 0; i < NR_CPUS; i++) |
| _raw_write_lock(&__brlock_array[i][idx]); |
| } |
| |
| void __br_write_unlock (enum brlock_indices idx) |
| { |
| int i; |
| |
| for (i = 0; i < NR_CPUS; i++) |
| _raw_write_unlock(&__brlock_array[i][idx]); |
| preempt_enable(); |
| } |
| |
| #else /* ! __BRLOCK_USE_ATOMICS */ |
| |
| brlock_read_lock_t __brlock_array[NR_CPUS][__BR_IDX_MAX] = |
| { [0 ... NR_CPUS-1] = { [0 ... __BR_IDX_MAX-1] = 0 } }; |
| |
| struct br_wrlock __br_write_locks[__BR_IDX_MAX] = |
| { [0 ... __BR_IDX_MAX-1] = { SPIN_LOCK_UNLOCKED } }; |
| |
| void __br_write_lock (enum brlock_indices idx) |
| { |
| int i; |
| |
| preempt_disable(); |
| again: |
| _raw_spin_lock(&__br_write_locks[idx].lock); |
| for (i = 0; i < NR_CPUS; i++) |
| if (__brlock_array[i][idx] != 0) { |
| _raw_spin_unlock(&__br_write_locks[idx].lock); |
| barrier(); |
| cpu_relax(); |
| goto again; |
| } |
| } |
| |
| void __br_write_unlock (enum brlock_indices idx) |
| { |
| spin_unlock(&__br_write_locks[idx].lock); |
| } |
| |
| #endif /* __BRLOCK_USE_ATOMICS */ |
| |
| #endif /* CONFIG_SMP */ |