/* * 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_draw.c * Description: APIs for colour format conversion * * $Date: 08. Aug 2022 * $Revision: V.1.0.2 * * Target Processor: Cortex-M cores * * -------------------------------------------------------------------- */ /*============================ INCLUDES ======================================*/ #define __ARM_2D_IMPL__ #include "arm_2d.h" #include "__arm_2d_impl.h" #ifdef __cplusplus extern "C" { #endif #if defined(__clang__) # pragma clang diagnostic ignored "-Wunknown-warning-option" # pragma clang diagnostic ignored "-Wreserved-identifier" # pragma clang diagnostic ignored "-Wincompatible-pointer-types-discards-qualifiers" # pragma clang diagnostic ignored "-Wmissing-variable-declarations" # 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" # pragma clang diagnostic ignored "-Wgnu-zero-variadic-macro-arguments" # pragma clang diagnostic ignored "-Wswitch-enum" # pragma clang diagnostic ignored "-Wswitch" # pragma clang diagnostic ignored "-Wdeclaration-after-statement" #elif defined(__IS_COMPILER_ARM_COMPILER_5__) # pragma diag_suppress 174,177,188,68,513,144 #endif /*============================ MACROS ========================================*/ /*============================ MACROFIED FUNCTIONS ===========================*/ /*============================ TYPES =========================================*/ /*============================ GLOBAL VARIABLES ==============================*/ /*============================ PROTOTYPES ====================================*/ void __arm_2d_impl_gray8_to_rgb565( uint8_t *__RESTRICT pchSourceBase, int16_t iSourceStride, uint16_t *__RESTRICT phwTargetBase, int16_t iTargetStride, arm_2d_size_t *__RESTRICT ptCopySize); void __arm_2d_impl_cccn888_to_rgb565(uint32_t *__RESTRICT pwSource, int16_t iSourceStride, uint16_t *__RESTRICT phwTarget, int16_t iTargetStride, arm_2d_size_t *__RESTRICT ptCopySize); void __arm_2d_impl_rgb565_to_cccn888(uint16_t *__RESTRICT phwSourceBase, int16_t iSourceStride, uint32_t *__RESTRICT pwTargetBase, int16_t iTargetStride, arm_2d_size_t *__RESTRICT ptCopySize); void __arm_2d_impl_gray8_to_cccn888(uint8_t *__RESTRICT pchSourceBase, int16_t iSourceStride, uint32_t *__RESTRICT pwTargetBase, int16_t iTargetStride, arm_2d_size_t *__RESTRICT ptCopySize); void __arm_2d_impl_rgb565_to_gray8( uint16_t *__RESTRICT phwSourceBase, int16_t iSourceStride, uint8_t *__RESTRICT pchTargetBase, int16_t iTargetStride, arm_2d_size_t *__RESTRICT ptCopySize); void __arm_2d_impl_cccn888_to_gray8(uint32_t *__RESTRICT pwSource, int16_t iSourceStride, uint8_t *__RESTRICT pchwTarget, int16_t iTargetStride, arm_2d_size_t *__RESTRICT ptCopySize); /*============================ LOCAL VARIABLES ===============================*/ /*============================ IMPLEMENTATION ================================*/ /*----------------------------------------------------------------------------* * Convert Colour format * *----------------------------------------------------------------------------*/ /*! * \brief convert the colour format of a given tile to rgb888 * \param[in] ptOP the control block, NULL means using the default control block * \param[in] ptSource the source tile * \param[out] ptTarget the output tile (holding a buffer) * \return arm_fsm_rt_t the operation result */ ARM_NONNULL(2,3) arm_fsm_rt_t arm_2dp_convert_colour_to_rgb888( arm_2d_op_cl_convt_t *ptOP, const arm_2d_tile_t *ptSource, const arm_2d_tile_t *ptTarget) { assert(NULL != ptSource); assert(NULL != ptTarget); ARM_2D_IMPL(arm_2d_op_cl_convt_t, ptOP); 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_CONVERT_TO_RGB888; this.Target.ptTile = ptTarget; this.Target.ptRegion = NULL; this.Source.ptTile = ptSource; return __arm_2d_op_invoke((arm_2d_op_core_t *)ptThis); } /*! * \brief convert the colour format of a given tile to rgb565 * \param[in] ptOP the control block, NULL means using the default control block * \param[in] ptSource the source tile * \param[out] ptTarget the output tile (holding a buffer) * \return arm_fsm_rt_t the operation result */ ARM_NONNULL(2,3) arm_fsm_rt_t arm_2dp_convert_colour_to_rgb565( arm_2d_op_cl_convt_t *ptOP, const arm_2d_tile_t *ptSource, const arm_2d_tile_t *ptTarget) { assert(NULL != ptSource); assert(NULL != ptTarget); ARM_2D_IMPL(arm_2d_op_cl_convt_t, ptOP); 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_CONVERT_TO_RGB565; this.Target.ptTile = ptTarget; this.Target.ptRegion = NULL; this.Source.ptTile = ptSource; return __arm_2d_op_invoke((arm_2d_op_core_t *)ptThis); } /*! * \brief convert the colour format of a given tile to gray8 * \param[in] ptOP the control block, NULL means using the default control block * \param[in] ptSource the source tile * \param[out] ptTarget the output tile (holding a buffer) * \return arm_fsm_rt_t the operation result */ ARM_NONNULL(2,3) arm_fsm_rt_t arm_2dp_convert_colour_to_gray8( arm_2d_op_cl_convt_t *ptOP, const arm_2d_tile_t *ptSource, const arm_2d_tile_t *ptTarget) { assert(NULL != ptSource); assert(NULL != ptTarget); ARM_2D_IMPL(arm_2d_op_cl_convt_t, ptOP); 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_CONVERT_TO_GRAY8; this.Target.ptTile = ptTarget; this.Target.ptRegion = NULL; this.Source.ptTile = ptSource; return __arm_2d_op_invoke((arm_2d_op_core_t *)ptThis); } arm_fsm_rt_t __arm_2d_sw_convert_colour_to_gray8(__arm_2d_sub_task_t *ptTask) { ARM_2D_IMPL(arm_2d_op_cl_convt_t, ptTask->ptOP); if (!this.Source.ptTile->bHasEnforcedColour) { return (arm_fsm_rt_t)ARM_2D_ERR_MISSING_PARAM; } switch ( this.Source.ptTile->tInfo.tColourInfo.u3ColourSZ) { case ARM_2D_COLOUR_8BIT: /* no need to convert, return cpl directly */ break; case ARM_2D_COLOUR_SZ_16BIT: __arm_2d_impl_rgb565_to_gray8( ptTask->Param.tCopy.tSource.pBuffer, ptTask->Param.tCopy.tSource.iStride, ptTask->Param.tCopy.tTarget.pBuffer, ptTask->Param.tCopy.tTarget.iStride, &(ptTask->Param.tCopy.tCopySize)); break; case ARM_2D_COLOUR_SZ_32BIT: __arm_2d_impl_cccn888_to_gray8( ptTask->Param.tCopy.tSource.pBuffer, ptTask->Param.tCopy.tSource.iStride, ptTask->Param.tCopy.tTarget.pBuffer, ptTask->Param.tCopy.tTarget.iStride, &(ptTask->Param.tCopy.tCopySize)); break; default: return (arm_fsm_rt_t)ARM_2D_ERR_UNSUPPORTED_COLOUR; } return arm_fsm_rt_cpl; } arm_fsm_rt_t __arm_2d_sw_convert_colour_to_rgb565(__arm_2d_sub_task_t *ptTask) { ARM_2D_IMPL(arm_2d_op_cl_convt_t, ptTask->ptOP); if (!this.Source.ptTile->bHasEnforcedColour) { return (arm_fsm_rt_t)ARM_2D_ERR_MISSING_PARAM; } switch ( this.Source.ptTile->tInfo.tColourInfo.u3ColourSZ) { case ARM_2D_COLOUR_8BIT: __arm_2d_impl_gray8_to_rgb565( ptTask->Param.tCopy.tSource.pBuffer, ptTask->Param.tCopy.tSource.iStride, ptTask->Param.tCopy.tTarget.pBuffer, ptTask->Param.tCopy.tTarget.iStride, &(ptTask->Param.tCopy.tCopySize)); break; case ARM_2D_COLOUR_SZ_16BIT: /* no need to convert, return cpl directly */ break; case ARM_2D_COLOUR_SZ_32BIT: __arm_2d_impl_cccn888_to_rgb565( ptTask->Param.tCopy.tSource.pBuffer, ptTask->Param.tCopy.tSource.iStride, ptTask->Param.tCopy.tTarget.pBuffer, ptTask->Param.tCopy.tTarget.iStride, &(ptTask->Param.tCopy.tCopySize)); break; default: return (arm_fsm_rt_t)ARM_2D_ERR_UNSUPPORTED_COLOUR; } return arm_fsm_rt_cpl; } arm_fsm_rt_t __arm_2d_sw_convert_colour_to_rgb888(__arm_2d_sub_task_t *ptTask) { ARM_2D_IMPL(arm_2d_op_cl_convt_t, ptTask->ptOP); if (!this.Source.ptTile->bHasEnforcedColour) { return (arm_fsm_rt_t)ARM_2D_ERR_MISSING_PARAM; } switch ( this.Source.ptTile->tInfo.tColourInfo.u3ColourSZ) { case ARM_2D_COLOUR_8BIT: __arm_2d_impl_gray8_to_cccn888( ptTask->Param.tCopy.tSource.pBuffer, ptTask->Param.tCopy.tSource.iStride, ptTask->Param.tCopy.tTarget.pBuffer, ptTask->Param.tCopy.tTarget.iStride, &(ptTask->Param.tCopy.tCopySize)); break; case ARM_2D_COLOUR_SZ_16BIT: __arm_2d_impl_rgb565_to_cccn888( ptTask->Param.tCopy.tSource.pBuffer, ptTask->Param.tCopy.tSource.iStride, ptTask->Param.tCopy.tTarget.pBuffer, ptTask->Param.tCopy.tTarget.iStride, &(ptTask->Param.tCopy.tCopySize)); break; case ARM_2D_COLOUR_SZ_32BIT: /* no need to convert, return cpl directly */ break; default: return (arm_fsm_rt_t)ARM_2D_ERR_UNSUPPORTED_COLOUR; } return arm_fsm_rt_cpl; } /*----------------------------------------------------------------------------* * Accelerable Low Level APIs * *----------------------------------------------------------------------------*/ __WEAK void __arm_2d_impl_gray8_to_rgb565( uint8_t *__RESTRICT pchSourceBase, int16_t iSourceStride, uint16_t *__RESTRICT phwTargetBase, int16_t iTargetStride, arm_2d_size_t *__RESTRICT ptCopySize) { for (int_fast16_t y = 0; y < ptCopySize->iHeight; y++) { const uint8_t *__RESTRICT pchSource = pchSourceBase; uint16_t *__RESTRICT phwTarget = phwTargetBase; for (int_fast16_t x = 0; x < ptCopySize->iWidth; x++) { __arm_2d_color_fast_rgb_t hwTargetPixel; uint_fast8_t chColour = *pchSource++; hwTargetPixel.R = chColour; hwTargetPixel.G = chColour; hwTargetPixel.B = chColour; *phwTarget++ = __arm_2d_rgb565_pack(&hwTargetPixel); } pchSourceBase += iSourceStride; phwTargetBase += iTargetStride; } } __WEAK void __arm_2d_impl_cccn888_to_rgb565(uint32_t *__RESTRICT pwSourceBase, int16_t iSourceStride, uint16_t *__RESTRICT phwTargetBase, int16_t iTargetStride, arm_2d_size_t *__RESTRICT ptCopySize) { for (int_fast16_t y = 0; y < ptCopySize->iHeight; y++) { const uint32_t *__RESTRICT pwSource = pwSourceBase; uint16_t *__RESTRICT phwTarget = phwTargetBase; for (int_fast16_t x = 0; x < ptCopySize->iWidth; x++) { arm_2d_color_bgra8888_t wSrcPixel; __arm_2d_color_fast_rgb_t hwTargetPixel; wSrcPixel.tValue = *pwSource++; hwTargetPixel.R = wSrcPixel.u8R; hwTargetPixel.G = wSrcPixel.u8G; hwTargetPixel.B = wSrcPixel.u8B; *phwTarget++ = __arm_2d_rgb565_pack(&hwTargetPixel); } pwSourceBase += iSourceStride; phwTargetBase += iTargetStride; } } __WEAK void __arm_2d_impl_gray8_to_cccn888(uint8_t *__RESTRICT pchSourceBase, int16_t iSourceStride, uint32_t *__RESTRICT pwTargetBase, int16_t iTargetStride, arm_2d_size_t *__RESTRICT ptCopySize) { for (int_fast16_t y = 0; y < ptCopySize->iHeight; y++) { const uint8_t *__RESTRICT pchSource = pchSourceBase; uint32_t *__RESTRICT pwTarget = pwTargetBase; for (int_fast16_t x = 0; x < ptCopySize->iWidth; x++) { uint_fast8_t chPixel = *pchSource++; arm_2d_color_bgra8888_t wTargetPixel; wTargetPixel.u8R = chPixel; wTargetPixel.u8G = chPixel; wTargetPixel.u8B = chPixel; wTargetPixel.u8A = 0xFF; *pwTarget++ = wTargetPixel.tValue; } pchSourceBase += iSourceStride; pwTargetBase += iTargetStride; } } __WEAK void __arm_2d_impl_rgb565_to_cccn888(uint16_t *__RESTRICT phwSourceBase, int16_t iSourceStride, uint32_t *__RESTRICT pwTargetBase, int16_t iTargetStride, arm_2d_size_t *__RESTRICT ptCopySize) { for (int_fast16_t y = 0; y < ptCopySize->iHeight; y++) { const uint16_t *__RESTRICT phwSource = phwSourceBase; uint32_t *__RESTRICT pwTarget = pwTargetBase; for (int_fast16_t x = 0; x < ptCopySize->iWidth; x++) { __arm_2d_color_fast_rgb_t hwSrcPixel; arm_2d_color_bgra8888_t wTargetPixel; __arm_2d_rgb565_unpack(*phwSource++, &hwSrcPixel); wTargetPixel.u8R = hwSrcPixel.R; wTargetPixel.u8G = hwSrcPixel.G; wTargetPixel.u8B = hwSrcPixel.B; wTargetPixel.u8A = 0xFF; *pwTarget++ = wTargetPixel.tValue; } phwSourceBase += iSourceStride; pwTargetBase += iTargetStride; } } __WEAK void __arm_2d_impl_rgb565_to_gray8( uint16_t *__RESTRICT phwSourceBase, int16_t iSourceStride, uint8_t *__RESTRICT pchTargetBase, int16_t iTargetStride, arm_2d_size_t *__RESTRICT ptCopySize) { for (int_fast16_t y = 0; y < ptCopySize->iHeight; y++) { const uint16_t *__RESTRICT phwSource = phwSourceBase; uint8_t *__RESTRICT pchTarget = pchTargetBase; for (int_fast16_t x = 0; x < ptCopySize->iWidth; x++) { __arm_2d_color_fast_rgb_t hwSrcPixel; uint_fast16_t hwPixel = 0; __arm_2d_rgb565_unpack(*phwSource++, &hwSrcPixel); hwPixel += hwSrcPixel.R; hwPixel += hwSrcPixel.G; hwPixel += hwSrcPixel.B; *pchTarget++ = hwPixel / 3; } phwSourceBase += iSourceStride; pchTargetBase += iTargetStride; } } __WEAK void __arm_2d_impl_cccn888_to_gray8(uint32_t *__RESTRICT pwSourceBase, int16_t iSourceStride, uint8_t *__RESTRICT pchTargetBase, int16_t iTargetStride, arm_2d_size_t *__RESTRICT ptCopySize) { for (int_fast16_t y = 0; y < ptCopySize->iHeight; y++) { const uint32_t *__RESTRICT pwSource = pwSourceBase; uint8_t *__RESTRICT pchTarget = pchTargetBase; for (int_fast16_t x = 0; x < ptCopySize->iWidth; x++) { arm_2d_color_bgra8888_t wSrcPixel; uint_fast16_t hwPixel = 0; wSrcPixel.tValue = *pwSource++; hwPixel += wSrcPixel.u8R; hwPixel += wSrcPixel.u8G; hwPixel += wSrcPixel.u8B; *pchTarget++ = hwPixel / 3; } pwSourceBase += iSourceStride; pchTargetBase += iTargetStride; } } /*----------------------------------------------------------------------------* * Low Level IO Interfaces * *----------------------------------------------------------------------------*/ __WEAK def_low_lv_io( __ARM_2D_IO_COLOUR_CONVERT_TO_GRAY8, __arm_2d_sw_convert_colour_to_gray8); __WEAK def_low_lv_io( __ARM_2D_IO_COLOUR_CONVERT_TO_RGB565, __arm_2d_sw_convert_colour_to_rgb565); __WEAK def_low_lv_io( __ARM_2D_IO_COLOUR_CONVERT_TO_RGB888, __arm_2d_sw_convert_colour_to_rgb888); const __arm_2d_op_info_t ARM_2D_OP_CONVERT_TO_GRAY8 = { .Info = { .Colour = { .chScheme = ARM_2D_COLOUR_GRAY8, }, .Param = { .bHasSource = true, .bHasTarget = true, .bAllowEnforcedColour = true, }, .chOpIndex = __ARM_2D_OP_IDX_COLOUR_FORMAT_CONVERSION, .LowLevelIO = { .ptCopyLike = ref_low_lv_io(__ARM_2D_IO_COLOUR_CONVERT_TO_GRAY8), .ptFillLike = NULL, }, }, }; const __arm_2d_op_info_t ARM_2D_OP_CONVERT_TO_RGB565 = { .Info = { .Colour = { .chScheme = ARM_2D_COLOUR_RGB565, }, .Param = { .bHasSource = true, .bHasTarget = true, .bAllowEnforcedColour = true, }, .chOpIndex = __ARM_2D_OP_IDX_COLOUR_FORMAT_CONVERSION, .LowLevelIO = { .ptCopyLike = ref_low_lv_io(__ARM_2D_IO_COLOUR_CONVERT_TO_RGB565), .ptFillLike = NULL, }, }, }; const __arm_2d_op_info_t ARM_2D_OP_CONVERT_TO_RGB888 = { .Info = { .Colour = { .chScheme = ARM_2D_COLOUR_CCCN888, }, .Param = { .bHasSource = true, .bHasTarget = true, .bAllowEnforcedColour = true, }, .chOpIndex = __ARM_2D_OP_IDX_COLOUR_FORMAT_CONVERSION, .LowLevelIO = { .ptCopyLike = ref_low_lv_io(__ARM_2D_IO_COLOUR_CONVERT_TO_RGB888), .ptFillLike = NULL, }, }, }; #ifdef __cplusplus } #endif