| /****************************************************************************** |
| * |
| * Copyright (C) 2012 Ittiam Systems Pvt Ltd, Bangalore |
| * |
| * 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: |
| * |
| * http://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. |
| * |
| ******************************************************************************/ |
| /** |
| ******************************************************************************* |
| * @file |
| * ihevc_intra_pred_filters.c |
| * |
| * @brief |
| * Contains function Definition for intra prediction interpolation filters |
| * |
| * |
| * @author |
| * Srinivas T |
| * |
| * @par List of Functions: |
| * - ihevc_intra_pred_luma_planar() |
| * - ihevc_intra_pred_luma_dc() |
| * - ihevc_intra_pred_luma_horz() |
| * - ihevc_intra_pred_luma_ver() |
| * - ihevc_intra_pred_luma_mode2() |
| * - ihevc_intra_pred_luma_mode_18_34() |
| * - ihevc_intra_pred_luma_mode_3_to_9() |
| * - ihevc_intra_pred_luma_mode_11_to_17() |
| * - ihevc_intra_pred_luma_mode_19_to_25() |
| * - ihevc_intra_pred_luma_mode_27_to_33() |
| * - ihevc_intra_pred_luma_ref_substitution() |
| * |
| * @remarks |
| * None |
| * |
| ******************************************************************************* |
| */ |
| |
| |
| /*****************************************************************************/ |
| /* File Includes */ |
| /*****************************************************************************/ |
| |
| #include <assert.h> |
| #include "ihevc_typedefs.h" |
| #include "ihevc_intra_pred.h" |
| #include "ihevc_macros.h" |
| #include "ihevc_func_selector.h" |
| #include "ihevc_platform_macros.h" |
| #include "ihevc_common_tables.h" |
| #include "ihevc_defs.h" |
| #include "ihevc_mem_fns.h" |
| #include "ihevc_debug.h" |
| |
| /****************************************************************************/ |
| /* Constant Macros */ |
| /****************************************************************************/ |
| #define MAX_CU_SIZE 64 |
| #define BIT_DEPTH 8 |
| #define T32_4NT 128 |
| #define T16_4NT 64 |
| |
| |
| /****************************************************************************/ |
| /* Function Macros */ |
| /****************************************************************************/ |
| #define GET_BITS(y,x) ((y) & (1 << x)) && (1 << x) |
| |
| /*****************************************************************************/ |
| /* global tables Definition */ |
| /*****************************************************************************/ |
| |
| |
| /*****************************************************************************/ |
| /* Function Definition */ |
| /*****************************************************************************/ |
| |
| /** |
| ******************************************************************************* |
| * |
| * @brief |
| * Intra prediction interpolation filter for pu1_ref substitution |
| * |
| * |
| * @par Description: |
| * Reference substitution process for samples unavailable for prediction |
| * Refer to section 8.4.4.2.2 |
| * |
| * @param[in] pu1_top_left |
| * UWORD8 pointer to the top-left |
| * |
| * @param[in] pu1_top |
| * UWORD8 pointer to the top |
| * |
| * @param[in] pu1_left |
| * UWORD8 pointer to the left |
| * |
| * @param[in] src_strd |
| * WORD32 Source stride |
| * |
| * @param[in] nbr_flags |
| * WORD32 neighbor availability flags |
| * |
| * @param[in] nt |
| * WORD32 transform Block size |
| * |
| * @param[in] dst_strd |
| * WORD32 Destination stride |
| * |
| * @returns |
| * |
| * @remarks |
| * None |
| * |
| ******************************************************************************* |
| */ |
| void ihevc_intra_pred_luma_ref_subst_all_avlble(UWORD8 *pu1_top_left, |
| UWORD8 *pu1_top, |
| UWORD8 *pu1_left, |
| WORD32 src_strd, |
| WORD32 nt, |
| WORD32 nbr_flags, |
| UWORD8 *pu1_dst, |
| WORD32 dst_strd) |
| { |
| |
| WORD32 i; |
| WORD32 two_nt = 2 * nt; |
| UNUSED(nbr_flags); |
| UNUSED(dst_strd); |
| |
| /* Neighbor Flag Structure*/ |
| /* MSB ---> LSB */ |
| /* Top-Left | Top-Right | Top | Left | Bottom-Left |
| 1 4 4 4 4 |
| */ |
| ASSERT((nbr_flags == 0x11188) || (nbr_flags == 0x133CC) || (nbr_flags == 0x1FFFF)); |
| { |
| |
| if(nt == 4) |
| { |
| /* 1 bit extraction for all the neighboring blocks */ |
| |
| |
| /* Else fill the corresponding samples */ |
| pu1_dst[two_nt] = *pu1_top_left; |
| //if(left) |
| { |
| for(i = 0; i < nt; i++) |
| pu1_dst[two_nt - 1 - i] = pu1_left[i * src_strd]; |
| } |
| // if(bot_left) |
| { |
| for(i = nt; i < two_nt; i++) |
| pu1_dst[two_nt - 1 - i] = pu1_left[i * src_strd]; |
| } |
| // if(top) |
| { |
| ihevc_memcpy(&pu1_dst[two_nt + 1], pu1_top, nt); |
| } |
| // if(tp_right) |
| { |
| ihevc_memcpy(&pu1_dst[two_nt + 1 + nt], pu1_top + nt, nt); |
| } |
| |
| |
| } |
| else |
| |
| { |
| |
| /* Else fill the corresponding samples */ |
| ASSERT((nt == 8) || (nt == 16) || (nt == 32)); |
| pu1_dst[two_nt] = *pu1_top_left; |
| |
| for(i = 0; i < nt; i++) |
| pu1_dst[two_nt - 1 - i] = pu1_left[i * src_strd]; |
| |
| for(i = nt; i < two_nt; i++) |
| pu1_dst[two_nt - 1 - i] = pu1_left[i * src_strd]; |
| |
| ihevc_memcpy_mul_8(&pu1_dst[two_nt + 1], pu1_top, nt); |
| |
| ihevc_memcpy_mul_8(&pu1_dst[two_nt + 1 + nt], pu1_top + nt, nt); |
| } |
| |
| } |
| } |
| |
| |
| void ihevc_intra_pred_luma_ref_substitution(UWORD8 *pu1_top_left, |
| UWORD8 *pu1_top, |
| UWORD8 *pu1_left, |
| WORD32 src_strd, |
| WORD32 nt, |
| WORD32 nbr_flags, |
| UWORD8 *pu1_dst, |
| WORD32 dst_strd) |
| { |
| UWORD8 pu1_ref; |
| WORD32 dc_val, i; |
| WORD32 total_samples = (4 * nt) + 1; |
| WORD32 two_nt = 2 * nt; |
| |
| WORD32 three_nt = 3 * nt; |
| WORD32 get_bits; |
| WORD32 next; |
| WORD32 bot_left, left, top, tp_right, tp_left; |
| |
| WORD32 idx, nbr_id_from_bl, frwd_nbr_flag; |
| UNUSED(dst_strd); |
| /*dc_val = 1 << (BIT_DEPTH - 1);*/ |
| dc_val = 1 << (8 - 1); |
| |
| |
| /* Neighbor Flag Structure*/ |
| /* MSB ---> LSB */ |
| /* Top-Left | Top-Right | Top | Left | Bottom-Left |
| 1 4 4 4 4 |
| */ |
| /* If no neighbor flags are present, fill the neighbor samples with DC value */ |
| if(nbr_flags == 0) |
| { |
| for(i = 0; i < total_samples; i++) |
| { |
| pu1_dst[i] = dc_val; |
| } |
| } |
| else |
| { |
| if(nt <= 8) |
| { |
| /* 1 bit extraction for all the neighboring blocks */ |
| tp_left = (nbr_flags & 0x10000) >> 16; |
| bot_left = (nbr_flags & 0x8) >> 3; |
| left = (nbr_flags & 0x80) >> 7; |
| top = (nbr_flags & 0x100) >> 8; |
| tp_right = (nbr_flags & 0x1000) >> 12; |
| |
| /* Else fill the corresponding samples */ |
| if(tp_left) |
| pu1_dst[two_nt] = *pu1_top_left; |
| else |
| pu1_dst[two_nt] = 0; |
| |
| |
| if(left) |
| { |
| for(i = 0; i < nt; i++) |
| pu1_dst[two_nt - 1 - i] = pu1_left[i * src_strd]; |
| } |
| else |
| { |
| ihevc_memset(&pu1_dst[two_nt - 1 - (nt - 1)], 0, nt); |
| } |
| |
| |
| if(bot_left) |
| { |
| for(i = nt; i < two_nt; i++) |
| pu1_dst[two_nt - 1 - i] = pu1_left[i * src_strd]; |
| } |
| else |
| { |
| ihevc_memset(&pu1_dst[two_nt - 1 - (two_nt - 1)], 0, nt); |
| } |
| |
| |
| if(top) |
| { |
| ihevc_memcpy(&pu1_dst[two_nt + 1], pu1_top, nt); |
| } |
| else |
| { |
| ihevc_memset(&pu1_dst[two_nt + 1], 0, nt); |
| } |
| |
| if(tp_right) |
| { |
| ihevc_memcpy(&pu1_dst[two_nt + 1 + nt], pu1_top + nt, nt); |
| } |
| else |
| { |
| ihevc_memset(&pu1_dst[two_nt + 1 + nt], 0, nt); |
| } |
| next = 1; |
| |
| /* If bottom -left is not available, reverse substitution process*/ |
| if(bot_left == 0) |
| { |
| WORD32 a_nbr_flag[5]; |
| a_nbr_flag[0] = bot_left; |
| a_nbr_flag[1] = left; |
| a_nbr_flag[2] = tp_left; |
| a_nbr_flag[3] = top; |
| a_nbr_flag[4] = tp_right; |
| |
| /* Check for the 1st available sample from bottom-left*/ |
| while(!a_nbr_flag[next]) |
| next++; |
| |
| /* If Left, top-left are available*/ |
| if(next <= 2) |
| { |
| idx = nt * next; |
| pu1_ref = pu1_dst[idx]; |
| for(i = 0; i < idx; i++) |
| pu1_dst[i] = pu1_ref; |
| } |
| else /* If top, top-right are available */ |
| { |
| /* Idx is changed to copy 1 pixel value for top-left ,if top-left is not available*/ |
| idx = (nt * (next - 1)) + 1; |
| pu1_ref = pu1_dst[idx]; |
| for(i = 0; i < idx; i++) |
| pu1_dst[i] = pu1_ref; |
| } |
| } |
| |
| /* Forward Substitution Process */ |
| /* If left is Unavailable, copy the last bottom-left value */ |
| if(left == 0) |
| { |
| ihevc_memset(&pu1_dst[nt], pu1_dst[nt - 1], nt); |
| |
| } |
| /* If top-left is Unavailable, copy the last left value */ |
| if(tp_left == 0) |
| pu1_dst[two_nt] = pu1_dst[two_nt - 1]; |
| /* If top is Unavailable, copy the last top-left value */ |
| if(top == 0) |
| { |
| ihevc_memset(&pu1_dst[two_nt + 1], pu1_dst[two_nt], nt); |
| } |
| /* If to right is Unavailable, copy the last top value */ |
| if(tp_right == 0) |
| { |
| ihevc_memset(&pu1_dst[three_nt + 1], pu1_dst[three_nt], nt); |
| |
| } |
| } |
| |
| if(nt == 16) |
| { |
| WORD32 nbr_flags_temp = 0; |
| nbr_flags_temp = ((nbr_flags & 0xC) >> 2) + ((nbr_flags & 0xC0) >> 4) |
| + ((nbr_flags & 0x300) >> 4) |
| + ((nbr_flags & 0x3000) >> 6) |
| + ((nbr_flags & 0x10000) >> 8); |
| |
| /* Else fill the corresponding samples */ |
| if(nbr_flags & 0x10000) |
| pu1_dst[two_nt] = *pu1_top_left; |
| else |
| pu1_dst[two_nt] = 0; |
| |
| if(nbr_flags & 0xC0) |
| { |
| for(i = 0; i < nt; i++) |
| pu1_dst[two_nt - 1 - i] = pu1_left[i * src_strd]; |
| } |
| else |
| { |
| ihevc_memset_mul_8(&pu1_dst[two_nt - 1 - (nt - 1)], 0, nt); |
| } |
| |
| /* Bottom - left availability is checked for every 8x8 TU position and set accordingly */ |
| { |
| if(nbr_flags & 0x8) |
| { |
| for(i = nt; i < (nt + 8); i++) |
| pu1_dst[two_nt - 1 - i] = pu1_left[i * src_strd]; |
| } |
| else |
| { |
| ihevc_memset_mul_8(&pu1_dst[nt - 8], 0, 8); |
| } |
| |
| if(nbr_flags & 0x4) |
| { |
| for(i = (nt + 8); i < two_nt; i++) |
| pu1_dst[two_nt - 1 - i] = pu1_left[i * src_strd]; |
| } |
| else |
| { |
| ihevc_memset_mul_8(&pu1_dst[0], 0, 8); |
| } |
| } |
| |
| if(nbr_flags & 0x300) |
| { |
| ihevc_memcpy_mul_8(&pu1_dst[two_nt + 1], pu1_top, nt); |
| } |
| else |
| { |
| ihevc_memset_mul_8(&pu1_dst[two_nt + 1], 0, nt); |
| } |
| |
| if(nbr_flags & 0x3000) |
| { |
| ihevc_memcpy_mul_8(&pu1_dst[two_nt + 1 + nt], pu1_top + nt, nt); |
| } |
| else |
| { |
| ihevc_memset_mul_8(&pu1_dst[two_nt + 1 + nt], 0, nt); |
| } |
| /* compute trailing zeors based on nbr_flag for substitution process of below left see section .*/ |
| /* as each bit in nbr flags corresponds to 8 pels for bot_left, left, top and topright but 1 pel for topleft */ |
| { |
| nbr_id_from_bl = look_up_trailing_zeros(nbr_flags_temp & 0XF) * 8; /* for below left and left */ |
| |
| if(nbr_id_from_bl == 64) |
| nbr_id_from_bl = 32; |
| |
| if(nbr_id_from_bl == 32) |
| { |
| /* for top left : 1 pel per nbr bit */ |
| if(!((nbr_flags_temp >> 8) & 0x1)) |
| { |
| nbr_id_from_bl++; |
| nbr_id_from_bl += look_up_trailing_zeros((nbr_flags_temp >> 4) & 0xF) * 8; /* top and top right; 8 pels per nbr bit */ |
| //nbr_id_from_bl += idx * 8; |
| } |
| } |
| /* Reverse Substitution Process*/ |
| if(nbr_id_from_bl) |
| { |
| /* Replicate the bottom-left and subsequent unavailable pixels with the 1st available pixel above */ |
| pu1_ref = pu1_dst[nbr_id_from_bl]; |
| for(i = (nbr_id_from_bl - 1); i >= 0; i--) |
| { |
| pu1_dst[i] = pu1_ref; |
| } |
| } |
| } |
| |
| /* for the loop of 4*Nt+1 pixels (excluding pixels computed from reverse substitution) */ |
| while(nbr_id_from_bl < ((T16_4NT)+1)) |
| { |
| /* To Obtain the next unavailable idx flag after reverse neighbor substitution */ |
| /* Devide by 8 to obtain the original index */ |
| frwd_nbr_flag = (nbr_id_from_bl >> 3); /*+ (nbr_id_from_bl & 0x1);*/ |
| |
| /* The Top-left flag is at the last bit location of nbr_flags*/ |
| if(nbr_id_from_bl == (T16_4NT / 2)) |
| { |
| get_bits = GET_BITS(nbr_flags_temp, 8); |
| |
| /* only pel substitution for TL */ |
| if(!get_bits) |
| pu1_dst[nbr_id_from_bl] = pu1_dst[nbr_id_from_bl - 1]; |
| } |
| else |
| { |
| get_bits = GET_BITS(nbr_flags_temp, frwd_nbr_flag); |
| if(!get_bits) |
| { |
| /* 8 pel substitution (other than TL) */ |
| pu1_ref = pu1_dst[nbr_id_from_bl - 1]; |
| ihevc_memset_mul_8(pu1_dst + nbr_id_from_bl, pu1_ref, 8); |
| |
| |
| } |
| |
| } |
| nbr_id_from_bl += (nbr_id_from_bl == (T16_4NT / 2)) ? 1 : 8; |
| } |
| |
| |
| } |
| |
| if(nt == 32) |
| { |
| /* Else fill the corresponding samples */ |
| if(nbr_flags & 0x10000) |
| pu1_dst[two_nt] = *pu1_top_left; |
| else |
| pu1_dst[two_nt] = 0; |
| |
| if(nbr_flags & 0xF0) |
| { |
| for(i = 0; i < nt; i++) |
| pu1_dst[two_nt - 1 - i] = pu1_left[i * src_strd]; |
| } |
| else |
| { |
| ihevc_memset_mul_8(&pu1_dst[two_nt - 1 - (nt - 1)], 0, nt); |
| } |
| |
| /* Bottom - left availability is checked for every 8x8 TU position and set accordingly */ |
| { |
| if(nbr_flags & 0x8) |
| { |
| for(i = nt; i < (nt + 8); i++) |
| pu1_dst[two_nt - 1 - i] = pu1_left[i * src_strd]; |
| } |
| else |
| { |
| ihevc_memset_mul_8(&pu1_dst[24], 0, 8); |
| } |
| |
| if(nbr_flags & 0x4) |
| { |
| for(i = (nt + 8); i < (nt + 16); i++) |
| pu1_dst[two_nt - 1 - i] = pu1_left[i * src_strd]; |
| } |
| else |
| { |
| ihevc_memset_mul_8(&pu1_dst[16], 0, 8); |
| } |
| |
| if(nbr_flags & 0x2) |
| { |
| for(i = (nt + 16); i < (nt + 24); i++) |
| pu1_dst[two_nt - 1 - i] = pu1_left[i * src_strd]; |
| } |
| else |
| { |
| ihevc_memset_mul_8(&pu1_dst[8], 0, 8); |
| } |
| |
| if(nbr_flags & 0x1) |
| { |
| for(i = (nt + 24); i < (two_nt); i++) |
| pu1_dst[two_nt - 1 - i] = pu1_left[i * src_strd]; |
| } |
| else |
| { |
| ihevc_memset_mul_8(&pu1_dst[0], 0, 8); |
| } |
| } |
| |
| |
| if(nbr_flags & 0xF00) |
| { |
| ihevc_memcpy_mul_8(&pu1_dst[two_nt + 1], pu1_top, nt); |
| } |
| else |
| { |
| ihevc_memset_mul_8(&pu1_dst[two_nt + 1], 0, nt); |
| } |
| |
| if(nbr_flags & 0xF000) |
| { |
| ihevc_memcpy_mul_8(&pu1_dst[two_nt + 1 + nt], pu1_top + nt, nt); |
| } |
| else |
| { |
| ihevc_memset_mul_8(&pu1_dst[two_nt + 1 + nt], 0, nt); |
| } |
| /* compute trailing ones based on mbr_flag for substitution process of below left see section .*/ |
| /* as each bit in nbr flags corresponds to 8 pels for bot_left, left, top and topright but 1 pel for topleft */ |
| { |
| nbr_id_from_bl = look_up_trailing_zeros((nbr_flags & 0XFF)) * 8; /* for below left and left */ |
| |
| if(nbr_id_from_bl == 64) |
| { |
| /* for top left : 1 pel per nbr bit */ |
| if(!((nbr_flags >> 16) & 0x1)) |
| { |
| /* top left not available */ |
| nbr_id_from_bl++; |
| /* top and top right; 8 pels per nbr bit */ |
| nbr_id_from_bl += look_up_trailing_zeros((nbr_flags >> 8) & 0xFF) * 8; |
| } |
| } |
| /* Reverse Substitution Process*/ |
| if(nbr_id_from_bl) |
| { |
| /* Replicate the bottom-left and subsequent unavailable pixels with the 1st available pixel above */ |
| pu1_ref = pu1_dst[nbr_id_from_bl]; |
| for(i = (nbr_id_from_bl - 1); i >= 0; i--) |
| pu1_dst[i] = pu1_ref; |
| } |
| } |
| |
| /* for the loop of 4*Nt+1 pixels (excluding pixels computed from reverse substitution) */ |
| while(nbr_id_from_bl < ((T32_4NT)+1)) |
| { |
| /* To Obtain the next unavailable idx flag after reverse neighbor substitution */ |
| /* Devide by 8 to obtain the original index */ |
| frwd_nbr_flag = (nbr_id_from_bl >> 3); /*+ (nbr_id_from_bl & 0x1);*/ |
| |
| /* The Top-left flag is at the last bit location of nbr_flags*/ |
| if(nbr_id_from_bl == (T32_4NT / 2)) |
| { |
| get_bits = GET_BITS(nbr_flags, 16); |
| /* only pel substitution for TL */ |
| if(!get_bits) |
| pu1_dst[nbr_id_from_bl] = pu1_dst[nbr_id_from_bl - 1]; |
| } |
| else |
| { |
| get_bits = GET_BITS(nbr_flags, frwd_nbr_flag); |
| if(!get_bits) |
| { |
| /* 8 pel substitution (other than TL) */ |
| pu1_ref = pu1_dst[nbr_id_from_bl - 1]; |
| ihevc_memset_mul_8(&pu1_dst[nbr_id_from_bl], pu1_ref, 8); |
| |
| } |
| |
| } |
| nbr_id_from_bl += (nbr_id_from_bl == (T32_4NT / 2)) ? 1 : 8; |
| } |
| } |
| |
| } |
| } |
| |
| |
| /** |
| ******************************************************************************* |
| * |
| * @brief |
| * Intra prediction interpolation filter for ref_filtering |
| * |
| * |
| * @par Description: |
| * Reference DC filtering for neighboring samples dependent on TU size and |
| * mode Refer to section 8.4.4.2.3 in the standard |
| * |
| * @param[in] pu1_src |
| * UWORD8 pointer to the source |
| * |
| * @param[out] pu1_dst |
| * UWORD8 pointer to the destination |
| * |
| * @param[in] nt |
| * integer Transform Block size |
| * |
| * @param[in] mode |
| * integer intraprediction mode |
| * |
| * @returns |
| * |
| * @remarks |
| * None |
| * |
| ******************************************************************************* |
| */ |
| |
| |
| void ihevc_intra_pred_ref_filtering(UWORD8 *pu1_src, |
| WORD32 nt, |
| UWORD8 *pu1_dst, |
| WORD32 mode, |
| WORD32 strong_intra_smoothing_enable_flag) |
| { |
| WORD32 filter_flag; |
| WORD32 i; /* Generic indexing variable */ |
| WORD32 four_nt = 4 * nt; |
| UWORD8 au1_flt[(4 * MAX_CU_SIZE) + 1]; |
| WORD32 bi_linear_int_flag = 0; |
| WORD32 abs_cond_left_flag = 0; |
| WORD32 abs_cond_top_flag = 0; |
| /*WORD32 dc_val = 1 << (BIT_DEPTH - 5);*/ |
| WORD32 dc_val = 1 << (8 - 5); |
| //WORD32 strong_intra_smoothing_enable_flag = 1; |
| |
| filter_flag = gau1_intra_pred_ref_filter[mode] & (1 << (CTZ(nt) - 2)); |
| if(0 == filter_flag) |
| { |
| if(pu1_src == pu1_dst) |
| { |
| return; |
| } |
| else |
| { |
| for(i = 0; i < (four_nt + 1); i++) |
| pu1_dst[i] = pu1_src[i]; |
| } |
| } |
| |
| else |
| { |
| /* If strong intra smoothin is enabled and transform size is 32 */ |
| if((1 == strong_intra_smoothing_enable_flag) && (32 == nt)) |
| { |
| /* Strong Intra Filtering */ |
| abs_cond_top_flag = (ABS(pu1_src[2 * nt] + pu1_src[4 * nt] |
| - (2 * pu1_src[3 * nt]))) < dc_val; |
| abs_cond_left_flag = (ABS(pu1_src[2 * nt] + pu1_src[0] |
| - (2 * pu1_src[nt]))) < dc_val; |
| |
| bi_linear_int_flag = ((1 == abs_cond_left_flag) |
| && (1 == abs_cond_top_flag)); |
| } |
| /* Extremities Untouched*/ |
| au1_flt[0] = pu1_src[0]; |
| au1_flt[4 * nt] = pu1_src[4 * nt]; |
| |
| /* Strong filtering of reference samples */ |
| if(1 == bi_linear_int_flag) |
| { |
| au1_flt[2 * nt] = pu1_src[2 * nt]; |
| |
| for(i = 1; i < (2 * nt); i++) |
| au1_flt[i] = (((2 * nt) - i) * pu1_src[0] + i * pu1_src[2 * nt] + 32) >> 6; |
| |
| for(i = 1; i < (2 * nt); i++) |
| au1_flt[i + (2 * nt)] = (((2 * nt) - i) * pu1_src[2 * nt] + i * pu1_src[4 * nt] + 32) >> 6; |
| |
| } |
| else |
| { |
| /* Perform bilinear filtering of Reference Samples */ |
| for(i = 0; i < (four_nt - 1); i++) |
| { |
| au1_flt[i + 1] = (pu1_src[i] + 2 * pu1_src[i + 1] |
| + pu1_src[i + 2] + 2) >> 2; |
| } |
| } |
| |
| |
| for(i = 0; i < (four_nt + 1); i++) |
| pu1_dst[i] = au1_flt[i]; |
| } |
| |
| } |
| |
| |
| /** |
| ******************************************************************************* |
| * |
| * @brief |
| * Intra prediction interpolation filter for luma planar |
| * |
| * @par Description: |
| * Planar Intraprediction with reference neighboring samples location |
| * pointed by 'pu1_ref' to the TU block location pointed by 'pu1_dst' Refer |
| * to section 8.4.4.2.4 in the standard |
| * |
| * @param[in] pu1_src |
| * UWORD8 pointer to the source |
| * |
| * @param[out] pu1_dst |
| * UWORD8 pointer to the destination |
| * |
| * @param[in] src_strd |
| * integer source stride |
| * |
| * @param[in] dst_strd |
| * integer destination stride |
| * |
| * @param[in] nt |
| * integer Transform Block size |
| * |
| * @param[in] mode |
| * integer intraprediction mode |
| * |
| * @returns |
| * |
| * @remarks |
| * None |
| * |
| ******************************************************************************* |
| */ |
| |
| |
| void ihevc_intra_pred_luma_planar(UWORD8 *pu1_ref, |
| WORD32 src_strd, |
| UWORD8 *pu1_dst, |
| WORD32 dst_strd, |
| WORD32 nt, |
| WORD32 mode) |
| { |
| |
| |
| WORD32 row, col; |
| WORD32 log2nt = 5; |
| WORD32 two_nt, three_nt; |
| UNUSED(src_strd); |
| UNUSED(mode); |
| switch(nt) |
| { |
| case 32: |
| log2nt = 5; |
| break; |
| case 16: |
| log2nt = 4; |
| break; |
| case 8: |
| log2nt = 3; |
| break; |
| case 4: |
| log2nt = 2; |
| break; |
| default: |
| break; |
| } |
| two_nt = 2 * nt; |
| three_nt = 3 * nt; |
| /* Planar filtering */ |
| for(row = 0; row < nt; row++) |
| { |
| for(col = 0; col < nt; col++) |
| { |
| pu1_dst[row * dst_strd + col] = ((nt - 1 - col) |
| * pu1_ref[two_nt - 1 - row] |
| + (col + 1) * pu1_ref[three_nt + 1] |
| + (nt - 1 - row) * pu1_ref[two_nt + 1 + col] |
| + (row + 1) * pu1_ref[nt - 1] + nt) >> (log2nt + 1); |
| } |
| } |
| } |
| |
| |
| /** |
| ******************************************************************************* |
| * |
| * @brief |
| * Intra prediction interpolation filter for luma dc |
| * |
| * @par Description: |
| * Intraprediction for DC mode with reference neighboring samples location |
| * pointed by 'pu1_ref' to the TU block location pointed by 'pu1_dst' Refer |
| * to section 8.4.4.2.5 in the standard |
| * |
| * @param[in] pu1_src |
| * UWORD8 pointer to the source |
| * |
| * @param[out] pu1_dst |
| * UWORD8 pointer to the destination |
| * |
| * @param[in] src_strd |
| * integer source stride |
| * |
| * @param[in] dst_strd |
| * integer destination stride |
| * |
| * @param[in] nt |
| * integer Transform Block size |
| * |
| * @param[in] mode |
| * integer intraprediction mode |
| * |
| * @returns |
| * |
| * @remarks |
| * None |
| * |
| ******************************************************************************* |
| */ |
| |
| |
| void ihevc_intra_pred_luma_dc(UWORD8 *pu1_ref, |
| WORD32 src_strd, |
| UWORD8 *pu1_dst, |
| WORD32 dst_strd, |
| WORD32 nt, |
| WORD32 mode) |
| { |
| |
| WORD32 acc_dc; |
| WORD32 dc_val, two_dc_val, three_dc_val; |
| WORD32 i; |
| WORD32 row, col; |
| WORD32 log2nt = 5; |
| WORD32 two_nt, three_nt; |
| UNUSED(mode); |
| UNUSED(src_strd); |
| switch(nt) |
| { |
| case 32: |
| log2nt = 5; |
| break; |
| case 16: |
| log2nt = 4; |
| break; |
| case 8: |
| log2nt = 3; |
| break; |
| case 4: |
| log2nt = 2; |
| break; |
| default: |
| break; |
| } |
| two_nt = 2 * nt; |
| three_nt = 3 * nt; |
| |
| acc_dc = 0; |
| /* Calculate DC value for the transform block */ |
| for(i = nt; i < two_nt; i++) |
| acc_dc += pu1_ref[i]; |
| |
| for(i = (two_nt + 1); i <= three_nt; i++) |
| acc_dc += pu1_ref[i]; |
| |
| dc_val = (acc_dc + nt) >> (log2nt + 1); |
| |
| two_dc_val = 2 * dc_val; |
| three_dc_val = 3 * dc_val; |
| |
| |
| if(nt == 32) |
| { |
| for(row = 0; row < nt; row++) |
| for(col = 0; col < nt; col++) |
| pu1_dst[(row * dst_strd) + col] = dc_val; |
| } |
| else |
| { |
| /* DC filtering for the first top row and first left column */ |
| pu1_dst[0] = ((pu1_ref[two_nt - 1] + two_dc_val + pu1_ref[two_nt + 1] + 2) |
| >> 2); |
| |
| for(col = 1; col < nt; col++) |
| pu1_dst[col] = (pu1_ref[two_nt + 1 + col] + three_dc_val + 2) >> 2; |
| |
| for(row = 1; row < nt; row++) |
| pu1_dst[row * dst_strd] = (pu1_ref[two_nt - 1 - row] + three_dc_val + 2) |
| >> 2; |
| |
| /* Fill the remaining rows with DC value*/ |
| for(row = 1; row < nt; row++) |
| for(col = 1; col < nt; col++) |
| pu1_dst[(row * dst_strd) + col] = dc_val; |
| } |
| } |
| |
| |
| |
| /** |
| ******************************************************************************* |
| * |
| * @brief |
| * Intra prediction interpolation filter for horizontal luma variable. |
| * |
| * @par Description: |
| * Horizontal intraprediction(mode 10) with reference samples location |
| * pointed by 'pu1_ref' to the TU block location pointed by 'pu1_dst' Refer |
| * to section 8.4.4.2.6 in the standard (Special case) |
| * |
| * @param[in] pu1_src |
| * UWORD8 pointer to the source |
| * |
| * @param[out] pu1_dst |
| * UWORD8 pointer to the destination |
| * |
| * @param[in] src_strd |
| * integer source stride |
| * |
| * @param[in] dst_strd |
| * integer destination stride |
| * |
| * @param[in] nt |
| * integer Transform Block size |
| * |
| * @param[in] mode |
| * integer intraprediction mode |
| * |
| * @returns |
| * |
| * @remarks |
| * None |
| * |
| ******************************************************************************* |
| */ |
| |
| |
| void ihevc_intra_pred_luma_horz(UWORD8 *pu1_ref, |
| WORD32 src_strd, |
| UWORD8 *pu1_dst, |
| WORD32 dst_strd, |
| WORD32 nt, |
| WORD32 mode) |
| { |
| |
| WORD32 row, col; |
| WORD32 two_nt; |
| WORD16 s2_predpixel; |
| UNUSED(mode); |
| UNUSED(src_strd); |
| two_nt = 2 * nt; |
| |
| if(nt == 32) |
| { |
| for(row = 0; row < nt; row++) |
| for(col = 0; col < nt; col++) |
| pu1_dst[(row * dst_strd) + col] = pu1_ref[two_nt - 1 - row]; |
| } |
| else |
| { |
| /*Filtering done for the 1st row */ |
| for(col = 0; col < nt; col++) |
| { |
| s2_predpixel = pu1_ref[two_nt - 1] |
| + ((pu1_ref[two_nt + 1 + col] - pu1_ref[two_nt]) >> 1); |
| pu1_dst[col] = CLIP_U8(s2_predpixel); |
| } |
| |
| /* Replication to next rows*/ |
| for(row = 1; row < nt; row++) |
| for(col = 0; col < nt; col++) |
| pu1_dst[(row * dst_strd) + col] = pu1_ref[two_nt - 1 - row]; |
| } |
| } |
| |
| |
| |
| |
| |
| /** |
| ******************************************************************************* |
| * |
| * @brief |
| * Intra prediction interpolation filter for vertical luma variable. |
| * |
| * @par Description: |
| * Horizontal intraprediction with reference neighboring samples location |
| * pointed by 'pu1_ref' to the TU block location pointed by 'pu1_dst' Refer |
| * to section 8.4.4.2.6 in the standard (Special case) |
| * |
| * @param[in] pu1_src |
| * UWORD8 pointer to the source |
| * |
| * @param[out] pu1_dst |
| * UWORD8 pointer to the destination |
| * |
| * @param[in] src_strd |
| * integer source stride |
| * |
| * @param[in] dst_strd |
| * integer destination stride |
| * |
| * @param[in] nt |
| * integer Transform Block size |
| * |
| * @param[in] mode |
| * integer intraprediction mode |
| * |
| * @returns |
| * |
| * @remarks |
| * None |
| * |
| ******************************************************************************* |
| */ |
| |
| |
| void ihevc_intra_pred_luma_ver(UWORD8 *pu1_ref, |
| WORD32 src_strd, |
| UWORD8 *pu1_dst, |
| WORD32 dst_strd, |
| WORD32 nt, |
| WORD32 mode) |
| { |
| WORD32 row, col; |
| WORD16 s2_predpixel; |
| WORD32 two_nt = 2 * nt; |
| UNUSED(mode); |
| UNUSED(src_strd); |
| |
| if(nt == 32) |
| { |
| /* Replication to next columns*/ |
| for(row = 0; row < nt; row++) |
| for(col = 0; col < nt; col++) |
| pu1_dst[(row * dst_strd) + col] = pu1_ref[two_nt + 1 + col]; |
| } |
| else |
| { |
| /*Filtering done for the 1st column */ |
| for(row = 0; row < nt; row++) |
| { |
| s2_predpixel = pu1_ref[two_nt + 1] |
| + ((pu1_ref[two_nt - 1 - row] - pu1_ref[two_nt]) >> 1); |
| pu1_dst[row * dst_strd] = CLIP_U8(s2_predpixel); |
| } |
| |
| /* Replication to next columns*/ |
| for(row = 0; row < nt; row++) |
| for(col = 1; col < nt; col++) |
| pu1_dst[(row * dst_strd) + col] = pu1_ref[two_nt + 1 + col]; |
| } |
| } |
| |
| |
| |
| |
| /** |
| ******************************************************************************* |
| * |
| * @brief |
| * Intra prediction interpolation filter for luma mode2. |
| * |
| * @par Description: |
| * Intraprediction for mode 2 (sw angle) with reference neighboring samples |
| * location pointed by 'pu1_ref' to the TU block location pointed by |
| * 'pu1_dst' Refer to section 8.4.4.2.6 in the standard |
| * |
| * @param[in] pu1_src |
| * UWORD8 pointer to the source |
| * |
| * @param[out] pu1_dst |
| * UWORD8 pointer to the destination |
| * |
| * @param[in] src_strd |
| * integer source stride |
| * |
| * @param[in] dst_strd |
| * integer destination stride |
| * |
| * @param[in] nt |
| * integer Transform Block size |
| * |
| * @param[in] mode |
| * integer intraprediction mode |
| * |
| * @returns |
| * |
| * @remarks |
| * None |
| * |
| ******************************************************************************* |
| */ |
| |
| |
| void ihevc_intra_pred_luma_mode2(UWORD8 *pu1_ref, |
| WORD32 src_strd, |
| UWORD8 *pu1_dst, |
| WORD32 dst_strd, |
| WORD32 nt, |
| WORD32 mode) |
| { |
| WORD32 row, col; |
| WORD32 two_nt = 2 * nt; |
| WORD32 intra_pred_ang = 32; |
| WORD32 idx = 0; |
| UNUSED(mode); |
| UNUSED(src_strd); |
| /* For the angle 45, replication is done from the corresponding angle */ |
| /* intra_pred_ang = tan(angle) in q5 format */ |
| for(col = 0; col < nt; col++) |
| { |
| idx = ((col + 1) * intra_pred_ang) >> 5; /* Use idx++ */ |
| |
| for(row = 0; row < nt; row++) |
| pu1_dst[col + (row * dst_strd)] = pu1_ref[two_nt - row - idx - 1]; |
| } |
| |
| } |
| |
| |
| /** |
| ******************************************************************************* |
| * |
| * @brief |
| * Intra prediction interpolation filter for luma mode 18 & mode 34. |
| * |
| * @par Description: |
| * Intraprediction for mode 34 (ne angle) and mode 18 (nw angle) with |
| * reference neighboring samples location pointed by 'pu1_ref' to the TU |
| * block location pointed by 'pu1_dst' |
| * |
| * @param[in] pu1_src |
| * UWORD8 pointer to the source |
| * |
| * @param[out] pu1_dst |
| * UWORD8 pointer to the destination |
| * |
| * @param[in] src_strd |
| * integer source stride |
| * |
| * @param[in] dst_strd |
| * integer destination stride |
| * |
| * @param[in] nt |
| * integer Transform Block size |
| * |
| * @param[in] mode |
| * integer intraprediction mode |
| * |
| * @returns |
| * |
| * @remarks |
| * None |
| * |
| ******************************************************************************* |
| */ |
| |
| |
| void ihevc_intra_pred_luma_mode_18_34(UWORD8 *pu1_ref, |
| WORD32 src_strd, |
| UWORD8 *pu1_dst, |
| WORD32 dst_strd, |
| WORD32 nt, |
| WORD32 mode) |
| { |
| WORD32 row, col; |
| WORD32 intra_pred_ang; |
| WORD32 idx = 0; |
| WORD32 two_nt = 2 * nt; |
| UNUSED(src_strd); |
| intra_pred_ang = 32; /*Default value*/ |
| |
| /* For mode 18, angle is -45degree */ |
| if(mode == 18) |
| intra_pred_ang = -32; |
| /* For mode 34, angle is 45degree */ |
| else if(mode == 34) |
| intra_pred_ang = 32; |
| /* For the angle 45 and -45, replication is done from the corresponding angle */ |
| /* No interpolation is done for 45 degree*/ |
| for(row = 0; row < nt; row++) |
| { |
| idx = ((row + 1) * intra_pred_ang) >> 5; |
| #if OPT |
| if(mode == 18) |
| idx--; |
| if(mode == 34) |
| idx++; |
| #endif |
| for(col = 0; col < nt; col++) |
| pu1_dst[col + (row * dst_strd)] = pu1_ref[two_nt + col + idx + 1]; |
| |
| } |
| |
| } |
| |
| |
| /** |
| ******************************************************************************* |
| * |
| * @brief |
| * Intra prediction interpolation filter for luma mode 3 to mode 9 |
| * |
| * @par Description: |
| * Intraprediction for mode 3 to 9 (positive angle, horizontal mode ) with |
| * reference neighboring samples location pointed by 'pu1_ref' to the TU |
| * block location pointed by 'pu1_dst' |
| * |
| * @param[in] pu1_src |
| * UWORD8 pointer to the source |
| * |
| * @param[out] pu1_dst |
| * UWORD8 pointer to the destination |
| * |
| * @param[in] src_strd |
| * integer source stride |
| * |
| * @param[in] dst_strd |
| * integer destination stride |
| * |
| * @param[in] nt |
| * integer Transform Block size |
| * |
| * @param[in] mode |
| * integer intraprediction mode |
| * |
| * @returns |
| * |
| * @remarks |
| * None |
| * |
| ******************************************************************************* |
| */ |
| |
| |
| void ihevc_intra_pred_luma_mode_3_to_9(UWORD8 *pu1_ref, |
| WORD32 src_strd, |
| UWORD8 *pu1_dst, |
| WORD32 dst_strd, |
| WORD32 nt, |
| WORD32 mode) |
| { |
| WORD32 row, col; |
| WORD32 two_nt = 2 * nt; |
| WORD32 intra_pred_ang; |
| WORD32 idx, ref_main_idx; |
| WORD32 pos, fract; |
| UNUSED(src_strd); |
| /* Intra Pred Angle according to the mode */ |
| intra_pred_ang = gai4_ihevc_ang_table[mode]; |
| |
| /* For the angles other then 45 degree, interpolation btw 2 neighboring */ |
| /* samples dependent on distance to obtain destination sample */ |
| |
| for(col = 0; col < nt; col++) |
| { |
| pos = ((col + 1) * intra_pred_ang); |
| idx = pos >> 5; |
| fract = pos & (31); |
| |
| // Do linear filtering |
| for(row = 0; row < nt; row++) |
| { |
| ref_main_idx = two_nt - row - idx - 1; |
| pu1_dst[col + (row * dst_strd)] = (((32 - fract) |
| * pu1_ref[ref_main_idx] |
| + fract * pu1_ref[ref_main_idx - 1] + 16) >> 5); |
| } |
| |
| } |
| |
| } |
| |
| |
| /** |
| ******************************************************************************* |
| * |
| * @brief |
| * Intra prediction interpolation filter for luma mode 11 to mode 17 |
| * |
| * @par Description: |
| * Intraprediction for mode 11 to 17 (negative angle, horizontal mode ) |
| * with reference neighboring samples location pointed by 'pu1_ref' to the |
| * TU block location pointed by 'pu1_dst' |
| * |
| * @param[in] pu1_src |
| * UWORD8 pointer to the source |
| * |
| * @param[out] pu1_dst |
| * UWORD8 pointer to the destination |
| * |
| * @param[in] src_strd |
| * integer source stride |
| * |
| * @param[in] dst_strd |
| * integer destination stride |
| * |
| * @param[in] nt |
| * integer Transform Block size |
| * |
| * @param[in] mode |
| * integer intraprediction mode |
| * |
| * @returns |
| * |
| * @remarks |
| * None |
| * |
| ******************************************************************************* |
| */ |
| |
| |
| void ihevc_intra_pred_luma_mode_11_to_17(UWORD8 *pu1_ref, |
| WORD32 src_strd, |
| UWORD8 *pu1_dst, |
| WORD32 dst_strd, |
| WORD32 nt, |
| WORD32 mode) |
| { |
| /* This function and ihevc_intra_pred_luma_mode_19_to_25 are same except*/ |
| /* for ref main & side samples assignment,can be combined for */ |
| /* optimzation*/ |
| |
| WORD32 row, col, k; |
| WORD32 two_nt; |
| WORD32 intra_pred_ang, inv_ang, inv_ang_sum; |
| WORD32 idx, ref_main_idx, ref_idx; |
| WORD32 pos, fract; |
| |
| UWORD8 ref_temp[2 * MAX_CU_SIZE + 1]; |
| UWORD8 *ref_main; |
| UNUSED(src_strd); |
| inv_ang_sum = 128; |
| two_nt = 2 * nt; |
| |
| intra_pred_ang = gai4_ihevc_ang_table[mode]; |
| |
| inv_ang = gai4_ihevc_inv_ang_table[mode - 11]; |
| /* Intermediate reference samples for negative angle modes */ |
| /* This have to be removed during optimization*/ |
| /* For horizontal modes, (ref main = ref left) (ref side = ref above) */ |
| |
| ref_main = ref_temp + nt - 1; |
| for(k = 0; k < nt + 1; k++) |
| ref_temp[k + nt - 1] = pu1_ref[two_nt - k]; |
| |
| ref_main = ref_temp + nt - 1; |
| ref_idx = (nt * intra_pred_ang) >> 5; |
| |
| /* SIMD Optimization can be done using look-up table for the loop */ |
| /* For negative angled derive the main reference samples from side */ |
| /* reference samples refer to section 8.4.4.2.6 */ |
| for(k = -1; k > ref_idx; k--) |
| { |
| inv_ang_sum += inv_ang; |
| ref_main[k] = pu1_ref[two_nt + (inv_ang_sum >> 8)]; |
| } |
| |
| /* For the angles other then 45 degree, interpolation btw 2 neighboring */ |
| /* samples dependent on distance to obtain destination sample */ |
| for(col = 0; col < nt; col++) |
| { |
| pos = ((col + 1) * intra_pred_ang); |
| idx = pos >> 5; |
| fract = pos & (31); |
| |
| // Do linear filtering |
| for(row = 0; row < nt; row++) |
| { |
| ref_main_idx = row + idx + 1; |
| pu1_dst[col + (dst_strd * row)] = (UWORD8)(((32 - fract) |
| * ref_main[ref_main_idx] |
| + fract * ref_main[ref_main_idx + 1] + 16) >> 5); |
| |
| } |
| |
| } |
| |
| } |
| |
| |
| |
| /** |
| ******************************************************************************* |
| * |
| * @brief |
| * Intra prediction interpolation filter for luma mode 19 to mode 25 |
| * |
| * @par Description: |
| * Intraprediction for mode 19 to 25 (negative angle, vertical mode ) with |
| * reference neighboring samples location pointed by 'pu1_ref' to the TU |
| * block location pointed by 'pu1_dst' |
| * |
| * @param[in] pu1_src |
| * UWORD8 pointer to the source |
| * |
| * @param[out] pu1_dst |
| * UWORD8 pointer to the destination |
| * |
| * @param[in] src_strd |
| * integer source stride |
| * |
| * @param[in] dst_strd |
| * integer destination stride |
| * |
| * @param[in] nt |
| * integer Transform Block size |
| * |
| * @param[in] mode |
| * integer intraprediction mode |
| * |
| * @returns |
| * |
| * @remarks |
| * None |
| * |
| ******************************************************************************* |
| */ |
| |
| |
| void ihevc_intra_pred_luma_mode_19_to_25(UWORD8 *pu1_ref, |
| WORD32 src_strd, |
| UWORD8 *pu1_dst, |
| WORD32 dst_strd, |
| WORD32 nt, |
| WORD32 mode) |
| { |
| |
| WORD32 row, col, k; |
| WORD32 two_nt, intra_pred_ang, idx; |
| WORD32 inv_ang, inv_ang_sum, pos, fract; |
| WORD32 ref_main_idx, ref_idx; |
| UWORD8 ref_temp[(2 * MAX_CU_SIZE) + 1]; |
| UWORD8 *ref_main; |
| UNUSED(src_strd); |
| two_nt = 2 * nt; |
| intra_pred_ang = gai4_ihevc_ang_table[mode]; |
| inv_ang = gai4_ihevc_inv_ang_table[mode - 12]; |
| |
| /* Intermediate reference samples for negative angle modes */ |
| /* This have to be removed during optimization*/ |
| /* For horizontal modes, (ref main = ref above) (ref side = ref left) */ |
| ref_main = ref_temp + nt - 1; |
| for(k = 0; k < (nt + 1); k++) |
| ref_temp[k + nt - 1] = pu1_ref[two_nt + k]; |
| |
| ref_idx = (nt * intra_pred_ang) >> 5; |
| inv_ang_sum = 128; |
| |
| /* SIMD Optimization can be done using look-up table for the loop */ |
| /* For negative angled derive the main reference samples from side */ |
| /* reference samples refer to section 8.4.4.2.6 */ |
| for(k = -1; k > ref_idx; k--) |
| { |
| inv_ang_sum += inv_ang; |
| ref_main[k] = pu1_ref[two_nt - (inv_ang_sum >> 8)]; |
| } |
| |
| for(row = 0; row < nt; row++) |
| { |
| pos = ((row + 1) * intra_pred_ang); |
| idx = pos >> 5; |
| fract = pos & (31); |
| |
| // Do linear filtering |
| for(col = 0; col < nt; col++) |
| { |
| ref_main_idx = col + idx + 1; |
| pu1_dst[(row * dst_strd) + col] = (UWORD8)(((32 - fract) |
| * ref_main[ref_main_idx] |
| + fract * ref_main[ref_main_idx + 1] + 16) >> 5); |
| |
| } |
| |
| } |
| |
| } |
| |
| |
| |
| /** |
| ******************************************************************************* |
| * |
| * @brief |
| * Intra prediction interpolation filter for luma mode 27 to mode 33 |
| * |
| * @par Description: |
| * Intraprediction for mode 27 to 33 (positive angle, vertical mode ) with |
| * reference neighboring samples location pointed by 'pu1_ref' to the TU |
| * block location pointed by 'pu1_dst' |
| * |
| * @param[in] pu1_src |
| * UWORD8 pointer to the source |
| * |
| * @param[out] pu1_dst |
| * UWORD8 pointer to the destination |
| * |
| * @param[in] src_strd |
| * integer source stride |
| * |
| * @param[in] dst_strd |
| * integer destination stride |
| * |
| * @param[in] nt |
| * integer Transform Block size |
| * |
| * @param[in] mode |
| * integer intraprediction mode |
| * |
| * @returns |
| * |
| * @remarks |
| * None |
| * |
| ******************************************************************************* |
| */ |
| |
| |
| void ihevc_intra_pred_luma_mode_27_to_33(UWORD8 *pu1_ref, |
| WORD32 src_strd, |
| UWORD8 *pu1_dst, |
| WORD32 dst_strd, |
| WORD32 nt, |
| WORD32 mode) |
| { |
| WORD32 row, col; |
| WORD32 two_nt, pos, fract; |
| WORD32 intra_pred_ang; |
| WORD32 idx, ref_main_idx; |
| UNUSED(src_strd); |
| two_nt = 2 * nt; |
| intra_pred_ang = gai4_ihevc_ang_table[mode]; |
| |
| for(row = 0; row < nt; row++) |
| { |
| pos = ((row + 1) * intra_pred_ang); |
| idx = pos >> 5; |
| fract = pos & (31); |
| |
| // Do linear filtering |
| for(col = 0; col < nt; col++) |
| { |
| ref_main_idx = two_nt + col + idx + 1; |
| pu1_dst[col + (row * dst_strd)] = (((32 - fract) |
| * pu1_ref[ref_main_idx] |
| + fract * pu1_ref[ref_main_idx + 1] + 16) >> 5); |
| } |
| |
| } |
| |
| } |
| |