123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375 |
- /* This Source Code Form is subject to the terms of the Mozilla Public
- * License, v. 2.0. If a copy of the MPL was not distributed with this
- * file, You can obtain one at http://mozilla.org/MPL/2.0/.
- *
- * Copyright 2017 (c) Fraunhofer IOSB (Author: Julius Pfrommer)
- * Copyright 2017 (c) Stefan Profanter, fortiss GmbH
- * Copyright 2018 (c) Jose Cabral, fortiss GmbH
- */
- #ifndef ARCH_UA_ARCHITECTURE_DEFINITIONS_H_
- #define ARCH_UA_ARCHITECTURE_DEFINITIONS_H_
- /**
- * C99 Definitions
- * --------------- */
- #include <string.h>
- #include <stddef.h>
- /* Include stdint.h and stdbool.h or workaround for older Visual Studios */
- #ifdef UNDER_CE
- # include "stdint.h"
- #endif
- #if !defined(_MSC_VER) || _MSC_VER >= 1600
- # include <stdint.h>
- # include <stdbool.h> /* C99 Boolean */
- #else
- # include "ms_stdint.h"
- # if !defined(__bool_true_false_are_defined)
- # define bool unsigned char
- # define true 1
- # define false 0
- # define __bool_true_false_are_defined
- # endif
- #endif
- /**
- * Assertions
- * ----------
- * The assert macro is disabled by defining NDEBUG. It is often forgotten to
- * include -DNDEBUG in the compiler flags when using the single-file release. So
- * we make assertions dependent on the UA_DEBUG definition handled by CMake. */
- #ifdef UA_DEBUG
- # include <assert.h>
- # define UA_assert(ignore) assert(ignore)
- #else
- # define UA_assert(ignore)
- #endif
- /* Outputs an error message at compile time if the assert fails.
- * Example usage:
- * UA_STATIC_ASSERT(sizeof(long)==7, use_another_compiler_luke)
- * See: https://stackoverflow.com/a/4815532/869402 */
- #if defined(__cplusplus) && __cplusplus >= 201103L /* C++11 or above */
- # define UA_STATIC_ASSERT(cond,msg) static_assert(cond, #msg)
- #elif defined(__STDC_VERSION__) && __STDC_VERSION__ >= 201112L /* C11 or above */
- # define UA_STATIC_ASSERT(cond,msg) _Static_assert(cond, #msg)
- #elif defined(__GNUC__) || defined(__clang__) || defined(_MSC_VER) /* GCC, Clang, MSC */
- # define UA_CTASTR2(pre,post) pre ## post
- # define UA_CTASTR(pre,post) UA_CTASTR2(pre,post)
- # ifndef __COUNTER__ /* PPC GCC fix */
- # define __COUNTER__ __LINE__
- # endif
- # define UA_STATIC_ASSERT(cond,msg) \
- typedef struct { \
- int UA_CTASTR(static_assertion_failed_,msg) : !!(cond); \
- } UA_CTASTR(static_assertion_failed_,__COUNTER__)
- #else /* Everybody else */
- # define UA_STATIC_ASSERT(cond,msg) typedef char static_assertion_##msg[(cond)?1:-1]
- #endif
- #if defined(_WIN32) && defined(UA_DYNAMIC_LINKING)
- # ifdef UA_DYNAMIC_LINKING_EXPORT /* export dll */
- # ifdef __GNUC__
- # define UA_EXPORT __attribute__ ((dllexport))
- # else
- # define UA_EXPORT __declspec(dllexport)
- # endif
- # else /* import dll */
- # ifdef __GNUC__
- # define UA_EXPORT __attribute__ ((dllimport))
- # else
- # define UA_EXPORT __declspec(dllimport)
- # endif
- # endif
- #else /* non win32 */
- # if __GNUC__ || __clang__
- # define UA_EXPORT __attribute__ ((visibility ("default")))
- # endif
- #endif
- #ifndef UA_EXPORT
- # define UA_EXPORT /* fallback to default */
- #endif
- /**
- * Inline Functions
- * ---------------- */
- #ifdef _MSC_VER
- # define UA_INLINE __inline
- #else
- # define UA_INLINE inline
- #endif
- /**
- * Non-aliasing pointers
- * -------------------- */
- #ifdef _MSC_VER
- # define UA_RESTRICT __restrict
- #elif defined(__GNUC__)
- # define UA_RESTRICT __restrict__
- #else
- # define UA_RESTRICT restrict
- #endif
- /**
- * Function attributes
- * ------------------- */
- #if defined(__GNUC__) || defined(__clang__)
- # define UA_FUNC_ATTR_MALLOC __attribute__((malloc))
- # define UA_FUNC_ATTR_PURE __attribute__ ((pure))
- # define UA_FUNC_ATTR_CONST __attribute__((const))
- # define UA_FUNC_ATTR_WARN_UNUSED_RESULT __attribute__((warn_unused_result))
- # define UA_FORMAT(X,Y) __attribute__ ((format (printf, X, Y)))
- #else
- # define UA_FUNC_ATTR_MALLOC
- # define UA_FUNC_ATTR_PURE
- # define UA_FUNC_ATTR_CONST
- # define UA_FUNC_ATTR_WARN_UNUSED_RESULT
- # define UA_FORMAT(X,Y)
- #endif
- #if defined(__GNUC__) || defined(__clang__)
- # define UA_DEPRECATED __attribute__((deprecated))
- #elif defined(_MSC_VER)
- # define UA_DEPRECATED __declspec(deprecated)
- #else
- # define UA_DEPRECATED
- #endif
- /**
- * Internal Attributes
- * -------------------
- * These attributes are only defined if the macro UA_INTERNAL is defined. That
- * way public methods can be annotated (e.g. to warn for unused results) but
- * warnings are only triggered for internal code. */
- #if defined(UA_INTERNAL) && (defined(__GNUC__) || defined(__clang__))
- # define UA_INTERNAL_DEPRECATED _Pragma ("GCC warning \"Macro is deprecated for internal use\"")
- #else
- # define UA_INTERNAL_DEPRECATED
- #endif
- #if defined(UA_INTERNAL) && (defined(__GNUC__) || defined(__clang__))
- # define UA_INTERNAL_FUNC_ATTR_WARN_UNUSED_RESULT __attribute__((warn_unused_result))
- #else
- # define UA_INTERNAL_FUNC_ATTR_WARN_UNUSED_RESULT
- #endif
- /**
- * Detect Endianness and IEEE 754 floating point
- * ---------------------------------------------
- * Integers and floating point numbers are transmitted in little-endian (IEEE
- * 754 for floating point) encoding. If the target architecture uses the same
- * format, numeral datatypes can be memcpy'd (overlayed) on the network buffer.
- * Otherwise, a slow default encoding routine is used that works for every
- * architecture.
- *
- * Integer Endianness
- * ^^^^^^^^^^^^^^^^^^
- * The definition ``UA_LITTLE_ENDIAN`` is true when the integer representation
- * of the target architecture is little-endian. */
- #if defined(_WIN32)
- # define UA_LITTLE_ENDIAN 1
- #elif defined(__i386__) || defined(__x86_64__) || defined(__amd64__)
- # define UA_LITTLE_ENDIAN 1
- #elif (defined(__BYTE_ORDER__) && defined(__ORDER_LITTLE_ENDIAN__) && \
- (__BYTE_ORDER__ == __ORDER_LITTLE_ENDIAN__))
- # define UA_LITTLE_ENDIAN 1
- #elif defined(__linux__) /* Linux (including Android) */
- # include <endian.h>
- # if __BYTE_ORDER == __LITTLE_ENDIAN
- # define UA_LITTLE_ENDIAN 1
- # endif
- #elif defined(__OpenBSD__) /* OpenBSD */
- # include <sys/endian.h>
- # if BYTE_ORDER == LITTLE_ENDIAN
- # define UA_LITTLE_ENDIAN 1
- # endif
- #elif defined(__NetBSD__) || defined(__FreeBSD__) || defined(__DragonFly__) /* Other BSD */
- # include <sys/endian.h>
- # if _BYTE_ORDER == _LITTLE_ENDIAN
- # define UA_LITTLE_ENDIAN 1
- # endif
- #elif defined(__APPLE__) /* Apple (MacOS, iOS) */
- # include <libkern/OSByteOrder.h>
- # if defined(__LITTLE_ENDIAN__)
- # define UA_LITTLE_ENDIAN 1
- # endif
- #elif defined(__QNX__) || defined(__QNXNTO__) /* QNX */
- # include <gulliver.h>
- # if defined(__LITTLEENDIAN__)
- # define UA_LITTLE_ENDIAN 1
- # endif
- #elif defined(_OS9000) /* OS-9 */
- # if defined(_LIL_END)
- # define UA_LITTLE_ENDIAN 1
- # endif
- #endif
- #ifndef UA_LITTLE_ENDIAN
- # define UA_LITTLE_ENDIAN 0
- #endif
- /* Can the integers be memcpy'd onto the network buffer? Add additional checks
- * here. Some platforms (e.g. QNX) have sizeof(bool) > 1. Manually disable
- * overlayed integer encoding if that is the case. */
- #if (UA_LITTLE_ENDIAN == 1)
- UA_STATIC_ASSERT(sizeof(bool) == 1, cannot_overlay_integers_with_large_bool);
- # define UA_BINARY_OVERLAYABLE_INTEGER 1
- #else
- # define UA_BINARY_OVERLAYABLE_INTEGER 0
- #endif
- /**
- * Float Endianness
- * ^^^^^^^^^^^^^^^^
- * The definition ``UA_FLOAT_IEEE754`` is set to true when the floating point
- * number representation of the target architecture is IEEE 754. The definition
- * ``UA_FLOAT_LITTLE_ENDIAN`` is set to true when the floating point number
- * representation is in little-endian encoding. */
- #if defined(_WIN32)
- # define UA_FLOAT_IEEE754 1
- #elif defined(__i386__) || defined(__x86_64__) || defined(__amd64__) || \
- defined(__ia64__) || defined(__powerpc__) || defined(__sparc__) || \
- defined(__arm__)
- # define UA_FLOAT_IEEE754 1
- #elif defined(__STDC_IEC_559__)
- # define UA_FLOAT_IEEE754 1
- #else
- # define UA_FLOAT_IEEE754 0
- #endif
- /* Wikipedia says (https://en.wikipedia.org/wiki/Endianness): Although the
- * ubiquitous x86 processors of today use little-endian storage for all types of
- * data (integer, floating point, BCD), there are a number of hardware
- * architectures where floating-point numbers are represented in big-endian form
- * while integers are represented in little-endian form. */
- #if defined(_WIN32)
- # define UA_FLOAT_LITTLE_ENDIAN 1
- #elif defined(__i386__) || defined(__x86_64__) || defined(__amd64__)
- # define UA_FLOAT_LITTLE_ENDIAN 1
- #elif defined(__FLOAT_WORD_ORDER__) && defined(__ORDER_LITTLE_ENDIAN__) && \
- (__FLOAT_WORD_ORDER__ == __ORDER_LITTLE_ENDIAN__) /* Defined only in GCC */
- # define UA_FLOAT_LITTLE_ENDIAN 1
- #elif defined(__FLOAT_WORD_ORDER) && defined(__LITTLE_ENDIAN) && \
- (__FLOAT_WORD_ORDER == __LITTLE_ENDIAN) /* Defined only in GCC */
- # define UA_FLOAT_LITTLE_ENDIAN 1
- #endif
- #ifndef UA_FLOAT_LITTLE_ENDIAN
- # define UA_FLOAT_LITTLE_ENDIAN 0
- #endif
- /* Only if the floating points are litle-endian **and** in IEEE 754 format can
- * we memcpy directly onto the network buffer. */
- #if (UA_FLOAT_IEEE754 == 1) && (UA_FLOAT_LITTLE_ENDIAN == 1)
- # define UA_BINARY_OVERLAYABLE_FLOAT 1
- #else
- # define UA_BINARY_OVERLAYABLE_FLOAT 0
- #endif
- /* Atomic Operations
- * -----------------
- * Atomic operations that synchronize across processor cores (for
- * multithreading). Only the inline-functions defined next are used. Replace
- * with architecture-specific operations if necessary. */
- #ifndef UA_ENABLE_MULTITHREADING
- # define UA_atomic_sync()
- #else
- # ifdef _MSC_VER /* Visual Studio */
- # define UA_atomic_sync() _ReadWriteBarrier()
- # else /* GCC/Clang */
- # define UA_atomic_sync() __sync_synchronize()
- # endif
- #endif
- static UA_INLINE void *
- UA_atomic_xchg(void * volatile * addr, void *newptr) {
- #ifndef UA_ENABLE_MULTITHREADING
- void *old = *addr;
- *addr = newptr;
- return old;
- #else
- # ifdef _MSC_VER /* Visual Studio */
- return _InterlockedExchangePointer(addr, newptr);
- # else /* GCC/Clang */
- return __sync_lock_test_and_set(addr, newptr);
- # endif
- #endif
- }
- static UA_INLINE void *
- UA_atomic_cmpxchg(void * volatile * addr, void *expected, void *newptr) {
- #ifndef UA_ENABLE_MULTITHREADING
- void *old = *addr;
- if(old == expected) {
- *addr = newptr;
- }
- return old;
- #else
- # ifdef _MSC_VER /* Visual Studio */
- return _InterlockedCompareExchangePointer(addr, expected, newptr);
- # else /* GCC/Clang */
- return __sync_val_compare_and_swap(addr, expected, newptr);
- # endif
- #endif
- }
- static UA_INLINE uint32_t
- UA_atomic_addUInt32(volatile uint32_t *addr, uint32_t increase) {
- #ifndef UA_ENABLE_MULTITHREADING
- *addr += increase;
- return *addr;
- #else
- # ifdef _MSC_VER /* Visual Studio */
- return _InterlockedExchangeAdd(addr, increase) + increase;
- # else /* GCC/Clang */
- return __sync_add_and_fetch(addr, increase);
- # endif
- #endif
- }
- static UA_INLINE size_t
- UA_atomic_addSize(volatile size_t *addr, size_t increase) {
- #ifndef UA_ENABLE_MULTITHREADING
- *addr += increase;
- return *addr;
- #else
- # ifdef _MSC_VER /* Visual Studio */
- return _InterlockedExchangeAdd(addr, increase) + increase;
- # else /* GCC/Clang */
- return __sync_add_and_fetch(addr, increase);
- # endif
- #endif
- }
- static UA_INLINE uint32_t
- UA_atomic_subUInt32(volatile uint32_t *addr, uint32_t decrease) {
- #ifndef UA_ENABLE_MULTITHREADING
- *addr -= decrease;
- return *addr;
- #else
- # ifdef _MSC_VER /* Visual Studio */
- return _InterlockedExchangeSub(addr, decrease) - decrease;
- # else /* GCC/Clang */
- return __sync_sub_and_fetch(addr, decrease);
- # endif
- #endif
- }
- static UA_INLINE size_t
- UA_atomic_subSize(volatile size_t *addr, size_t decrease) {
- #ifndef UA_ENABLE_MULTITHREADING
- *addr -= decrease;
- return *addr;
- #else
- # ifdef _MSC_VER /* Visual Studio */
- return _InterlockedExchangeSub(addr, decrease) - decrease;
- # else /* GCC/Clang */
- return __sync_sub_and_fetch(addr, decrease);
- # endif
- #endif
- }
- #endif /* ARCH_UA_ARCHITECTURE_DEFINITIONS_H_ */
|