Coverage Report

Created: 2023-06-07 06:31

/src/aom/av1/common/x86/cfl_ssse3.c
Line
Count
Source (jump to first uncovered line)
1
/*
2
 * Copyright (c) 2017, 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 <tmmintrin.h>
13
14
#include "config/av1_rtcd.h"
15
16
#include "av1/common/cfl.h"
17
18
#include "av1/common/x86/cfl_simd.h"
19
20
// Load 32-bit integer from memory into the first element of dst.
21
8.04M
static INLINE __m128i _mm_loadh_epi32(__m128i const *mem_addr) {
22
8.04M
  return _mm_cvtsi32_si128(*((int *)mem_addr));
23
8.04M
}
24
25
// Store 32-bit integer from the first element of a into memory.
26
7.55M
static INLINE void _mm_storeh_epi32(__m128i const *mem_addr, __m128i a) {
27
7.55M
  *((int *)mem_addr) = _mm_cvtsi128_si32(a);
28
7.55M
}
29
30
/**
31
 * Adds 4 pixels (in a 2x2 grid) and multiplies them by 2. Resulting in a more
32
 * precise version of a box filter 4:2:0 pixel subsampling in Q3.
33
 *
34
 * The CfL prediction buffer is always of size CFL_BUF_SQUARE. However, the
35
 * active area is specified using width and height.
36
 *
37
 * Note: We don't need to worry about going over the active area, as long as we
38
 * stay inside the CfL prediction buffer.
39
 */
40
static INLINE void cfl_luma_subsampling_420_lbd_ssse3(const uint8_t *input,
41
                                                      int input_stride,
42
                                                      uint16_t *pred_buf_q3,
43
1.73M
                                                      int width, int height) {
44
1.73M
  const __m128i twos = _mm_set1_epi8(2);
45
1.73M
  __m128i *pred_buf_m128i = (__m128i *)pred_buf_q3;
46
1.73M
  const __m128i *end = pred_buf_m128i + (height >> 1) * CFL_BUF_LINE_I128;
47
1.73M
  const int luma_stride = input_stride << 1;
48
6.36M
  do {
49
6.36M
    if (width == 4) {
50
3.57M
      __m128i top = _mm_loadh_epi32((__m128i *)input);
51
3.57M
      top = _mm_maddubs_epi16(top, twos);
52
3.57M
      __m128i bot = _mm_loadh_epi32((__m128i *)(input + input_stride));
53
3.57M
      bot = _mm_maddubs_epi16(bot, twos);
54
3.57M
      const __m128i sum = _mm_add_epi16(top, bot);
55
3.57M
      _mm_storeh_epi32(pred_buf_m128i, sum);
56
3.57M
    } else if (width == 8) {
57
1.50M
      __m128i top = _mm_loadl_epi64((__m128i *)input);
58
1.50M
      top = _mm_maddubs_epi16(top, twos);
59
1.50M
      __m128i bot = _mm_loadl_epi64((__m128i *)(input + input_stride));
60
1.50M
      bot = _mm_maddubs_epi16(bot, twos);
61
1.50M
      const __m128i sum = _mm_add_epi16(top, bot);
62
1.50M
      _mm_storel_epi64(pred_buf_m128i, sum);
63
1.50M
    } else {
64
1.28M
      __m128i top = _mm_loadu_si128((__m128i *)input);
65
1.28M
      top = _mm_maddubs_epi16(top, twos);
66
1.28M
      __m128i bot = _mm_loadu_si128((__m128i *)(input + input_stride));
67
1.28M
      bot = _mm_maddubs_epi16(bot, twos);
68
1.28M
      const __m128i sum = _mm_add_epi16(top, bot);
69
1.28M
      _mm_storeu_si128(pred_buf_m128i, sum);
70
1.28M
      if (width == 32) {
71
0
        __m128i top_1 = _mm_loadu_si128(((__m128i *)input) + 1);
72
0
        __m128i bot_1 =
73
0
            _mm_loadu_si128(((__m128i *)(input + input_stride)) + 1);
74
0
        top_1 = _mm_maddubs_epi16(top_1, twos);
75
0
        bot_1 = _mm_maddubs_epi16(bot_1, twos);
76
0
        __m128i sum_1 = _mm_add_epi16(top_1, bot_1);
77
0
        _mm_storeu_si128(pred_buf_m128i + 1, sum_1);
78
0
      }
79
1.28M
    }
80
6.36M
    input += luma_stride;
81
6.36M
    pred_buf_m128i += CFL_BUF_LINE_I128;
82
6.36M
  } while (pred_buf_m128i < end);
83
1.73M
}
84
85
/**
86
 * Adds 2 pixels (in a 2x1 grid) and multiplies them by 4. Resulting in a more
87
 * precise version of a box filter 4:2:2 pixel subsampling in Q3.
88
 *
89
 * The CfL prediction buffer is always of size CFL_BUF_SQUARE. However, the
90
 * active area is specified using width and height.
91
 *
92
 * Note: We don't need to worry about going over the active area, as long as we
93
 * stay inside the CfL prediction buffer.
94
 */
