Coverage Report

Created: 2026-02-14 07:09

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