Zephyr API Documentation 4.3.99
A Scalable Open Source RTOS
Loading...
Searching...
No Matches
atomic.h
Go to the documentation of this file.
1/*
2 * Copyright (c) 1997-2015, Wind River Systems, Inc.
3 * Copyright (c) 2021 Intel Corporation
4 * Copyright (c) 2023 Nordic Semiconductor ASA
5 *
6 * SPDX-License-Identifier: Apache-2.0
7 */
8
9#ifndef ZEPHYR_INCLUDE_SYS_ATOMIC_H_
10#define ZEPHYR_INCLUDE_SYS_ATOMIC_H_
11
12#include <stdbool.h>
13#include <zephyr/toolchain.h>
14#include <stddef.h>
15
16#include <zephyr/sys/atomic_types.h> /* IWYU pragma: export */
17#include <zephyr/types.h>
18#include <zephyr/sys/util.h>
19
20#ifdef __cplusplus
21extern "C" {
22#endif
23
24/* Low-level primitives come in several styles: */
25
26#if defined(CONFIG_ATOMIC_OPERATIONS_C)
27/* Generic-but-slow implementation based on kernel locking and syscalls */
28#include <zephyr/sys/atomic_c.h>
29#elif defined(CONFIG_ATOMIC_OPERATIONS_ARCH)
30/* Some architectures need their own implementation */
31# ifdef CONFIG_XTENSA
32/* Not all Xtensa toolchains support GCC-style atomic intrinsics */
34# else
35/* Other arch specific implementation */
37# endif /* CONFIG_XTENSA */
38#elif defined(CONFIG_ATOMIC_OPERATIONS_BUILTIN)
39/* Default. See this file for the Doxygen reference: */
41#else
42#error "CONFIG_ATOMIC_OPERATIONS_* not defined"
43#endif
44
45/* Portable higher-level utilities: */
46
52
61#define ATOMIC_INIT(i) (i)
62
72#define ATOMIC_PTR_INIT(p) (p)
73
77
78#define ATOMIC_BITS (sizeof(atomic_val_t) * BITS_PER_BYTE)
79#define ATOMIC_MASK(bit) BIT((unsigned long)(bit) & (ATOMIC_BITS - 1U))
80#define ATOMIC_ELEM(addr, bit) ((addr) + ((bit) / ATOMIC_BITS))
81
85
92#define ATOMIC_BITMAP_SIZE(num_bits) (ROUND_UP(num_bits, ATOMIC_BITS) / ATOMIC_BITS)
93
113#define ATOMIC_DEFINE(name, num_bits) \
114 atomic_t name[ATOMIC_BITMAP_SIZE(num_bits)]
115
129static inline bool atomic_test_bit(const atomic_t *target, int bit)
130{
131 atomic_val_t val = atomic_get(ATOMIC_ELEM(target, bit));
132
133 return (1 & (val >> (bit & (ATOMIC_BITS - 1)))) != 0;
134}
135
149static inline bool atomic_test_and_clear_bit(atomic_t *target, int bit)
150{
151 atomic_val_t mask = ATOMIC_MASK(bit);
152 atomic_val_t old;
153
154 old = atomic_and(ATOMIC_ELEM(target, bit), ~mask);
155
156 return (old & mask) != 0;
157}
158
172static inline bool atomic_test_and_set_bit(atomic_t *target, int bit)
173{
174 atomic_val_t mask = ATOMIC_MASK(bit);
175 atomic_val_t old;
176
177 old = atomic_or(ATOMIC_ELEM(target, bit), mask);
178
179 return (old & mask) != 0;
180}
181
193static inline void atomic_clear_bit(atomic_t *target, int bit)
194{
195 atomic_val_t mask = ATOMIC_MASK(bit);
196
197 (void)atomic_and(ATOMIC_ELEM(target, bit), ~mask);
198}
199
211static inline void atomic_set_bit(atomic_t *target, int bit)
212{
213 atomic_val_t mask = ATOMIC_MASK(bit);
214
215 (void)atomic_or(ATOMIC_ELEM(target, bit), mask);
216}
217
230static inline void atomic_set_bit_to(atomic_t *target, int bit, bool val)
231{
232 atomic_val_t mask = ATOMIC_MASK(bit);
233
234 if (val) {
235 (void)atomic_or(ATOMIC_ELEM(target, bit), mask);
236 } else {
237 (void)atomic_and(ATOMIC_ELEM(target, bit), ~mask);
238 }
239}
240
256bool atomic_cas(atomic_t *target, atomic_val_t old_value, atomic_val_t new_value);
257
274 atomic_ptr_val_t new_value);
275
289
303
316
329
342
355
370
385
399
413
428
443
458
473
477
478#ifdef __cplusplus
479} /* extern "C" */
480#endif
481
482#endif /* ZEPHYR_INCLUDE_SYS_ATOMIC_H_ */
long atomic_t
Definition atomic_types.h:15
atomic_t atomic_val_t
Definition atomic_types.h:16
atomic_ptr_t atomic_ptr_val_t
Definition atomic_types.h:18
void * atomic_ptr_t
Definition atomic_types.h:17
static _Bool atomic_test_and_set_bit(atomic_t *target, int bit)
Atomically set a bit and test it.
Definition atomic.h:172
static _Bool atomic_test_bit(const atomic_t *target, int bit)
Atomically get and test a bit.
Definition atomic.h:129
atomic_val_t atomic_or(atomic_t *target, atomic_val_t value)
Atomic bitwise inclusive OR.
static void atomic_set_bit(atomic_t *target, int bit)
Atomically set a bit.
Definition atomic.h:211
atomic_val_t atomic_xor(atomic_t *target, atomic_val_t value)
Atomic bitwise exclusive OR (XOR).
static void atomic_clear_bit(atomic_t *target, int bit)
Atomically clear a bit.
Definition atomic.h:193
atomic_ptr_val_t atomic_ptr_get(const atomic_ptr_t *target)
Atomic get a pointer value.
atomic_val_t atomic_get(const atomic_t *target)
Atomic get.
atomic_ptr_val_t atomic_ptr_set(atomic_ptr_t *target, atomic_ptr_val_t value)
Atomic get-and-set for pointer values.
atomic_val_t atomic_nand(atomic_t *target, atomic_val_t value)
Atomic bitwise NAND.
static _Bool atomic_test_and_clear_bit(atomic_t *target, int bit)
Atomically clear a bit and test it.
Definition atomic.h:149
atomic_val_t atomic_and(atomic_t *target, atomic_val_t value)
Atomic bitwise AND.
atomic_val_t atomic_add(atomic_t *target, atomic_val_t value)
Atomic addition.
atomic_ptr_val_t atomic_ptr_clear(atomic_ptr_t *target)
Atomic clear of a pointer value.
atomic_val_t atomic_set(atomic_t *target, atomic_val_t value)
Atomic get-and-set.
static void atomic_set_bit_to(atomic_t *target, int bit, _Bool val)
Atomically set a bit to a given value.
Definition atomic.h:230
atomic_val_t atomic_sub(atomic_t *target, atomic_val_t value)
Atomic subtraction.
atomic_val_t atomic_clear(atomic_t *target)
Atomic clear.
atomic_val_t atomic_inc(atomic_t *target)
Atomic increment.
atomic_val_t atomic_dec(atomic_t *target)
Atomic decrement.
_Bool atomic_ptr_cas(atomic_ptr_t *target, atomic_ptr_val_t old_value, atomic_ptr_val_t new_value)
Atomic compare-and-set with pointer values.
_Bool atomic_cas(atomic_t *target, atomic_val_t old_value, atomic_val_t new_value)
Atomic compare-and-set.
Misc utilities.
Macros to abstract toolchain specific capabilities.