blob: 782e6e7778dcb7a3ded37348d41a5911b9e2a0c5 [file] [log] [blame]
/*
* Copyright (c) 2023, 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 "av1/common/hr_coding.h"
#include "aom/internal/aom_codec_internal.h"
void write_exp_golomb(aom_writer *w, int level, int k) {
int x = level + (1 << k);
int length = 0;
length = get_msb(x) + 1;
assert(length > k);
#if CONFIG_BYPASS_IMPROVEMENT
aom_write_literal(w, 0, length - 1 - k);
aom_write_literal(w, x, length);
#else
for (i = 0; i < length - 1 - k; ++i) aom_write_bit(w, 0);
for (i = length - 1; i >= 0; --i) aom_write_bit(w, (x >> i) & 0x01);
#endif // CONFIG_BYPASS_IMPROVEMENT
}
int read_exp_golomb(MACROBLOCKD *xd, aom_reader *r, int k) {
#if CONFIG_BYPASS_IMPROVEMENT
int length = aom_read_unary(r, 21, ACCT_INFO("hr"));
if (length > 20) {
aom_internal_error(xd->error_info, AOM_CODEC_CORRUPT_FRAME,
"Invalid length in read_exp_golomb");
}
length += k;
int x = 1 << length;
x += aom_read_literal(r, length, ACCT_INFO("hr"));
#else
int x = 1;
int length = 0;
int i = 0;
while (!i) {
i = aom_read_bit(r, ACCT_INFO("hr"));
++length;
if (length > 20) {
aom_internal_error(xd->error_info, AOM_CODEC_CORRUPT_FRAME,
"Invalid length in read_exp_golomb");
break;
}
}
length += k;
for (i = 0; i < length - 1; ++i) {
x <<= 1;
x += aom_read_bit(r, ACCT_INFO("hr"));
}
#endif // CONFIG_BYPASS_IMPROVEMENT
return x - (1 << k);
}
#if CONFIG_ADAPTIVE_HR
typedef struct adaptive_table {
int *table;
int initial_param;
unsigned int table_size;
} adaptive_table;
int intra_table[] = { 10, 20, 40, 75, 135 };
int inter_table[] = { 20, 35, 55, 95, 170 };
int idtx_table[] = { 25, 50, 130 };
adaptive_table tables[] = {
{ .table = intra_table, .initial_param = 1, .table_size = 5 }, // Intra table
{ .table = inter_table, .initial_param = 1, .table_size = 5 }, // Inter table
{ .table = idtx_table,
.initial_param = 1,
.table_size = 3 }, // IDTX table (only for inter)
};
int get_adaptive_param(adaptive_hr_info *info) {
adaptive_table *adp_table;
adp_table = &tables[0];
int m = adp_table->initial_param;
for (unsigned int i = 0; i < adp_table->table_size; ++i) {
if (info->context < adp_table->table[i]) break;
++m;
}
return m;
}
void write_truncated_rice(aom_writer *w, int level, int m, int k, int cmax) {
const int mask = (1 << m) - 1;
int q = level >> m;
if (q >= cmax) {
aom_write_literal(w, 0, cmax);
write_exp_golomb(w, level - (cmax << m), k);
} else {
aom_write_literal(w, 0, q);
aom_write_literal(w, 1, 1);
aom_write_literal(w, level & mask, m);
}
}
int read_truncated_rice(MACROBLOCKD *xd, aom_reader *r, int m, int k,
int cmax) {
int q = aom_read_unary(r, cmax, ACCT_INFO("hr"));
if (q == cmax) return read_exp_golomb(xd, r, k) + (cmax << m);
int rem = aom_read_literal(r, m, ACCT_INFO("hr"));
return rem + (q << m);
}
int get_truncated_rice_length(int level, int m, int k, int cmax) {
int q = level >> m;
if (q >= cmax) return cmax + get_exp_golomb_length(level - (cmax << m), k);
return q + 1 + m;
}
int get_truncated_rice_length_diff(int level, int m, int k, int cmax,
int *diff) {
int q = level >> m;
if (q >= cmax) {
int lshifted = level - (cmax << m);
if (lshifted == 0) {
int golomb_len0 = k + 1;
*diff = golomb_len0 - m;
return cmax + golomb_len0;
}
return cmax + get_exp_golomb_length_diff(lshifted, k, diff);
}
if (level == 0) {
*diff = m + 1;
return m + 1;
}
*diff = level == (q << m);
return q + 1 + m;
}
void write_adaptive_hr(aom_writer *w, int level, adaptive_hr_info *info) {
int m = get_adaptive_param(info);
write_truncated_rice(w, level, m, m + 1, AOMMIN(m + 4, 6));
}
int read_adaptive_hr(MACROBLOCKD *xd, aom_reader *r, adaptive_hr_info *info) {
int m = get_adaptive_param(info);
return read_truncated_rice(xd, r, m, m + 1, AOMMIN(m + 4, 6));
}
int get_adaptive_hr_length(int level, adaptive_hr_info *info) {
int m = get_adaptive_param(info);
return get_truncated_rice_length(level, m, m + 1, AOMMIN(m + 4, 6));
}
int get_adaptive_hr_length_diff(int level, adaptive_hr_info *info, int *diff) {
int m = get_adaptive_param(info);
return get_truncated_rice_length_diff(level, m, m + 1, AOMMIN(m + 4, 6),
diff);
}
#endif // CONFIG_ADAPTIVE_HR