Coverage Report

Created: 2025-09-08 07:52

/src/libavif/third_party/libyuv/source/scale.c
Line
Count
Source (jump to first uncovered line)
1
/*
2
 *  Copyright 2011 The LibYuv Project Authors. All rights reserved.
3
 *
4
 *  Use of this source code is governed by a BSD-style license
5
 *  that can be found in the LICENSE file in the root of the source
6
 *  tree. An additional intellectual property rights grant can be found
7
 *  in the file PATENTS. All contributing project authors may
8
 *  be found in the AUTHORS file in the root of the source tree.
9
 */
10
11
#include "libyuv/scale.h"
12
13
#include <assert.h>
14
#include <string.h>
15
16
#include "libyuv/planar_functions.h"  // For CopyPlane
17
#include "libyuv/row.h"
18
#include "libyuv/scale_row.h"
19
20
3.50k
static __inline int Abs(int v) {
21
3.50k
  return v >= 0 ? v : -v;
22
3.50k
}
23
24
255
#define CENTERSTART(dx, s) (dx < 0) ? -((-dx >> 1) + s) : ((dx >> 1) + s)
25
26
231k
#define MIN1(x) ((x) < 1 ? 1 : (x))
27
28
48.9k
static __inline uint32_t SumPixels(int iboxwidth, const uint16_t* src_ptr) {
29
48.9k
  uint32_t sum = 0u;
30
48.9k
  int x;
31
48.9k
  assert(iboxwidth > 0);
32
339k
  for (x = 0; x < iboxwidth; ++x) {
33
290k
    sum += src_ptr[x];
34
290k
  }
35
48.9k
  return sum;
36
48.9k
}
37
38
175k
static __inline uint32_t SumPixels_16(int iboxwidth, const uint32_t* src_ptr) {
39
175k
  uint32_t sum = 0u;
40
175k
  int x;
41
175k
  assert(iboxwidth > 0);
42
666k
  for (x = 0; x < iboxwidth; ++x) {
43
491k
    sum += src_ptr[x];
44
491k
  }
45
175k
  return sum;
46
175k
}
47
48
static void ScaleAddCols2_C(int dst_width,
49
                            int boxheight,
50
                            int x,
51
                            int dx,
52
                            const uint16_t* src_ptr,
53
1.70k
                            uint8_t* dst_ptr) {
54
1.70k
  int i;
55
1.70k
  int scaletbl[2];
56
1.70k
  int minboxwidth = dx >> 16;
57
1.70k
  int boxwidth;
58
1.70k
  scaletbl[0] = 65536 / (MIN1(minboxwidth) * boxheight);
59
1.70k
  scaletbl[1] = 65536 / (MIN1(minboxwidth + 1) * boxheight);
60
48.2k
  for (i = 0; i < dst_width; ++i) {
61
46.5k
    int ix = x >> 16;
62
46.5k
    x += dx;
63
46.5k
    boxwidth = MIN1((x >> 16) - ix);
64
46.5k
    int scaletbl_index = boxwidth - minboxwidth;
65
46.5k
    assert((scaletbl_index == 0) || (scaletbl_index == 1));
66
46.5k
    *dst_ptr++ = (uint8_t)(SumPixels(boxwidth, src_ptr + ix) *
67
46.5k
                               scaletbl[scaletbl_index] >>
68
46.5k
                           16);
69
46.5k
  }
70
1.70k
}
71
72
static void ScaleAddCols2_16_C(int dst_width,
73
                               int boxheight,
74
                               int x,
75
                               int dx,
76
                               const uint32_t* src_ptr,
77
2.40k
                               uint16_t* dst_ptr) {
78
2.40k
  int i;
79
2.40k
  int scaletbl[2];
80
2.40k
  int minboxwidth = dx >> 16;
81
2.40k
  int boxwidth;
82
2.40k
  scaletbl[0] = 65536 / (MIN1(minboxwidth) * boxheight);
83
2.40k
  scaletbl[1] = 65536 / (MIN1(minboxwidth + 1) * boxheight);
84
172k
  for (i = 0; i < dst_width; ++i) {
85
170k
    int ix = x >> 16;
86
170k
    x += dx;
87
170k
    boxwidth = MIN1((x >> 16) - ix);
88
170k
    int scaletbl_index = boxwidth - minboxwidth;
89
170k
    assert((scaletbl_index == 0) || (scaletbl_index == 1));
90
170k
    *dst_ptr++ =
91
170k
        SumPixels_16(boxwidth, src_ptr + ix) * scaletbl[scaletbl_index] >> 16;
92
170k
  }
93
2.40k
}
94
95
static void ScaleAddCols0_C(int dst_width,
96
                            int boxheight,
97
                            int x,
98
                            int dx,
99
                            const uint16_t* src_ptr,
100
0
                            uint8_t* dst_ptr) {
101
0
  int scaleval = 65536 / boxheight;
102
0
  int i;
103
0
  (void)dx;
104
0
  src_ptr += (x >> 16);
105
0
  for (i = 0; i < dst_width; ++i) {
106
0
    *dst_ptr++ = (uint8_t)(src_ptr[i] * scaleval >> 16);
107
0
  }
108
0
}
109
110
static void ScaleAddCols1_C(int dst_width,
111
                            int boxheight,
112
                            int x,
113
                            int dx,
114
                            const uint16_t* src_ptr,
115
934
                            uint8_t* dst_ptr) {
116
934
  int boxwidth = MIN1(dx >> 16);
117
934
  int scaleval = 65536 / (boxwidth * boxheight);
118
934
  int i;
119
934
  x >>= 16;
120
3.33k
  for (i = 0; i < dst_width; ++i) {
121
2.40k
    *dst_ptr++ = (uint8_t)(SumPixels(boxwidth, src_ptr + x) * scaleval >> 16);
122
2.40k
    x += boxwidth;
123
2.40k
  }
124
934
}
125
126
static void ScaleAddCols1_16_C(int dst_width,
127
                               int boxheight,
128
                               int x,
129
                               int dx,
130
                               const uint32_t* src_ptr,
131
417
                               uint16_t* dst_ptr) {
132
417
  int boxwidth = MIN1(dx >> 16);
133
417
  int scaleval = 65536 / (boxwidth * boxheight);
134
417
  int i;
135
5.45k
  for (i = 0; i < dst_width; ++i) {
136
5.03k
    *dst_ptr++ = SumPixels_16(boxwidth, src_ptr + x) * scaleval >> 16;
137
5.03k
    x += boxwidth;
138
5.03k
  }
139
417
}
140
141
// Scale plane down to any dimensions, with interpolation.
142
// (boxfilter).
143
//
144
// Same method as SimpleScale, which is fixed point, outputting
145
// one pixel of destination using fixed point (16.16) to step
146
// through source, sampling a box of pixel with simple
147
// averaging.
148
static int ScalePlaneBox(int src_width,
149
                         int src_height,
150
                         int dst_width,
151
                         int dst_height,
152
                         int src_stride,
153
                         int dst_stride,
154
                         const uint8_t* src_ptr,
155
109
                         uint8_t* dst_ptr) {
156
109
  int j, k;
157
  // Initial source x/y coordinate and step values as 16.16 fixed point.
158
109
  int x = 0;
159
109
  int y = 0;
160
109
  int dx = 0;
161
109
  int dy = 0;
162
109
  const int max_y = (src_height << 16);
163
109
  ScaleSlope(src_width, src_height, dst_width, dst_height, kFilterBox, &x, &y,
164
109
             &dx, &dy);
165
109
  src_width = Abs(src_width);
166
109
  {
167
    // Allocate a row buffer of uint16_t.
168
109
    align_buffer_64(row16, src_width * 2);
169
109
    if (!row16)
170
0
      return 1;
171
109
    void (*ScaleAddCols)(int dst_width, int boxheight, int x, int dx,
172
109
                         const uint16_t* src_ptr, uint8_t* dst_ptr) =
173
109
        (dx & 0xffff) ? ScaleAddCols2_C
174
109
                      : ((dx != 0x10000) ? ScaleAddCols1_C : ScaleAddCols0_C);
175
109
    void (*ScaleAddRow)(const uint8_t* src_ptr, uint16_t* dst_ptr,
176
109
                        int src_width) = ScaleAddRow_C;
177
178
2.75k
    for (j = 0; j < dst_height; ++j) {
179
2.64k
      int boxheight;
180
2.64k
      int iy = y >> 16;
181
2.64k
      const uint8_t* src = src_ptr + iy * (int64_t)src_stride;
182
2.64k
      y += dy;
183
2.64k
      if (y > max_y) {
184
0
        y = max_y;
185
0
      }
186
2.64k
      boxheight = MIN1((y >> 16) - iy);
187
2.64k
      memset(row16, 0, src_width * 2);
188
25.9k
      for (k = 0; k < boxheight; ++k) {
189
23.2k
        ScaleAddRow(src, (uint16_t*)(row16), src_width);
190
23.2k
        src += src_stride;
191
23.2k
      }
192
2.64k
      ScaleAddCols(dst_width, boxheight, x, dx, (uint16_t*)(row16), dst_ptr);
193
2.64k
      dst_ptr += dst_stride;
194
2.64k
    }
195
109
    free_aligned_buffer_64(row16);
196
109
  }
197
0
  return 0;
198
109
}
199
200
static int ScalePlaneBox_16(int src_width,
201
                            int src_height,
202
                            int dst_width,
203
                            int dst_height,
204
                            int src_stride,
205
                            int dst_stride,
206
                            const uint16_t* src_ptr,
207
87
                            uint16_t* dst_ptr) {
208
87
  int j, k;
209
  // Initial source x/y coordinate and step values as 16.16 fixed point.
210
87
  int x = 0;
211
87
  int y = 0;
212
87
  int dx = 0;
213
87
  int dy = 0;
214
87
  const int max_y = (src_height << 16);
215
87
  ScaleSlope(src_width, src_height, dst_width, dst_height, kFilterBox, &x, &y,
216
87
             &dx, &dy);
217
87
  src_width = Abs(src_width);
218
87
  {
219
    // Allocate a row buffer of uint32_t.
220
87
    align_buffer_64(row32, src_width * 4);
221
87
    if (!row32)
222
0
      return 1;
223
87
    void (*ScaleAddCols)(int dst_width, int boxheight, int x, int dx,
224
87
                         const uint32_t* src_ptr, uint16_t* dst_ptr) =
225
87
        (dx & 0xffff) ? ScaleAddCols2_16_C : ScaleAddCols1_16_C;
226
87
    void (*ScaleAddRow)(const uint16_t* src_ptr, uint32_t* dst_ptr,
227
87
                        int src_width) = ScaleAddRow_16_C;
228
229
#if defined(HAS_SCALEADDROW_16_SSE2)
230
    if (TestCpuFlag(kCpuHasSSE2) && IS_ALIGNED(src_width, 16)) {
231
      ScaleAddRow = ScaleAddRow_16_SSE2;
232
    }
233
#endif
234
235
2.90k
    for (j = 0; j < dst_height; ++j) {
236
2.81k
      int boxheight;
237
2.81k
      int iy = y >> 16;
238
2.81k
      const uint16_t* src = src_ptr + iy * (int64_t)src_stride;
239
2.81k
      y += dy;
240
2.81k
      if (y > max_y) {
241
0
        y = max_y;
242
0
      }
243
2.81k
      boxheight = MIN1((y >> 16) - iy);
244
2.81k
      memset(row32, 0, src_width * 4);
245
25.1k
      for (k = 0; k < boxheight; ++k) {
246
22.3k
        ScaleAddRow(src, (uint32_t*)(row32), src_width);
247
22.3k
        src += src_stride;
248
22.3k
      }
249
2.81k
      ScaleAddCols(dst_width, boxheight, x, dx, (uint32_t*)(row32), dst_ptr);
250
2.81k
      dst_ptr += dst_stride;
251
2.81k
    }
252
87
    free_aligned_buffer_64(row32);
253
87
  }
254
0
  return 0;
255
87
}
256
257
// Scale plane down with bilinear interpolation.
258
static int ScalePlaneBilinearDown(int src_width,
259
                                  int src_height,
260
                                  int dst_width,
261
                                  int dst_height,
262
                                  int src_stride,
263
                                  int dst_stride,
264
                                  const uint8_t* src_ptr,
265
                                  uint8_t* dst_ptr,
266
459
                                  enum FilterMode filtering) {
267
  // Initial source x/y coordinate and step values as 16.16 fixed point.
268
459
  int x = 0;
269
459
  int y = 0;
270
459
  int dx = 0;
271
459
  int dy = 0;
272
  // TODO(fbarchard): Consider not allocating row buffer for kFilterLinear.
273
  // Allocate a row buffer.
274
459
  align_buffer_64(row, src_width);
275
459
  if (!row)
276
0
    return 1;
277
278
459
  const int max_y = (src_height - 1) << 16;
279
459
  int j;
280
459
  void (*ScaleFilterCols)(uint8_t* dst_ptr, const uint8_t* src_ptr,
281
459
                          int dst_width, int x, int dx) =
282
459
      (src_width >= 32768) ? ScaleFilterCols64_C : ScaleFilterCols_C;
283
459
  void (*InterpolateRow)(uint8_t* dst_ptr, const uint8_t* src_ptr,
284
459
                         ptrdiff_t src_stride, int dst_width,
285
459
                         int source_y_fraction) = InterpolateRow_C;
286
459
  ScaleSlope(src_width, src_height, dst_width, dst_height, filtering, &x, &y,
287
459
             &dx, &dy);
288
459
  src_width = Abs(src_width);
289
290
459
  if (y > max_y) {
291
31
    y = max_y;
292
31
  }
293
294
41.6k
  for (j = 0; j < dst_height; ++j) {
295
41.2k
    int yi = y >> 16;
296
41.2k
    const uint8_t* src = src_ptr + yi * (int64_t)src_stride;
297
41.2k
    if (filtering == kFilterLinear) {
298
3.44k
      ScaleFilterCols(dst_ptr, src, dst_width, x, dx);
299
37.7k
    } else {
300
37.7k
      int yf = (y >> 8) & 255;
301
37.7k
      InterpolateRow(row, src, src_stride, src_width, yf);
302
37.7k
      ScaleFilterCols(dst_ptr, row, dst_width, x, dx);
303
37.7k
    }
304
41.2k
    dst_ptr += dst_stride;
305
41.2k
    y += dy;
306
41.2k
    if (y > max_y) {
307
495
      y = max_y;
308
495
    }
309
41.2k
  }
310
459
  free_aligned_buffer_64(row);
311
459
  return 0;
312
459
}
313
314
static int ScalePlaneBilinearDown_16(int src_width,
315
                                     int src_height,
316
                                     int dst_width,
317
                                     int dst_height,
318
                                     int src_stride,
319
                                     int dst_stride,
320
                                     const uint16_t* src_ptr,
321
                                     uint16_t* dst_ptr,
322
479
                                     enum FilterMode filtering) {
323
  // Initial source x/y coordinate and step values as 16.16 fixed point.
324
479
  int x = 0;
325
479
  int y = 0;
326
479
  int dx = 0;
327
479
  int dy = 0;
328
  // TODO(fbarchard): Consider not allocating row buffer for kFilterLinear.
329
  // Allocate a row buffer.
330
479
  align_buffer_64(row, src_width * 2);
331
479
  if (!row)
332
0
    return 1;
333
334
479
  const int max_y = (src_height - 1) << 16;
335
479
  int j;
336
479
  void (*ScaleFilterCols)(uint16_t* dst_ptr, const uint16_t* src_ptr,
337
479
                          int dst_width, int x, int dx) =
338
479
      (src_width >= 32768) ? ScaleFilterCols64_16_C : ScaleFilterCols_16_C;
339
479
  void (*InterpolateRow)(uint16_t* dst_ptr, const uint16_t* src_ptr,
340
479
                         ptrdiff_t src_stride, int dst_width,
341
479
                         int source_y_fraction) = InterpolateRow_16_C;
342
479
  ScaleSlope(src_width, src_height, dst_width, dst_height, filtering, &x, &y,
343
479
             &dx, &dy);
344
479
  src_width = Abs(src_width);
345
346
479
  if (y > max_y) {
347
17
    y = max_y;
348
17
  }
349
350
49.0k
  for (j = 0; j < dst_height; ++j) {
351
48.5k
    int yi = y >> 16;
352
48.5k
    const uint16_t* src = src_ptr + yi * (int64_t)src_stride;
353
48.5k
    if (filtering == kFilterLinear) {
354
648
      ScaleFilterCols(dst_ptr, src, dst_width, x, dx);
355
47.8k
    } else {
356
47.8k
      int yf = (y >> 8) & 255;
357
47.8k
      InterpolateRow((uint16_t*)row, src, src_stride, src_width, yf);
358
47.8k
      ScaleFilterCols(dst_ptr, (uint16_t*)row, dst_width, x, dx);
359
47.8k
    }
360
48.5k
    dst_ptr += dst_stride;
361
48.5k
    y += dy;
362
48.5k
    if (y > max_y) {
363
496
      y = max_y;
364
496
    }
365
48.5k
  }
366
479
  free_aligned_buffer_64(row);
367
479
  return 0;
368
479
}
369
370
// Scale up down with bilinear interpolation.
371
static int ScalePlaneBilinearUp(int src_width,
372
                                int src_height,
373
                                int dst_width,
374
                                int dst_height,
375
                                int src_stride,
376
                                int dst_stride,
377
                                const uint8_t* src_ptr,
378
                                uint8_t* dst_ptr,
379
646
                                enum FilterMode filtering) {
380
646
  int j;
381
  // Initial source x/y coordinate and step values as 16.16 fixed point.
382
646
  int x = 0;
383
646
  int y = 0;
384
646
  int dx = 0;
385
646
  int dy = 0;
386
646
  const int max_y = (src_height - 1) << 16;
387
646
  void (*InterpolateRow)(uint8_t* dst_ptr, const uint8_t* src_ptr,
388
646
                         ptrdiff_t src_stride, int dst_width,
389
646
                         int source_y_fraction) = InterpolateRow_C;
390
646
  void (*ScaleFilterCols)(uint8_t* dst_ptr, const uint8_t* src_ptr,
391
646
                          int dst_width, int x, int dx) =
392
646
      filtering ? ScaleFilterCols_C : ScaleCols_C;
393
646
  ScaleSlope(src_width, src_height, dst_width, dst_height, filtering, &x, &y,
394
646
             &dx, &dy);
395
646
  src_width = Abs(src_width);
396
397
646
  if (filtering && src_width >= 32768) {
398
0
    ScaleFilterCols = ScaleFilterCols64_C;
399
0
  }
400
646
  if (!filtering && src_width * 2 == dst_width && x < 0x8000) {
401
0
    ScaleFilterCols = ScaleColsUp2_C;
402
0
  }
403
404
646
  if (y > max_y) {
405
98
    y = max_y;
406
98
  }
407
646
  {
408
646
    int yi = y >> 16;
409
646
    const uint8_t* src = src_ptr + yi * (int64_t)src_stride;
410
411
    // Allocate 2 row buffers.
412
646
    const int row_size = (dst_width + 31) & ~31;
413
646
    align_buffer_64(row, row_size * 2);
414
646
    if (!row)
415
0
      return 1;
416
417
646
    uint8_t* rowptr = row;
418
646
    int rowstride = row_size;
419
646
    int lasty = yi;
420
421
646
    ScaleFilterCols(rowptr, src, dst_width, x, dx);
422
646
    if (src_height > 1) {
423
538
      src += src_stride;
424
538
    }
425
646
    ScaleFilterCols(rowptr + rowstride, src, dst_width, x, dx);
426
646
    if (src_height > 2) {
427
487
      src += src_stride;
428
487
    }
429
430
8.37M
    for (j = 0; j < dst_height; ++j) {
431
8.37M
      yi = y >> 16;
432
8.37M
      if (yi != lasty) {
433
111k
        if (y > max_y) {
434
0
          y = max_y;
435
0
          yi = y >> 16;
436
0
          src = src_ptr + yi * (int64_t)src_stride;
437
0
        }
438
111k
        if (yi != lasty) {
439
111k
          ScaleFilterCols(rowptr, src, dst_width, x, dx);
440
111k
          rowptr += rowstride;
441
111k
          rowstride = -rowstride;
442
111k
          lasty = yi;
443
111k
          if ((y + 65536) < max_y) {
444
111k
            src += src_stride;
445
111k
          }
446
111k
        }
447
111k
      }
448
8.37M
      if (filtering == kFilterLinear) {
449
564k
        InterpolateRow(dst_ptr, rowptr, 0, dst_width, 0);
450
7.81M
      } else {
451
7.81M
        int yf = (y >> 8) & 255;
452
7.81M
        InterpolateRow(dst_ptr, rowptr, rowstride, dst_width, yf);
453
7.81M
      }
454
8.37M
      dst_ptr += dst_stride;
455
8.37M
      y += dy;
456
8.37M
    }
457
646
    free_aligned_buffer_64(row);
458
646
  }
459
0
  return 0;
460
646
}
461
462
// Scale plane, horizontally up by 2 times.
463
// Uses linear filter horizontally, nearest vertically.
464
// This is an optimized version for scaling up a plane to 2 times of
465
// its original width, using linear interpolation.
466
// This is used to scale U and V planes of I422 to I444.
467
static void ScalePlaneUp2_Linear(int src_width,
468
                                 int src_height,
469
                                 int dst_width,
470
                                 int dst_height,
471
                                 int src_stride,
472
                                 int dst_stride,
473
                                 const uint8_t* src_ptr,
474
66
                                 uint8_t* dst_ptr) {
475
66
  void (*ScaleRowUp)(const uint8_t* src_ptr, uint8_t* dst_ptr, int dst_width) =
476
66
      ScaleRowUp2_Linear_Any_C;
477
66
  int i;
478
66
  int y;
479
66
  int dy;
480
481
66
  (void)src_width;
482
  // This function can only scale up by 2 times horizontally.
483
66
  assert(src_width == ((dst_width + 1) / 2));
484
485
66
  if (dst_height == 1) {
486
12
    ScaleRowUp(src_ptr + ((src_height - 1) / 2) * (int64_t)src_stride, dst_ptr,
487
12
               dst_width);
488
54
  } else {
489
54
    dy = FixedDiv(src_height - 1, dst_height - 1);
490
54
    y = (1 << 15) - 1;
491
266k
    for (i = 0; i < dst_height; ++i) {
492
266k
      ScaleRowUp(src_ptr + (y >> 16) * (int64_t)src_stride, dst_ptr, dst_width);
493
266k
      dst_ptr += dst_stride;
494
266k
      y += dy;
495
266k
    }
496
54
  }
497
66
}
498
499
// Scale plane, up by 2 times.
500
// This is an optimized version for scaling up a plane to 2 times of
501
// its original size, using bilinear interpolation.
502
// This is used to scale U and V planes of I420 to I444.
503
static void ScalePlaneUp2_Bilinear(int src_width,
504
                                   int src_height,
505
                                   int dst_width,
506
                                   int dst_height,
507
                                   int src_stride,
508
                                   int dst_stride,
509
                                   const uint8_t* src_ptr,
510
34
                                   uint8_t* dst_ptr) {
511
34
  void (*Scale2RowUp)(const uint8_t* src_ptr, ptrdiff_t src_stride,
512
34
                      uint8_t* dst_ptr, ptrdiff_t dst_stride, int dst_width) =
513
34
      ScaleRowUp2_Bilinear_Any_C;
514
34
  int x;
515
516
34
  (void)src_width;
517
  // This function can only scale up by 2 times.
518
34
  assert(src_width == ((dst_width + 1) / 2));
519
34
  assert(src_height == ((dst_height + 1) / 2));
520
521
34
  Scale2RowUp(src_ptr, 0, dst_ptr, 0, dst_width);
522
34
  dst_ptr += dst_stride;
523
306
  for (x = 0; x < src_height - 1; ++x) {
524
272
    Scale2RowUp(src_ptr, src_stride, dst_ptr, dst_stride, dst_width);
525
272
    src_ptr += src_stride;
526
    // TODO(fbarchard): Test performance of writing one row of destination at a
527
    // time.
528
272
    dst_ptr += 2 * dst_stride;
529
272
  }
530
34
  if (!(dst_height & 1)) {
531
13
    Scale2RowUp(src_ptr, 0, dst_ptr, 0, dst_width);
532
13
  }
533
34
}
534
535
// Scale at most 14 bit plane, horizontally up by 2 times.
536
// This is an optimized version for scaling up a plane to 2 times of
537
// its original width, using linear interpolation.
538
// stride is in count of uint16_t.
539
// This is used to scale U and V planes of I210 to I410 and I212 to I412.
540
static void ScalePlaneUp2_12_Linear(int src_width,
541
                                    int src_height,
542
                                    int dst_width,
543
                                    int dst_height,
544
                                    int src_stride,
545
                                    int dst_stride,
546
                                    const uint16_t* src_ptr,
547
38
                                    uint16_t* dst_ptr) {
548
38
  void (*ScaleRowUp)(const uint16_t* src_ptr, uint16_t* dst_ptr,
549
38
                     int dst_width) = ScaleRowUp2_Linear_16_Any_C;
550
38
  int i;
551
38
  int y;
552
38
  int dy;
553
554
38
  (void)src_width;
555
  // This function can only scale up by 2 times horizontally.
556
38
  assert(src_width == ((dst_width + 1) / 2));
557
558
38
  if (dst_height == 1) {
559
10
    ScaleRowUp(src_ptr + ((src_height - 1) / 2) * (int64_t)src_stride, dst_ptr,
560
10
               dst_width);
561
28
  } else {
562
28
    dy = FixedDiv(src_height - 1, dst_height - 1);
563
28
    y = (1 << 15) - 1;
564
280k
    for (i = 0; i < dst_height; ++i) {
565
280k
      ScaleRowUp(src_ptr + (y >> 16) * (int64_t)src_stride, dst_ptr, dst_width);
566
280k
      dst_ptr += dst_stride;
567
280k
      y += dy;
568
280k
    }
569
28
  }
570
38
}
571
572
// Scale at most 12 bit plane, up by 2 times.
573
// This is an optimized version for scaling up a plane to 2 times of
574
// its original size, using bilinear interpolation.
575
// stride is in count of uint16_t.
576
// This is used to scale U and V planes of I010 to I410 and I012 to I412.
577
static void ScalePlaneUp2_12_Bilinear(int src_width,
578
                                      int src_height,
579
                                      int dst_width,
580
                                      int dst_height,
581
                                      int src_stride,
582
                                      int dst_stride,
583
                                      const uint16_t* src_ptr,
584
29
                                      uint16_t* dst_ptr) {
585
29
  void (*Scale2RowUp)(const uint16_t* src_ptr, ptrdiff_t src_stride,
586
29
                      uint16_t* dst_ptr, ptrdiff_t dst_stride, int dst_width) =
587
29
      ScaleRowUp2_Bilinear_16_Any_C;
588
29
  int x;
589
590
29
  (void)src_width;
591
  // This function can only scale up by 2 times.
592
29
  assert(src_width == ((dst_width + 1) / 2));
593
29
  assert(src_height == ((dst_height + 1) / 2));
594
595
29
  Scale2RowUp(src_ptr, 0, dst_ptr, 0, dst_width);
596
29
  dst_ptr += dst_stride;
597
106
  for (x = 0; x < src_height - 1; ++x) {
598
77
    Scale2RowUp(src_ptr, src_stride, dst_ptr, dst_stride, dst_width);
599
77
    src_ptr += src_stride;
600
77
    dst_ptr += 2 * dst_stride;
601
77
  }
602
29
  if (!(dst_height & 1)) {
603
20
    Scale2RowUp(src_ptr, 0, dst_ptr, 0, dst_width);
604
20
  }
605
29
}
606
607
static void ScalePlaneUp2_16_Linear(int src_width,
608
                                    int src_height,
609
                                    int dst_width,
610
                                    int dst_height,
611
                                    int src_stride,
612
                                    int dst_stride,
613
                                    const uint16_t* src_ptr,
614
0
                                    uint16_t* dst_ptr) {
615
0
  void (*ScaleRowUp)(const uint16_t* src_ptr, uint16_t* dst_ptr,
616
0
                     int dst_width) = ScaleRowUp2_Linear_16_Any_C;
617
0
  int i;
618
0
  int y;
619
0
  int dy;
620
621
0
  (void)src_width;
622
  // This function can only scale up by 2 times horizontally.
623
0
  assert(src_width == ((dst_width + 1) / 2));
624
625
0
  if (dst_height == 1) {
626
0
    ScaleRowUp(src_ptr + ((src_height - 1) / 2) * (int64_t)src_stride, dst_ptr,
627
0
               dst_width);
628
0
  } else {
629
0
    dy = FixedDiv(src_height - 1, dst_height - 1);
630
0
    y = (1 << 15) - 1;
631
0
    for (i = 0; i < dst_height; ++i) {
632
0
      ScaleRowUp(src_ptr + (y >> 16) * (int64_t)src_stride, dst_ptr, dst_width);
633
0
      dst_ptr += dst_stride;
634
0
      y += dy;
635
0
    }
636
0
  }
637
0
}
638
639
static void ScalePlaneUp2_16_Bilinear(int src_width,
640
                                      int src_height,
641
                                      int dst_width,
642
                                      int dst_height,
643
                                      int src_stride,
644
                                      int dst_stride,
645
                                      const uint16_t* src_ptr,
646
0
                                      uint16_t* dst_ptr) {
647
0
  void (*Scale2RowUp)(const uint16_t* src_ptr, ptrdiff_t src_stride,
648
0
                      uint16_t* dst_ptr, ptrdiff_t dst_stride, int dst_width) =
649
0
      ScaleRowUp2_Bilinear_16_Any_C;
650
0
  int x;
651
652
0
  (void)src_width;
653
  // This function can only scale up by 2 times.
654
0
  assert(src_width == ((dst_width + 1) / 2));
655
0
  assert(src_height == ((dst_height + 1) / 2));
656
657
0
  Scale2RowUp(src_ptr, 0, dst_ptr, 0, dst_width);
658
0
  dst_ptr += dst_stride;
659
0
  for (x = 0; x < src_height - 1; ++x) {
660
0
    Scale2RowUp(src_ptr, src_stride, dst_ptr, dst_stride, dst_width);
661
0
    src_ptr += src_stride;
662
0
    dst_ptr += 2 * dst_stride;
663
0
  }
664
0
  if (!(dst_height & 1)) {
665
0
    Scale2RowUp(src_ptr, 0, dst_ptr, 0, dst_width);
666
0
  }
667
0
}
668
669
static int ScalePlaneBilinearUp_16(int src_width,
670
                                   int src_height,
671
                                   int dst_width,
672
                                   int dst_height,
673
                                   int src_stride,
674
                                   int dst_stride,
675
                                   const uint16_t* src_ptr,
676
                                   uint16_t* dst_ptr,
677
920
                                   enum FilterMode filtering) {
678
920
  int j;
679
  // Initial source x/y coordinate and step values as 16.16 fixed point.
680
920
  int x = 0;
681
920
  int y = 0;
682
920
  int dx = 0;
683
920
  int dy = 0;
684
920
  const int max_y = (src_height - 1) << 16;
685
920
  void (*InterpolateRow)(uint16_t* dst_ptr, const uint16_t* src_ptr,
686
920
                         ptrdiff_t src_stride, int dst_width,
687
920
                         int source_y_fraction) = InterpolateRow_16_C;
688
920
  void (*ScaleFilterCols)(uint16_t* dst_ptr, const uint16_t* src_ptr,
689
920
                          int dst_width, int x, int dx) =
690
920
      filtering ? ScaleFilterCols_16_C : ScaleCols_16_C;
691
920
  ScaleSlope(src_width, src_height, dst_width, dst_height, filtering, &x, &y,
692
920
             &dx, &dy);
693
920
  src_width = Abs(src_width);
694
695
920
  if (filtering && src_width >= 32768) {
696
0
    ScaleFilterCols = ScaleFilterCols64_16_C;
697
0
  }
698
920
  if (!filtering && src_width * 2 == dst_width && x < 0x8000) {
699
0
    ScaleFilterCols = ScaleColsUp2_16_C;
700
0
  }
701
920
  if (y > max_y) {
702
233
    y = max_y;
703
233
  }
704
920
  {
705
920
    int yi = y >> 16;
706
920
    const uint16_t* src = src_ptr + yi * (int64_t)src_stride;
707
708
    // Allocate 2 row buffers.
709
920
    const int row_size = (dst_width + 31) & ~31;
710
920
    align_buffer_64(row, row_size * 4);
711
920
    int rowstride = row_size;
712
920
    int lasty = yi;
713
920
    uint16_t* rowptr = (uint16_t*)row;
714
920
    if (!row)
715
0
      return 1;
716
717
920
    ScaleFilterCols(rowptr, src, dst_width, x, dx);
718
920
    if (src_height > 1) {
719
677
      src += src_stride;
720
677
    }
721
920
    ScaleFilterCols(rowptr + rowstride, src, dst_width, x, dx);
722
920
    if (src_height > 2) {
723
548
      src += src_stride;
724
548
    }
725
726
8.08M
    for (j = 0; j < dst_height; ++j) {
727
8.08M
      yi = y >> 16;
728
8.08M
      if (yi != lasty) {
729
118k
        if (y > max_y) {
730
0
          y = max_y;
731
0
          yi = y >> 16;
732
0
          src = src_ptr + yi * (int64_t)src_stride;
733
0
        }
734
118k
        if (yi != lasty) {
735
118k
          ScaleFilterCols(rowptr, src, dst_width, x, dx);
736
118k
          rowptr += rowstride;
737
118k
          rowstride = -rowstride;
738
118k
          lasty = yi;
739
118k
          if ((y + 65536) < max_y) {
740
117k
            src += src_stride;
741
117k
          }
742
118k
        }
743
118k
      }
744
8.08M
      if (filtering == kFilterLinear) {
745
850k
        InterpolateRow(dst_ptr, rowptr, 0, dst_width, 0);
746
7.23M
      } else {
747
7.23M
        int yf = (y >> 8) & 255;
748
7.23M
        InterpolateRow(dst_ptr, rowptr, rowstride, dst_width, yf);
749
7.23M
      }
750
8.08M
      dst_ptr += dst_stride;
751
8.08M
      y += dy;
752
8.08M
    }
753
920
    free_aligned_buffer_64(row);
754
920
  }
755
0
  return 0;
756
920
}
757
758
// Scale Plane to/from any dimensions, without interpolation.
759
// Fixed point math is used for performance: The upper 16 bits
760
// of x and dx is the integer part of the source position and
761
// the lower 16 bits are the fixed decimal part.
762
763
static void ScalePlaneSimple(int src_width,
764
                             int src_height,
765
                             int dst_width,
766
                             int dst_height,
767
                             int src_stride,
768
                             int dst_stride,
769
                             const uint8_t* src_ptr,
770
437
                             uint8_t* dst_ptr) {
771
437
  int i;
772
437
  void (*ScaleCols)(uint8_t* dst_ptr, const uint8_t* src_ptr, int dst_width,
773
437
                    int x, int dx) = ScaleCols_C;
774
  // Initial source x/y coordinate and step values as 16.16 fixed point.
775
437
  int x = 0;
776
437
  int y = 0;
777
437
  int dx = 0;
778
437
  int dy = 0;
779
437
  ScaleSlope(src_width, src_height, dst_width, dst_height, kFilterNone, &x, &y,
780
437
             &dx, &dy);
781
437
  src_width = Abs(src_width);
782
783
437
  if (src_width * 2 == dst_width && x < 0x8000) {
784
77
    ScaleCols = ScaleColsUp2_C;
785
77
  }
786
787
386k
  for (i = 0; i < dst_height; ++i) {
788
385k
    ScaleCols(dst_ptr, src_ptr + (y >> 16) * (int64_t)src_stride, dst_width, x,
789
385k
              dx);
790
385k
    dst_ptr += dst_stride;
791
385k
    y += dy;
792
385k
  }
793
437
}
794
795
static void ScalePlaneSimple_16(int src_width,
796
                                int src_height,
797
                                int dst_width,
798
                                int dst_height,
799
                                int src_stride,
800
                                int dst_stride,
801
                                const uint16_t* src_ptr,
802
364
                                uint16_t* dst_ptr) {
803
364
  int i;
804
364
  void (*ScaleCols)(uint16_t* dst_ptr, const uint16_t* src_ptr, int dst_width,
805
364
                    int x, int dx) = ScaleCols_16_C;
806
  // Initial source x/y coordinate and step values as 16.16 fixed point.
807
364
  int x = 0;
808
364
  int y = 0;
809
364
  int dx = 0;
810
364
  int dy = 0;
811
364
  ScaleSlope(src_width, src_height, dst_width, dst_height, kFilterNone, &x, &y,
812
364
             &dx, &dy);
813
364
  src_width = Abs(src_width);
814
815
364
  if (src_width * 2 == dst_width && x < 0x8000) {
816
113
    ScaleCols = ScaleColsUp2_16_C;
817
113
  }
818
819
984k
  for (i = 0; i < dst_height; ++i) {
820
984k
    ScaleCols(dst_ptr, src_ptr + (y >> 16) * (int64_t)src_stride, dst_width, x,
821
984k
              dx);
822
984k
    dst_ptr += dst_stride;
823
984k
    y += dy;
824
984k
  }
825
364
}
826
827
// Scale a plane.
828
// This function dispatches to a specialized scaler based on scale factor.
829
int ScalePlane(const uint8_t* src,
830
               int src_stride,
831
               int src_width,
832
               int src_height,
833
               uint8_t* dst,
834
               int dst_stride,
835
               int dst_width,
836
               int dst_height,
837
2.03k
               enum FilterMode filtering) {
838
  // Simplify filtering when possible.
839
2.03k
  filtering = ScaleFilterReduce(src_width, src_height, dst_width, dst_height,
840
2.03k
                                filtering);
841
842
  // Negative height means invert the image.
843
2.03k
  if (src_height < 0) {
844
0
    src_height = -src_height;
845
0
    src = src + (src_height - 1) * (int64_t)src_stride;
846
0
    src_stride = -src_stride;
847
0
  }
848
  // Use specialized scales to improve performance for common resolutions.
849
  // For example, all the 1/2 scalings will use ScalePlaneDown2()
850
2.03k
  if (dst_width == src_width && dst_height == src_height) {
851
    // Straight copy.
852
12
    CopyPlane(src, src_stride, dst, dst_stride, dst_width, dst_height);
853
12
    return 0;
854
12
  }
855
2.01k
  if (dst_width == src_width && filtering != kFilterBox) {
856
268
    int dy = 0;
857
268
    int y = 0;
858
    // When scaling down, use the center 2 rows to filter.
859
    // When scaling up, last row of destination uses the last 2 source rows.
860
268
    if (dst_height <= src_height) {
861
170
      dy = FixedDiv(src_height, dst_height);
862
170
      y = CENTERSTART(dy, -32768);  // Subtract 0.5 (32768) to center filter.
863
170
    } else if (src_height > 1 && dst_height > 1) {
864
46
      dy = FixedDiv1(src_height, dst_height);
865
46
    }
866
    // Arbitrary scale vertically, but unscaled horizontally.
867
268
    ScalePlaneVertical(src_height, dst_width, dst_height, src_stride,
868
268
                       dst_stride, src, dst, 0, y, dy, /*bpp=*/1, filtering);
869
268
    return 0;
870
268
  }
871
1.75k
  if (filtering == kFilterBox && dst_height * 2 < src_height) {
872
109
    return ScalePlaneBox(src_width, src_height, dst_width, dst_height,
873
109
                         src_stride, dst_stride, src, dst);
874
109
  }
875
1.64k
  if ((dst_width + 1) / 2 == src_width && filtering == kFilterLinear) {
876
66
    ScalePlaneUp2_Linear(src_width, src_height, dst_width, dst_height,
877
66
                         src_stride, dst_stride, src, dst);
878
66
    return 0;
879
66
  }
880
1.57k
  if ((dst_height + 1) / 2 == src_height && (dst_width + 1) / 2 == src_width &&
881
1.57k
      (filtering == kFilterBilinear || filtering == kFilterBox)) {
882
34
    ScalePlaneUp2_Bilinear(src_width, src_height, dst_width, dst_height,
883
34
                           src_stride, dst_stride, src, dst);
884
34
    return 0;
885
34
  }
886
1.54k
  if (filtering && dst_height > src_height) {
887
646
    return ScalePlaneBilinearUp(src_width, src_height, dst_width, dst_height,
888
646
                                src_stride, dst_stride, src, dst, filtering);
889
646
  }
890
896
  if (filtering) {
891
459
    return ScalePlaneBilinearDown(src_width, src_height, dst_width, dst_height,
892
459
                                  src_stride, dst_stride, src, dst, filtering);
893
459
  }
894
437
  ScalePlaneSimple(src_width, src_height, dst_width, dst_height, src_stride,
895
437
                   dst_stride, src, dst);
896
437
  return 0;
897
896
}
898
899
int ScalePlane_16(const uint16_t* src,
900
                  int src_stride,
901
                  int src_width,
902
                  int src_height,
903
                  uint16_t* dst,
904
                  int dst_stride,
905
                  int dst_width,
906
                  int dst_height,
907
2.01k
                  enum FilterMode filtering) {
908
  // Simplify filtering when possible.
909
2.01k
  filtering = ScaleFilterReduce(src_width, src_height, dst_width, dst_height,
910
2.01k
                                filtering);
911
912
  // Negative height means invert the image.
913
2.01k
  if (src_height < 0) {
914
0
    src_height = -src_height;
915
0
    src = src + (src_height - 1) * (int64_t)src_stride;
916
0
    src_stride = -src_stride;
917
0
  }
918
  // Use specialized scales to improve performance for common resolutions.
919
  // For example, all the 1/2 scalings will use ScalePlaneDown2()
920
2.01k
  if (dst_width == src_width && dst_height == src_height) {
921
    // Straight copy.
922
14
    CopyPlane_16(src, src_stride, dst, dst_stride, dst_width, dst_height);
923
14
    return 0;
924
14
  }
925
1.99k
  if (dst_width == src_width && filtering != kFilterBox) {
926
149
    int dy = 0;
927
149
    int y = 0;
928
    // When scaling down, use the center 2 rows to filter.
929
    // When scaling up, last row of destination uses the last 2 source rows.
930
149
    if (dst_height <= src_height) {
931
85
      dy = FixedDiv(src_height, dst_height);
932
85
      y = CENTERSTART(dy, -32768);  // Subtract 0.5 (32768) to center filter.
933
      // When scaling up, ensure the last row of destination uses the last
934
      // source. Avoid divide by zero for dst_height but will do no scaling
935
      // later.
936
85
    } else if (src_height > 1 && dst_height > 1) {
937
49
      dy = FixedDiv1(src_height, dst_height);
938
49
    }
939
    // Arbitrary scale vertically, but unscaled horizontally.
940
149
    ScalePlaneVertical_16(src_height, dst_width, dst_height, src_stride,
941
149
                          dst_stride, src, dst, 0, y, dy, /*bpp=*/1, filtering);
942
149
    return 0;
943
149
  }
944
1.85k
  if (filtering == kFilterBox && dst_height * 2 < src_height) {
945
87
    return ScalePlaneBox_16(src_width, src_height, dst_width, dst_height,
946
87
                            src_stride, dst_stride, src, dst);
947
87
  }
948
1.76k
  if ((dst_width + 1) / 2 == src_width && filtering == kFilterLinear) {
949
0
    ScalePlaneUp2_16_Linear(src_width, src_height, dst_width, dst_height,
950
0
                            src_stride, dst_stride, src, dst);
951
0
    return 0;
952
0
  }
953
1.76k
  if ((dst_height + 1) / 2 == src_height && (dst_width + 1) / 2 == src_width &&
954
1.76k
      (filtering == kFilterBilinear || filtering == kFilterBox)) {
955
0
    ScalePlaneUp2_16_Bilinear(src_width, src_height, dst_width, dst_height,
956
0
                              src_stride, dst_stride, src, dst);
957
0
    return 0;
958
0
  }
959
1.76k
  if (filtering && dst_height > src_height) {
960
920
    return ScalePlaneBilinearUp_16(src_width, src_height, dst_width, dst_height,
961
920
                                   src_stride, dst_stride, src, dst, filtering);
962
920
  }
963
843
  if (filtering) {
964
479
    return ScalePlaneBilinearDown_16(src_width, src_height, dst_width,
965
479
                                     dst_height, src_stride, dst_stride, src,
966
479
                                     dst, filtering);
967
479
  }
968
364
  ScalePlaneSimple_16(src_width, src_height, dst_width, dst_height, src_stride,
969
364
                      dst_stride, src, dst);
970
364
  return 0;
971
843
}
972
973
int ScalePlane_12(const uint16_t* src,
974
                  int src_stride,
975
                  int src_width,
976
                  int src_height,
977
                  uint16_t* dst,
978
                  int dst_stride,
979
                  int dst_width,
980
                  int dst_height,
981
2.08k
                  enum FilterMode filtering) {
982
  // Simplify filtering when possible.
983
2.08k
  filtering = ScaleFilterReduce(src_width, src_height, dst_width, dst_height,
984
2.08k
                                filtering);
985
986
  // Negative height means invert the image.
987
2.08k
  if (src_height < 0) {
988
0
    src_height = -src_height;
989
0
    src = src + (src_height - 1) * (int64_t)src_stride;
990
0
    src_stride = -src_stride;
991
0
  }
992
993
2.08k
  if ((dst_width + 1) / 2 == src_width && filtering == kFilterLinear) {
994
38
    ScalePlaneUp2_12_Linear(src_width, src_height, dst_width, dst_height,
995
38
                            src_stride, dst_stride, src, dst);
996
38
    return 0;
997
38
  }
998
2.04k
  if ((dst_height + 1) / 2 == src_height && (dst_width + 1) / 2 == src_width &&
999
2.04k
      (filtering == kFilterBilinear || filtering == kFilterBox)) {
1000
29
    ScalePlaneUp2_12_Bilinear(src_width, src_height, dst_width, dst_height,
1001
29
                              src_stride, dst_stride, src, dst);
1002
29
    return 0;
1003
29
  }
1004
1005
2.01k
  return ScalePlane_16(src, src_stride, src_width, src_height, dst, dst_stride,
1006
2.01k
                       dst_width, dst_height, filtering);
1007
2.04k
}