/* * Copyright (C) 2010-2020 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_tile.c * Description: Basic Tile operations * * $Date: 01. December 2020 * $Revision: V.0.9.0 * * Target Processor: Cortex-M cores * * -------------------------------------------------------------------- */ #define __ARM_2D_IMPL__ #include "arm_2d.h" #include "__arm_2d_impl.h" #include "__arm_2d_paving.h" #ifdef __cplusplus extern "C" { #endif #if defined(__clang__) # pragma clang diagnostic push # pragma clang diagnostic ignored "-Wincompatible-pointer-types-discards-qualifiers" # pragma clang diagnostic ignored "-Wcast-qual" # pragma clang diagnostic ignored "-Wcast-align" # pragma clang diagnostic ignored "-Wextra-semi-stmt" # pragma clang diagnostic ignored "-Wsign-conversion" # pragma clang diagnostic ignored "-Wunused-function" # pragma clang diagnostic ignored "-Wimplicit-int-float-conversion" # pragma clang diagnostic ignored "-Wdouble-promotion" # pragma clang diagnostic ignored "-Wunused-parameter" # pragma clang diagnostic ignored "-Wimplicit-float-conversion" # pragma clang diagnostic ignored "-Wimplicit-int-conversion" # pragma clang diagnostic ignored "-Wtautological-pointer-compare" # pragma clang diagnostic ignored "-Wsign-compare" # pragma clang diagnostic ignored "-Wmissing-prototypes" #elif defined(__IS_COMPILER_ARM_COMPILER_5__) # pragma diag_suppress 174,177,188,68,513,144 #elif defined(__IS_COMPILER_GCC__) # pragma GCC diagnostic push # pragma GCC diagnostic ignored "-Wdiscarded-qualifiers" #endif /*----------------------------------------------------------------------------* * Code Template * *----------------------------------------------------------------------------*/ #define __API_COLOUR rgb16 #define __API_INT_TYPE uint16_t #define __API_INT_TYPE_BIT_NUM 16 #include "__arm_2d_copy.inc" #define __API_COLOUR rgb32 #define __API_INT_TYPE uint32_t #define __API_INT_TYPE_BIT_NUM 32 #include "__arm_2d_copy.inc" /*----------------------------------------------------------------------------* * Tile Operations * *----------------------------------------------------------------------------*/ /* HOW IT WORKS: Input Region 0 +------------------------------------------------------+ | | | | | | | +------------------------------+---------+ | | |/////////| | | Output Region |/////////| | | |/////////| +-----------------------+------------------------------+/////////| |////////////////////////////////////////| |////////////////////////////////////////| +----------------------------------------+ Input Region 1 */ ARM_NONNULL(1,2) bool arm_2d_region_intersect( const arm_2d_region_t *ptRegionIn0, const arm_2d_region_t *ptRegionIn1, arm_2d_region_t *ptRegionOut) { assert(ptRegionIn0 != NULL); assert(ptRegionIn1 != NULL); do { arm_2d_location_t tLocationIn0End = { .iX = ptRegionIn0->tLocation.iX + ptRegionIn0->tSize.iWidth - 1, .iY = ptRegionIn0->tLocation.iY + ptRegionIn0->tSize.iHeight - 1, }; arm_2d_location_t tLocationIn1End = { .iX = ptRegionIn1->tLocation.iX + ptRegionIn1->tSize.iWidth - 1, .iY = ptRegionIn1->tLocation.iY + ptRegionIn1->tSize.iHeight - 1, }; arm_2d_location_t tLocationOutStart = { .iX = MAX( ptRegionIn0->tLocation.iX, ptRegionIn1->tLocation.iX), .iY = MAX( ptRegionIn0->tLocation.iY, ptRegionIn1->tLocation.iY), }; arm_2d_location_t tLocationOutEnd = { .iX = MIN( tLocationIn0End.iX, tLocationIn1End.iX), .iY = MIN( tLocationIn0End.iY, tLocationIn1End.iY), }; if ( (tLocationOutStart.iX > tLocationOutEnd.iX) || (tLocationOutStart.iY > tLocationOutEnd.iY)) { return false; } if (NULL != ptRegionOut) { ptRegionOut->tLocation = tLocationOutStart; ptRegionOut->tSize.iWidth = tLocationOutEnd.iX - tLocationOutStart.iX + 1; ptRegionOut->tSize.iHeight = tLocationOutEnd.iY - tLocationOutStart.iY + 1; } } while(0); return true; } ARM_NONNULL(1,2) bool arm_2d_is_point_inside_region( const arm_2d_region_t *ptRegion, const arm_2d_location_t *ptPoint) { assert(ptRegion != NULL); assert(ptPoint != NULL); do { if (ptPoint->iX < ptRegion->tLocation.iX) { break; } else if (ptPoint->iY < ptRegion->tLocation.iY) { break; } else if (ptPoint->iX >= ptRegion->tLocation.iX + ptRegion->tSize.iWidth) { break; } else if (ptPoint->iY >= ptRegion->tLocation.iY + ptRegion->tSize.iHeight) { break; } return true; } while(0); return false; } /* HOW IT WORKS: Root Tile (Output Tile) +------------------------------------------------------------------------+ | ... ... | | | | Parent Tile | | +------------------------------------+ | | | Child Tile | | | | +------------------------------+---------+ | | | | |/////////| | | | | Valid Region |/////////| | | | | |/////////| | | +-----+------------------------------+/////////| | | |////////////////////////////////////////| | | |////////////////////////////////////////| | | +----------------------------------------+ | | | +------------------------------------------------------------------------+ */ ARM_NONNULL(1,2) const arm_2d_tile_t *arm_2d_tile_get_root( const arm_2d_tile_t *ptTile, arm_2d_region_t *ptValidRegion, arm_2d_location_t *ptOffset) { assert(NULL != ptTile); assert(NULL != ptValidRegion); *ptValidRegion = ptTile->tRegion; if (NULL != ptOffset) { ptOffset->iX = 0; ptOffset->iY = 0; } if (arm_2d_is_root_tile(ptTile)) { return ptTile; } do { //! get parent ptTile = (const arm_2d_tile_t *)ptTile->ptParent; if (NULL == ptTile) { break; } /*! \note Calculate the relative position between valid region and *! the tile's original region. Usually, the tile's region *! is inside the parent tile, but when the tile's location is *! out of the parent's region with one or more negative axies, *! the offset will be non-zero. *! The offset is used to indicate the tile's view, and the *! valid region is seen as inside the tile's region. *! *! Figure: What's the meaning of offset location *! *! The special case, where the child tile has a negative coordinates, *! hence, the offset is (a,b) **as if** the valid region is inside *! the child tile. *! *! (-a,-b) Child Tile *! +------------------------------------+ *! |///(0,0) Parent Tile ///////////////| *! |/////+------------------------------+---------+ *! |/////| | | *! |/////| Valid Region | | *! |/////| | | *! +-----+------------------------------+ | *! | | *! | | *! +----------------------------------------+ *! */ if (NULL != ptOffset) { arm_2d_location_t tOffset = ptValidRegion->tLocation; tOffset.iX = MAX(0, -tOffset.iX); tOffset.iY = MAX(0, -tOffset.iY); ptOffset->iX += tOffset.iX; ptOffset->iY += tOffset.iY; } /*! calculate the valid range in parent tile *! *! \note the location of the parent tile is used to indicate its *! relative location between the it and its parent. *! when calculate the valid range in parent, we have to assume *! that the location is always (0,0) *! *! \note the location of a root tile is always (0,0) */ arm_2d_region_t tParentRegion = { .tSize = ptTile->tRegion.tSize, }; /*! make sure the output region is valid */ if (!arm_2d_region_intersect( &tParentRegion, ptValidRegion, ptValidRegion)) { /* out of range */ return NULL; } if (arm_2d_is_root_tile(ptTile)) { break; } ptValidRegion->tLocation.iX += ptTile->tRegion.tLocation.iX; ptValidRegion->tLocation.iY += ptTile->tRegion.tLocation.iY; } while(true); return ptTile; } /* HOW IT WORKS: Parent Tile (Are NOT necessarily a ROOT tile ) +------------------------------------------------------+ | | | | | Target Region | | +------------------------------+---------+ | | |/////////| | | New Child Tile (Output) |/////////| | | |/////////| +-----------------------+------------------------------+/////////| |////////////////////////////////////////| |////////////////////////////////////////| +----------------------------------------+ */ ARM_NONNULL(1,2,3) arm_2d_tile_t *arm_2d_tile_generate_child( const arm_2d_tile_t *ptParentTile, const arm_2d_region_t *ptRegion, arm_2d_tile_t *ptOutput, bool bClipRegion) { assert(NULL != ptParentTile); assert(NULL != ptRegion); assert(NULL != ptOutput); memset(ptOutput, 0, sizeof(arm_2d_tile_t)); ptOutput->tRegion = *ptRegion; arm_2d_region_t tParentRegion = { .tSize = ptParentTile->tRegion.tSize, }; if (bClipRegion) { if (!arm_2d_region_intersect( &tParentRegion, &(ptOutput->tRegion), &(ptOutput->tRegion) )) { /* out of range */ return NULL; } } else { /*! \note since you are allowed to generate a child tile whose region is *! bigger than its parent, so we don't have to clip it, see **note** *! below. */ if (!arm_2d_region_intersect( &tParentRegion, &(ptOutput->tRegion), NULL //&(ptOutput->tRegion) //!< **note** )) { /* out of range */ return NULL; } } ptOutput->ptParent = (arm_2d_tile_t *)ptParentTile; return ptOutput; } /*----------------------------------------------------------------------------* * Copy tile to destination directly * *----------------------------------------------------------------------------*/ ARM_NONNULL(1,2) arm_fsm_rt_t arm_2d_rgb16_tile_copy( const arm_2d_tile_t *ptSource, const arm_2d_tile_t *ptTarget, const arm_2d_region_t *ptRegion, uint32_t wMode) { assert(NULL != ptSource); assert(NULL != ptTarget); ARM_2D_IMPL(arm_2d_op_cp_t); if (!__arm_2d_op_acquire((arm_2d_op_core_t *)ptThis)) { return arm_fsm_rt_on_going; } //memset(ptThis, 0, sizeof(*ptThis)); OP_CORE.ptOp = &ARM_2D_OP_TILE_COPY_RGB16; this.Target.ptTile = ptTarget; this.Target.ptRegion = ptRegion; this.Source.ptTile = ptSource; this.wMode = wMode; return __arm_2d_op_invoke(NULL); } ARM_NONNULL(1,2) arm_fsm_rt_t arm_2d_rgb32_tile_copy( const arm_2d_tile_t *ptSource, const arm_2d_tile_t *ptTarget, const arm_2d_region_t *ptRegion, uint32_t wMode) { assert(NULL != ptSource); assert(NULL != ptTarget); ARM_2D_IMPL(arm_2d_op_cp_t); if (!__arm_2d_op_acquire((arm_2d_op_core_t *)ptThis)) { return arm_fsm_rt_on_going; } //memset(ptThis, 0, sizeof(*ptThis)); OP_CORE.ptOp = &ARM_2D_OP_TILE_COPY_RGB32; this.Target.ptTile = ptTarget; this.Target.ptRegion = ptRegion; this.Source.ptTile = ptSource; this.wMode = wMode; return __arm_2d_op_invoke(NULL); } /*! \brief copy source tile to destination tile and use destination tile as *! background. When encountering specified mask colour, the background *! pixel should be used, otherwise the foreground pixel from source tile *! is used. *! *! \note All color formats which using 16bits per pixel are treated equally. *! *! \note alpha channel is not handled, i.e. rgba5551 */ ARM_NONNULL(1,2) arm_fsm_rt_t arm_2d_rgb16_tile_copy_with_colour_masking( const arm_2d_tile_t *ptSource, const arm_2d_tile_t *ptTarget, const arm_2d_region_t *ptRegion, uint16_t hwMaskColour, uint32_t wMode) { assert(NULL != ptSource); assert(NULL != ptTarget); ARM_2D_IMPL(arm_2d_op_cp_cl_msk_t); if (!__arm_2d_op_acquire((arm_2d_op_core_t *)ptThis)) { return arm_fsm_rt_on_going; } OP_CORE.ptOp = &ARM_2D_OP_TILE_COPY_WITH_COLOUR_MASKING_RGB16; this.Target.ptTile = ptTarget; this.Target.ptRegion = ptRegion; this.Source.ptTile = ptSource; this.wMode = wMode; this.hwColour = hwMaskColour; return __arm_2d_op_invoke(NULL); } /*! \brief copy source tile to destination tile and use destination tile as *! background. When encountering specified mask colour, the background *! pixel should be used, otherwise the foreground pixel from source tile *! is used. *! *! \note All color formats which using 32bits per pixel are treated equally. *! *! \note alpha channel is not handled. */ ARM_NONNULL(1,2) arm_fsm_rt_t arm_2d_rgb32_tile_copy_with_colour_masking( const arm_2d_tile_t *ptSource, const arm_2d_tile_t *ptTarget, const arm_2d_region_t *ptRegion, uint32_t wMaskColour, uint32_t wMode) { assert(NULL != ptSource); assert(NULL != ptTarget); ARM_2D_IMPL(arm_2d_op_cp_cl_msk_t); if (!__arm_2d_op_acquire((arm_2d_op_core_t *)ptThis)) { return arm_fsm_rt_on_going; } OP_CORE.ptOp = &ARM_2D_OP_TILE_COPY_WITH_COLOUR_MASKING_RGB32; this.Target.ptTile = ptTarget; this.Target.ptRegion = ptRegion; this.Source.ptTile = ptSource; this.wMode = wMode; this.wColour = wMaskColour; return __arm_2d_op_invoke(NULL); } arm_fsm_rt_t __arm_2d_rgb16_sw_tile_fill( __arm_2d_sub_task_t *ptTask) { ARM_2D_IMPL(arm_2d_op_cp_t, ptTask->ptOP); assert(ARM_2D_COLOUR_SZ_16BIT == OP_CORE.ptOp->Info.Colour.u3ColourSZ); uint32_t wMode = this.wMode; if (wMode & (ARM_2D_CP_MODE_Y_MIRROR | ARM_2D_CP_MODE_X_MIRROR)) { __arm_2d_impl_rgb16_fill_mirror(ptTask->Param.tFill.tSource.pBuffer, ptTask->Param.tFill.tSource.iStride, &ptTask->Param.tFill.tSource.tValidRegion.tSize, ptTask->Param.tFill.tTarget.pBuffer, ptTask->Param.tFill.tTarget.iStride, &ptTask->Param.tFill.tTarget.tValidRegion.tSize, wMode); } else { __arm_2d_impl_rgb16_fill( ptTask->Param.tFill.tSource.pBuffer, ptTask->Param.tFill.tSource.iStride, &ptTask->Param.tFill.tSource.tValidRegion.tSize, ptTask->Param.tFill.tTarget.pBuffer, ptTask->Param.tFill.tTarget.iStride, &ptTask->Param.tFill.tTarget.tValidRegion.tSize); } return arm_fsm_rt_cpl; } arm_fsm_rt_t __arm_2d_rgb32_sw_tile_fill( __arm_2d_sub_task_t *ptTask) { ARM_2D_IMPL(arm_2d_op_cp_t, ptTask->ptOP); assert(ARM_2D_COLOUR_SZ_32BIT == OP_CORE.ptOp->Info.Colour.u3ColourSZ); uint32_t wMode = this.wMode; if (wMode & (ARM_2D_CP_MODE_Y_MIRROR | ARM_2D_CP_MODE_X_MIRROR)) { __arm_2d_impl_rgb32_fill_mirror(ptTask->Param.tFill.tSource.pBuffer, ptTask->Param.tFill.tSource.iStride, &ptTask->Param.tFill.tSource.tValidRegion.tSize, ptTask->Param.tFill.tTarget.pBuffer, ptTask->Param.tFill.tTarget.iStride, &ptTask->Param.tFill.tTarget.tValidRegion.tSize, wMode); } else { __arm_2d_impl_rgb32_fill( ptTask->Param.tFill.tSource.pBuffer, ptTask->Param.tFill.tSource.iStride, &ptTask->Param.tFill.tSource.tValidRegion.tSize, ptTask->Param.tFill.tTarget.pBuffer, ptTask->Param.tFill.tTarget.iStride, &ptTask->Param.tFill.tTarget.tValidRegion.tSize); } return arm_fsm_rt_cpl; } arm_fsm_rt_t __arm_2d_rgb16_sw_tile_copy( __arm_2d_sub_task_t *ptTask) { ARM_2D_IMPL(arm_2d_op_cp_t, ptTask->ptOP); assert(ARM_2D_COLOUR_SZ_16BIT == OP_CORE.ptOp->Info.Colour.u3ColourSZ); uint32_t wMode = this.wMode; if (wMode & (ARM_2D_CP_MODE_Y_MIRROR | ARM_2D_CP_MODE_X_MIRROR)) { __arm_2d_impl_rgb16_copy_mirror( ptTask->Param.tCopy.tSource.pBuffer, ptTask->Param.tCopy.tSource.iStride, ptTask->Param.tCopy.tTarget.pBuffer, ptTask->Param.tCopy.tTarget.iStride, &ptTask->Param.tCopy.tCopySize, wMode); } else { __arm_2d_impl_rgb16_copy( ptTask->Param.tCopy.tSource.pBuffer, ptTask->Param.tCopy.tSource.iStride, ptTask->Param.tCopy.tTarget.pBuffer, ptTask->Param.tCopy.tTarget.iStride, &ptTask->Param.tCopy.tCopySize); } return arm_fsm_rt_cpl; } arm_fsm_rt_t __arm_2d_rgb32_sw_tile_copy( __arm_2d_sub_task_t *ptTask) { ARM_2D_IMPL(arm_2d_op_cp_t, ptTask->ptOP); assert(ARM_2D_COLOUR_SZ_32BIT == OP_CORE.ptOp->Info.Colour.u3ColourSZ); uint32_t wMode = this.wMode; if (wMode & (ARM_2D_CP_MODE_Y_MIRROR | ARM_2D_CP_MODE_X_MIRROR)) { __arm_2d_impl_rgb32_copy_mirror( ptTask->Param.tCopy.tSource.pBuffer, ptTask->Param.tCopy.tSource.iStride, ptTask->Param.tCopy.tTarget.pBuffer, ptTask->Param.tCopy.tTarget.iStride, &ptTask->Param.tCopy.tCopySize, wMode); } else { __arm_2d_impl_rgb32_copy( ptTask->Param.tCopy.tSource.pBuffer, ptTask->Param.tCopy.tSource.iStride, ptTask->Param.tCopy.tTarget.pBuffer, ptTask->Param.tCopy.tTarget.iStride, &ptTask->Param.tCopy.tCopySize); } return arm_fsm_rt_cpl; } arm_fsm_rt_t __arm_2d_rgb16_sw_tile_copy_with_colour_masking( __arm_2d_sub_task_t *ptTask) { ARM_2D_IMPL(arm_2d_op_cp_cl_msk_t, ptTask->ptOP) assert(ARM_2D_COLOUR_SZ_16BIT == OP_CORE.ptOp->Info.Colour.u3ColourSZ); uint32_t wMode = this.wMode; if (wMode & (ARM_2D_CP_MODE_Y_MIRROR | ARM_2D_CP_MODE_X_MIRROR)) { __arm_2d_impl_rgb16_copy_colour_masking_mirror( ptTask->Param.tCopy.tSource.pBuffer, ptTask->Param.tCopy.tSource.iStride, ptTask->Param.tCopy.tTarget.pBuffer, ptTask->Param.tCopy.tTarget.iStride, &ptTask->Param.tCopy.tCopySize, wMode, this.hwColour); } else { __arm_2d_impl_rgb16_copy_colour_masking( ptTask->Param.tCopy.tSource.pBuffer, ptTask->Param.tCopy.tSource.iStride, ptTask->Param.tCopy.tTarget.pBuffer, ptTask->Param.tCopy.tTarget.iStride, &ptTask->Param.tCopy.tCopySize, this.hwColour); } return arm_fsm_rt_cpl; } arm_fsm_rt_t __arm_2d_rgb32_sw_tile_copy_with_colour_masking( __arm_2d_sub_task_t *ptTask) { ARM_2D_IMPL(arm_2d_op_cp_cl_msk_t, ptTask->ptOP) assert(ARM_2D_COLOUR_SZ_32BIT == OP_CORE.ptOp->Info.Colour.u3ColourSZ); uint32_t wMode = this.wMode; if (wMode & (ARM_2D_CP_MODE_Y_MIRROR | ARM_2D_CP_MODE_X_MIRROR)) { __arm_2d_impl_rgb32_copy_colour_masking_mirror( ptTask->Param.tCopy.tSource.pBuffer, ptTask->Param.tCopy.tSource.iStride, ptTask->Param.tCopy.tTarget.pBuffer, ptTask->Param.tCopy.tTarget.iStride, &ptTask->Param.tCopy.tCopySize, wMode, this.wColour); } else { __arm_2d_impl_rgb32_copy_colour_masking( ptTask->Param.tCopy.tSource.pBuffer, ptTask->Param.tCopy.tSource.iStride, ptTask->Param.tCopy.tTarget.pBuffer, ptTask->Param.tCopy.tTarget.iStride, &ptTask->Param.tCopy.tCopySize, this.wColour); } return arm_fsm_rt_cpl; } arm_fsm_rt_t __arm_2d_rgb16_sw_tile_fill_with_colour_masking( __arm_2d_sub_task_t *ptTask) { ARM_2D_IMPL(arm_2d_op_cp_cl_msk_t, ptTask->ptOP); assert(ARM_2D_COLOUR_SZ_16BIT == OP_CORE.ptOp->Info.Colour.u3ColourSZ); uint32_t wMode = this.wMode; if (wMode & (ARM_2D_CP_MODE_Y_MIRROR | ARM_2D_CP_MODE_X_MIRROR)) { __arm_2d_impl_rgb16_fill_colour_masking_mirror( ptTask->Param.tFill.tSource.pBuffer, ptTask->Param.tFill.tSource.iStride, &ptTask->Param.tFill.tSource.tValidRegion.tSize, ptTask->Param.tFill.tTarget.pBuffer, ptTask->Param.tFill.tTarget.iStride, &ptTask->Param.tFill.tTarget.tValidRegion.tSize, wMode, this.hwColour); } else { __arm_2d_impl_rgb16_fill_colour_masking( ptTask->Param.tFill.tSource.pBuffer, ptTask->Param.tFill.tSource.iStride, &ptTask->Param.tFill.tSource.tValidRegion.tSize, ptTask->Param.tFill.tTarget.pBuffer, ptTask->Param.tFill.tTarget.iStride, &ptTask->Param.tFill.tTarget.tValidRegion.tSize, this.hwColour); } return arm_fsm_rt_cpl; } arm_fsm_rt_t __arm_2d_rgb32_sw_tile_fill_with_colour_masking( __arm_2d_sub_task_t *ptTask) { ARM_2D_IMPL(arm_2d_op_cp_cl_msk_t, ptTask->ptOP); assert(ARM_2D_COLOUR_SZ_32BIT == OP_CORE.ptOp->Info.Colour.u3ColourSZ); uint32_t wMode = this.wMode; if (wMode & (ARM_2D_CP_MODE_Y_MIRROR | ARM_2D_CP_MODE_X_MIRROR)) { __arm_2d_impl_rgb32_fill_colour_masking_mirror( ptTask->Param.tFill.tSource.pBuffer, ptTask->Param.tFill.tSource.iStride, &ptTask->Param.tFill.tSource.tValidRegion.tSize, ptTask->Param.tFill.tTarget.pBuffer, ptTask->Param.tFill.tTarget.iStride, &ptTask->Param.tFill.tTarget.tValidRegion.tSize, wMode, this.wColour); } else { __arm_2d_impl_rgb32_fill_colour_masking( ptTask->Param.tFill.tSource.pBuffer, ptTask->Param.tFill.tSource.iStride, &ptTask->Param.tFill.tSource.tValidRegion.tSize, ptTask->Param.tFill.tTarget.pBuffer, ptTask->Param.tFill.tTarget.iStride, &ptTask->Param.tFill.tTarget.tValidRegion.tSize, this.wColour); } return arm_fsm_rt_cpl; } /*----------------------------------------------------------------------------* * Accelerable Low Level APIs * *----------------------------------------------------------------------------*/ #if defined(__clang__) # pragma clang diagnostic pop #elif defined(__IS_COMPILER_ARM_COMPILER_5__) # pragma diag_warning 174,177,188,68,513,144 #elif defined(__IS_COMPILER_GCC__) # pragma GCC diagnostic pop #endif #ifdef __cplusplus } #endif