blob: 07df9a184e51ec999b12669b5a62ec07ef5c0fd4 [file] [log] [blame] [edit]
/*
* Copyright (c) 2025, 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/.
*/
#include <assert.h>
#include <limits.h>
#include <stdio.h>
#include "aom/aom_encoder.h"
#include "aom_dsp/aom_dsp_common.h"
#include "aom_dsp/binary_codes_writer.h"
#include "aom_dsp/bitwriter_buffer.h"
#include "aom_mem/aom_mem.h"
#include "aom_ports/bitops.h"
#include "aom_ports/mem_ops.h"
#include "aom_ports/system_state.h"
#include "av1/common/av1_common_int.h"
#include "av1/common/blockd.h"
#include "av1/common/enums.h"
#if CONFIG_BITSTREAM_DEBUG
#include "aom_util/debug_util.h"
#endif // CONFIG_BITSTREAM_DEBUG
#include "common/md5_utils.h"
#include "common/rawenc.h"
#include "av1/common/blockd.h"
#include "av1/common/cdef.h"
#include "av1/common/ccso.h"
#include "av1/common/cfl.h"
#include "av1/common/entropy.h"
#include "av1/common/entropymode.h"
#include "av1/common/entropymv.h"
#include "av1/common/intra_dip.h"
#include "av1/common/mvref_common.h"
#include "av1/common/pred_common.h"
#include "av1/common/quant_common.h"
#include "av1/common/reconinter.h"
#include "av1/common/reconintra.h"
#include "av1/common/seg_common.h"
#include "av1/common/tile_common.h"
#include "av1/encoder/bitstream.h"
#include "av1/common/cost.h"
#include "av1/encoder/encodemv.h"
#include "av1/encoder/encodetxb.h"
#include "av1/encoder/mcomp.h"
#include "av1/encoder/palette.h"
#include "av1/encoder/pickrst.h"
#include "av1/encoder/segmentation.h"
#include "av1/encoder/tokenize.h"
#if CONFIG_F153_FGM_OBU
void set_film_grain_model(const AV1_COMP *const cpi,
struct film_grain_model *fgm_current) {
const aom_film_grain_t *const pars = &cpi->common.film_grain_params;
const SequenceHeader *const seq_params = &cpi->common.seq_params;
fgm_current->fgm_id = 0;
fgm_current->fgm_chroma_idc = CHROMA_FORMAT_420;
if (seq_params->monochrome)
fgm_current->fgm_chroma_idc = CHROMA_FORMAT_400;
else if (seq_params->subsampling_x == 0 && seq_params->subsampling_y == 1)
fgm_current->fgm_chroma_idc = CHROMA_FORMAT_444;
else if (seq_params->subsampling_x == 1 && seq_params->subsampling_y == 0)
fgm_current->fgm_chroma_idc = CHROMA_FORMAT_422;
#if CONFIG_CWG_F298_REC11
for (int c = 0; c < 3; c++) {
fgm_current->fgm_points[c] = pars->fgm_points[c];
}
for (int i = 0; i < 14; i++) {
for (int j = 0; j < 2; j++) {
fgm_current->fgm_scaling_points[0][i][j] =
pars->fgm_scaling_points_0[i][j];
fgm_current->fgm_scaling_points[1][i][j] =
pars->fgm_scaling_points_1[i][j];
fgm_current->fgm_scaling_points[2][i][j] =
pars->fgm_scaling_points_2[i][j];
}
}
#else
for (int i = 0; i < 14; i++) {
fgm_current->scaling_points_y[i][0] = pars->scaling_points_y[i][0];
fgm_current->scaling_points_y[i][1] = pars->scaling_points_y[i][1];
}
fgm_current->num_y_points = pars->num_y_points; // value: 0..14
for (int i = 0; i < 10; i++) {
for (int j = 0; j < 2; j++) {
fgm_current->scaling_points_cb[i][j] = pars->scaling_points_cb[i][j];
fgm_current->scaling_points_cr[i][j] = pars->scaling_points_cr[i][j];
}
}
fgm_current->num_cb_points = pars->num_cb_points;
fgm_current->num_cr_points = pars->num_cr_points;
#endif // CONFIG_CWG_F298_REC11
fgm_current->scaling_shift = pars->scaling_shift;
fgm_current->ar_coeff_lag = pars->ar_coeff_lag;
// 8 bit values
for (int i = 0; i < 24; i++)
fgm_current->ar_coeffs_y[i] = pars->ar_coeffs_y[i];
for (int i = 0; i < 25; i++) {
fgm_current->ar_coeffs_cb[i] = pars->ar_coeffs_cb[i];
fgm_current->ar_coeffs_cr[i] = pars->ar_coeffs_cr[i];
}
fgm_current->ar_coeff_shift = pars->ar_coeff_shift;
fgm_current->cb_mult = pars->cb_mult;
fgm_current->cb_luma_mult = pars->cb_luma_mult;
fgm_current->cb_offset = pars->cb_offset;
fgm_current->cr_mult = pars->cr_mult;
fgm_current->cr_luma_mult = pars->cr_luma_mult;
fgm_current->cr_offset = pars->cr_offset;
fgm_current->overlap_flag = pars->overlap_flag;
fgm_current->clip_to_restricted_range = pars->clip_to_restricted_range;
#if CONFIG_CWG_F298_REC11
fgm_current->fgm_scale_from_channel0_flag =
pars->fgm_scale_from_channel0_flag;
#else
fgm_current->chroma_scaling_from_luma = pars->chroma_scaling_from_luma;
#endif
fgm_current->grain_scale_shift = pars->grain_scale_shift;
fgm_current->block_size = pars->block_size;
}
int film_grain_model_decision(int fgm_pos, struct film_grain_model *fgm_in_list,
struct film_grain_model *fgm_current) {
// 0 the contents of both memory blocks are equal
// int is_diff = memcmp(fgm_in_list, fgm_current, sizeof(struct
// film_grain_model)); return is_diff? -1 : fgm_pos;
// //comparison
#if CONFIG_CWG_F298_REC11
for (int c = 0; c < 3; c++) {
if (fgm_current->fgm_points[c] != fgm_in_list->fgm_points[c]) return -1;
}
for (int i = 0; i < 14; i++) {
for (int j = 0; j < 2; j++) {
if (fgm_current->fgm_scaling_points[0][i][j] !=
fgm_in_list->fgm_scaling_points[0][i][j])
return -1;
}
}
for (int c = 1; c < 3; c++) {
for (int i = 0; i < 10; i++) {
for (int j = 0; j < 2; j++) {
if (fgm_current->fgm_scaling_points[c][i][j] !=
fgm_in_list->fgm_scaling_points[c][i][j])
return -1;
}
}
}
#else
if (fgm_current->num_y_points != fgm_in_list->num_y_points) return -1;
if (fgm_current->num_cb_points != fgm_in_list->num_cb_points) return -1;
if (fgm_current->num_cr_points != fgm_in_list->num_cr_points) return -1;
for (int i = 0; i < 14; i++) {
if (fgm_current->scaling_points_y[i][0] !=
fgm_in_list->scaling_points_y[i][0])
return -1;
if (fgm_current->scaling_points_y[i][1] !=
fgm_in_list->scaling_points_y[i][1])
return -1;
}
for (int i = 0; i < 10; i++) {
for (int j = 0; j < 2; j++) {
if (fgm_current->scaling_points_cb[i][j] !=
fgm_in_list->scaling_points_cb[i][j])
return -1;
if (fgm_current->scaling_points_cr[i][j] !=
fgm_in_list->scaling_points_cr[i][j])
return -1;
}
}
#endif
if (fgm_current->scaling_shift != fgm_in_list->scaling_shift) return -1;
if (fgm_current->ar_coeff_lag != fgm_in_list->ar_coeff_lag) return -1;
for (int i = 0; i < 24; i++)
if (fgm_current->ar_coeffs_y[i] != fgm_in_list->ar_coeffs_y[i]) return -1;
for (int i = 0; i < 25; i++) {
if (fgm_current->ar_coeffs_cb[i] != fgm_in_list->ar_coeffs_cb[i]) return -1;
if (fgm_current->ar_coeffs_cr[i] != fgm_in_list->ar_coeffs_cr[i]) return -1;
}
if (fgm_current->ar_coeff_shift != fgm_in_list->ar_coeff_shift) return -1;
if (fgm_current->cb_mult != fgm_in_list->cb_mult) return -1;
if (fgm_current->cb_luma_mult != fgm_in_list->cb_luma_mult) return -1;
if (fgm_current->cb_offset != fgm_in_list->cb_offset) return -1;
if (fgm_current->cr_mult != fgm_in_list->cr_mult) return -1;
if (fgm_current->cr_luma_mult != fgm_in_list->cr_luma_mult) return -1;
if (fgm_current->cr_offset != fgm_in_list->cr_offset) return -1;
if (fgm_current->overlap_flag != fgm_in_list->overlap_flag) return -1;
if (fgm_current->clip_to_restricted_range !=
fgm_in_list->clip_to_restricted_range)
return -1;
#if CONFIG_CWG_F298_REC11
if (fgm_current->fgm_scale_from_channel0_flag !=
fgm_in_list->fgm_scale_from_channel0_flag)
#else
if (fgm_current->chroma_scaling_from_luma !=
fgm_in_list->chroma_scaling_from_luma)
#endif
return -1;
if (fgm_current->grain_scale_shift != fgm_in_list->grain_scale_shift)
return -1;
if (fgm_current->block_size != fgm_in_list->block_size) return -1;
return fgm_pos;
}
int write_fgm_obu(AV1_COMP *cpi, struct film_grain_model *fgm,
uint8_t *const dst) {
struct aom_write_bit_buffer wb = { dst, 0 };
uint32_t size = 0;
AV1_COMMON *cm = &cpi->common;
int fgm_bit_map = 1 << (fgm->fgm_id);
aom_wb_write_literal(&wb, fgm_bit_map, MAX_FGM_NUM);
int chroma_format_idc = CHROMA_FORMAT_420;
if (cm->seq_params.monochrome)
chroma_format_idc = CHROMA_FORMAT_400;
else if (cm->seq_params.subsampling_x == 0 &&
cm->seq_params.subsampling_y == 1)
chroma_format_idc = CHROMA_FORMAT_444;
else if (cm->seq_params.subsampling_x == 1 &&
cm->seq_params.subsampling_y == 0)
chroma_format_idc = CHROMA_FORMAT_422;
aom_wb_write_uvlc(&wb, chroma_format_idc);
for (int fgm_pos = 0; fgm_pos < 1; fgm_pos++) {
// Scaling functions parameters
#if CONFIG_CWG_F298_REC11
int fgmNumChannels = cm->seq_params.monochrome ? 1 : 3;
if (fgmNumChannels > 1) {
aom_wb_write_bit(&wb, fgm->fgm_scale_from_channel0_flag);
} else {
assert(!fgm->fgm_scale_from_channel0_flag);
}
int fgmNumScalingChannels =
fgm->fgm_scale_from_channel0_flag ? 1 : fgmNumChannels;
for (int c = 0; c < fgmNumScalingChannels; c++) {
aom_wb_write_literal(&wb, fgm->fgm_points[c], 4); // max 14
if (fgm->fgm_points[c]) {
// search for the max
int maxIncr = -1, maxScal = -1;
for (int i = 0; i < fgm->fgm_points[c]; i++) {
if (maxIncr < fgm->fgm_scaling_points[c][i][0])
maxIncr = fgm->fgm_scaling_points[c][i][0];
if (maxScal < fgm->fgm_scaling_points[c][i][1])
maxScal = fgm->fgm_scaling_points[c][i][1];
}
// ceillog2
int bitsIncr = maxIncr == -1 ? 0 : aom_ceil_log2(maxIncr + 1);
int bitsScal = maxScal == -1 ? 0 : aom_ceil_log2(maxScal + 1);
aom_wb_write_literal(&wb, bitsIncr - 1, 3);
aom_wb_write_literal(&wb, bitsScal - 5, 2);
for (int i = 0; i < fgm->fgm_points[c]; i++) {
if (i == 0)
aom_wb_write_literal(&wb, fgm->fgm_scaling_points[c][i][0],
bitsIncr);
else {
aom_wb_write_literal(&wb,
fgm->fgm_scaling_points[c][i][0] -
fgm->fgm_scaling_points[c][i - 1][0],
bitsIncr);
}
aom_wb_write_literal(&wb, fgm->fgm_scaling_points[c][i][1], bitsScal);
}
}
}
#else // CONFIG_CWG_F298_REC11
aom_wb_write_literal(&wb, fgm->num_y_points, 4); // max 14
if (fgm->num_y_points) {
// search for the max
int maxIncr = -1, maxScal = -1;
for (int i = 0; i < fgm->num_y_points; i++) {
if (maxIncr < fgm->scaling_points_y[i][0])
maxIncr = fgm->scaling_points_y[i][0];
if (maxScal < fgm->scaling_points_y[i][1])
maxScal = fgm->scaling_points_y[i][1];
}
// ceillog2
int bitsIncr = maxIncr == -1 ? 0 : aom_ceil_log2(maxIncr + 1);
int bitsScal = maxScal == -1 ? 0 : aom_ceil_log2(maxScal + 1);
aom_wb_write_literal(&wb, bitsIncr - 1, 3);
aom_wb_write_literal(&wb, bitsScal - 5, 2);
for (int i = 0; i < fgm->num_y_points; i++) {
aom_wb_write_literal(&wb, fgm->scaling_points_y[i][0], bitsIncr);
aom_wb_write_literal(&wb, fgm->scaling_points_y[i][1], bitsScal);
}
}
if (!cm->seq_params.monochrome) {
aom_wb_write_bit(&wb, fgm->chroma_scaling_from_luma);
} else {
assert(!fgm->chroma_scaling_from_luma);
}
if (cm->seq_params.monochrome || fgm->chroma_scaling_from_luma ||
((cm->seq_params.subsampling_x == 1) &&
(cm->seq_params.subsampling_y == 1) && (fgm->num_y_points == 0))) {
assert(fgm->num_cb_points == 0 && fgm->num_cr_points == 0);
} else {
aom_wb_write_literal(&wb, fgm->num_cb_points, 4); // max 10
if (fgm->num_cb_points) {
int maxIncr = -1, maxScal = -1;
for (int i = 0; i < fgm->num_cb_points; i++) {
if (maxIncr < fgm->scaling_points_cb[i][0])
maxIncr = fgm->scaling_points_cb[i][0];
if (maxScal < fgm->scaling_points_cb[i][1])
maxScal = fgm->scaling_points_cb[i][1];
}
// ceillog2
int bitsIncr = maxIncr == -1 ? 0 : aom_ceil_log2(maxIncr + 1);
int bitsScal = maxScal == -1 ? 0 : aom_ceil_log2(maxScal + 1);
aom_wb_write_literal(&wb, bitsIncr - 1, 3);
aom_wb_write_literal(&wb, bitsScal - 5, 2);
for (int i = 0; i < fgm->num_cb_points; i++) {
aom_wb_write_literal(&wb, fgm->scaling_points_cb[i][0], bitsIncr);
aom_wb_write_literal(&wb, fgm->scaling_points_cb[i][1], bitsScal);
}
}
aom_wb_write_literal(&wb, fgm->num_cr_points, 4); // max 10
if (fgm->num_cr_points) {
int maxIncr = -1, maxScal = -1;
for (int i = 0; i < fgm->num_cr_points; i++) {
if (maxIncr < fgm->scaling_points_cr[i][0])
maxIncr = fgm->scaling_points_cr[i][0];
if (maxScal < fgm->scaling_points_cr[i][1])
maxScal = fgm->scaling_points_cr[i][1];
}
// ceillog2
int bitsIncr = maxIncr == -1 ? 0 : aom_ceil_log2(maxIncr + 1);
int bitsScal = maxScal == -1 ? 0 : aom_ceil_log2(maxScal + 1);
aom_wb_write_literal(&wb, bitsIncr - 1, 3);
aom_wb_write_literal(&wb, bitsScal - 5, 2);
for (int i = 0; i < fgm->num_cr_points; i++) {
aom_wb_write_literal(&wb, fgm->scaling_points_cr[i][0], bitsIncr);
aom_wb_write_literal(&wb, fgm->scaling_points_cr[i][1], bitsScal);
}
}
}
#endif // #endif // CONFIG_CWG_F298_REC11
aom_wb_write_literal(&wb, fgm->scaling_shift - 8, 2); // 8 + value
// AR coefficients
// Only sent if the corresponsing scaling function has
// more than 0 points
aom_wb_write_literal(&wb, fgm->ar_coeff_lag, 2);
int num_pos_luma = 2 * fgm->ar_coeff_lag * (fgm->ar_coeff_lag + 1);
int num_pos_chroma = num_pos_luma;
#if CONFIG_CWG_F298_REC11
if (fgm->fgm_points[0] > 0)
#else
if (fgm->num_y_points > 0)
#endif // CONFIG_CWG_F298_REC11
++num_pos_chroma;
#if CONFIG_CWG_F298_REC11
if (fgm->fgm_points[0])
#else
if (fgm->num_y_points)
#endif
{
int maxAr = -1;
for (int i = 0; i < num_pos_luma; i++) {
if (maxAr < fgm->ar_coeffs_y[i]) maxAr = fgm->ar_coeffs_y[i];
}
// ceillog2
int bitArY = maxAr == -1 ? 0 : aom_ceil_log2(maxAr + 1 + 128);
aom_wb_write_literal(&wb, bitArY - 5, 2);
for (int i = 0; i < num_pos_luma; i++)
aom_wb_write_literal(&wb, fgm->ar_coeffs_y[i] + 128, bitArY);
}
#if CONFIG_CWG_F298_REC11
if (fgm->fgm_points[1] || fgm->fgm_scale_from_channel0_flag)
#else
if (fgm->num_cb_points || fgm->chroma_scaling_from_luma)
#endif
{
int maxAr = -1;
for (int i = 0; i < num_pos_chroma; i++) {
if (maxAr < fgm->ar_coeffs_cb[i]) maxAr = fgm->ar_coeffs_cb[i];
}
// ceillog2
int bitsArCb = maxAr == -1 ? 0 : aom_ceil_log2(maxAr + 1 + 128);
aom_wb_write_literal(&wb, bitsArCb - 5, 2);
for (int i = 0; i < num_pos_chroma; i++)
aom_wb_write_literal(&wb, fgm->ar_coeffs_cb[i] + 128, bitsArCb);
}
#if CONFIG_CWG_F298_REC11
if (fgm->fgm_points[2] || fgm->fgm_scale_from_channel0_flag)
#else
if (fgm->num_cr_points || fgm->chroma_scaling_from_luma)
#endif
{
int maxAr = -1;
for (int i = 0; i < num_pos_chroma; i++) {
if (maxAr < fgm->ar_coeffs_cr[i]) maxAr = fgm->ar_coeffs_cr[i];
}
// ceillog2
int bitsArCr = maxAr == -1 ? 0 : aom_ceil_log2(maxAr + 1 + 128);
aom_wb_write_literal(&wb, bitsArCr - 5, 2);
for (int i = 0; i < num_pos_chroma; i++)
aom_wb_write_literal(&wb, fgm->ar_coeffs_cr[i] + 128, bitsArCr);
}
aom_wb_write_literal(&wb, fgm->ar_coeff_shift - 6, 2); // 8 + value
aom_wb_write_literal(&wb, fgm->grain_scale_shift, 2);
#if CONFIG_CWG_F298_REC11
if (fgm->fgm_points[1])
#else
if (fgm->num_cb_points)
#endif
{
aom_wb_write_literal(&wb, fgm->cb_mult, 8);
aom_wb_write_literal(&wb, fgm->cb_luma_mult, 8);
aom_wb_write_literal(&wb, fgm->cb_offset, 9);
}
#if CONFIG_CWG_F298_REC11
if (fgm->fgm_points[2])
#else
if (fgm->num_cr_points)
#endif
{
aom_wb_write_literal(&wb, fgm->cr_mult, 8);
aom_wb_write_literal(&wb, fgm->cr_luma_mult, 8);
aom_wb_write_literal(&wb, fgm->cr_offset, 9);
}
aom_wb_write_bit(&wb, fgm->overlap_flag);
aom_wb_write_bit(&wb, fgm->clip_to_restricted_range);
aom_wb_write_bit(&wb, fgm->block_size);
}
av1_add_trailing_bits(&wb);
size = aom_wb_bytes_written(&wb);
return size;
}
#endif // CONFIG_F153_FGM_OBU