95
static INLINE void cfl_luma_subsampling_422_lbd_ssse3(const uint8_t *input,
96
                                                      int input_stride,
97
                                                      uint16_t *pred_buf_q3,
98
3.90k
                                                      int width, int height) {
99
3.90k
  const __m128i fours = _mm_set1_epi8(4);
100
3.90k
  __m128i *pred_buf_m128i = (__m128i *)pred_buf_q3;
101
3.90k
  const __m128i *end = pred_buf_m128i + height * CFL_BUF_LINE_I128;
102
25.4k
  do {
103
25.4k
    if (width == 4) {
104
4.00k
      __m128i top = _mm_loadh_epi32((__m128i *)input);
105
4.00k
      top = _mm_maddubs_epi16(top, fours);
106
4.00k
      _mm_storeh_epi32(pred_buf_m128i, top);
107
21.4k
    } else if (width == 8) {
108
3.52k
      __m128i top = _mm_loadl_epi64((__m128i *)input);
109
3.52k
      top = _mm_maddubs_epi16(top, fours);
110
3.52k
      _mm_storel_epi64(pred_buf_m128i, top);
111
17.9k
    } else {
112
17.9k
      __m128i top = _mm_loadu_si128((__m128i *)input);
113
17.9k
      top = _mm_maddubs_epi16(top, fours);
114
17.9k
      _mm_storeu_si128(pred_buf_m128i, top);
115
17.9k
      if (width == 32) {
116
0
        __m128i top_1 = _mm_loadu_si128(((__m128i *)input) + 1);
117
0
        top_1 = _mm_maddubs_epi16(top_1, fours);
118
0
        _mm_storeu_si128(pred_buf_m128i + 1, top_1);
119
0
      }
120
17.9k
    }
121
25.4k
    input += input_stride;
122
25.4k
    pred_buf_m128i += CFL_BUF_LINE_I128;
123
25.4k
  } while (pred_buf_m128i < end);
124
3.90k
}
125
126
/**
127
 * Multiplies the pixels by 8 (scaling in Q3).
128
 *
129
 * The CfL prediction buffer is always of size CFL_BUF_SQUARE. However, the
130
 * active area is specified using width and height.
131
 *
132
 * Note: We don't need to worry about going over the active area, as long as we
133
 * stay inside the CfL prediction buffer.
134
 */
