/* * Copyright (C) 2010-2022 Arm Limited or its affiliates. All rights reserved. * * SPDX-License-Identifier: Apache-2.0 * * Licensed under the Apache License, Version 2.0 (the License); you may * not use this file except in compliance with the License. * You may obtain a copy of the License at * * www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an AS IS BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ /* ---------------------------------------------------------------------- * Project: Arm-2D Library * Title: arm_2d_utils.h * Description: Public header file for Arm-2D Library * * $Date: 18. July 2022 * $Revision: V.1.0.3 * * -------------------------------------------------------------------- */ #ifndef __ARM_2D_UTILS_H__ #define __ARM_2D_UTILS_H__ /*============================ INCLUDES ======================================*/ #if defined(__clang__) # pragma clang diagnostic push # pragma clang diagnostic ignored "-Wunknown-warning-option" # pragma clang diagnostic ignored "-Wreserved-identifier" # pragma clang diagnostic ignored "-Wgnu-zero-variadic-macro-arguments" # pragma clang diagnostic ignored "-Wpadded" # pragma clang diagnostic ignored "-Wsign-conversion" # pragma clang diagnostic ignored "-Wimplicit-int-conversion" # pragma clang diagnostic ignored "-Wdollar-in-identifier-extension" # pragma clang diagnostic ignored "-Wundef" #elif defined(__IS_COMPILER_GCC__) # pragma GCC diagnostic push # pragma GCC diagnostic ignored "-Wpedantic" # pragma GCC diagnostic ignored "-Wstrict-aliasing" #endif #ifdef __ARM_2D_HAS_USER_HEADER__ # include __ARM_2D_HAS_USER_HEADER__ #endif /*! \note arm-2d relies on CMSIS 5.4.0 and above. */ #include "cmsis_compiler.h" #ifdef __cplusplus extern "C" { #endif /*! * \addtogroup gKernel 1 Kernel * @{ */ /*============================ MACROS ========================================*/ /*----------------------------------------------------------------------------* * Environment Detection * *----------------------------------------------------------------------------*/ /* The macros to identify compilers */ /*! * \brief to detect IAR */ #undef __IS_COMPILER_IAR__ #if defined(__IAR_SYSTEMS_ICC__) # define __IS_COMPILER_IAR__ 1 #endif /*! * \brief to detect arm compiler 5 * */ #undef __IS_COMPILER_ARM_COMPILER_5__ #if ((__ARMCC_VERSION >= 5000000) && (__ARMCC_VERSION < 6000000)) # define __IS_COMPILER_ARM_COMPILER_5__ 1 #endif /*! * \brief to detect arm compiler 6 * */ #undef __IS_COMPILER_ARM_COMPILER_6__ #if defined(__ARMCC_VERSION) && (__ARMCC_VERSION >= 6010050) # define __IS_COMPILER_ARM_COMPILER_6__ 1 #endif /*! * \brief to detect arm compilers * */ #undef __IS_COMPILER_ARM_COMPILER__ #if defined(__IS_COMPILER_ARM_COMPILER_5__) && __IS_COMPILER_ARM_COMPILER_5__ \ || defined(__IS_COMPILER_ARM_COMPILER_6__) && __IS_COMPILER_ARM_COMPILER_6__ # define __IS_COMPILER_ARM_COMPILER__ 1 #endif /*! * \brief to detect clang (llvm) * */ #undef __IS_COMPILER_LLVM__ #if defined(__clang__) && !__IS_COMPILER_ARM_COMPILER_6__ # define __IS_COMPILER_LLVM__ 1 #else /*! * \brief to detect gcc * */ # undef __IS_COMPILER_GCC__ # if defined(__GNUC__) && !( defined(__IS_COMPILER_ARM_COMPILER__) \ || defined(__IS_COMPILER_LLVM__) \ || defined(__IS_COMPILER_IAR__)) # define __IS_COMPILER_GCC__ 1 # endif #endif /*----------------------------------------------------------------------------* * OOC and Private Protection * *----------------------------------------------------------------------------*/ /* minimal support for OOPC */ #undef __implement_ex #undef __implement #undef implement #undef implement_ex #undef inherit #undef inherit_ex #define __implement_ex(__type, __name) \ union { \ __type __name; \ __type; \ } #define __inherit_ex(__type, __name) \ __type __name \ #define __implement(__type) __implement_ex( __type, \ use_as__##__type) #define __inherit(__type) __inherit_ex(__type,use_as__##__type) #define implement(__type) __implement(__type) #define implement_ex(__type, __name) __implement_ex(__type, __name) #define inherit(__type) __inherit(__type) #define inherit_ex(__type, __name) __inherit_ex(__type, __name) /*----------------------------------------------------------------------------* * Misc * *----------------------------------------------------------------------------*/ #ifndef ARM_2D_UNUSED # define ARM_2D_UNUSED(__VAR) (void)(__VAR) #endif #ifndef ARM_TEST_BITS # define ARM_TEST_BITS(__VALUE, __BITS) ((__BITS) == ((__VALUE) & (__BITS))) #endif #ifndef dimof # define dimof(__array) (sizeof(__array)/sizeof(__array[0])) #endif #ifndef offsetof # define offsetof(__type, __member) \ ((uintptr_t)&(((__type *)NULL)->__member)) #endif #define __ARM_TO_STRING(__STR) #__STR #define ARM_TO_STRING(__STR) __ARM_TO_STRING(__STR) #define __ARM_VA_NUM_ARGS_IMPL( _0,_1,_2,_3,_4,_5,_6,_7,_8,_9,_10,_11,_12, \ _13,_14,_15,_16,__N,...) __N /*! * \brief A macro to count the number of parameters * * \note if GNU extension is not supported or enabled, the following express will * be false: (__ARM_VA_NUM_ARGS() != 0) * This might cause problems when in this library. */ #define __ARM_VA_NUM_ARGS(...) \ __ARM_VA_NUM_ARGS_IMPL( 0,##__VA_ARGS__,16,15,14,13,12,11,10,9, \ 8,7,6,5,4,3,2,1,0) /*! * \brief detect whether GNU extension is enabled in compilation or not */ #if __ARM_VA_NUM_ARGS() != 0 # warning Please enable GNU extensions, it is required by the Arm-2D. #endif #ifndef ARM_2D_INVOKE /*! * \brief A macro to safely invode a function pointer * * \param[in] __FUNC_PTR the target function pointer * \param[in] ... an optional parameter list */ # define ARM_2D_INVOKE(__FUNC_PTR, ...) \ do { \ if (NULL != (__FUNC_PTR)) { \ (*(__FUNC_PTR))(__VA_ARGS__); \ } \ } while(0) #endif #define __ARM_CONNECT2(__A, __B) __A##__B #define __ARM_CONNECT2_ALT(__A, __B) __A##__B #define __ARM_CONNECT3(__A, __B, __C) __A##__B##__C #define __ARM_CONNECT4(__A, __B, __C, __D) __A##__B##__C##__D #define __ARM_CONNECT5(__A, __B, __C, __D, __E) __A##__B##__C##__D##__E #define __ARM_CONNECT6(__A, __B, __C, __D, __E, __F) \ __A##__B##__C##__D##__E##__F #define __ARM_CONNECT7(__A, __B, __C, __D, __E, __F, __G) \ __A##__B##__C##__D##__E##__F##__G #define __ARM_CONNECT8(__A, __B, __C, __D, __E, __F, __G, __H) \ __A##__B##__C##__D##__E##__F##__G##__H #define __ARM_CONNECT9(__A, __B, __C, __D, __E, __F, __G, __H, __I) \ __A##__B##__C##__D##__E##__F##__G##__H##__I #define ARM_CONNECT2(__A, __B) __ARM_CONNECT2(__A, __B) #define ARM_CONNECT2_ALT(__A, __B) __ARM_CONNECT2_ALT(__A, __B) #define ARM_CONNECT3(__A, __B, __C) __ARM_CONNECT3(__A, __B, __C) #define ARM_CONNECT4(__A, __B, __C, __D) __ARM_CONNECT4(__A, __B, __C, __D) #define ARM_CONNECT5(__A, __B, __C, __D, __E) \ __ARM_CONNECT5(__A, __B, __C, __D, __E) #define ARM_CONNECT6(__A, __B, __C, __D, __E, __F) \ __ARM_CONNECT6(__A, __B, __C, __D, __E, __F) #define ARM_CONNECT7(__A, __B, __C, __D, __E, __F, __G) \ __ARM_CONNECT7(__A, __B, __C, __D, __E, __F, __G) #define ARM_CONNECT8(__A, __B, __C, __D, __E, __F, __G, __H) \ __ARM_CONNECT8(__A, __B, __C, __D, __E, __F, __G, __H) #define ARM_CONNECT9(__A, __B, __C, __D, __E, __F, __G, __H, __I) \ __ARM_CONNECT9(__A, __B, __C, __D, __E, __F, __G, __H, __I) #define arm_connect(...) \ ARM_CONNECT2_ALT(ARM_CONNECT, __ARM_VA_NUM_ARGS(__VA_ARGS__)) \ (__VA_ARGS__) #define ARM_CONNECT(...) arm_connect(__VA_ARGS__) #define __ARM_USING1(__declare) \ for (__declare, *ARM_CONNECT3(__ARM_USING_, __LINE__,_ptr) = NULL; \ ARM_CONNECT3(__ARM_USING_, __LINE__,_ptr)++ == NULL; \ ) #define __ARM_USING2(__declare, __on_leave_expr) \ for (__declare, *ARM_CONNECT3(__ARM_USING_, __LINE__,_ptr) = NULL; \ ARM_CONNECT3(__ARM_USING_, __LINE__,_ptr)++ == NULL; \ __on_leave_expr \ ) #define __ARM_USING3(__declare, __on_enter_expr, __on_leave_expr) \ for (__declare, *ARM_CONNECT3(__ARM_USING_, __LINE__,_ptr) = NULL; \ ARM_CONNECT3(__ARM_USING_, __LINE__,_ptr)++ == NULL ? \ ((__on_enter_expr),1) : 0; \ __on_leave_expr \ ) #define __ARM_USING4(__dcl1, __dcl2, __on_enter_expr, __on_leave_expr) \ for (__dcl1,__dcl2,*ARM_CONNECT3(__ARM_USING_, __LINE__,_ptr)= NULL;\ ARM_CONNECT3(__ARM_USING_, __LINE__,_ptr)++ == NULL ? \ ((__on_enter_expr),1) : 0; \ __on_leave_expr \ ) #define arm_using(...) \ ARM_CONNECT2(__ARM_USING, __ARM_VA_NUM_ARGS(__VA_ARGS__))(__VA_ARGS__) #define __ARM_WITH2(__type, __addr) \ ARM_USING(__type *_p=(__addr)) #define __ARM_WITH3(__type, __addr, __item) \ ARM_USING(__type *_p=(__addr), *__item = _p, _p=_p, ) #define arm_with(...) \ ARM_CONNECT2(__ARM_WITH, __ARM_VA_NUM_ARGS(__VA_ARGS__))(__VA_ARGS__) #define ARM_FOREACH2(__type, __array) \ arm_using(__type *_ = __array) \ for ( uint_fast32_t ARM_CONNECT2(count,__LINE__) = dimof(__array);\ ARM_CONNECT2(count,__LINE__) > 0; \ _++, ARM_CONNECT2(count,__LINE__)-- \ ) #define ARM_FOREACH3(__type, __array, __item) \ arm_using(__type *_ = __array, *__item = _, _ = _, ) \ for ( uint_fast32_t ARM_CONNECT2(count,__LINE__) = dimof(__array);\ ARM_CONNECT2(count,__LINE__) > 0; \ _++, __item = _, ARM_CONNECT2(count,__LINE__)-- \ ) #define ARM_FOREACH4(__type, __array, __count, __item) \ arm_using(__type *_ = __array, *__item = _, _ = _, ) \ for ( uint_fast32_t ARM_CONNECT2(count,__LINE__) = (__count); \ ARM_CONNECT2(count,__LINE__) > 0; \ _++, __item = _, ARM_CONNECT2(count,__LINE__)-- \ ) #define arm_foreach(...) \ ARM_CONNECT2(ARM_FOREACH, __ARM_VA_NUM_ARGS(__VA_ARGS__))(__VA_ARGS__) #ifndef ARM_NONNULL # if defined(__IS_COMPILER_ARM_COMPILER_5__) ||\ defined(__IS_COMPILER_ARM_COMPILER_6__) ||\ defined(__IS_COMPILER_GCC__) ||\ defined(__IS_COMPILER_LLVM__) # define ARM_NONNULL(...) __attribute__((nonnull(__VA_ARGS__))) # else # define ARM_NONNULL(...) # endif #endif #ifndef ARM_NOINIT # if defined(__IS_COMPILER_ARM_COMPILER_5__) # define ARM_NOINIT __attribute__(( section( ".bss.noinit"),zero_init)) # elif defined(__IS_COMPILER_ARM_COMPILER_6__) # define ARM_NOINIT __attribute__(( section( ".bss.noinit"))) # elif defined(__IS_COMPILER_IAR__) # define ARM_NOINIT __no_init # elif defined(__IS_COMPILER_GCC__) || defined(__IS_COMPILER_LLVM__) # define ARM_NOINIT __attribute__(( section( ".bss.noinit"))) # else # define ARM_NOINIT # endif #endif #ifndef ARM_ALIGN # define __ARM_ALIGN(__N) __attribute__((aligned(__N))) # define ARM_ALIGN(__N) __ARM_ALIGN(__N) #endif #ifndef __RESTRICT # define __RESTRICT __restrict #endif #ifndef __OVERRIDE_WEAK # define __OVERRIDE_WEAK #endif /*! * \brief A macro to generate a safe name, usually used in macro template as the * name of local variables * */ #define ARM_2D_SAFE_NAME(...) ARM_CONNECT(__,__LINE__,##__VA_ARGS__) #define arm_2d_safe_name(...) ARM_2D_SAFE_NAME(__VA_ARGS__) #undef arm_irq_safe /*! * \brief a decoration to make the immediate following code irq-safe * * \code arm_irq_safe { // code inside the brackets are IRQ safe ... } // the printf is IRQ safe arm_irq_safe printf("IRQ safe printf\n"); \endcode */ #define arm_irq_safe \ arm_using( uint32_t ARM_2D_SAFE_NAME(temp) = \ ({uint32_t temp=__get_PRIMASK();__disable_irq();temp;}),\ __set_PRIMASK(ARM_2D_SAFE_NAME(temp))) #undef ARM_2D_WRAP_FUNC #undef __ARM_2D_WRAP_FUNC #undef ARM_2D_ORIG_FUNC #undef __ARM_2D_ORIG_FUNC #if defined(__IS_COMPILER_ARM_COMPILER_6__) # define __ARM_2D_WRAP_FUNC(__FUNC) $Sub$$##__FUNC # define __ARM_2D_ORIG_FUNC(__FUNC) $Super$$## __FUNC #else # define __ARM_2D_WRAP_FUNC(x) __wrap_ ## x # define __ARM_2D_ORIG_FUNC(x) __real_ ## x #endif #define ARM_2D_WRAP_FUNC(__FUNC) __ARM_2D_WRAP_FUNC(__FUNC) #define ARM_2D_ORIG_FUNC(__FUNC) __ARM_2D_ORIG_FUNC(__FUNC) /*----------------------------------------------------------------------------* * List Operations * *----------------------------------------------------------------------------*/ /* ALL the parameters passed to following macros must be pointer variables. */ #define __ARM_LIST_STACK_PUSH(__P_TOP, __P_NODE) \ do { \ ((__arm_slist_node_t *)(__P_NODE))->ptNext = \ (__arm_slist_node_t *)(__P_TOP); \ (*(__arm_slist_node_t **)&(__P_TOP)) = \ (__arm_slist_node_t *)(__P_NODE); \ } while(0) #define ARM_LIST_STACK_PUSH(__P_TOP, __P_NODE) \ __ARM_LIST_STACK_PUSH((__P_TOP), (__P_NODE)) #define ARM_LIST_INSERT_AFTER(__P_TARGET, __P_NODE) \ __ARM_LIST_STACK_PUSH((__P_TARGET), (__P_NODE)) #define __ARM_LIST_STACK_POP(__P_TOP, __P_NODE) \ do { \ (*(__arm_slist_node_t **)&(__P_NODE)) = \ (__arm_slist_node_t *)(__P_TOP); \ if (NULL != (__P_TOP)) { \ (*(__arm_slist_node_t **)&(__P_TOP)) = \ ((__arm_slist_node_t *)(__P_NODE))->ptNext; \ ((__arm_slist_node_t *)(__P_NODE))->ptNext = NULL; \ } \ } while(0) #define ARM_LIST_STACK_POP(__P_TOP, __P_NODE) \ __ARM_LIST_STACK_POP((__P_TOP), (__P_NODE)) #define ARM_LIST_REMOVE_AFTER(__P_TARGET, __P_NODE) \ ARM_LIST_STACK_POP((__P_TARGET), (__P_NODE)) #define __ARM_LIST_QUEUE_ENQUEUE(__HEAD, __TAIL, __ITEM) \ do { \ if (NULL == (__TAIL)) { \ (*((__arm_slist_node_t **)&(__TAIL))) = \ (__arm_slist_node_t *)(__ITEM); \ (*((__arm_slist_node_t **)&(__HEAD))) = \ (__arm_slist_node_t *)(__ITEM); \ } else { \ ((__arm_slist_node_t *)(__TAIL))->ptNext = \ (__arm_slist_node_t *)(__ITEM); \ (*(__arm_slist_node_t **)&(__TAIL)) = \ (__arm_slist_node_t *)(__ITEM); \ } \ ((__arm_slist_node_t *)(__ITEM))->ptNext = NULL; \ } while(0) #define ARM_LIST_QUEUE_ENQUEUE(__HEAD, __TAIL, __ITEM) \ __ARM_LIST_QUEUE_ENQUEUE((__HEAD), (__TAIL), (__ITEM)) #define __ARM_LIST_QUEUE_DEQUEUE(__HEAD, __TAIL, __ITEM) \ do { \ (*(__arm_slist_node_t **)&(__ITEM)) = (__arm_slist_node_t *)(__HEAD); \ if (NULL != (__HEAD)) { \ (*(__arm_slist_node_t **)&(__HEAD)) = \ ((__arm_slist_node_t *)(__HEAD))->ptNext; \ if (NULL == (__HEAD)) { \ (__TAIL) = NULL; \ } \ } \ } while(0) #define ARM_LIST_QUEUE_DEQUEUE(__HEAD, __TAIL, __ITEM) \ __ARM_LIST_QUEUE_DEQUEUE((__HEAD), (__TAIL), (__ITEM)) #define __ARM_LIST_QUEUE_PEEK(__HEAD, __TAIL, __ITEM) \ do { \ (*(__arm_slist_node_t **)&(__ITEM)) = (__arm_slist_node_t *)(__HEAD); \ } while(0) #define ARM_LIST_QUEUE_PEEK(__HEAD, __TAIL, __ITEM) \ __ARM_LIST_QUEUE_PEEK((__HEAD), (__TAIL), (__ITEM)) \ /*----------------------------------------------------------------------------* * Definition Template * *----------------------------------------------------------------------------*/ #define __def_low_lv_io(__NAME, __SW, ...) \ const __arm_2d_low_level_io_t LOW_LEVEL_IO##__NAME = { \ .SW = (__arm_2d_io_func_t *)&(__SW), \ .HW = (NULL, ##__VA_ARGS__) \ } #define def_low_lv_io(__NAME, __SW, ...) \ __def_low_lv_io(__NAME, __SW, ##__VA_ARGS__) #define __ref_low_lv_io(__NAME) &LOW_LEVEL_IO##__NAME #define ref_low_lv_io(__NAME) __ref_low_lv_io(__NAME) /*============================ TYPES =========================================*/ /*! * \brief a type for generic list * * \note to avoid using container_of() operand, please put __arm_slist_node_t * at the beginning of a class/structure */ typedef struct __arm_slist_node_t __arm_slist_node_t; struct __arm_slist_node_t { __arm_slist_node_t *ptNext; //!< the next node }; /*============================ GLOBAL VARIABLES ==============================*/ /*============================ PROTOTYPES ====================================*/ #if defined(__clang__) # pragma clang diagnostic pop #elif __IS_COMPILER_ARM_COMPILER_5__ #elif __IS_COMPILER_GCC__ # pragma GCC diagnostic pop #endif #ifdef __cplusplus } #endif #endif /* end of __ARM_2D_UTILS_H__ */ /*! @} */ /*============================ MACROS ========================================*/ /*----------------------------------------------------------------------------* * Reentrant Macros * *----------------------------------------------------------------------------*/ /* un-define macros */ #undef ARM_PRIVATE /* redefine macros */ #if defined(__cplusplus) # define ARM_PRIVATE(...) \ struct { \ __VA_ARGS__ \ }; #elif defined(__ARM_2D_IMPL__) || defined(__IS_COMPILER_IAR__) # define ARM_PRIVATE(...) \ struct { \ __VA_ARGS__ \ } __ALIGNED(__alignof__(struct {__VA_ARGS__})); #else # define ARM_PRIVATE(...) \ uint8_t ARM_CONNECT3(chMask,__LINE__,__COUNTER__) \ [sizeof(struct {__VA_ARGS__})] \ __ALIGNED(__alignof__(struct {__VA_ARGS__})); #endif /* post un-define macros */ #undef __ARM_2D_IMPL__