diff options
author | Tony Luck <tony.luck@intel.com> | 2009-10-24 02:59:06 +0200 |
---|---|---|
committer | Tony Luck <tony.luck@intel.com> | 2009-10-24 02:59:06 +0200 |
commit | 8add570a70d2ea6e9e15fce001c6503589c833d1 (patch) | |
tree | 2d9c3e75bbd0a9673f9a82d5a8cb98ef67585752 | |
parent | Pull ticket4byte into release branch (diff) | |
parent | [IA64] Re-implement spinaphores using ticket lock concepts (diff) | |
download | linux-8add570a70d2ea6e9e15fce001c6503589c833d1.tar.xz linux-8add570a70d2ea6e9e15fce001c6503589c833d1.zip |
Pull ticket-spinaphore into release branch
-rw-r--r-- | arch/ia64/mm/tlb.c | 24 |
1 files changed, 18 insertions, 6 deletions
diff --git a/arch/ia64/mm/tlb.c b/arch/ia64/mm/tlb.c index f426dc78d959..ee09d261f2e6 100644 --- a/arch/ia64/mm/tlb.c +++ b/arch/ia64/mm/tlb.c @@ -100,24 +100,36 @@ wrap_mmu_context (struct mm_struct *mm) * this primitive it can be moved up to a spinaphore.h header. */ struct spinaphore { - atomic_t cur; + unsigned long ticket; + unsigned long serve; }; static inline void spinaphore_init(struct spinaphore *ss, int val) { - atomic_set(&ss->cur, val); + ss->ticket = 0; + ss->serve = val; } static inline void down_spin(struct spinaphore *ss) { - while (unlikely(!atomic_add_unless(&ss->cur, -1, 0))) - while (atomic_read(&ss->cur) == 0) - cpu_relax(); + unsigned long t = ia64_fetchadd(1, &ss->ticket, acq), serve; + + if (time_before(t, ss->serve)) + return; + + ia64_invala(); + + for (;;) { + asm volatile ("ld4.c.nc %0=[%1]" : "=r"(serve) : "r"(&ss->serve) : "memory"); + if (time_before(t, serve)) + return; + cpu_relax(); + } } static inline void up_spin(struct spinaphore *ss) { - atomic_add(1, &ss->cur); + ia64_fetchadd(1, &ss->serve, rel); } static struct spinaphore ptcg_sem; |