135
static INLINE void cfl_luma_subsampling_444_lbd_ssse3(const uint8_t *input,
136
                                                      int input_stride,
137
                                                      uint16_t *pred_buf_q3,
138
281k
                                                      int width, int height) {
139
281k
  const __m128i zeros = _mm_setzero_si128();
140
281k
  const int luma_stride = input_stride;
141
281k
  __m128i *pred_buf_m128i = (__m128i *)pred_buf_q3;
142
281k
  const __m128i *end = pred_buf_m128i + height * CFL_BUF_LINE_I128;
143
3.10M
  do {
144
3.10M
    if (width == 4) {
145
895k
      __m128i row = _mm_loadh_epi32((__m128i *)input);
146
895k
      row = _mm_unpacklo_epi8(row, zeros);
147
895k
      _mm_storel_epi64(pred_buf_m128i, _mm_slli_epi16(row, 3));
148
2.21M
    } else if (width == 8) {
149
1.16M
      __m128i row = _mm_loadl_epi64((__m128i *)input);
150
1.16M
      row = _mm_unpacklo_epi8(row, zeros);
151
1.16M
      _mm_storeu_si128(pred_buf_m128i, _mm_slli_epi16(row, 3));
152
1.16M
    } else {
153
1.04M
      __m128i row = _mm_loadu_si128((__m128i *)input);
154
1.04M
      const __m128i row_lo = _mm_unpacklo_epi8(row, zeros);
155
1.04M
      const __m128i row_hi = _mm_unpackhi_epi8(row, zeros);
156
1.04M
      _mm_storeu_si128(pred_buf_m128i, _mm_slli_epi16(row_lo, 3));
157
1.04M
      _mm_storeu_si128(pred_buf_m128i + 1, _mm_slli_epi16(row_hi, 3));
158
1.04M
      if (width == 32) {
159
0
        __m128i row_1 = _mm_loadu_si128(((__m128i *)input) + 1);
160
0
        const __m128i row_1_lo = _mm_unpacklo_epi8(row_1, zeros);
161
0
        const __m128i row_1_hi = _mm_unpackhi_epi8(row_1, zeros);
162
0
        _mm_storeu_si128(pred_buf_m128i + 2, _mm_slli_epi16(row_1_lo, 3));
163
0
        _mm_storeu_si128(pred_buf_m128i + 3, _mm_slli_epi16(row_1_hi, 3));
164
0
      }
165
1.04M
    }
166
3.10M
    input += luma_stride;
167
3.10M
    pred_buf_m128i += CFL_BUF_LINE_I128;
168
3.10M
  } while (pred_buf_m128i < end);
169
281k
}
170
171
#if CONFIG_AV1_HIGHBITDEPTH
172
/**
173
 * Adds 4 pixels (in a 2x2 grid) and multiplies them by 2. Resulting in a more
174
 * precise version of a box filter 4:2:0 pixel subsampling in Q3.
175
 *
176
 * The CfL prediction buffer is always of size CFL_BUF_SQUARE. However, the
177
 * active area is specified using width and height.
178
 *
179
 * Note: We don't need to worry about going over the active area, as long as we
180
 * stay inside the CfL prediction buffer.
181
 */
182
static INLINE void cfl_luma_subsampling_420_hbd_ssse3(const uint16_t *input,
183
                                                      int input_stride,
184
                                                      uint16_t *pred_buf_q3,
185
505k
                                                      int width, int height) {
186
505k
  const uint16_t *end = pred_buf_q3 + (height >> 1) * CFL_BUF_LINE;
187
505k
  const int luma_stride = input_stride << 1;
188
1.91M
  do {
189
1.91M
    if (width == 4) {
190
1.04M
      const __m128i top = _mm_loadl_epi64((__m128i *)input);
191
1.04M
      const __m128i bot = _mm_loadl_epi64((__m128i *)(input + input_stride));
192
1.04M
      __m128i sum = _mm_add_epi16(top, bot);
193
1.04M
      sum = _mm_hadd_epi16(sum, sum);
194
1.04M
      *((int *)pred_buf_q3) = _mm_cvtsi128_si32(_mm_add_epi16(sum, sum));
195
1.04M
    } else {
196
869k
      const __m128i top = _mm_loadu_si128((__m128i *)input);
197
869k
      const __m128i bot = _mm_loadu_si128((__m128i *)(input + input_stride));
198
869k
      __m128i sum = _mm_add_epi16(top, bot);
199
869k
      if (width == 8) {
200
436k
        sum = _mm_hadd_epi16(sum, sum);
201
436k
        _mm_storel_epi64((__m128i *)pred_buf_q3, _mm_add_epi16(sum, sum));
202
436k
      } else {
203
433k
        const __m128i top_1 = _mm_loadu_si128(((__m128i *)input) + 1);
204
433k
        const __m128i bot_1 =
205
433k
            _mm_loadu_si128(((__m128i *)(input + input_stride)) + 1);
206
433k
        sum = _mm_hadd_epi16(sum, _mm_add_epi16(top_1, bot_1));
207
433k
        _mm_storeu_si128((__m128i *)pred_buf_q3, _mm_add_epi16(sum, sum));
208
433k
        if (width == 32) {
209
0
          const __m128i top_2 = _mm_loadu_si128(((__m128i *)input) + 2);
210
0
          const __m128i bot_2 =
211
0
              _mm_loadu_si128(((__m128i *)(input + input_stride)) + 2);
212
0
          const __m128i top_3 = _mm_loadu_si128(((__m128i *)input) + 3);
213
0
          const __m128i bot_3 =
214
0
              _mm_loadu_si128(((__m128i *)(input + input_stride)) + 3);
215
0
          const __m128i sum_2 = _mm_add_epi16(top_2, bot_2);
216
0
          const __m128i sum_3 = _mm_add_epi16(top_3, bot_3);
217
0
          __m128i next_sum = _mm_hadd_epi16(sum_2, sum_3);
218
0
          _mm_storeu_si128(((__m128i *)pred_buf_q3) + 1,
219
0
                           _mm_add_epi16(next_sum, next_sum));
220
0
        }
221
433k
      }
222
869k
    }
223
1.91M
    input += luma_stride;
224
1.91M
  } while ((pred_buf_q3 += CFL_BUF_LINE) < end);
225
505k
}
226
227
/**
228
 * Adds 2 pixels (in a 2x1 grid) and multiplies them by 4. Resulting in a more
229
 * precise version of a box filter 4:2:2 pixel subsampling in Q3.
230
 *
231
 * The CfL prediction buffer is always of size CFL_BUF_SQUARE. However, the
232
 * active area is specified using width and height.
233
 *
234
 * Note: We don't need to worry about going over the active area, as long as we
235
 * stay inside the CfL prediction buffer.
236
 */
