blob: 67f4bc4140830065e8d53f586bfdc765eb007059 [file] [log] [blame] [edit]
/*
* Copyright (c) 2021, 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_CDEF_H_
#define AOM_AV1_COMMON_CDEF_H_
#define CDEF_STRENGTH_BITS 6
#define CDEF_PRI_STRENGTHS 16
#define CDEF_SEC_STRENGTHS 4
#include "config/aom_config.h"
#include "aom/aom_integer.h"
#include "aom_ports/mem.h"
#include "av1/common/av1_common_int.h"
#include "av1/common/cdef_block.h"
enum { TOP, LEFT, BOTTOM, RIGHT, BOUNDARIES } UENUM1BYTE(BOUNDARY);
struct AV1CdefSyncData;
/*!\brief Parameters related to CDEF Block
* Stores buffers and parameters used while filtering a block. Unlike CdefInfo
*(frame-level data), this is a temporary structure and only used during block
* processing.
*/
typedef struct {
uint16_t *src; /*!< CDEF intermediate source buffer */
uint16_t *top_linebuf[MAX_MB_PLANE]; /*!< CDEF top line buffer */
uint16_t *bot_linebuf[MAX_MB_PLANE]; /*!< CDEF bottom line buffer */
uint16_t *dst; /*!< CDEF destination buffer */
cdef_list
dlist[MI_SIZE_64X64 * MI_SIZE_64X64]; /*!< CDEF 8x8 block positions */
int xdec; /*!< Sub-sampling X */
int ydec; /*!< Sub-sampling Y */
int mi_wide_l2; /*!< Pixels per mi unit in width */
int mi_high_l2; /*!< Pixels per mi unit in height */
int frame_boundary[BOUNDARIES]; /*!< Flags to indicate if the block is at a
frame boundary */
int damping; /*!< CDEF damping factor */
int coeff_shift; /*!< Bit-depth based shift for calculating filter strength */
int level; /*!< CDEF filtering level */
int sec_strength; /*!< CDEF secondary filter strength */
int cdef_count; /*!< Number of CDEF sub-blocks in a filter block unit */
int dir[CDEF_NBLOCKS]
[CDEF_NBLOCKS]; /*!< CDEF filter direction for all 8x8 sub-blocks*/
int var[CDEF_NBLOCKS][CDEF_NBLOCKS]; /*!< variance of all 8x8 sub-blocks */
int dst_stride; /*!< CDEF destination buffer stride */
int coffset; /*!< current filter block offset in a row */
int roffset; /*!< current filter block row offset */
} CdefBlockInfo;
static INLINE int sign_int(int i) { return i < 0 ? -1 : 1; }
static INLINE int constrain(int diff, int threshold, int damping) {
if (!threshold) return 0;
const int shift = AOMMAX(0, damping - get_msb(threshold));
return sign_int(diff) *
AOMMIN(abs(diff), AOMMAX(0, threshold - (abs(diff) >> shift)));
}
#if defined(__clang__) && defined(__has_attribute)
#if __has_attribute(no_sanitize)
#define AOM_NO_UNSIGNED_OVERFLOW_CHECK \
__attribute__(( \
no_sanitize("unsigned-integer-overflow", "unsigned-shift-base")))
#endif
#endif
#ifndef AOM_NO_UNSIGNED_OVERFLOW_CHECK
#define AOM_NO_UNSIGNED_OVERFLOW_CHECK
#endif
AOM_NO_UNSIGNED_OVERFLOW_CHECK static AOM_INLINE int
av1_get_cdef_transmitted_index(int mi_row, int mi_col) {
// Find index of this CDEF unit in this superblock.
const int index_mask = (UINT32_MAX << (32 - CDEF_SB_SHIFT)) >>
(32 - CDEF_SB_SHIFT - MI_IN_CDEF_LINEAR_LOG2);
const int cdef_unit_row_in_sb =
((mi_row & index_mask) >> MI_IN_CDEF_LINEAR_LOG2);
const int cdef_unit_col_in_sb =
((mi_col & index_mask) >> MI_IN_CDEF_LINEAR_LOG2);
return CDEF_IN_SB_STRIDE * cdef_unit_row_in_sb + cdef_unit_col_in_sb;
}
#undef AOM_NO_UNSIGNED_OVERFLOW_CHECK
#ifdef __cplusplus
extern "C" {
#endif
int av1_cdef_compute_sb_list(const AV1_COMMON *const cm,
const CommonModeInfoParams *const mi_params,
int mi_row, int mi_col, cdef_list *dlist,
BLOCK_SIZE bsize
#if CONFIG_DISABLE_LOOP_FILTERS_LOSSLESS
,
int num_planes
#endif // CONFIG_DISABLE_LOOP_FILTERS_LOSSLESS
);
typedef void (*cdef_init_fb_row_t)(AV1_COMMON *cm, MACROBLOCKD *const xd,
CdefBlockInfo *fb_info,
uint16_t **const linebuf,
uint16_t *const src,
struct AV1CdefSyncData *cdef_sync, int fbr);
static INLINE int fetch_cdef_mi_grid_index(const AV1_COMMON *const cm,
const MACROBLOCKD *const xd) {
const CommonModeInfoParams *const mi_params = &cm->mi_params;
const int mi_row = xd->mi_row;
const int mi_col = xd->mi_col;
// CDEF unit size is 64x64 irrespective of the superblock size.
const int cdef_size = 1 << MI_IN_CDEF_LINEAR_LOG2;
// CDEF strength for this CDEF unit needs to be read into the MB_MODE_INFO
// of the 1st block in this CDEF unit.
const int block_mask = ~(cdef_size - 1);
const int grid_idx =
get_mi_grid_idx(mi_params, mi_row & block_mask, mi_col & block_mask);
return grid_idx;
}
/*!\brief Function for applying CDEF to a frame
*
* \ingroup in_loop_cdef
* This function applies CDEF to a frame.
*
* \param[in, out] frame Compressed frame buffer
* \param[in, out] cm Pointer to top level common structure
* \param[in] xd Pointer to common current coding block structure
* \param[in] cdef_init_fb_row_fn Function Pointer to initialize filter
* block row info
*
* Nothing is returned. Instead, the filtered frame is output in \c frame.
*/
void av1_cdef_frame(YV12_BUFFER_CONFIG *frame, AV1_COMMON *cm, MACROBLOCKD *xd,
cdef_init_fb_row_t cdef_init_fb_row_fn);
/*!\brief Apply CDEF filtering for one row of 64x64 filter blocks
*
* \ingroup in_loop_cdef
* This function applies CDEF filtering for a single row of 64x64 filter
* blocks in the frame. It sets up frame boundary flags (LEFT/RIGHT) for the
* row and iterates over each filter block column, invoking ref cdef_fb_col
* to apply the filter.
*
* \param[in, out] cm Pointer to top level common structure
* \param[in, out] xd Pointer to common current coding block structure
* \param[in, out] linebuf Line buffer used for top/bottom row extension
* \param[in, out] colbuf Column buffer used for left/right border
* extension
* \param[in, out] src Pointer to source buffer for the
* \param[in] fbr Filter block row index current row
* \param[in] cdef_init_fb_row_fn Function pointer to initialize filter
block row info
* \param[in, out] cdef_sync Synchronization data for multi-threaded CDEF
*
* Nothing is returned. Instead, the filtered row is written into \c src
* and intermediate buffers (\c linebuf and \c colbuf).
*/
void av1_cdef_fb_row(AV1_COMMON *const cm, MACROBLOCKD *const xd,
uint16_t **const linebuf, uint16_t **const colbuf,
uint16_t *const src, int fbr,
cdef_init_fb_row_t cdef_init_fb_row_fn,
struct AV1CdefSyncData *const cdef_sync);
/*!\brief Initialize parameters for one row of 64x64 filter blocks
*
* \param[in, out] cm Pointer to top level common structure
* \param[in, out] xd Pointer to common current coding block structure
* \param[out] fb_info Structure holding filter block row parameters
* \param[in, out] linebuf Line buffer used for top/bottom row extension
* \param[in] src Pointer to source buffer for the frame
* \param[in, out] cdef_sync Synchronization data (unused in this function)
* \param[in] fbr Filter block row index of the current row
*
* Nothing is returned. Instead, the row-level filter block information is
* stored in \c fb_info and supporting line buffers are updated.
*/
void av1_cdef_init_fb_row(AV1_COMMON *const cm, MACROBLOCKD *const xd,
CdefBlockInfo *const fb_info,
uint16_t **const linebuf, uint16_t *const src,
struct AV1CdefSyncData *const cdef_sync, int fbr);
#ifdef __cplusplus
} // extern "C"
#endif
#endif // AOM_AV1_COMMON_CDEF_H_