2021-10-06 12:03:09 +01:00
|
|
|
/*
|
2024-02-27 10:00:06 +08:00
|
|
|
* SPDX-FileCopyrightText: 2015-2024 Espressif Systems (Shanghai) CO LTD
|
2021-10-06 12:03:09 +01:00
|
|
|
*
|
|
|
|
* SPDX-License-Identifier: Apache-2.0
|
|
|
|
*/
|
2019-06-13 14:15:10 +08:00
|
|
|
|
|
|
|
//replacement for gcc built-in functions
|
|
|
|
|
2019-12-26 18:20:54 +08:00
|
|
|
#include <stdbool.h>
|
2021-02-19 21:41:26 +08:00
|
|
|
#include <stdint.h>
|
2022-03-23 21:07:26 +03:00
|
|
|
#include <string.h>
|
2024-01-01 14:13:02 +04:00
|
|
|
#include "esp_stdatomic.h"
|
2021-05-06 19:26:21 +02:00
|
|
|
#include "freertos/FreeRTOS.h"
|
2024-01-01 14:13:02 +04:00
|
|
|
#include "sdkconfig.h"
|
2021-02-19 21:41:26 +08:00
|
|
|
|
2024-01-01 14:13:02 +04:00
|
|
|
#if SOC_CPU_CORES_NUM > 1
|
|
|
|
#if !CONFIG_STDATOMIC_S32C1I_SPIRAM_WORKAROUND
|
2021-05-06 19:26:21 +02:00
|
|
|
_Static_assert(HAS_ATOMICS_32, "32-bit atomics should be supported if SOC_CPU_CORES_NUM > 1");
|
2024-01-01 14:13:02 +04:00
|
|
|
#endif // CONFIG_STDATOMIC_S32C1I_SPIRAM_WORKAROUND
|
2021-05-06 19:26:21 +02:00
|
|
|
// Only need to implement 64-bit atomics here. Use a single global portMUX_TYPE spinlock
|
|
|
|
// to emulate the atomics.
|
|
|
|
static portMUX_TYPE s_atomic_lock = portMUX_INITIALIZER_UNLOCKED;
|
2024-01-01 14:13:02 +04:00
|
|
|
#endif
|
2021-10-06 12:03:09 +01:00
|
|
|
|
2021-05-06 19:26:21 +02:00
|
|
|
#if !HAS_ATOMICS_32
|
2021-02-19 21:41:26 +08:00
|
|
|
|
2022-01-20 09:27:52 +07:00
|
|
|
_Static_assert(sizeof(unsigned char) == 1, "atomics require a 1-byte type");
|
|
|
|
_Static_assert(sizeof(short unsigned int) == 2, "atomics require a 2-bytes type");
|
|
|
|
_Static_assert(sizeof(unsigned int) == 4, "atomics require a 4-bytes type");
|
2021-04-22 14:43:54 +05:30
|
|
|
|
2024-01-01 14:13:02 +04:00
|
|
|
ATOMIC_FUNCTIONS(1, unsigned char)
|
|
|
|
ATOMIC_FUNCTIONS(2, short unsigned int)
|
|
|
|
ATOMIC_FUNCTIONS(4, unsigned int)
|
2021-10-06 12:03:09 +01:00
|
|
|
|
2023-01-31 23:19:21 +07:00
|
|
|
#elif __riscv_atomic == 1
|
|
|
|
|
2024-02-27 10:00:06 +08:00
|
|
|
bool CLANG_ATOMIC_SUFFIX(__atomic_always_lock_free)(unsigned int size, const volatile void *)
|
|
|
|
{
|
|
|
|
return size <= sizeof(int);
|
2023-01-31 23:19:21 +07:00
|
|
|
}
|
2024-02-27 10:00:06 +08:00
|
|
|
CLANG_DECLARE_ALIAS(__atomic_always_lock_free)
|
2023-01-31 23:19:21 +07:00
|
|
|
|
2024-02-27 10:00:06 +08:00
|
|
|
bool CLANG_ATOMIC_SUFFIX(__atomic_is_lock_free)(unsigned int size, const volatile void *)
|
|
|
|
{
|
|
|
|
return size <= sizeof(int);
|
2023-01-31 23:19:21 +07:00
|
|
|
}
|
2024-02-27 10:00:06 +08:00
|
|
|
CLANG_DECLARE_ALIAS(__atomic_is_lock_free)
|
2023-01-31 23:19:21 +07:00
|
|
|
|
2021-05-06 19:26:21 +02:00
|
|
|
#endif // !HAS_ATOMICS_32
|
|
|
|
|
|
|
|
#if !HAS_ATOMICS_64
|
|
|
|
|
2024-01-01 14:13:02 +04:00
|
|
|
#if CONFIG_STDATOMIC_S32C1I_SPIRAM_WORKAROUND
|
|
|
|
#undef _ATOMIC_HW_STUB_OP_FUNCTION
|
|
|
|
#undef _ATOMIC_HW_STUB_EXCHANGE
|
|
|
|
#undef _ATOMIC_HW_STUB_STORE
|
|
|
|
#undef _ATOMIC_HW_STUB_CMP_EXCHANGE
|
|
|
|
#undef _ATOMIC_HW_STUB_LOAD
|
|
|
|
#undef _ATOMIC_HW_STUB_SYNC_BOOL_CMP_EXCHANGE
|
|
|
|
#undef _ATOMIC_HW_STUB_SYNC_VAL_CMP_EXCHANGE
|
|
|
|
#undef _ATOMIC_HW_STUB_SYNC_LOCK_TEST_AND_SET
|
|
|
|
#undef _ATOMIC_HW_STUB_SYNC_LOCK_RELEASE
|
|
|
|
|
|
|
|
#define _ATOMIC_HW_STUB_OP_FUNCTION(n, type, name_1, name_2)
|
|
|
|
#define _ATOMIC_HW_STUB_EXCHANGE(n, type)
|
|
|
|
#define _ATOMIC_HW_STUB_STORE(n, type)
|
|
|
|
#define _ATOMIC_HW_STUB_CMP_EXCHANGE(n, type)
|
|
|
|
#define _ATOMIC_HW_STUB_LOAD(n, type)
|
|
|
|
#define _ATOMIC_HW_STUB_SYNC_BOOL_CMP_EXCHANGE(n, type)
|
|
|
|
#define _ATOMIC_HW_STUB_SYNC_VAL_CMP_EXCHANGE(n, type)
|
|
|
|
#define _ATOMIC_HW_STUB_SYNC_LOCK_TEST_AND_SET(n, type)
|
|
|
|
#define _ATOMIC_HW_STUB_SYNC_LOCK_RELEASE(n, type)
|
|
|
|
#endif // CONFIG_STDATOMIC_S32C1I_SPIRAM_WORKAROUND
|
2021-12-14 16:06:12 +01:00
|
|
|
|
2024-01-01 14:13:02 +04:00
|
|
|
_Static_assert(sizeof(long long unsigned int) == 8, "atomics require a 8-bytes type");
|
2021-10-06 12:03:09 +01:00
|
|
|
|
2024-01-01 14:13:02 +04:00
|
|
|
ATOMIC_FUNCTIONS(8, long long unsigned int)
|
2021-10-06 12:03:09 +01:00
|
|
|
|
2021-05-06 19:26:21 +02:00
|
|
|
#endif // !HAS_ATOMICS_64
|
2022-03-23 21:07:26 +03:00
|
|
|
|
|
|
|
// Clang generates calls to the __atomic_load/__atomic_store functions for object size more then 4 bytes
|
2024-02-27 10:00:06 +08:00
|
|
|
void CLANG_ATOMIC_SUFFIX(__atomic_load)(size_t size, const volatile void *src, void *dest, int model)
|
|
|
|
{
|
2024-01-01 14:13:02 +04:00
|
|
|
_ATOMIC_ENTER_CRITICAL();
|
2022-05-13 13:33:04 +07:00
|
|
|
memcpy(dest, (const void *)src, size);
|
2024-01-01 14:13:02 +04:00
|
|
|
_ATOMIC_EXIT_CRITICAL();
|
2022-03-23 21:07:26 +03:00
|
|
|
}
|
2024-02-27 10:00:06 +08:00
|
|
|
CLANG_DECLARE_ALIAS(__atomic_load)
|
2022-03-23 21:07:26 +03:00
|
|
|
|
2024-02-27 10:00:06 +08:00
|
|
|
void CLANG_ATOMIC_SUFFIX(__atomic_store)(size_t size, volatile void *dest, void *src, int model)
|
|
|
|
{
|
2024-01-01 14:13:02 +04:00
|
|
|
_ATOMIC_ENTER_CRITICAL();
|
2022-05-13 13:33:04 +07:00
|
|
|
memcpy((void *)dest, (const void *)src, size);
|
2024-01-01 14:13:02 +04:00
|
|
|
_ATOMIC_EXIT_CRITICAL();
|
2022-03-23 21:07:26 +03:00
|
|
|
}
|
2024-02-27 10:00:06 +08:00
|
|
|
CLANG_DECLARE_ALIAS(__atomic_store)
|
2023-01-31 19:35:10 +07:00
|
|
|
|
2024-02-27 10:00:06 +08:00
|
|
|
bool CLANG_ATOMIC_SUFFIX(__atomic_compare_exchange)(size_t size, volatile void *ptr, void *expected, void *desired, int success_memorder, int failure_memorder)
|
|
|
|
{
|
2023-01-31 19:35:10 +07:00
|
|
|
bool ret = false;
|
2024-01-01 14:13:02 +04:00
|
|
|
_ATOMIC_ENTER_CRITICAL();
|
2023-01-31 19:35:10 +07:00
|
|
|
if (!memcmp((void *)ptr, expected, size)) {
|
|
|
|
memcpy((void *)ptr, (const void *)desired, size);
|
|
|
|
ret = true;
|
|
|
|
} else {
|
|
|
|
memcpy((void *)expected, (const void *)ptr, size);
|
|
|
|
}
|
2024-01-01 14:13:02 +04:00
|
|
|
_ATOMIC_EXIT_CRITICAL();
|
2023-01-31 19:35:10 +07:00
|
|
|
return ret;
|
|
|
|
}
|
2024-02-27 10:00:06 +08:00
|
|
|
CLANG_DECLARE_ALIAS(__atomic_compare_exchange)
|