| /* |
| * Copyright (c) 2020, Alliance for Open Media. All rights reserved. |
| * |
| * This source code is subject to the terms of the BSD 2 Clause License and |
| * the Alliance for Open Media Patent License 1.0. If the BSD 2 Clause License |
| * was not distributed with this source code in the LICENSE file, you can |
| * obtain it at www.aomedia.org/license/software. 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 www.aomedia.org/license/patent. |
| */ |
| |
| #ifndef AOM_AV1_ENCODER_MOTION_SEARCH_H_ |
| #define AOM_AV1_ENCODER_MOTION_SEARCH_H_ |
| |
| #include "av1/encoder/encoder.h" |
| |
| #ifdef __cplusplus |
| extern "C" { |
| #endif |
| |
| #define NUM_JOINT_ME_REFINE_ITER 2 |
| #define REDUCED_JOINT_ME_REFINE_ITER 1 |
| // TODO(any): rename this struct to something else. There is already another |
| // struct called inter_modes_info, which makes this terribly confusing. |
| typedef struct { |
| int drl_cost; |
| int_mv full_search_mv; |
| int full_mv_rate; |
| int full_mv_bestsme; |
| int skip; |
| } inter_mode_info; |
| |
| struct HandleInterModeArgs; |
| void av1_single_motion_search(const AV1_COMP *const cpi, MACROBLOCK *x, |
| BLOCK_SIZE bsize, int ref_idx, int *rate_mv, |
| int search_range, inter_mode_info *mode_info, |
| int_mv *best_mv, |
| struct HandleInterModeArgs *const args); |
| |
| int av1_joint_motion_search(const AV1_COMP *cpi, MACROBLOCK *x, |
| BLOCK_SIZE bsize, int_mv *cur_mv, |
| const uint8_t *mask, int mask_stride, int *rate_mv, |
| int allow_second_mv, int joint_me_num_refine_iter); |
| |
| int av1_interinter_compound_motion_search(const AV1_COMP *const cpi, |
| MACROBLOCK *x, |
| const int_mv *const cur_mv, |
| const BLOCK_SIZE bsize, |
| const PREDICTION_MODE this_mode); |
| |
| int av1_compound_single_motion_search_interinter( |
| const AV1_COMP *cpi, MACROBLOCK *x, BLOCK_SIZE bsize, int_mv *cur_mv, |
| const uint8_t *mask, int mask_stride, int *rate_mv, int ref_idx); |
| |
| int av1_compound_single_motion_search(const AV1_COMP *cpi, MACROBLOCK *x, |
| BLOCK_SIZE bsize, MV *this_mv, |
| const uint8_t *second_pred, |
| const uint8_t *mask, int mask_stride, |
| int *rate_mv, int ref_idx); |
| |
| // Performs a motion search in SIMPLE_TRANSLATION mode using reference frame |
| // ref and calculates the sse and var of the residue. Note that this sets the |
| // offset of mbmi, so we will need to reset it after calling this function. |
| int_mv av1_simple_motion_search_sse_var(struct AV1_COMP *cpi, MACROBLOCK *x, |
| int mi_row, int mi_col, |
| BLOCK_SIZE bsize, int ref, |
| const FULLPEL_MV start_mv, |
| int num_planes, int use_subpixel, |
| unsigned int *sse, unsigned int *var); |
| |
| static AOM_INLINE const search_site_config *av1_get_search_site_config( |
| const AV1_COMP *cpi, MACROBLOCK *x, SEARCH_METHODS search_method) { |
| const int ref_stride = x->e_mbd.plane[0].pre[0].stride; |
| |
| // AV1_COMP::mv_search_params.search_site_config is a compressor level cache |
| // that's shared by multiple threads. In most cases where all frames have the |
| // same resolution, the cache contains the search site config that we need. |
| const MotionVectorSearchParams *mv_search_params = &cpi->mv_search_params; |
| if (ref_stride == mv_search_params->search_site_cfg[SS_CFG_SRC]->stride) { |
| return mv_search_params->search_site_cfg[SS_CFG_SRC]; |
| } else if (ref_stride == |
| mv_search_params->search_site_cfg[SS_CFG_LOOKAHEAD]->stride) { |
| return mv_search_params->search_site_cfg[SS_CFG_LOOKAHEAD]; |
| } |
| |
| // If the cache does not contain the correct stride, then we will need to rely |
| // on the thread level config MACROBLOCK::search_site_cfg_buf. If even the |
| // thread level config doesn't match, then we need to update it. |
| search_method = search_method_lookup[search_method]; |
| assert(search_method_lookup[search_method] == search_method && |
| "The search_method_lookup table should be idempotent."); |
| if (ref_stride != x->search_site_cfg_buf[search_method].stride) { |
| av1_refresh_search_site_config(x->search_site_cfg_buf, search_method, |
| ref_stride); |
| } |
| |
| return x->search_site_cfg_buf; |
| } |
| |
| static AOM_INLINE SEARCH_METHODS |
| av1_get_faster_search_method(SEARCH_METHODS search_method) { |
| // Note on search method's accuracy: |
| // 1. NSTEP |
| // 2. DIAMOND |
| // 3. BIGDIA \approx SQUARE |
| // 4. HEX. |
| // 5. FAST_HEX \approx FAST_DIAMOND |
| switch (search_method) { |
| case NSTEP: return DIAMOND; |
| case NSTEP_8PT: return DIAMOND; |
| case DIAMOND: return BIGDIA; |
| case CLAMPED_DIAMOND: return BIGDIA; |
| case BIGDIA: return HEX; |
| case SQUARE: return HEX; |
| case HEX: return FAST_HEX; |
| case FAST_HEX: return FAST_HEX; |
| case FAST_DIAMOND: return VFAST_DIAMOND; |
| case FAST_BIGDIA: return FAST_BIGDIA; |
| case VFAST_DIAMOND: return VFAST_DIAMOND; |
| default: assert(0 && "Invalid search method!"); return DIAMOND; |
| } |
| } |
| |
| static AOM_INLINE SEARCH_METHODS av1_get_default_mv_search_method( |
| const MACROBLOCK *x, const MV_SPEED_FEATURES *mv_sf, BLOCK_SIZE bsize) { |
| SEARCH_METHODS search_method = mv_sf->search_method; |
| const int sf_blk_search_method = mv_sf->use_bsize_dependent_search_method; |
| const int min_dim = AOMMIN(block_size_wide[bsize], block_size_high[bsize]); |
| const int qband = x->qindex >> (QINDEX_BITS - 2); |
| const bool use_faster_search_method = |
| (sf_blk_search_method == 1 && min_dim >= 32) || |
| (sf_blk_search_method >= 2 && min_dim >= 16 && |
| x->content_state_sb.source_sad_nonrd <= kMedSad && qband < 3); |
| |
| if (use_faster_search_method) { |
| search_method = av1_get_faster_search_method(search_method); |
| } |
| return search_method; |
| } |
| |
| #ifdef __cplusplus |
| } // extern "C" |
| #endif |
| |
| #endif // AOM_AV1_ENCODER_MOTION_SEARCH_H_ |