Coverage Report

Created: 2026-06-14 06:57

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/libavif/third_party/libyuv/source/scale.c
Line
Count
Source
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
2.84k
static __inline int Abs(int v) {
21
2.84k
  return v >= 0 ? v : -v;
22
2.84k
}
23
24
278
#define CENTERSTART(dx, s) (dx < 0) ? -((-dx >> 1) + s) : ((dx >> 1) + s)
25
26
215k
#define MIN1(x) ((x) < 1 ? 1 : (x))
27
28
77.3k
static __inline uint32_t SumPixels(int iboxwidth, const uint16_t* src_ptr) {
29
77.3k
  uint32_t sum = 0u;
30
77.3k
  int x;
31
77.3k
  assert(iboxwidth > 0);
32
330k
  for (x = 0; x < iboxwidth; ++x) {
33
253k
    sum += src_ptr[x];
34
253k
  }
35
77.3k
  return sum;
36
77.3k
}
37
38
141k
static __inline uint32_t SumPixels_16(int iboxwidth, const uint32_t* src_ptr) {
39
141k
  uint32_t sum = 0u;
40
141k
  int x;
41
141k
  assert(iboxwidth > 0);
42
741k
  for (x = 0; x < iboxwidth; ++x) {
43
600k
    sum += src_ptr[x];
44
600k
  }
45
141k
  return sum;
46
141k
}
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
2.36k
                            uint8_t* dst_ptr) {
54
2.36k
  int i;
55
2.36k
  int scaletbl[2];
56
2.36k
  int minboxwidth = dx >> 16;
57
2.36k
  int boxwidth;
58
2.36k
  scaletbl[0] = 65536 / (MIN1(minboxwidth) * boxheight);
59
2.36k
  scaletbl[1] = 65536 / (MIN1(minboxwidth + 1) * boxheight);
60
57.5k
  for (i = 0; i < dst_width; ++i) {
61
55.1k
    int ix = x >> 16;
62
55.1k
    x += dx;
63
55.1k
    boxwidth = MIN1((x >> 16) - ix);
64
55.1k
    int scaletbl_index = boxwidth - minboxwidth;
65
55.1k
    assert((scaletbl_index == 0) || (scaletbl_index == 1));
66
55.1k
    *dst_ptr++ = (uint8_t)(SumPixels(boxwidth, src_ptr + ix) *
67
55.1k
                               scaletbl[scaletbl_index] >>
68
55.1k
                           16);
69
55.1k
  }
70
2.36k
}
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
4.04k
                               uint16_t* dst_ptr) {
78
4.04k
  int i;
79
4.04k
  int scaletbl[2];
80
4.04k
  int minboxwidth = dx >> 16;
81
4.04k
  int boxwidth;
82
4.04k
  scaletbl[0] = 65536 / (MIN1(minboxwidth) * boxheight);
83
4.04k
  scaletbl[1] = 65536 / (MIN1(minboxwidth + 1) * boxheight);
84
142k
  for (i = 0; i < dst_width; ++i) {
85
138k
    int ix = x >> 16;
86
138k
    x += dx;
87
138k
    boxwidth = MIN1((x >> 16) - ix);
88
138k
    int scaletbl_index = boxwidth - minboxwidth;
89
138k
    assert((scaletbl_index == 0) || (scaletbl_index == 1));
90
138k
    *dst_ptr++ =
91
138k
        SumPixels_16(boxwidth, src_ptr + ix) * scaletbl[scaletbl_index] >> 16;
92
138k
  }
93
4.04k
}
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
863
                            uint8_t* dst_ptr) {
116
863
  int boxwidth = MIN1(dx >> 16);
117
863
  int scaleval = 65536 / (boxwidth * boxheight);
118
863
  int i;
119
863
  x >>= 16;
120
23.1k
  for (i = 0; i < dst_width; ++i) {
121
22.2k
    *dst_ptr++ = (uint8_t)(SumPixels(boxwidth, src_ptr + x) * scaleval >> 16);
122
22.2k
    x += boxwidth;
123
22.2k
  }
124
863
}
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
327
                               uint16_t* dst_ptr) {
132
327
  int boxwidth = MIN1(dx >> 16);
133
327
  int scaleval = 65536 / (boxwidth * boxheight);
134
327
  int i;
135
3.31k
  for (i = 0; i < dst_width; ++i) {
136
2.98k
    *dst_ptr++ = SumPixels_16(boxwidth, src_ptr + x) * scaleval >> 16;
137
2.98k
    x += boxwidth;
138
2.98k
  }
139
327
}
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
100
                         uint8_t* dst_ptr) {
156
100
  int j, k;
157
  // Initial source x/y coordinate and step values as 16.16 fixed point.
158
100
  int x = 0;
159
100
  int y = 0;
160
100
  int dx = 0;
161
100
  int dy = 0;
162
100
  const int max_y = (src_height << 16);
163
100
  ScaleSlope(src_width, src_height, dst_width, dst_height, kFilterBox, &x, &y,
164
100
             &dx, &dy);
165
100
  src_width = Abs(src_width);
166
100
  {
167
    // Allocate a row buffer of uint16_t.
168
100
    align_buffer_64(row16, src_width * 2);
169
100
    if (!row16)
170
0
      return 1;
171
100
    void (*ScaleAddCols)(int dst_width, int boxheight, int x, int dx,
172
100
                         const uint16_t* src_ptr, uint8_t* dst_ptr) =
173
100
        (dx & 0xffff) ? ScaleAddCols2_C
174
100
                      : ((dx != 0x10000) ? ScaleAddCols1_C : ScaleAddCols0_C);
175
100
    void (*ScaleAddRow)(const uint8_t* src_ptr, uint16_t* dst_ptr,
176
100
                        int src_width) = ScaleAddRow_C;
177
178
3.32k
    for (j = 0; j < dst_height; ++j) {
179
3.22k
      int boxheight;
180
3.22k
      int iy = y >> 16;
181
3.22k
      const uint8_t* src = src_ptr + iy * (int64_t)src_stride;
182
3.22k
      y += dy;
183
3.22k
      if (y > max_y) {
184
0
        y = max_y;
185
0
      }
186
3.22k
      boxheight = MIN1((y >> 16) - iy);
187
3.22k
      memset(row16, 0, src_width * 2);
188
36.8k
      for (k = 0; k < boxheight; ++k) {
189
33.5k
        ScaleAddRow(src, (uint16_t*)(row16), src_width);
190
33.5k
        src += src_stride;
191
33.5k
      }
192
3.22k
      ScaleAddCols(dst_width, boxheight, x, dx, (uint16_t*)(row16), dst_ptr);
193
3.22k
      dst_ptr += dst_stride;
194
3.22k
    }
195
100
    free_aligned_buffer_64(row16);
196
100
  }
197
0
  return 0;
198
100
}
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
160
                            uint16_t* dst_ptr) {
208
160
  int j, k;
209
  // Initial source x/y coordinate and step values as 16.16 fixed point.
210
160
  int x = 0;
211
160
  int y = 0;
212
160
  int dx = 0;
213
160
  int dy = 0;
214
160
  const int max_y = (src_height << 16);
215
160
  ScaleSlope(src_width, src_height, dst_width, dst_height, kFilterBox, &x, &y,
216
160
             &dx, &dy);
217
160
  src_width = Abs(src_width);
218
160
  {
219
    // Allocate a row buffer of uint32_t.
220
160
    align_buffer_64(row32, src_width * 4);
221
160
    if (!row32)
222
0
      return 1;
223
160
    void (*ScaleAddCols)(int dst_width, int boxheight, int x, int dx,
224
160
                         const uint32_t* src_ptr, uint16_t* dst_ptr) =
225
160
        (dx & 0xffff) ? ScaleAddCols2_16_C : ScaleAddCols1_16_C;
226
160
    void (*ScaleAddRow)(const uint16_t* src_ptr, uint32_t* dst_ptr,
227
160
                        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
4.53k
    for (j = 0; j < dst_height; ++j) {
236
4.37k
      int boxheight;
237
4.37k
      int iy = y >> 16;
238
4.37k
      const uint16_t* src = src_ptr + iy * (int64_t)src_stride;
239
4.37k
      y += dy;
240
4.37k
      if (y > max_y) {
241
0
        y = max_y;
242
0
      }
243
4.37k
      boxheight = MIN1((y >> 16) - iy);
244
4.37k
      memset(row32, 0, src_width * 4);
245
33.3k
      for (k = 0; k < boxheight; ++k) {
246
28.9k
        ScaleAddRow(src, (uint32_t*)(row32), src_width);
247
28.9k
        src += src_stride;
248
28.9k
      }
249
4.37k
      ScaleAddCols(dst_width, boxheight, x, dx, (uint32_t*)(row32), dst_ptr);
250
4.37k
      dst_ptr += dst_stride;
251
4.37k
    }
252
160
    free_aligned_buffer_64(row32);
253
160
  }
254
0
  return 0;
255
160
}
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
370
                                  enum FilterMode filtering) {
267
  // Initial source x/y coordinate and step values as 16.16 fixed point.
268
370
  int x = 0;
269
370
  int y = 0;
270
370
  int dx = 0;
271
370
  int dy = 0;
272
  // TODO(fbarchard): Consider not allocating row buffer for kFilterLinear.
273
  // Allocate a row buffer.
274
370
  align_buffer_64(row, src_width);
275
370
  if (!row)
276
0
    return 1;
277
278
370
  const int max_y = (src_height - 1) << 16;
279
370
  int j;
280
370
  void (*ScaleFilterCols)(uint8_t* dst_ptr, const uint8_t* src_ptr,
281
370
                          int dst_width, int x, int dx) =
282
370
      (src_width >= 32768) ? ScaleFilterCols64_C : ScaleFilterCols_C;
283
370
  void (*InterpolateRow)(uint8_t* dst_ptr, const uint8_t* src_ptr,
284
370
                         ptrdiff_t src_stride, int dst_width,
285
370
                         int source_y_fraction) = InterpolateRow_C;
286
370
  ScaleSlope(src_width, src_height, dst_width, dst_height, filtering, &x, &y,
287
370
             &dx, &dy);
288
370
  src_width = Abs(src_width);
289
290
370
  if (y > max_y) {
291
26
    y = max_y;
292
26
  }
293
294
34.6k
  for (j = 0; j < dst_height; ++j) {
295
34.2k
    int yi = y >> 16;
296
34.2k
    const uint8_t* src = src_ptr + yi * (int64_t)src_stride;
297
34.2k
    if (filtering == kFilterLinear) {
298
3.87k
      ScaleFilterCols(dst_ptr, src, dst_width, x, dx);
299
30.4k
    } else {
300
30.4k
      int yf = (y >> 8) & 255;
301
30.4k
      InterpolateRow(row, src, src_stride, src_width, yf);
302
30.4k
      ScaleFilterCols(dst_ptr, row, dst_width, x, dx);
303
30.4k
    }
304
34.2k
    dst_ptr += dst_stride;
305
34.2k
    y += dy;
306
34.2k
    if (y > max_y) {
307
404
      y = max_y;
308
404
    }
309
34.2k
  }
310
370
  free_aligned_buffer_64(row);
311
370
  return 0;
312
370
}
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
409
                                     enum FilterMode filtering) {
323
  // Initial source x/y coordinate and step values as 16.16 fixed point.
324
409
  int x = 0;
325
409
  int y = 0;
326
409
  int dx = 0;
327
409
  int dy = 0;
328
  // TODO(fbarchard): Consider not allocating row buffer for kFilterLinear.
329
  // Allocate a row buffer.
330
409
  align_buffer_64(row, src_width * 2);
331
409
  if (!row)
332
0
    return 1;
333
334
409
  const int max_y = (src_height - 1) << 16;
335
409
  int j;
336
409
  void (*ScaleFilterCols)(uint16_t* dst_ptr, const uint16_t* src_ptr,
337
409
                          int dst_width, int x, int dx) =
338
409
      (src_width >= 32768) ? ScaleFilterCols64_16_C : ScaleFilterCols_16_C;
339
409
  void (*InterpolateRow)(uint16_t* dst_ptr, const uint16_t* src_ptr,
340
409
                         ptrdiff_t src_stride, int dst_width,
341
409
                         int source_y_fraction) = InterpolateRow_16_C;
342
409
  ScaleSlope(src_width, src_height, dst_width, dst_height, filtering, &x, &y,
343
409
             &dx, &dy);
344
409
  src_width = Abs(src_width);
345
346
409
  if (y > max_y) {
347
8
    y = max_y;
348
8
  }
349
350
32.4k
  for (j = 0; j < dst_height; ++j) {
351
32.0k
    int yi = y >> 16;
352
32.0k
    const uint16_t* src = src_ptr + yi * (int64_t)src_stride;
353
32.0k
    if (filtering == kFilterLinear) {
354
780
      ScaleFilterCols(dst_ptr, src, dst_width, x, dx);
355
31.2k
    } else {
356
31.2k
      int yf = (y >> 8) & 255;
357
31.2k
      InterpolateRow((uint16_t*)row, src, src_stride, src_width, yf);
358
31.2k
      ScaleFilterCols(dst_ptr, (uint16_t*)row, dst_width, x, dx);
359
31.2k
    }
360
32.0k
    dst_ptr += dst_stride;
361
32.0k
    y += dy;
362
32.0k
    if (y > max_y) {
363
425
      y = max_y;
364
425
    }
365
32.0k
  }
366
409
  free_aligned_buffer_64(row);
367
409
  return 0;
368
409
}
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
377
                                enum FilterMode filtering) {
380
377
  int j;
381
  // Initial source x/y coordinate and step values as 16.16 fixed point.
382
377
  int x = 0;
383
377
  int y = 0;
384
377
  int dx = 0;
385
377
  int dy = 0;
386
377
  const int max_y = (src_height - 1) << 16;
387
377
  void (*InterpolateRow)(uint8_t* dst_ptr, const uint8_t* src_ptr,
388
377
                         ptrdiff_t src_stride, int dst_width,
389
377
                         int source_y_fraction) = InterpolateRow_C;
390
377
  void (*ScaleFilterCols)(uint8_t* dst_ptr, const uint8_t* src_ptr,
391
377
                          int dst_width, int x, int dx) =
392
377
      filtering ? ScaleFilterCols_C : ScaleCols_C;
393
377
  ScaleSlope(src_width, src_height, dst_width, dst_height, filtering, &x, &y,
394
377
             &dx, &dy);
395
377
  src_width = Abs(src_width);
396
397
377
  if (filtering && src_width >= 32768) {
398
0
    ScaleFilterCols = ScaleFilterCols64_C;
399
0
  }
400
377
  if (!filtering && src_width * 2 == dst_width && x < 0x8000) {
401
0
    ScaleFilterCols = ScaleColsUp2_C;
402
0
  }
403
404
377
  if (y > max_y) {
405
68
    y = max_y;
406
68
  }
407
377
  {
408
377
    int yi = y >> 16;
409
377
    const uint8_t* src = src_ptr + yi * (int64_t)src_stride;
410
411
    // Allocate 2 row buffers.
412
377
    const int row_size = (dst_width + 31) & ~31;
413
377
    align_buffer_64(row, row_size * 2);
414
377
    if (!row)
415
0
      return 1;
416
417
377
    uint8_t* rowptr = row;
418
377
    int rowstride = row_size;
419
377
    int lasty = yi;
420
421
377
    ScaleFilterCols(rowptr, src, dst_width, x, dx);
422
377
    if (src_height > 1) {
423
305
      src += src_stride;
424
305
    }
425
377
    ScaleFilterCols(rowptr + rowstride, src, dst_width, x, dx);
426
377
    if (src_height > 2) {
427
269
      src += src_stride;
428
269
    }
429
430
3.07M
    for (j = 0; j < dst_height; ++j) {
431
3.07M
      yi = y >> 16;
432
3.07M
      if (yi != lasty) {
433
57.1k
        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
57.1k
        if (yi != lasty) {
439
57.1k
          ScaleFilterCols(rowptr, src, dst_width, x, dx);
440
57.1k
          rowptr += rowstride;
441
57.1k
          rowstride = -rowstride;
442
57.1k
          lasty = yi;
443
57.1k
          if ((y + 65536) < max_y) {
444
56.9k
            src += src_stride;
445
56.9k
          }
446
57.1k
        }
447
57.1k
      }
448
3.07M
      if (filtering == kFilterLinear) {
449
362k
        InterpolateRow(dst_ptr, rowptr, 0, dst_width, 0);
450
2.70M
      } else {
451
2.70M
        int yf = (y >> 8) & 255;
452
2.70M
        InterpolateRow(dst_ptr, rowptr, rowstride, dst_width, yf);
453
2.70M
      }
454
3.07M
      dst_ptr += dst_stride;
455
3.07M
      y += dy;
456
3.07M
    }
457
377
    free_aligned_buffer_64(row);
458
377
  }
459
0
  return 0;
460
377
}
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
43
                                 uint8_t* dst_ptr) {
475
43
  void (*ScaleRowUp)(const uint8_t* src_ptr, uint8_t* dst_ptr, int dst_width) =
476
43
      ScaleRowUp2_Linear_Any_C;
477
43
  int i;
478
43
  int y;
479
43
  int dy;
480
481
43
  (void)src_width;
482
  // This function can only scale up by 2 times horizontally.
483
43
  assert(src_width == ((dst_width + 1) / 2));
484
485
43
  if (dst_height == 1) {
486
9
    ScaleRowUp(src_ptr + ((src_height - 1) / 2) * (int64_t)src_stride, dst_ptr,
487
9
               dst_width);
488
34
  } else {
489
34
    dy = FixedDiv(src_height - 1, dst_height - 1);
490
34
    y = (1 << 15) - 1;
491
118k
    for (i = 0; i < dst_height; ++i) {
492
118k
      ScaleRowUp(src_ptr + (y >> 16) * (int64_t)src_stride, dst_ptr, dst_width);
493
118k
      dst_ptr += dst_stride;
494
118k
      y += dy;
495
118k
    }
496
34
  }
497
43
}
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
33
                                   uint8_t* dst_ptr) {
511
33
  void (*Scale2RowUp)(const uint8_t* src_ptr, ptrdiff_t src_stride,
512
33
                      uint8_t* dst_ptr, ptrdiff_t dst_stride, int dst_width) =
513
33
      ScaleRowUp2_Bilinear_Any_C;
514
33
  int x;
515
516
33
  (void)src_width;
517
  // This function can only scale up by 2 times.
518
33
  assert(src_width == ((dst_width + 1) / 2));
519
33
  assert(src_height == ((dst_height + 1) / 2));
520
521
33
  Scale2RowUp(src_ptr, 0, dst_ptr, 0, dst_width);
522
33
  dst_ptr += dst_stride;
523
340
  for (x = 0; x < src_height - 1; ++x) {
524
307
    Scale2RowUp(src_ptr, src_stride, dst_ptr, dst_stride, dst_width);
525
307
    src_ptr += src_stride;
526
    // TODO(fbarchard): Test performance of writing one row of destination at a
527
    // time.
528
307
    dst_ptr += 2 * dst_stride;
529
307
  }
530
33
  if (!(dst_height & 1)) {
531
12
    Scale2RowUp(src_ptr, 0, dst_ptr, 0, dst_width);
532
12
  }
533
33
}
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
34
                                    uint16_t* dst_ptr) {
548
34
  void (*ScaleRowUp)(const uint16_t* src_ptr, uint16_t* dst_ptr,
549
34
                     int dst_width) = ScaleRowUp2_Linear_16_Any_C;
550
34
  int i;
551
34
  int y;
552
34
  int dy;
553
554
34
  (void)src_width;
555
  // This function can only scale up by 2 times horizontally.
556
34
  assert(src_width == ((dst_width + 1) / 2));
557
558
34
  if (dst_height == 1) {
559
6
    ScaleRowUp(src_ptr + ((src_height - 1) / 2) * (int64_t)src_stride, dst_ptr,
560
6
               dst_width);
561
28
  } else {
562
28
    dy = FixedDiv(src_height - 1, dst_height - 1);
563
28
    y = (1 << 15) - 1;
564
184k
    for (i = 0; i < dst_height; ++i) {
565
184k
      ScaleRowUp(src_ptr + (y >> 16) * (int64_t)src_stride, dst_ptr, dst_width);
566
184k
      dst_ptr += dst_stride;
567
184k
      y += dy;
568
184k
    }
569
28
  }
570
34
}
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
25
                                      uint16_t* dst_ptr) {
585
25
  void (*Scale2RowUp)(const uint16_t* src_ptr, ptrdiff_t src_stride,
586
25
                      uint16_t* dst_ptr, ptrdiff_t dst_stride, int dst_width) =
587
25
      ScaleRowUp2_Bilinear_16_Any_C;
588
25
  int x;
589
590
25
  (void)src_width;
591
  // This function can only scale up by 2 times.
592
25
  assert(src_width == ((dst_width + 1) / 2));
593
25
  assert(src_height == ((dst_height + 1) / 2));
594
595
25
  Scale2RowUp(src_ptr, 0, dst_ptr, 0, dst_width);
596
25
  dst_ptr += dst_stride;
597
77
  for (x = 0; x < src_height - 1; ++x) {
598
52
    Scale2RowUp(src_ptr, src_stride, dst_ptr, dst_stride, dst_width);
599
52
    src_ptr += src_stride;
600
52
    dst_ptr += 2 * dst_stride;
601
52
  }
602
25
  if (!(dst_height & 1)) {
603
19
    Scale2RowUp(src_ptr, 0, dst_ptr, 0, dst_width);
604
19
  }
605
25
}
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
732
                                   enum FilterMode filtering) {
678
732
  int j;
679
  // Initial source x/y coordinate and step values as 16.16 fixed point.
680
732
  int x = 0;
681
732
  int y = 0;
682
732
  int dx = 0;
683
732
  int dy = 0;
684
732
  const int max_y = (src_height - 1) << 16;
685
732
  void (*InterpolateRow)(uint16_t* dst_ptr, const uint16_t* src_ptr,
686
732
                         ptrdiff_t src_stride, int dst_width,
687
732
                         int source_y_fraction) = InterpolateRow_16_C;
688
732
  void (*ScaleFilterCols)(uint16_t* dst_ptr, const uint16_t* src_ptr,
689
732
                          int dst_width, int x, int dx) =
690
732
      filtering ? ScaleFilterCols_16_C : ScaleCols_16_C;
691
732
  ScaleSlope(src_width, src_height, dst_width, dst_height, filtering, &x, &y,
692
732
             &dx, &dy);
693
732
  src_width = Abs(src_width);
694
695
732
  if (filtering && src_width >= 32768) {
696
0
    ScaleFilterCols = ScaleFilterCols64_16_C;
697
0
  }
698
732
  if (!filtering && src_width * 2 == dst_width && x < 0x8000) {
699
0
    ScaleFilterCols = ScaleColsUp2_16_C;
700
0
  }
701
732
  if (y > max_y) {
702
168
    y = max_y;
703
168
  }
704
732
  {
705
732
    int yi = y >> 16;
706
732
    const uint16_t* src = src_ptr + yi * (int64_t)src_stride;
707
708
    // Allocate 2 row buffers.
709
732
    const int row_size = (dst_width + 31) & ~31;
710
732
    align_buffer_64(row, row_size * 4);
711
732
    int rowstride = row_size;
712
732
    int lasty = yi;
713
732
    uint16_t* rowptr = (uint16_t*)row;
714
732
    if (!row)
715
0
      return 1;
716
717
732
    ScaleFilterCols(rowptr, src, dst_width, x, dx);
718
732
    if (src_height > 1) {
719
556
      src += src_stride;
720
556
    }
721
732
    ScaleFilterCols(rowptr + rowstride, src, dst_width, x, dx);
722
732
    if (src_height > 2) {
723
447
      src += src_stride;
724
447
    }
725
726
5.27M
    for (j = 0; j < dst_height; ++j) {
727
5.27M
      yi = y >> 16;
728
5.27M
      if (yi != lasty) {
729
105k
        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
105k
        if (yi != lasty) {
735
105k
          ScaleFilterCols(rowptr, src, dst_width, x, dx);
736
105k
          rowptr += rowstride;
737
105k
          rowstride = -rowstride;
738
105k
          lasty = yi;
739
105k
          if ((y + 65536) < max_y) {
740
105k
            src += src_stride;
741
105k
          }
742
105k
        }
743
105k
      }
744
5.27M
      if (filtering == kFilterLinear) {
745
971k
        InterpolateRow(dst_ptr, rowptr, 0, dst_width, 0);
746
4.30M
      } else {
747
4.30M
        int yf = (y >> 8) & 255;
748
4.30M
        InterpolateRow(dst_ptr, rowptr, rowstride, dst_width, yf);
749
4.30M
      }
750
5.27M
      dst_ptr += dst_stride;
751
5.27M
      y += dy;
752
5.27M
    }
753
732
    free_aligned_buffer_64(row);
754
732
  }
755
0
  return 0;
756
732
}
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
420
                             uint8_t* dst_ptr) {
771
420
  int i;
772
420
  void (*ScaleCols)(uint8_t* dst_ptr, const uint8_t* src_ptr, int dst_width,
773
420
                    int x, int dx) = ScaleCols_C;
774
  // Initial source x/y coordinate and step values as 16.16 fixed point.
775
420
  int x = 0;
776
420
  int y = 0;
777
420
  int dx = 0;
778
420
  int dy = 0;
779
420
  ScaleSlope(src_width, src_height, dst_width, dst_height, kFilterNone, &x, &y,
780
420
             &dx, &dy);
781
420
  src_width = Abs(src_width);
782
783
420
  if (src_width * 2 == dst_width && x < 0x8000) {
784
89
    ScaleCols = ScaleColsUp2_C;
785
89
  }
786
787
728k
  for (i = 0; i < dst_height; ++i) {
788
728k
    ScaleCols(dst_ptr, src_ptr + (y >> 16) * (int64_t)src_stride, dst_width, x,
789
728k
              dx);
790
728k
    dst_ptr += dst_stride;
791
728k
    y += dy;
792
728k
  }
793
420
}
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
276
                                uint16_t* dst_ptr) {
803
276
  int i;
804
276
  void (*ScaleCols)(uint16_t* dst_ptr, const uint16_t* src_ptr, int dst_width,
805
276
                    int x, int dx) = ScaleCols_16_C;
806
  // Initial source x/y coordinate and step values as 16.16 fixed point.
807
276
  int x = 0;
808
276
  int y = 0;
809
276
  int dx = 0;
810
276
  int dy = 0;
811
276
  ScaleSlope(src_width, src_height, dst_width, dst_height, kFilterNone, &x, &y,
812
276
             &dx, &dy);
813
276
  src_width = Abs(src_width);
814
815
276
  if (src_width * 2 == dst_width && x < 0x8000) {
816
69
    ScaleCols = ScaleColsUp2_16_C;
817
69
  }
818
819
248k
  for (i = 0; i < dst_height; ++i) {
820
247k
    ScaleCols(dst_ptr, src_ptr + (y >> 16) * (int64_t)src_stride, dst_width, x,
821
247k
              dx);
822
247k
    dst_ptr += dst_stride;
823
247k
    y += dy;
824
247k
  }
825
276
}
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
1.69k
               enum FilterMode filtering) {
838
  // Simplify filtering when possible.
839
1.69k
  filtering = ScaleFilterReduce(src_width, src_height, dst_width, dst_height,
840
1.69k
                                filtering);
841
842
  // Negative height means invert the image.
843
1.69k
  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
1.69k
  if (dst_width == src_width && dst_height == src_height) {
851
    // Straight copy.
852
10
    CopyPlane(src, src_stride, dst, dst_stride, dst_width, dst_height);
853
10
    return 0;
854
10
  }
855
1.68k
  if (dst_width == src_width && filtering != kFilterBox) {
856
342
    int dy = 0;
857
342
    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
342
    if (dst_height <= src_height) {
861
218
      dy = FixedDiv(src_height, dst_height);
862
218
      y = CENTERSTART(dy, -32768);  // Subtract 0.5 (32768) to center filter.
863
218
    } else if (src_height > 1 && dst_height > 1) {
864
41
      dy = FixedDiv1(src_height, dst_height);
865
41
    }
866
    // Arbitrary scale vertically, but unscaled horizontally.
867
342
    ScalePlaneVertical(src_height, dst_width, dst_height, src_stride,
868
342
                       dst_stride, src, dst, 0, y, dy, /*bpp=*/1, filtering);
869
342
    return 0;
870
342
  }
871
1.34k
  if (filtering == kFilterBox && dst_height * 2 < src_height) {
872
100
    return ScalePlaneBox(src_width, src_height, dst_width, dst_height,
873
100
                         src_stride, dst_stride, src, dst);
874
100
  }
875
1.24k
  if ((dst_width + 1) / 2 == src_width && filtering == kFilterLinear) {
876
43
    ScalePlaneUp2_Linear(src_width, src_height, dst_width, dst_height,
877
43
                         src_stride, dst_stride, src, dst);
878
43
    return 0;
879
43
  }
880
1.20k
  if ((dst_height + 1) / 2 == src_height && (dst_width + 1) / 2 == src_width &&
881
59
      (filtering == kFilterBilinear || filtering == kFilterBox)) {
882
33
    ScalePlaneUp2_Bilinear(src_width, src_height, dst_width, dst_height,
883
33
                           src_stride, dst_stride, src, dst);
884
33
    return 0;
885
33
  }
886
1.16k
  if (filtering && dst_height > src_height) {
887
377
    return ScalePlaneBilinearUp(src_width, src_height, dst_width, dst_height,
888
377
                                src_stride, dst_stride, src, dst, filtering);
889
377
  }
890
790
  if (filtering) {
891
370
    return ScalePlaneBilinearDown(src_width, src_height, dst_width, dst_height,
892
370
                                  src_stride, dst_stride, src, dst, filtering);
893
370
  }
894
420
  ScalePlaneSimple(src_width, src_height, dst_width, dst_height, src_stride,
895
420
                   dst_stride, src, dst);
896
420
  return 0;
897
790
}
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
1.72k
                  enum FilterMode filtering) {
908
  // Simplify filtering when possible.
909
1.72k
  filtering = ScaleFilterReduce(src_width, src_height, dst_width, dst_height,
910
1.72k
                                filtering);
911
912
  // Negative height means invert the image.
913
1.72k
  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
1.72k
  if (dst_width == src_width && dst_height == src_height) {
921
    // Straight copy.
922
8
    CopyPlane_16(src, src_stride, dst, dst_stride, dst_width, dst_height);
923
8
    return 0;
924
8
  }
925
1.71k
  if (dst_width == src_width && filtering != kFilterBox) {
926
140
    int dy = 0;
927
140
    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
140
    if (dst_height <= src_height) {
931
60
      dy = FixedDiv(src_height, dst_height);
932
60
      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
80
    } else if (src_height > 1 && dst_height > 1) {
937
52
      dy = FixedDiv1(src_height, dst_height);
938
52
    }
939
    // Arbitrary scale vertically, but unscaled horizontally.
940
140
    ScalePlaneVertical_16(src_height, dst_width, dst_height, src_stride,
941
140
                          dst_stride, src, dst, 0, y, dy, /*bpp=*/1, filtering);
942
140
    return 0;
943
140
  }
944
1.57k
  if (filtering == kFilterBox && dst_height * 2 < src_height) {
945
160
    return ScalePlaneBox_16(src_width, src_height, dst_width, dst_height,
946
160
                            src_stride, dst_stride, src, dst);
947
160
  }
948
1.41k
  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.41k
  if ((dst_height + 1) / 2 == src_height && (dst_width + 1) / 2 == src_width &&
954
20
      (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.41k
  if (filtering && dst_height > src_height) {
960
732
    return ScalePlaneBilinearUp_16(src_width, src_height, dst_width, dst_height,
961
732
                                   src_stride, dst_stride, src, dst, filtering);
962
732
  }
963
685
  if (filtering) {
964
409
    return ScalePlaneBilinearDown_16(src_width, src_height, dst_width,
965
409
                                     dst_height, src_stride, dst_stride, src,
966
409
                                     dst, filtering);
967
409
  }
968
276
  ScalePlaneSimple_16(src_width, src_height, dst_width, dst_height, src_stride,
969
276
                      dst_stride, src, dst);
970
276
  return 0;
971
685
}
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
1.78k
                  enum FilterMode filtering) {
982
  // Simplify filtering when possible.
983
1.78k
  filtering = ScaleFilterReduce(src_width, src_height, dst_width, dst_height,
984
1.78k
                                filtering);
985
986
  // Negative height means invert the image.
987
1.78k
  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
1.78k
  if ((dst_width + 1) / 2 == src_width && filtering == kFilterLinear) {
994
34
    ScalePlaneUp2_12_Linear(src_width, src_height, dst_width, dst_height,
995
34
                            src_stride, dst_stride, src, dst);
996
34
    return 0;
997
34
  }
998
1.75k
  if ((dst_height + 1) / 2 == src_height && (dst_width + 1) / 2 == src_width &&
999
55
      (filtering == kFilterBilinear || filtering == kFilterBox)) {
1000
25
    ScalePlaneUp2_12_Bilinear(src_width, src_height, dst_width, dst_height,
1001
25
                              src_stride, dst_stride, src, dst);
1002
25
    return 0;
1003
25
  }
1004
1005
1.72k
  return ScalePlane_16(src, src_stride, src_width, src_height, dst, dst_stride,
1006
1.72k
                       dst_width, dst_height, filtering);
1007
1.75k
}