blob: 70ffbcf889b8e34f6e4195e98d2c494f190693a9 [file] [log] [blame]
Greg Kroah-Hartmanb2441312017-11-01 15:07:57 +01001// SPDX-License-Identifier: GPL-2.0
Linus Torvalds1da177e2005-04-16 15:20:36 -07002/*
3 * bitops.c: atomic operations which got too long to be inlined all over
4 * the place.
5 *
6 * Copyright 1999 Philipp Rumpf (prumpf@tux.org)
7 * Copyright 2000 Grant Grundler (grundler@cup.hp.com)
8 */
9
Linus Torvalds1da177e2005-04-16 15:20:36 -070010#include <linux/kernel.h>
11#include <linux/spinlock.h>
Arun Sharma600634972011-07-26 16:09:06 -070012#include <linux/atomic.h>
Linus Torvalds1da177e2005-04-16 15:20:36 -070013
14#ifdef CONFIG_SMP
Thomas Gleixner445c8952009-12-02 19:49:50 +010015arch_spinlock_t __atomic_hash[ATOMIC_HASH_SIZE] __lock_aligned = {
Thomas Gleixneredc35bd2009-12-03 12:38:57 +010016 [0 ... (ATOMIC_HASH_SIZE-1)] = __ARCH_SPIN_LOCK_UNLOCKED
Linus Torvalds1da177e2005-04-16 15:20:36 -070017};
18#endif
19
Helge Dellera8f44e32007-01-28 14:58:52 +010020#ifdef CONFIG_64BIT
Linus Torvalds1da177e2005-04-16 15:20:36 -070021unsigned long __xchg64(unsigned long x, unsigned long *ptr)
22{
23 unsigned long temp, flags;
24
25 _atomic_spin_lock_irqsave(ptr, flags);
26 temp = *ptr;
27 *ptr = x;
28 _atomic_spin_unlock_irqrestore(ptr, flags);
29 return temp;
30}
31#endif
32
33unsigned long __xchg32(int x, int *ptr)
34{
35 unsigned long flags;
36 long temp;
37
38 _atomic_spin_lock_irqsave(ptr, flags);
39 temp = (long) *ptr; /* XXX - sign extension wanted? */
40 *ptr = x;
41 _atomic_spin_unlock_irqrestore(ptr, flags);
42 return (unsigned long)temp;
43}
44
45
46unsigned long __xchg8(char x, char *ptr)
47{
48 unsigned long flags;
49 long temp;
50
51 _atomic_spin_lock_irqsave(ptr, flags);
52 temp = (long) *ptr; /* XXX - sign extension wanted? */
53 *ptr = x;
54 _atomic_spin_unlock_irqrestore(ptr, flags);
55 return (unsigned long)temp;
56}
57
58
Helge Deller54b668002016-04-20 21:34:15 +020059u64 __cmpxchg_u64(volatile u64 *ptr, u64 old, u64 new)
Linus Torvalds1da177e2005-04-16 15:20:36 -070060{
61 unsigned long flags;
Helge Deller54b668002016-04-20 21:34:15 +020062 u64 prev;
Linus Torvalds1da177e2005-04-16 15:20:36 -070063
64 _atomic_spin_lock_irqsave(ptr, flags);
65 if ((prev = *ptr) == old)
66 *ptr = new;
67 _atomic_spin_unlock_irqrestore(ptr, flags);
68 return prev;
69}
Linus Torvalds1da177e2005-04-16 15:20:36 -070070
71unsigned long __cmpxchg_u32(volatile unsigned int *ptr, unsigned int old, unsigned int new)
72{
73 unsigned long flags;
74 unsigned int prev;
75
76 _atomic_spin_lock_irqsave(ptr, flags);
77 if ((prev = *ptr) == old)
78 *ptr = new;
79 _atomic_spin_unlock_irqrestore(ptr, flags);
80 return (unsigned long)prev;
81}