Coverage Report

Created: 2025-06-22 08:04

/src/aom/av1/encoder/hash.c
Line
Count
Source (jump to first uncovered line)
1
/*
2
 * Copyright (c) 2016, Alliance for Open Media. All rights reserved.
3
 *
4
 * This source code is subject to the terms of the BSD 2 Clause License and
5
 * the Alliance for Open Media Patent License 1.0. If the BSD 2 Clause License
6
 * was not distributed with this source code in the LICENSE file, you can
7
 * obtain it at www.aomedia.org/license/software. If the Alliance for Open
8
 * Media Patent License 1.0 was not distributed with this source code in the
9
 * PATENTS file, you can obtain it at www.aomedia.org/license/patent.
10
 */
11
12
#include "av1/encoder/hash.h"
13
#include "config/av1_rtcd.h"
14
15
static void crc_calculator_process_data(CRC_CALCULATOR *p_crc_calculator,
16
0
                                        uint8_t *pData, uint32_t dataLength) {
17
0
  for (uint32_t i = 0; i < dataLength; i++) {
18
0
    const uint8_t index = (uint8_t)((p_crc_calculator->remainder >>
19
0
                                     (p_crc_calculator->bits - 8)) ^
20
0
                                    pData[i]);
21
0
    p_crc_calculator->remainder <<= 8;
22
0
    p_crc_calculator->remainder ^= p_crc_calculator->table[index];
23
0
  }
24
0
}
25
26
0
static void crc_calculator_reset(CRC_CALCULATOR *p_crc_calculator) {
27
0
  p_crc_calculator->remainder = 0;
28
0
}
29
30
0
static uint32_t crc_calculator_get_crc(CRC_CALCULATOR *p_crc_calculator) {
31
0
  return p_crc_calculator->remainder & p_crc_calculator->final_result_mask;
32
0
}
33
34
0
static void crc_calculator_init_table(CRC_CALCULATOR *p_crc_calculator) {
35
0
  const uint32_t high_bit = 1 << (p_crc_calculator->bits - 1);
36
0
  const uint32_t byte_high_bit = 1 << (8 - 1);
37
38
0
  for (uint32_t value = 0; value < 256; value++) {
39
0
    uint32_t remainder = 0;
40
0
    for (uint8_t mask = byte_high_bit; mask != 0; mask >>= 1) {
41
0
      if (value & mask) {
42
0
        remainder ^= high_bit;
43
0
      }
44
45
0
      if (remainder & high_bit) {
46
0
        remainder <<= 1;
47
0
        remainder ^= p_crc_calculator->trunc_poly;
48
0
      } else {
49
0
        remainder <<= 1;
50
0
      }
51
0
    }
52
0
    p_crc_calculator->table[value] = remainder;
53
0
  }
54
0
}
55
56
void av1_crc_calculator_init(CRC_CALCULATOR *p_crc_calculator, uint32_t bits,
57
0
                             uint32_t truncPoly) {
58
0
  p_crc_calculator->remainder = 0;
59
0
  p_crc_calculator->bits = bits;
60
0
  p_crc_calculator->trunc_poly = truncPoly;
61
0
  p_crc_calculator->final_result_mask = (1 << bits) - 1;
62
0
  crc_calculator_init_table(p_crc_calculator);
63
0
}
64
65
uint32_t av1_get_crc_value(CRC_CALCULATOR *p_crc_calculator, uint8_t *p,
66
0
                           int length) {
67
0
  crc_calculator_reset(p_crc_calculator);
68
0
  crc_calculator_process_data(p_crc_calculator, p, length);
69
0
  return crc_calculator_get_crc(p_crc_calculator);
70
0
}
71
72
/* CRC-32C (iSCSI) polynomial in reversed bit order. */
73
0
#define POLY 0x82f63b78
74
75
/* Construct table for software CRC-32C calculation. */
76
0
void av1_crc32c_calculator_init(CRC32C *p_crc32c) {
77
0
  uint32_t crc;
78
79
0
  for (int n = 0; n < 256; n++) {
80
0
    crc = n;
81
0
    crc = crc & 1 ? (crc >> 1) ^ POLY : crc >> 1;
82
0
    crc = crc & 1 ? (crc >> 1) ^ POLY : crc >> 1;
83
0
    crc = crc & 1 ? (crc >> 1) ^ POLY : crc >> 1;
84
0
    crc = crc & 1 ? (crc >> 1) ^ POLY : crc >> 1;
85
0
    crc = crc & 1 ? (crc >> 1) ^ POLY : crc >> 1;
86
0
    crc = crc & 1 ? (crc >> 1) ^ POLY : crc >> 1;
87
0
    crc = crc & 1 ? (crc >> 1) ^ POLY : crc >> 1;
88
0
    crc = crc & 1 ? (crc >> 1) ^ POLY : crc >> 1;
89
0
    p_crc32c->table[0][n] = crc;
90
0
  }
91
0
  for (int n = 0; n < 256; n++) {
92
0
    crc = p_crc32c->table[0][n];
93
0
    for (int k = 1; k < 8; k++) {
94
0
      crc = p_crc32c->table[0][crc & 0xff] ^ (crc >> 8);
95
0
      p_crc32c->table[k][n] = crc;
96
0
    }
97
0
  }
98
0
}
99
100
/* Table-driven software version as a fall-back.  This is about 15 times slower
101
 than using the hardware instructions.  This assumes little-endian integers,
102
 as is the case on Intel processors that the assembler code here is for. */
103
0
uint32_t av1_get_crc32c_value_c(void *c, uint8_t *buf, size_t len) {
104
0
  const uint8_t *next = (const uint8_t *)(buf);
105
0
  uint64_t crc;
106
0
  CRC32C *p = (CRC32C *)c;
107
0
  crc = 0 ^ 0xffffffff;
108
0
  while (len && ((uintptr_t)next & 7) != 0) {
109
0
    crc = p->table[0][(crc ^ *next++) & 0xff] ^ (crc >> 8);
110
0
    len--;
111
0
  }
112
0
  while (len >= 8) {
113
0
    crc ^= *(uint64_t *)next;
114
0
    crc = p->table[7][crc & 0xff] ^ p->table[6][(crc >> 8) & 0xff] ^
115
0
          p->table[5][(crc >> 16) & 0xff] ^ p->table[4][(crc >> 24) & 0xff] ^
116
0
          p->table[3][(crc >> 32) & 0xff] ^ p->table[2][(crc >> 40) & 0xff] ^
117
0
          p->table[1][(crc >> 48) & 0xff] ^ p->table[0][crc >> 56];
118
0
    next += 8;
119
0
    len -= 8;
120
0
  }
121
0
  while (len) {
122
0
    crc = p->table[0][(crc ^ *next++) & 0xff] ^ (crc >> 8);
123
0
    len--;
124
0
  }
125
0
  return (uint32_t)crc ^ 0xffffffff;
126
0
}