blob: fc207168896761f7d643c74379077d5f3b010d80 [file] [log] [blame]
/* 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_
#include <open62541/config.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 >= 1800
# 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) do {} while(0)
#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
#define UA_THREADSAFE
/**
* 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. */
#if UA_MULTITHREADING >= 200
#ifdef _MSC_VER /* Visual Studio */
#define UA_atomic_sync() _ReadWriteBarrier()
#else /* GCC/Clang */
#define UA_atomic_sync() __sync_synchronize()
#endif
#else
#define UA_atomic_sync()
#endif
static UA_INLINE void *
UA_atomic_xchg(void * volatile * addr, void *newptr) {
#if UA_MULTITHREADING >= 200
#ifdef _MSC_VER /* Visual Studio */
return _InterlockedExchangePointer(addr, newptr);
#else /* GCC/Clang */
return __sync_lock_test_and_set(addr, newptr);
#endif
#else
void *old = *addr;
*addr = newptr;
return old;
#endif
}
static UA_INLINE void *
UA_atomic_cmpxchg(void * volatile * addr, void *expected, void *newptr) {
#if UA_MULTITHREADING >= 200
#ifdef _MSC_VER /* Visual Studio */
return _InterlockedCompareExchangePointer(addr, expected, newptr);
#else /* GCC/Clang */
return __sync_val_compare_and_swap(addr, expected, newptr);
#endif
#else
void *old = *addr;
if(old == expected) {
*addr = newptr;
}
return old;
#endif
}
static UA_INLINE uint32_t
UA_atomic_addUInt32(volatile uint32_t *addr, uint32_t increase) {
#if UA_MULTITHREADING >= 200
#ifdef _MSC_VER /* Visual Studio */
return _InterlockedExchangeAdd(addr, increase) + increase;
#else /* GCC/Clang */
return __sync_add_and_fetch(addr, increase);
#endif
#else
*addr += increase;
return *addr;
#endif
}
static UA_INLINE size_t
UA_atomic_addSize(volatile size_t *addr, size_t increase) {
#if UA_MULTITHREADING >= 200
#ifdef _MSC_VER /* Visual Studio */
return _InterlockedExchangeAdd(addr, increase) + increase;
#else /* GCC/Clang */
return __sync_add_and_fetch(addr, increase);
#endif
#else
*addr += increase;
return *addr;
#endif
}
static UA_INLINE uint32_t
UA_atomic_subUInt32(volatile uint32_t *addr, uint32_t decrease) {
#if UA_MULTITHREADING >= 200
#ifdef _MSC_VER /* Visual Studio */
return _InterlockedExchangeSub(addr, decrease) - decrease;
#else /* GCC/Clang */
return __sync_sub_and_fetch(addr, decrease);
#endif
#else
*addr -= decrease;
return *addr;
#endif
}
static UA_INLINE size_t
UA_atomic_subSize(volatile size_t *addr, size_t decrease) {
#if UA_MULTITHREADING >= 200
#ifdef _MSC_VER /* Visual Studio */
return _InterlockedExchangeSub(addr, decrease) - decrease;
#else /* GCC/Clang */
return __sync_sub_and_fetch(addr, decrease);
#endif
#else
*addr -= decrease;
return *addr;
#endif
}
#endif /* ARCH_UA_ARCHITECTURE_DEFINITIONS_H_ */