x86, asm: Merge cmpxchg_486_u64() and cmpxchg8b_emu()

We have two functions for doing exactly the same thing -- emulating
cmpxchg8b on 486 and older hardware -- with different calling
conventions, and yet doing the same thing.  Drop the C version and use
the assembly version, via alternatives, for both the local and
non-local versions of cmpxchg8b.

Signed-off-by: H. Peter Anvin <hpa@linux.intel.com>
LKML-Reference: <AANLkTikAmaDPji-TVDarmG1yD=fwbffcsmEU=YEuP+8r@mail.gmail.com>
This commit is contained in:
H. Peter Anvin 2010-07-28 17:05:11 -07:00
parent 90c8f92f5c
commit a378d9338e
2 changed files with 14 additions and 34 deletions

View File

@ -246,8 +246,6 @@ static inline unsigned long cmpxchg_386(volatile void *ptr, unsigned long old,
* to simulate the cmpxchg8b on the 80386 and 80486 CPU. * to simulate the cmpxchg8b on the 80386 and 80486 CPU.
*/ */
extern unsigned long long cmpxchg_486_u64(volatile void *, u64, u64);
#define cmpxchg64(ptr, o, n) \ #define cmpxchg64(ptr, o, n) \
({ \ ({ \
__typeof__(*(ptr)) __ret; \ __typeof__(*(ptr)) __ret; \
@ -265,20 +263,20 @@ extern unsigned long long cmpxchg_486_u64(volatile void *, u64, u64);
__ret; }) __ret; })
#define cmpxchg64_local(ptr, o, n) \
#define cmpxchg64_local(ptr, o, n) \ ({ \
({ \ __typeof__(*(ptr)) __ret; \
__typeof__(*(ptr)) __ret; \ __typeof__(*(ptr)) __old = (o); \
if (likely(boot_cpu_data.x86 > 4)) \ __typeof__(*(ptr)) __new = (n); \
__ret = (__typeof__(*(ptr)))__cmpxchg64_local((ptr), \ alternative_io("call cmpxchg8b_emu", \
(unsigned long long)(o), \ "cmpxchg8b (%%esi)" , \
(unsigned long long)(n)); \ X86_FEATURE_CX8, \
else \ "=A" (__ret), \
__ret = (__typeof__(*(ptr)))cmpxchg_486_u64((ptr), \ "S" ((ptr)), "0" (__old), \
(unsigned long long)(o), \ "b" ((unsigned int)__new), \
(unsigned long long)(n)); \ "c" ((unsigned int)(__new>>32)) \
__ret; \ : "memory"); \
}) __ret; })
#endif #endif

View File

@ -52,21 +52,3 @@ unsigned long cmpxchg_386_u32(volatile void *ptr, u32 old, u32 new)
} }
EXPORT_SYMBOL(cmpxchg_386_u32); EXPORT_SYMBOL(cmpxchg_386_u32);
#endif #endif
#ifndef CONFIG_X86_CMPXCHG64
unsigned long long cmpxchg_486_u64(volatile void *ptr, u64 old, u64 new)
{
u64 prev;
unsigned long flags;
/* Poor man's cmpxchg8b for 386 and 486. Unsuitable for SMP */
local_irq_save(flags);
prev = *(u64 *)ptr;
if (prev == old)
*(u64 *)ptr = new;
local_irq_restore(flags);
return prev;
}
EXPORT_SYMBOL(cmpxchg_486_u64);
#endif