/* SPDX-License-Identifier: GPL-3.0-or-later * Copyright © 2016-2025 The TokTok team. */ /** * C language compatibility macros for varying compiler support. */ #ifndef C_TOXCORE_TOXCORE_CCOMPAT_H #define C_TOXCORE_TOXCORE_CCOMPAT_H #include // NULL, size_t #include "attributes.h" //!TOKSTYLE- // Variable length arrays. // VLA(type, name, size) allocates a variable length array with automatic // storage duration. VLA_SIZE(name) evaluates to the runtime size of that array // in bytes. // // If C99 VLAs are not available, an emulation using alloca (stack allocation // "function") is used. Note the semantic difference: alloca'd memory does not // get freed at the end of the declaration's scope. Do not use VLA() in loops or // you may run out of stack space. #if !defined(DISABLE_VLA) && !defined(_MSC_VER) && defined(__STDC_VERSION__) && __STDC_VERSION__ >= 199901L // C99 VLAs. #define ALLOC_VLA(type, name, size) type name[size] #else // Emulation using alloca. #ifdef _WIN32 #include #elif defined(__COMPCERT__) // TODO(iphydf): This leaks memory like crazy, so compcert is useless for now. // Once we're rid of VLAs, we can remove this and compcert becomes useful. #define alloca malloc #include #elif defined(__linux__) #include #else #include #if !defined(alloca) && defined(__GNUC__) #define alloca __builtin_alloca #endif #endif #define ALLOC_VLA(type, name, size) \ type *const name = (type *)alloca((size) * sizeof(type)) #endif #ifdef MAX_VLA_SIZE #include #define VLA(type, name, size) \ ALLOC_VLA(type, name, size); \ assert((size_t)(size) * sizeof(type) <= MAX_VLA_SIZE) #else #define VLA ALLOC_VLA #endif #if !defined(__cplusplus) || __cplusplus < 201103L #define nullptr NULL #ifndef static_assert #ifdef __GNUC__ // We'll just assume gcc and clang support C11 _Static_assert. #define static_assert _Static_assert #else // !__GNUC__ #define STATIC_ASSERT_(cond, msg, line) typedef int static_assert_##line[(cond) ? 1 : -1] #define STATIC_ASSERT(cond, msg, line) STATIC_ASSERT_(cond, msg, line) #define static_assert(cond, msg) STATIC_ASSERT(cond, msg, __LINE__) #endif /* !__GNUC__ */ #endif /* !static_assert */ #endif /* !__cplusplus */ //!TOKSTYLE+ #endif /* C_TOXCORE_TOXCORE_CCOMPAT_H */