| /* |
| * Copyright (c) 2022, Alliance for Open Media. All rights reserved |
| * |
| * This source code is subject to the terms of the BSD 3-Clause Clear License |
| * and the Alliance for Open Media Patent License 1.0. If the BSD 3-Clause Clear |
| * License was not distributed with this source code in the LICENSE file, you |
| * can obtain it at aomedia.org/license/software-license/bsd-3-c-c/. If the |
| * Alliance for Open Media Patent License 1.0 was not distributed with this |
| * source code in the PATENTS file, you can obtain it at |
| * aomedia.org/license/patent-license/. |
| */ |
| #ifndef AOM_AV1_COMMON_BRU_H_ |
| #define AOM_AV1_COMMON_BRU_H_ |
| #include "av1/common/av1_common_int.h" |
| #include "av1/common/pred_common.h" |
| #include "av1/common/blockd.h" |
| // Encoder only macros for BRU |
| #ifndef BRU_OFF_RATIO |
| #define BRU_OFF_RATIO 50 |
| #endif |
| #ifndef MAX_ACTIVE_REGION |
| #define MAX_ACTIVE_REGION 8 |
| #endif |
| // how far can BRU ref been picked |
| #ifndef BRU_ENC_LOOKAHEAD_DIST_MINUS_1 |
| #define BRU_ENC_LOOKAHEAD_DIST_MINUS_1 1 |
| #endif |
| #define BRU_ENC_REF_DELAY 1 |
| // Encoder will use cur order_hint - (BRU_ENC_LOOKAHEAD_DIST_MINUS_1 + 1) as BRU |
| // ref frame But it will wait BRU_ENC_REF_DELAY frame to start: e.g. |
| // BRU_ENC_REF_DELAY = 1 and BRU_ENC_LOOKAHEAD_DIST_MINUS_1 = 1 means first |
| // possible BRU frame is POC3 which is using POC1 as BRU ref. e.g. |
| // BRU_ENC_REF_DELAY = 0 and BRU_ENC_LOOKAHEAD_DIST_MINUS_1 = 2 means first |
| // possible BRU frame is POC3 which is using POC0 as BRU ref. |
| |
| /* This function test the reference frame used for inter prediction. |
| BRU conformance requires any inter prediction should not use any pixels in |
| BRU reference frame. |
| */ |
| static AOM_INLINE int bru_is_valid_inter(const AV1_COMMON *const cm, |
| MACROBLOCKD *const xd) { |
| // None-BRU frame does not need to check BRU inter |
| if (!cm->bru.enabled) return 1; |
| const MB_MODE_INFO *const mbmi = xd->mi[0]; |
| const BruActiveMode active_mode = xd->mi[0]->sb_active_mode; |
| const int tip_ref_frame = is_tip_ref_frame(mbmi->ref_frame[0]); |
| const int is_compound = has_second_ref(mbmi); |
| if (active_mode != BRU_ACTIVE_SB) { |
| if (tip_ref_frame || is_compound) return 0; |
| if (mbmi->ref_frame[0] != cm->bru.update_ref_idx) return 0; |
| const int_mv mi_mv = mbmi->mv[0]; |
| // MV must be (0,0) |
| if (mi_mv.as_int != 0) { |
| return 0; |
| } |
| } else { |
| if (tip_ref_frame) { |
| if ((cm->tip_ref.ref_frame[0] == cm->bru.update_ref_idx) || |
| (cm->tip_ref.ref_frame[1] == cm->bru.update_ref_idx)) |
| return 0; |
| } |
| for (int ref = 0; ref < 1 + is_compound; ++ref) { |
| if (mbmi->ref_frame[ref] == cm->bru.update_ref_idx) { |
| // if any ref is BRU ref, it is illegal |
| return 0; |
| } |
| } |
| } |
| return 1; |
| } |
| |
| /* Dynamic allocate active map and active region structure */ |
| static INLINE void realloc_bru_info(AV1_COMMON *cm) { |
| BruInfo *bru_info = &cm->bru; |
| uint32_t unit_rows = |
| (cm->mi_params.mi_rows + cm->mib_size - 1) / cm->mib_size; |
| uint32_t unit_cols = |
| (cm->mi_params.mi_cols + cm->mib_size - 1) / cm->mib_size; |
| if (unit_rows != bru_info->unit_rows || unit_cols != bru_info->unit_cols || |
| bru_info->unit_mi_size_log2 != (uint32_t)cm->mib_size_log2) { |
| bru_info->unit_rows = unit_rows; |
| bru_info->unit_cols = unit_cols; |
| bru_info->unit_mi_size_log2 = cm->mib_size_log2; |
| bru_info->total_units = bru_info->unit_rows * bru_info->unit_cols; |
| aom_free(bru_info->active_mode_map); |
| CHECK_MEM_ERROR(cm, bru_info->active_mode_map, |
| (uint8_t *)aom_calloc(bru_info->total_units, 1)); |
| bru_info->num_active_regions = 0; |
| aom_free(bru_info->active_region); |
| CHECK_MEM_ERROR( |
| cm, bru_info->active_region, |
| (AV1PixelRect *)aom_calloc( |
| (bru_info->unit_cols / 3 + 1) * (bru_info->unit_rows / 3 + 1), |
| sizeof(AV1PixelRect))); |
| |
| aom_free(bru_info->active_sb_in_region); |
| CHECK_MEM_ERROR(cm, bru_info->active_sb_in_region, |
| (uint32_t *)aom_calloc((bru_info->unit_cols / 3 + 1) * |
| (bru_info->unit_rows / 3 + 1), |
| sizeof(uint32_t))); |
| |
| aom_free(bru_info->ref_scores); |
| CHECK_MEM_ERROR( |
| cm, bru_info->ref_scores, |
| (RefScoreData *)aom_calloc(REF_FRAMES, sizeof(RefScoreData))); |
| } |
| return; |
| } |
| /* Free active map and active region structure */ |
| static INLINE void free_bru_info(AV1_COMMON const *cm) { |
| aom_free(cm->bru.active_mode_map); |
| aom_free(cm->bru.active_region); |
| aom_free(cm->bru.active_sb_in_region); |
| aom_free(cm->bru.ref_scores); |
| return; |
| } |
| /* Check if current mi is the start mi of the super block*/ |
| static INLINE int is_sb_start_mi(const AV1_COMMON *cm, const int mi_col, |
| const int mi_row) { |
| const int sb_mask = (cm->seq_params.mib_size - 1); |
| // Check if current block is SB start MI |
| if ((mi_row & sb_mask) == 0 && (mi_col & sb_mask) == 0) return 1; |
| return 0; |
| } |
| |
| /* determine region SB activity using mbmi, if any SB is ACTIVE, return false */ |
| static INLINE int bru_is_fu_skipped_mbmi(const AV1_COMMON *cm, const int mi_col, |
| const int mi_row, const int mi_width, |
| const int mi_height) { |
| if (!cm->bru.enabled) return 0; |
| if (mi_col < 0 || mi_row < 0) return 0; |
| if (mi_width <= 0 || mi_height <= 0) return 0; |
| const CommonModeInfoParams *const mi_params = &cm->mi_params; |
| const int mib_size = cm->mib_size; |
| MB_MODE_INFO **mbmi = |
| mi_params->mi_grid_base + mi_row * mi_params->mi_stride + mi_col; |
| const int stride = mi_params->mi_stride; |
| int mi_width_cur = mi_width; |
| int mi_height_cur = mi_height; |
| if (mi_col + mi_width >= mi_params->mi_cols) |
| mi_width_cur = mi_params->mi_cols - mi_col; |
| if (mi_row + mi_height >= mi_params->mi_rows) |
| mi_height_cur = mi_params->mi_rows - mi_row; |
| // if any sb in the region is active, this region is not inactive |
| for (int r = 0; r < mi_height_cur; r += mib_size, mbmi += mib_size * stride) { |
| for (int c = 0; c < mi_width_cur; c += mib_size) { |
| if (mbmi[c]->sb_active_mode == BRU_ACTIVE_SB) return 0; |
| } |
| } |
| return 1; |
| } |
| |
| /* Return SB activity based on SB_INFO */ |
| static INLINE int bru_is_sb_active(const AV1_COMMON *cm, const int mi_col, |
| const int mi_row) { |
| if (!cm->bru.enabled) return 1; |
| // treat padding region as active |
| if (mi_col < 0 || mi_row < 0) return 1; |
| SB_INFO *sbi = av1_get_sb_info(cm, mi_row, mi_col); |
| return (sbi->sb_active_mode == BRU_ACTIVE_SB); |
| } |
| |
| /* Check is SB pixels available. active and support SBs are available. */ |
| static INLINE int bru_is_sb_available(const AV1_COMMON *cm, const int mi_col, |
| const int mi_row) { |
| if (!cm->bru.enabled) return 1; |
| // treat padding region as active |
| if (mi_col < 0 || mi_row < 0) return 1; |
| SB_INFO *sbi = av1_get_sb_info(cm, mi_row, mi_col); |
| return (sbi->sb_active_mode != BRU_INACTIVE_SB); |
| } |
| |
| /* Return SB activity based on active map */ |
| static INLINE BruActiveMode enc_get_cur_sb_active_mode(const AV1_COMMON *cm, |
| const int mi_col, |
| const int mi_row) { |
| if (!cm->bru.enabled) return BRU_ACTIVE_SB; |
| uint8_t *const active_mode_map = cm->bru.active_mode_map; |
| const int mib_size_log2 = cm->seq_params.mib_size_log2; |
| const int sb_stride = cm->bru.unit_cols; |
| int sb_idx = |
| (mi_row >> mib_size_log2) * sb_stride + (mi_col >> mib_size_log2); |
| return (BruActiveMode)active_mode_map[sb_idx]; |
| } |
| |
| /* Update active map given SB activity */ |
| static INLINE BruActiveMode set_active_map(const AV1_COMMON *cm, |
| const int mi_col, const int mi_row, |
| int sb_active_mode) { |
| if (!cm->bru.enabled) return BRU_ACTIVE_SB; |
| uint8_t *const active_mode_map = cm->bru.active_mode_map; |
| const int mib_size_log2 = cm->seq_params.mib_size_log2; |
| const int sb_stride = cm->bru.unit_cols; |
| int sb_idx = |
| (mi_row >> mib_size_log2) * sb_stride + (mi_col >> mib_size_log2); |
| active_mode_map[sb_idx] = sb_active_mode; |
| return (BruActiveMode)sb_active_mode; |
| } |
| |
| /* Validate active map, for each active SB, it cannot has any inactive neighbor |
| */ |
| static INLINE int bru_active_map_validation(const AV1_COMMON *cm) { |
| // this can only be called after all the SBs are decoded |
| if (!cm->bru.enabled) return 1; |
| if (cm->bru.frame_inactive_flag) return 1; |
| const uint8_t *act = cm->bru.active_mode_map; |
| const int stride = cm->bru.unit_cols; |
| for (unsigned int row = 0; row < cm->bru.unit_rows; row++) { |
| for (unsigned int col = 0; col < cm->bru.unit_cols; col++) { |
| // if active must surrounded by active/support |
| if (*(act + col) == BRU_ACTIVE_SB) { |
| const uint8_t has_top = row > 0; |
| const uint8_t has_left = col > 0; |
| const uint8_t has_bottom = row + 1 < cm->bru.unit_rows; |
| const uint8_t has_right = col + 1 < cm->bru.unit_cols; |
| uint8_t top_inactive = |
| has_top ? *(act + col - stride) == BRU_INACTIVE_SB : 0; |
| uint8_t bot_inactive = |
| has_bottom ? *(act + col + stride) == BRU_INACTIVE_SB : 0; |
| uint8_t left_inactive = |
| has_left ? *(act + col - 1) == BRU_INACTIVE_SB : 0; |
| uint8_t right_inactive = |
| has_right ? *(act + col + 1) == BRU_INACTIVE_SB : 0; |
| uint8_t top_left_inactive = |
| has_top && has_left ? *(act + col - 1 - stride) == BRU_INACTIVE_SB |
| : 0; |
| uint8_t top_right_inactive = |
| has_top && has_right ? *(act + col + 1 - stride) == BRU_INACTIVE_SB |
| : 0; |
| uint8_t bot_left_inactive = |
| has_bottom && has_left |
| ? *(act + col - 1 + stride) == BRU_INACTIVE_SB |
| : 0; |
| uint8_t bot_right_inactive = |
| has_bottom && has_right |
| ? *(act + col + 1 + stride) == BRU_INACTIVE_SB |
| : 0; |
| if (top_inactive || bot_inactive || left_inactive || right_inactive || |
| top_left_inactive || top_right_inactive || bot_left_inactive || |
| bot_right_inactive) { |
| return 0; |
| } |
| } |
| } |
| act += stride; |
| } |
| return 1; |
| } |
| |
| /* Check if this SB is not active and not on the partial border */ |
| static AOM_INLINE bool is_bru_not_active_and_not_on_partial_border( |
| const AV1_COMMON *cm, int mi_col, int mi_row, BLOCK_SIZE bsize) { |
| (void)bsize; |
| if (!cm->bru.enabled) return false; |
| SB_INFO *sbi = av1_get_sb_info(cm, mi_row, mi_col); |
| BruActiveMode mode = sbi->sb_active_mode; |
| bool on_partion_border = |
| mi_row + mi_size_high[bsize] > cm->mi_params.mi_rows || |
| mi_col + mi_size_wide[bsize] > cm->mi_params.mi_cols; |
| return (mode != BRU_ACTIVE_SB) && (!on_partion_border); |
| } |
| |
| /* Check if all the pixels in the Rect are available */ |
| static INLINE bool is_ru_bru_skip(const AV1_COMMON *cm, AV1PixelRect *ru_rect) { |
| if (!cm->bru.enabled) return 0; |
| // convert to mi unit |
| // make sure all units are in luma mi size |
| const int sb_mi_size = cm->seq_params.mib_size; |
| const int mib_size_log2 = cm->seq_params.mib_size_log2; |
| bool bru_skip = true; |
| // adjust height and width according to frame size |
| const int mi_sb_x_start = (ru_rect->left >> (MI_SIZE_LOG2 + mib_size_log2)) |
| << mib_size_log2; |
| const int mi_sb_y_start = (ru_rect->top >> (MI_SIZE_LOG2 + mib_size_log2)) |
| << mib_size_log2; |
| const int mi_sb_x_end = |
| ((ru_rect->right - 1) >> (MI_SIZE_LOG2 + mib_size_log2)) << mib_size_log2; |
| const int mi_sb_y_end = |
| ((ru_rect->bottom - 1) >> (MI_SIZE_LOG2 + mib_size_log2)) |
| << mib_size_log2; |
| for (int mi_row = mi_sb_y_start; mi_row <= mi_sb_y_end; |
| mi_row += sb_mi_size) { |
| for (int mi_col = mi_sb_x_start; mi_col <= mi_sb_x_end; |
| mi_col += sb_mi_size) { |
| if (bru_is_sb_active(cm, mi_col, mi_row)) { |
| bru_skip = false; |
| return bru_skip; |
| } |
| } |
| } |
| return bru_skip; |
| } |
| /* Return the number of active region */ |
| static AOM_INLINE int bru_get_num_of_active_region(const AV1_COMMON *const cm) { |
| if (cm->bru.enabled) { |
| return cm->bru.num_active_regions; |
| } |
| return 1; |
| } |
| /* Init BRU off status*/ |
| static INLINE void init_bru_params(AV1_COMMON *cm) { |
| cm->bru.enabled = 0; |
| cm->bru.update_ref_idx = -1; |
| cm->bru.explicit_ref_idx = -1; |
| cm->bru.ref_disp_order = -1; |
| cm->bru.frame_inactive_flag = 0; |
| } |
| |
| void bru_extend_mc_border(const AV1_COMMON *const cm, int mi_row, int mi_col, |
| BLOCK_SIZE bsize, YV12_BUFFER_CONFIG *src); |
| BruActiveMode set_sb_mbmi_bru_mode(const AV1_COMMON *cm, MACROBLOCKD *const xd, |
| const int mi_col, const int mi_row, |
| const BLOCK_SIZE bsize, |
| const BruActiveMode bru_sb_mode); |
| void bru_copy_sb(const struct AV1Common *cm, const int mi_col, |
| const int mi_row); |
| void bru_update_sb(const struct AV1Common *cm, const int mi_col, |
| const int mi_row); |
| void bru_set_default_inter_mb_mode_info(const AV1_COMMON *const cm, |
| MACROBLOCKD *const xd, |
| MB_MODE_INFO *const mbmi, |
| BLOCK_SIZE bsize); |
| RefCntBuffer *bru_swap_common(AV1_COMMON *cm); |
| #endif // AOM_AV1_COMMON_ARD_H_ |