237
static INLINE void cfl_luma_subsampling_422_hbd_ssse3(const uint16_t *input,
238
                                                      int input_stride,
239
                                                      uint16_t *pred_buf_q3,
240
2.20k
                                                      int width, int height) {
241
2.20k
  __m128i *pred_buf_m128i = (__m128i *)pred_buf_q3;
242
2.20k
  const __m128i *end = pred_buf_m128i + height * CFL_BUF_LINE_I128;
243
11.0k
  do {
244
11.0k
    if (width == 4) {
245
6.54k
      const __m128i top = _mm_loadl_epi64((__m128i *)input);
246
6.54k
      const __m128i sum = _mm_slli_epi16(_mm_hadd_epi16(top, top), 2);
247
6.54k
      _mm_storeh_epi32(pred_buf_m128i, sum);
248
6.54k
    } else {
249
4.50k
      const __m128i top = _mm_loadu_si128((__m128i *)input);
250
4.50k
      if (width == 8) {
251
2.50k
        const __m128i sum = _mm_slli_epi16(_mm_hadd_epi16(top, top), 2);
252
2.50k
        _mm_storel_epi64(pred_buf_m128i, sum);
253
2.50k
      } else {
254
2.00k
        const __m128i top_1 = _mm_loadu_si128(((__m128i *)input) + 1);
255
2.00k
        const __m128i sum = _mm_slli_epi16(_mm_hadd_epi16(top, top_1), 2);
256
2.00k
        _mm_storeu_si128(pred_buf_m128i, sum);
257
2.00k
        if (width == 32) {
258
0
          const __m128i top_2 = _mm_loadu_si128(((__m128i *)input) + 2);
259
0
          const __m128i top_3 = _mm_loadu_si128(((__m128i *)input) + 3);
260
0
          const __m128i sum_1 = _mm_slli_epi16(_mm_hadd_epi16(top_2, top_3), 2);
261
0
          _mm_storeu_si128(pred_buf_m128i + 1, sum_1);
262
0
        }
263
2.00k
      }
264
4.50k
    }
265
11.0k
    pred_buf_m128i += CFL_BUF_LINE_I128;
266
11.0k
    input += input_stride;
267
11.0k
  } while (pred_buf_m128i < end);
268
2.20k
}
269
270
static INLINE void cfl_luma_subsampling_444_hbd_ssse3(const uint16_t *input,
271
                                                      int input_stride,
272
                                                      uint16_t *pred_buf_q3,
273
627k
                                                      int width, int height) {
274
627k
  const uint16_t *end = pred_buf_q3 + height * CFL_BUF_LINE;
275
5.59M
  do {
276
5.59M
    if (width == 4) {
277
1.28M
      const __m128i row = _mm_slli_epi16(_mm_loadl_epi64((__m128i *)input), 3);
278
1.28M
      _mm_storel_epi64((__m128i *)pred_buf_q3, row);
279
4.31M
    } else {
280
4.31M
      const __m128i row = _mm_slli_epi16(_mm_loadu_si128((__m128i *)input), 3);
281
4.31M
      _mm_storeu_si128((__m128i *)pred_buf_q3, row);
282
4.31M
      if (width >= 16) {
283
1.43M
        __m128i row_1 = _mm_loadu_si128(((__m128i *)input) + 1);
284
1.43M
        row_1 = _mm_slli_epi16(row_1, 3);
285
1.43M
        _mm_storeu_si128(((__m128i *)pred_buf_q3) + 1, row_1);
286
1.43M
        if (width == 32) {
287
0
          __m128i row_2 = _mm_loadu_si128(((__m128i *)input) + 2);
288
0
          row_2 = _mm_slli_epi16(row_2, 3);
289
0
          _mm_storeu_si128(((__m128i *)pred_buf_q3) + 2, row_2);
290
0
          __m128i row_3 = _mm_loadu_si128(((__m128i *)input) + 3);
291
0
          row_3 = _mm_slli_epi16(row_3, 3);
292
0
          _mm_storeu_si128(((__m128i *)pred_buf_q3) + 3, row_3);
293
0
        }
294
1.43M
      }
295
4.31M
    }
296
5.59M
    input += input_stride;
297
5.59M
    pred_buf_q3 += CFL_BUF_LINE;
298
5.59M
  } while (pred_buf_q3 < end);
299
627k
}
300
#endif  // CONFIG_AV1_HIGHBITDEPTH
301
302
CFL_GET_SUBSAMPLE_FUNCTION(ssse3)
303
304
static INLINE __m128i predict_unclipped(const __m128i *input, __m128i alpha_q12,
305
22.4M
                                        __m128i alpha_sign, __m128i dc_q0) {
306
22.4M
  __m128i ac_q3 = _mm_loadu_si128(input);
307
22.4M
  __m128i ac_sign = _mm_sign_epi16(alpha_sign, ac_q3);
308
22.4M
  __m128i scaled_luma_q0 = _mm_mulhrs_epi16(_mm_abs_epi16(ac_q3), alpha_q12);
309
22.4M
  scaled_luma_q0 = _mm_sign_epi16(scaled_luma_q0, ac_sign);
310
22.4M
  return _mm_add_epi16(scaled_luma_q0, dc_q0);
311
22.4M
}
312
313
static INLINE void cfl_predict_lbd_ssse3(const int16_t *pred_buf_q3,
314
                                         uint8_t *dst, int dst_stride,
315
1.27M
                                         int alpha_q3, int width, int height) {
316
1.27M
  const __m128i alpha_sign = _mm_set1_epi16(alpha_q3);
317
1.27M
  const __m128i alpha_q12 = _mm_slli_epi16(_mm_abs_epi16(alpha_sign), 9);
318
1.27M
  const __m128i dc_q0 = _mm_set1_epi16(*dst);
319
1.27M
  __m128i *row = (__m128i *)pred_buf_q3;
320
1.27M
  const __m128i *row_end = row + height * CFL_BUF_LINE_I128;
321
11.3M
  do {
322
11.3M
    __m128i res = predict_unclipped(row, alpha_q12, alpha_sign, dc_q0);
323
11.3M
    if (width < 16) {
324
7.96M
      res = _mm_packus_epi16(res, res);
325
7.96M
      if (width == 4)
326
3.97M
        _mm_storeh_epi32((__m128i *)dst, res);
327
3.99M
      else
328
3.99M
        _mm_storel_epi64((__m128i *)dst, res);
329
7.96M
    } else {
330
3.43M
      __m128i next = predict_unclipped(row + 1, alpha_q12, alpha_sign, dc_q0);
331
3.43M
      res = _mm_packus_epi16(res, next);
332
3.43M
      _mm_storeu_si128((__m128i *)dst, res);
333
3.43M
      if (width == 32) {
334
0
        res = predict_unclipped(row + 2, alpha_q12, alpha_sign, dc_q0);
335
0
        next = predict_unclipped(row + 3, alpha_q12, alpha_sign, dc_q0);
336
0
        res = _mm_packus_epi16(res, next);
337
0
        _mm_storeu_si128((__m128i *)(dst + 16), res);
338
0
      }
339
3.43M
    }
340
11.3M
    dst += dst_stride;
341
11.3M
  } while ((row += CFL_BUF_LINE_I128) < row_end);
342
1.27M
}
343
344
CFL_PREDICT_FN(ssse3, lbd)
345
346
#if CONFIG_AV1_HIGHBITDEPTH
347
785k
static INLINE __m128i highbd_max_epi16(int bd) {
348
785k
  const __m128i neg_one = _mm_set1_epi16(-1);
349
  // (1 << bd) - 1 => -(-1 << bd) -1 => -1 - (-1 << bd) => -1 ^ (-1 << bd)
350
785k
  return _mm_xor_si128(_mm_slli_epi16(neg_one, bd), neg_one);
351
785k
}
352
353
7.64M
static INLINE __m128i highbd_clamp_epi16(__m128i u, __m128i zero, __m128i max) {
354
7.64M
  return _mm_max_epi16(_mm_min_epi16(u, max), zero);
355
7.64M
}
356
357
static INLINE void cfl_predict_hbd_ssse3(const int16_t *pred_buf_q3,
358
                                         uint16_t *dst, int dst_stride,
359
                                         int alpha_q3, int bd, int width,
360
785k
                                         int height) {
361
785k
  const __m128i alpha_sign = _mm_set1_epi16(alpha_q3);
362
785k
  const __m128i alpha_q12 = _mm_slli_epi16(_mm_abs_epi16(alpha_sign), 9);
363
785k
  const __m128i dc_q0 = _mm_set1_epi16(*dst);
364
785k
  const __m128i max = highbd_max_epi16(bd);
365
785k
  const __m128i zeros = _mm_setzero_si128();
366
785k
  __m128i *row = (__m128i *)pred_buf_q3;
367
785k
  const __m128i *row_end = row + height * CFL_BUF_LINE_I128;
368
7.64M
  do {
369
7.64M
    __m128i res = predict_unclipped(row, alpha_q12, alpha_sign, dc_q0);
370
7.64M
    res = highbd_clamp_epi16(res, zeros, max);
371
7.64M
    if (width == 4) {
372
2.71M
      _mm_storel_epi64((__m128i *)dst, res);
373
4.92M
    } else {
374
4.92M
      _mm_storeu_si128((__m128i *)dst, res);
375
4.92M
    }
376
7.64M
    if (width >= 16) {
377
0
      const __m128i res_1 =
378
0
          predict_unclipped(row + 1, alpha_q12, alpha_sign, dc_q0);
379
0
      _mm_storeu_si128(((__m128i *)dst) + 1,
380
0
                       highbd_clamp_epi16(res_1, zeros, max));
381
0
    }
382
7.64M
    if (width == 32) {
383
0
      const __m128i res_2 =
384
0
          predict_unclipped(row + 2, alpha_q12, alpha_sign, dc_q0);
385
0
      _mm_storeu_si128((__m128i *)(dst + 16),
386
0
                       highbd_clamp_epi16(res_2, zeros, max));
387
0
      const __m128i res_3 =
388
0
          predict_unclipped(row + 3, alpha_q12, alpha_sign, dc_q0);
389
0
      _mm_storeu_si128((__m128i *)(dst + 24),
390
0
                       highbd_clamp_epi16(res_3, zeros, max));
391
0
    }
392
7.64M
    dst += dst_stride;
393
7.64M
  } while ((row += CFL_BUF_LINE_I128) < row_end);
394
785k
}
395
396
CFL_PREDICT_FN(ssse3, hbd)
397
#endif  // CONFIG_AV1_HIGHBITDEPTH