source: mainline/kernel/arch/arm32/src/atomic.c@ 4621d23

lfn serial ticket/834-toolchain-update topic/msim-upgrade topic/simplify-dev-export
Last change on this file since 4621d23 was 4621d23, checked in by Jiří Zárevúcky <jiri.zarevucky@…>, 7 years ago

Use compiler builtins for kernel atomics

  • Property mode set to 100644
File size: 3.5 KB
Line 
1/*
2 * Copyright (c) 2012 Adam Hraska
3 * All rights reserved.
4 *
5 * Redistribution and use in source and binary forms, with or without
6 * modification, are permitted provided that the following conditions
7 * are met:
8 *
9 * - Redistributions of source code must retain the above copyright
10 * notice, this list of conditions and the following disclaimer.
11 * - Redistributions in binary form must reproduce the above copyright
12 * notice, this list of conditions and the following disclaimer in the
13 * documentation and/or other materials provided with the distribution.
14 * - The name of the author may not be used to endorse or promote products
15 * derived from this software without specific prior written permission.
16 *
17 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
18 * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
19 * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
20 * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
21 * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
22 * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
23 * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
24 * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
25 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
26 * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
27 */
28
29/** @addtogroup arm32
30 * @{
31 */
32/** @file
33 * @brief Atomic operations emulation.
34 */
35
36#include <synch/spinlock.h>
37#include <arch/barrier.h>
38#include <arch/asm.h>
39
40unsigned __atomic_fetch_add_4(volatile unsigned *mem, unsigned val, int model)
41{
42 /*
43 * This implementation is for UP pre-ARMv6 systems where we do not have
44 * the LDREX and STREX instructions.
45 */
46 ipl_t ipl = interrupts_disable();
47 unsigned ret = *mem;
48 *mem += val;
49 interrupts_restore(ipl);
50 return ret;
51}
52
53unsigned __atomic_fetch_sub_4(volatile unsigned *mem, unsigned val, int model)
54{
55 ipl_t ipl = interrupts_disable();
56 unsigned ret = *mem;
57 *mem -= val;
58 interrupts_restore(ipl);
59 return ret;
60}
61
62IRQ_SPINLOCK_STATIC_INITIALIZE_NAME(cas_lock, "arm-cas-lock");
63
64/** Implements GCC's missing compare-and-swap intrinsic for ARM.
65 *
66 * Sets \a *ptr to \a new_val if it is equal to \a expected. In any case,
67 * returns the previous value of \a *ptr.
68 */
69void *__sync_val_compare_and_swap_4(void **ptr, void *expected, void *new_val)
70{
71 /*
72 * Using an interrupt disabling spinlock might still lead to deadlock
73 * if CAS() is used in an exception handler. Eg. if a CAS() results
74 * in a page fault exception and the exception handler again tries
75 * to invoke CAS() (even for a different memory location), the spinlock
76 * would deadlock.
77 */
78 irq_spinlock_lock(&cas_lock, true);
79
80 void *cur_val = *ptr;
81
82 if (cur_val == expected) {
83 *ptr = new_val;
84 }
85
86 irq_spinlock_unlock(&cas_lock, true);
87
88 return cur_val;
89}
90
91void __sync_synchronize(void)
92{
93 dsb();
94}
95
96/* Naive implementations of the newer intrinsics. */
97
98_Bool __atomic_compare_exchange_4(void **mem, void **expected, void *desired, _Bool weak, int success, int failure)
99{
100 (void) weak;
101 (void) success;
102 (void) failure;
103
104 void *old = *expected;
105 void *new = __sync_val_compare_and_swap_4(mem, old, desired);
106 if (old == new) {
107 return 1;
108 } else {
109 *expected = new;
110 return 0;
111 }
112}
113
114void *__atomic_exchange_4(void **mem, void *val, int model)
115{
116 (void) model;
117
118 irq_spinlock_lock(&cas_lock, true);
119 void *old = *mem;
120 *mem = val;
121 irq_spinlock_unlock(&cas_lock, true);
122
123 return old;
124}
125
126/** @}
127 */
Note: See TracBrowser for help on using the repository browser.