Coverage Report

Created: 2025-11-16 07:09

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/aom/aom_scale/generic/yv12extend.c
Line
Count
Source
1
/*
2
 * Copyright (c) 2016, Alliance for Open Media. All rights reserved.
3
 *
4
 * This source code is subject to the terms of the BSD 2 Clause License and
5
 * the Alliance for Open Media Patent License 1.0. If the BSD 2 Clause License
6
 * was not distributed with this source code in the LICENSE file, you can
7
 * obtain it at www.aomedia.org/license/software. If the Alliance for Open
8
 * Media Patent License 1.0 was not distributed with this source code in the
9
 * PATENTS file, you can obtain it at www.aomedia.org/license/patent.
10
 */
11
12
#include <assert.h>
13
14
#include "config/aom_config.h"
15
#include "config/aom_scale_rtcd.h"
16
17
#include "aom/aom_integer.h"
18
#include "aom_mem/aom_mem.h"
19
#include "aom_ports/mem.h"
20
#include "aom_scale/yv12config.h"
21
22
static void extend_plane(uint8_t *const src, int src_stride, int width,
23
                         int height, int extend_top, int extend_left,
24
                         int extend_bottom, int extend_right, int v_start,
25
31.0k
                         int v_end) {
26
31.0k
  assert(src != NULL);
27
31.0k
  int i;
28
31.0k
  const int linesize = extend_left + extend_right + width;
29
31.0k
  assert(linesize <= src_stride);
30
31
  /* copy the left and right most columns out */
32
31.0k
  uint8_t *src_ptr1 = src + v_start * src_stride;
33
31.0k
  uint8_t *src_ptr2 = src + v_start * src_stride + width - 1;
34
31.0k
  uint8_t *dst_ptr1 = src + v_start * src_stride - extend_left;
35
31.0k
  uint8_t *dst_ptr2 = src_ptr2 + 1;
36
37
2.53M
  for (i = v_start; i < v_end; ++i) {
38
2.50M
    memset(dst_ptr1, src_ptr1[0], extend_left);
39
2.50M
    memset(dst_ptr2, src_ptr2[0], extend_right);
40
2.50M
    src_ptr1 += src_stride;
41
2.50M
    src_ptr2 += src_stride;
42
2.50M
    dst_ptr1 += src_stride;
43
2.50M
    dst_ptr2 += src_stride;
44
2.50M
  }
45
46
  /* Now copy the top and bottom lines into each line of the respective
47
   * borders
48
   */
49
31.0k
  src_ptr1 = src - extend_left;
50
31.0k
  dst_ptr1 = src_ptr1 + src_stride * -extend_top;
51
52
7.70M
  for (i = 0; i < extend_top; ++i) {
53
7.67M
    memcpy(dst_ptr1, src_ptr1, linesize);
54
7.67M
    dst_ptr1 += src_stride;
55
7.67M
  }
56
57
31.0k
  src_ptr2 = src_ptr1 + src_stride * (height - 1);
58
31.0k
  dst_ptr2 = src_ptr2;
59
60
7.83M
  for (i = 0; i < extend_bottom; ++i) {
61
7.80M
    dst_ptr2 += src_stride;
62
7.80M
    memcpy(dst_ptr2, src_ptr2, linesize);
63
7.80M
  }
64
31.0k
}
65
66
#if CONFIG_AV1_HIGHBITDEPTH
67
static void extend_plane_high(uint8_t *const src8, int src_stride, int width,
68
                              int height, int extend_top, int extend_left,
69
                              int extend_bottom, int extend_right, int v_start,
70
17.8k
                              int v_end) {
71
17.8k
  int i;
72
17.8k
  const int linesize = extend_left + extend_right + width;
73
17.8k
  assert(linesize <= src_stride);
74
17.8k
  uint16_t *src = CONVERT_TO_SHORTPTR(src8);
75
76
  /* copy the left and right most columns out */
77
17.8k
  uint16_t *src_ptr1 = src + v_start * src_stride;
78
17.8k
  uint16_t *src_ptr2 = src + v_start * src_stride + width - 1;
79
17.8k
  uint16_t *dst_ptr1 = src + v_start * src_stride - extend_left;
80
17.8k
  uint16_t *dst_ptr2 = src_ptr2 + 1;
81
82
1.53M
  for (i = v_start; i < v_end; ++i) {
83
1.51M
    aom_memset16(dst_ptr1, src_ptr1[0], extend_left);
84
1.51M
    aom_memset16(dst_ptr2, src_ptr2[0], extend_right);
85
1.51M
    src_ptr1 += src_stride;
86
1.51M
    src_ptr2 += src_stride;
87
1.51M
    dst_ptr1 += src_stride;
88
1.51M
    dst_ptr2 += src_stride;
89
1.51M
  }
90
91
  /* Now copy the top and bottom lines into each line of the respective
92
   * borders
93
   */
94
17.8k
  src_ptr1 = src - extend_left;
95
17.8k
  dst_ptr1 = src_ptr1 + src_stride * -extend_top;
96
97
4.82M
  for (i = 0; i < extend_top; ++i) {
98
4.81M
    memcpy(dst_ptr1, src_ptr1, linesize * sizeof(uint16_t));
99
4.81M
    dst_ptr1 += src_stride;
100
4.81M
  }
101
102
17.8k
  src_ptr2 = src_ptr1 + src_stride * (height - 1);
103
17.8k
  dst_ptr2 = src_ptr2;
104
105
4.87M
  for (i = 0; i < extend_bottom; ++i) {
106
4.85M
    dst_ptr2 += src_stride;
107
4.85M
    memcpy(dst_ptr2, src_ptr2, linesize * sizeof(uint16_t));
108
4.85M
  }
109
17.8k
}
110
#endif  // CONFIG_AV1_HIGHBITDEPTH
111
112
void aom_extend_frame_borders_plane_row_c(const YV12_BUFFER_CONFIG *ybf,
113
0
                                          int plane, int v_start, int v_end) {
114
0
  const int ext_size = ybf->border;
115
0
  const int ss_x = ybf->subsampling_x;
116
0
  const int ss_y = ybf->subsampling_y;
117
118
0
  assert(ybf->y_height - ybf->y_crop_height < 16);
119
0
  assert(ybf->y_width - ybf->y_crop_width < 16);
120
0
  assert(ybf->y_height - ybf->y_crop_height >= 0);
121
0
  assert(ybf->y_width - ybf->y_crop_width >= 0);
122
123
0
  const int is_uv = plane > 0;
124
0
  const int top = ext_size >> (is_uv ? ss_y : 0);
125
0
  const int left = ext_size >> (is_uv ? ss_x : 0);
126
0
  const int bottom = top + ybf->heights[is_uv] - ybf->crop_heights[is_uv];
127
0
  const int right = left + ybf->widths[is_uv] - ybf->crop_widths[is_uv];
128
0
  const int extend_top_border = (v_start == 0);
129
0
  const int extend_bottom_border = (v_end == ybf->crop_heights[is_uv]);
130
131
0
#if CONFIG_AV1_HIGHBITDEPTH
132
0
  if (ybf->flags & YV12_FLAG_HIGHBITDEPTH) {
133
0
    extend_plane_high(ybf->buffers[plane], ybf->strides[is_uv],
134
0
                      ybf->crop_widths[is_uv], ybf->crop_heights[is_uv],
135
0
                      extend_top_border ? top : 0, left,
136
0
                      extend_bottom_border ? bottom : 0, right, v_start, v_end);
137
0
    return;
138
0
  }
139
0
#endif
140
141
0
  extend_plane(ybf->buffers[plane], ybf->strides[is_uv],
142
0
               ybf->crop_widths[is_uv], ybf->crop_heights[is_uv],
143
0
               extend_top_border ? top : 0, left,
144
0
               extend_bottom_border ? bottom : 0, right, v_start, v_end);
145
0
}
146
147
static void extend_frame(YV12_BUFFER_CONFIG *const ybf, int ext_size,
148
24.0k
                         const int num_planes) {
149
24.0k
  const int ss_x = ybf->subsampling_x;
150
24.0k
  const int ss_y = ybf->subsampling_y;
151
152
24.0k
  assert(ybf->y_height - ybf->y_crop_height < 16);
153
24.0k
  assert(ybf->y_width - ybf->y_crop_width < 16);
154
24.0k
  assert(ybf->y_height - ybf->y_crop_height >= 0);
155
24.0k
  assert(ybf->y_width - ybf->y_crop_width >= 0);
156
157
24.0k
#if CONFIG_AV1_HIGHBITDEPTH
158
24.0k
  if (ybf->flags & YV12_FLAG_HIGHBITDEPTH) {
159
30.9k
    for (int plane = 0; plane < num_planes; ++plane) {
160
17.8k
      const int is_uv = plane > 0;
161
17.8k
      const int top = ext_size >> (is_uv ? ss_y : 0);
162
17.8k
      const int left = ext_size >> (is_uv ? ss_x : 0);
163
17.8k
      const int bottom = top + ybf->heights[is_uv] - ybf->crop_heights[is_uv];
164
17.8k
      const int right = left + ybf->widths[is_uv] - ybf->crop_widths[is_uv];
165
17.8k
      extend_plane_high(ybf->buffers[plane], ybf->strides[is_uv],
166
17.8k
                        ybf->crop_widths[is_uv], ybf->crop_heights[is_uv], top,
167
17.8k
                        left, bottom, right, 0, ybf->crop_heights[is_uv]);
168
17.8k
    }
169
13.0k
    return;
170
13.0k
  }
171
11.0k
#endif
172
173
42.0k
  for (int plane = 0; plane < num_planes; ++plane) {
174
31.0k
    const int is_uv = plane > 0;
175
31.0k
    const int top = ext_size >> (is_uv ? ss_y : 0);
176
31.0k
    const int left = ext_size >> (is_uv ? ss_x : 0);
177
31.0k
    const int bottom = top + ybf->heights[is_uv] - ybf->crop_heights[is_uv];
178
31.0k
    const int right = left + ybf->widths[is_uv] - ybf->crop_widths[is_uv];
179
31.0k
    extend_plane(ybf->buffers[plane], ybf->strides[is_uv],
180
31.0k
                 ybf->crop_widths[is_uv], ybf->crop_heights[is_uv], top, left,
181
31.0k
                 bottom, right, 0, ybf->crop_heights[is_uv]);
182
31.0k
  }
183
11.0k
}
184
185
24.0k
void aom_extend_frame_borders_c(YV12_BUFFER_CONFIG *ybf, const int num_planes) {
186
24.0k
  extend_frame(ybf, ybf->border, num_planes);
187
24.0k
}
188
189
#if CONFIG_AV1_HIGHBITDEPTH
190
780k
static void memcpy_short_addr(uint8_t *dst8, const uint8_t *src8, int num) {
191
780k
  uint16_t *dst = CONVERT_TO_SHORTPTR(dst8);
192
780k
  uint16_t *src = CONVERT_TO_SHORTPTR(src8);
193
780k
  memcpy(dst, src, num * sizeof(uint16_t));
194
780k
}
195
#endif
196
197
// Copies the source image into the destination image and updates the
198
// destination's UMV borders.
199
// Note: The frames are assumed to be identical in size.
200
void aom_yv12_copy_frame_c(const YV12_BUFFER_CONFIG *src_bc,
201
12.0k
                           YV12_BUFFER_CONFIG *dst_bc, const int num_planes) {
202
12.0k
  assert(src_bc->y_width == dst_bc->y_width);
203
12.0k
  assert(src_bc->y_height == dst_bc->y_height);
204
205
12.0k
#if CONFIG_AV1_HIGHBITDEPTH
206
12.0k
  assert((src_bc->flags & YV12_FLAG_HIGHBITDEPTH) ==
207
12.0k
         (dst_bc->flags & YV12_FLAG_HIGHBITDEPTH));
208
209
12.0k
  if (src_bc->flags & YV12_FLAG_HIGHBITDEPTH) {
210
15.4k
    for (int plane = 0; plane < num_planes; ++plane) {
211
8.93k
      const uint8_t *plane_src = src_bc->buffers[plane];
212
8.93k
      uint8_t *plane_dst = dst_bc->buffers[plane];
213
8.93k
      const int is_uv = plane > 0;
214
215
789k
      for (int row = 0; row < src_bc->heights[is_uv]; ++row) {
216
780k
        memcpy_short_addr(plane_dst, plane_src, src_bc->widths[is_uv]);
217
780k
        plane_src += src_bc->strides[is_uv];
218
780k
        plane_dst += dst_bc->strides[is_uv];
219
780k
      }
220
8.93k
    }
221
6.53k
    aom_extend_frame_borders_c(dst_bc, num_planes);
222
6.53k
    return;
223
6.53k
  }
224
5.51k
#endif
225
21.0k
  for (int plane = 0; plane < num_planes; ++plane) {
226
15.5k
    const uint8_t *plane_src = src_bc->buffers[plane];
227
15.5k
    uint8_t *plane_dst = dst_bc->buffers[plane];
228
15.5k
    const int is_uv = plane > 0;
229
230
1.33M
    for (int row = 0; row < src_bc->heights[is_uv]; ++row) {
231
1.31M
      memcpy(plane_dst, plane_src, src_bc->widths[is_uv]);
232
1.31M
      plane_src += src_bc->strides[is_uv];
233
1.31M
      plane_dst += dst_bc->strides[is_uv];
234
1.31M
    }
235
15.5k
  }
236
5.51k
  aom_extend_frame_borders_c(dst_bc, num_planes);
237
5.51k
}
238
239
void aom_yv12_copy_y_c(const YV12_BUFFER_CONFIG *src_ybc,
240
0
                       YV12_BUFFER_CONFIG *dst_ybc, int use_crop) {
241
0
  int row;
242
0
  int width = use_crop ? src_ybc->y_crop_width : src_ybc->y_width;
243
0
  int height = use_crop ? src_ybc->y_crop_height : src_ybc->y_height;
244
0
  const uint8_t *src = src_ybc->y_buffer;
245
0
  uint8_t *dst = dst_ybc->y_buffer;
246
247
0
#if CONFIG_AV1_HIGHBITDEPTH
248
0
  if (src_ybc->flags & YV12_FLAG_HIGHBITDEPTH) {
249
0
    const uint16_t *src16 = CONVERT_TO_SHORTPTR(src);
250
0
    uint16_t *dst16 = CONVERT_TO_SHORTPTR(dst);
251
0
    for (row = 0; row < height; ++row) {
252
0
      memcpy(dst16, src16, width * sizeof(uint16_t));
253
0
      src16 += src_ybc->y_stride;
254
0
      dst16 += dst_ybc->y_stride;
255
0
    }
256
0
    return;
257
0
  }
258
0
#endif
259
260
0
  for (row = 0; row < height; ++row) {
261
0
    memcpy(dst, src, width);
262
0
    src += src_ybc->y_stride;
263
0
    dst += dst_ybc->y_stride;
264
0
  }
265
0
}
266
267
void aom_yv12_copy_u_c(const YV12_BUFFER_CONFIG *src_bc,
268
0
                       YV12_BUFFER_CONFIG *dst_bc, int use_crop) {
269
0
  int row;
270
0
  int width = use_crop ? src_bc->uv_crop_width : src_bc->uv_width;
271
0
  int height = use_crop ? src_bc->uv_crop_height : src_bc->uv_height;
272
0
  const uint8_t *src = src_bc->u_buffer;
273
0
  uint8_t *dst = dst_bc->u_buffer;
274
0
#if CONFIG_AV1_HIGHBITDEPTH
275
0
  if (src_bc->flags & YV12_FLAG_HIGHBITDEPTH) {
276
0
    const uint16_t *src16 = CONVERT_TO_SHORTPTR(src);
277
0
    uint16_t *dst16 = CONVERT_TO_SHORTPTR(dst);
278
0
    for (row = 0; row < height; ++row) {
279
0
      memcpy(dst16, src16, width * sizeof(uint16_t));
280
0
      src16 += src_bc->uv_stride;
281
0
      dst16 += dst_bc->uv_stride;
282
0
    }
283
0
    return;
284
0
  }
285
0
#endif
286
0
  for (row = 0; row < height; ++row) {
287
0
    memcpy(dst, src, width);
288
0
    src += src_bc->uv_stride;
289
0
    dst += dst_bc->uv_stride;
290
0
  }
291
0
}
292
293
void aom_yv12_copy_v_c(const YV12_BUFFER_CONFIG *src_bc,
294
0
                       YV12_BUFFER_CONFIG *dst_bc, int use_crop) {
295
0
  int row;
296
0
  int width = use_crop ? src_bc->uv_crop_width : src_bc->uv_width;
297
0
  int height = use_crop ? src_bc->uv_crop_height : src_bc->uv_height;
298
0
  const uint8_t *src = src_bc->v_buffer;
299
0
  uint8_t *dst = dst_bc->v_buffer;
300
0
#if CONFIG_AV1_HIGHBITDEPTH
301
0
  if (src_bc->flags & YV12_FLAG_HIGHBITDEPTH) {
302
0
    const uint16_t *src16 = CONVERT_TO_SHORTPTR(src);
303
0
    uint16_t *dst16 = CONVERT_TO_SHORTPTR(dst);
304
0
    for (row = 0; row < height; ++row) {
305
0
      memcpy(dst16, src16, width * sizeof(uint16_t));
306
0
      src16 += src_bc->uv_stride;
307
0
      dst16 += dst_bc->uv_stride;
308
0
    }
309
0
    return;
310
0
  }
311
0
#endif
312
0
  for (row = 0; row < height; ++row) {
313
0
    memcpy(dst, src, width);
314
0
    src += src_bc->uv_stride;
315
0
    dst += dst_bc->uv_stride;
316
0
  }
317
0
}
318
319
void aom_yv12_partial_copy_y_c(const YV12_BUFFER_CONFIG *src_ybc, int hstart1,
320
                               int hend1, int vstart1, int vend1,
321
                               YV12_BUFFER_CONFIG *dst_ybc, int hstart2,
322
24.4k
                               int vstart2) {
323
24.4k
  int row;
324
24.4k
  const uint8_t *src = src_ybc->y_buffer;
325
24.4k
  uint8_t *dst = dst_ybc->y_buffer;
326
24.4k
#if CONFIG_AV1_HIGHBITDEPTH
327
24.4k
  if (src_ybc->flags & YV12_FLAG_HIGHBITDEPTH) {
328
15.1k
    const uint16_t *src16 =
329
15.1k
        CONVERT_TO_SHORTPTR(src + vstart1 * src_ybc->y_stride + hstart1);
330
15.1k
    uint16_t *dst16 =
331
15.1k
        CONVERT_TO_SHORTPTR(dst + vstart2 * dst_ybc->y_stride + hstart2);
332
333
1.67M
    for (row = vstart1; row < vend1; ++row) {
334
1.65M
      memcpy(dst16, src16, (hend1 - hstart1) * sizeof(uint16_t));
335
1.65M
      src16 += src_ybc->y_stride;
336
1.65M
      dst16 += dst_ybc->y_stride;
337
1.65M
    }
338
15.1k
    return;
339
15.1k
  }
340
9.32k
#endif
341
9.32k
  src = (src + vstart1 * src_ybc->y_stride + hstart1);
342
9.32k
  dst = (dst + vstart2 * dst_ybc->y_stride + hstart2);
343
344
1.34M
  for (row = vstart1; row < vend1; ++row) {
345
1.33M
    memcpy(dst, src, (hend1 - hstart1));
346
1.33M
    src += src_ybc->y_stride;
347
1.33M
    dst += dst_ybc->y_stride;
348
1.33M
  }
349
9.32k
}
350
351
void aom_yv12_partial_coloc_copy_y_c(const YV12_BUFFER_CONFIG *src_ybc,
352
                                     YV12_BUFFER_CONFIG *dst_ybc, int hstart,
353
24.4k
                                     int hend, int vstart, int vend) {
354
24.4k
  aom_yv12_partial_copy_y_c(src_ybc, hstart, hend, vstart, vend, dst_ybc,
355
24.4k
                            hstart, vstart);
356
24.4k
}
357
358
void aom_yv12_partial_copy_u_c(const YV12_BUFFER_CONFIG *src_bc, int hstart1,
359
                               int hend1, int vstart1, int vend1,
360
                               YV12_BUFFER_CONFIG *dst_bc, int hstart2,
361
21.1k
                               int vstart2) {
362
21.1k
  int row;
363
21.1k
  const uint8_t *src = src_bc->u_buffer;
364
21.1k
  uint8_t *dst = dst_bc->u_buffer;
365
21.1k
#if CONFIG_AV1_HIGHBITDEPTH
366
21.1k
  if (src_bc->flags & YV12_FLAG_HIGHBITDEPTH) {
367
9.79k
    const uint16_t *src16 =
368
9.79k
        CONVERT_TO_SHORTPTR(src + vstart1 * src_bc->uv_stride + hstart1);
369
9.79k
    uint16_t *dst16 =
370
9.79k
        CONVERT_TO_SHORTPTR(dst + vstart2 * dst_bc->uv_stride + hstart2);
371
599k
    for (row = vstart1; row < vend1; ++row) {
372
589k
      memcpy(dst16, src16, (hend1 - hstart1) * sizeof(uint16_t));
373
589k
      src16 += src_bc->uv_stride;
374
589k
      dst16 += dst_bc->uv_stride;
375
589k
    }
376
9.79k
    return;
377
9.79k
  }
378
11.3k
#endif
379
11.3k
  src = (src + vstart1 * src_bc->uv_stride + hstart1);
380
11.3k
  dst = (dst + vstart2 * dst_bc->uv_stride + hstart2);
381
382
826k
  for (row = vstart1; row < vend1; ++row) {
383
814k
    memcpy(dst, src, (hend1 - hstart1));
384
814k
    src += src_bc->uv_stride;
385
814k
    dst += dst_bc->uv_stride;
386
814k
  }
387
11.3k
}
388
389
void aom_yv12_partial_coloc_copy_u_c(const YV12_BUFFER_CONFIG *src_bc,
390
                                     YV12_BUFFER_CONFIG *dst_bc, int hstart,
391
21.1k
                                     int hend, int vstart, int vend) {
392
21.1k
  aom_yv12_partial_copy_u_c(src_bc, hstart, hend, vstart, vend, dst_bc, hstart,
393
21.1k
                            vstart);
394
21.1k
}
395
396
void aom_yv12_partial_copy_v_c(const YV12_BUFFER_CONFIG *src_bc, int hstart1,
397
                               int hend1, int vstart1, int vend1,
398
                               YV12_BUFFER_CONFIG *dst_bc, int hstart2,
399
9.78k
                               int vstart2) {
400
9.78k
  int row;
401
9.78k
  const uint8_t *src = src_bc->v_buffer;
402
9.78k
  uint8_t *dst = dst_bc->v_buffer;
403
9.78k
#if CONFIG_AV1_HIGHBITDEPTH
404
9.78k
  if (src_bc->flags & YV12_FLAG_HIGHBITDEPTH) {
405
3.38k
    const uint16_t *src16 =
406
3.38k
        CONVERT_TO_SHORTPTR(src + vstart1 * src_bc->uv_stride + hstart1);
407
3.38k
    uint16_t *dst16 =
408
3.38k
        CONVERT_TO_SHORTPTR(dst + vstart2 * dst_bc->uv_stride + hstart2);
409
258k
    for (row = vstart1; row < vend1; ++row) {
410
255k
      memcpy(dst16, src16, (hend1 - hstart1) * sizeof(uint16_t));
411
255k
      src16 += src_bc->uv_stride;
412
255k
      dst16 += dst_bc->uv_stride;
413
255k
    }
414
3.38k
    return;
415
3.38k
  }
416
6.39k
#endif
417
6.39k
  src = (src + vstart1 * src_bc->uv_stride + hstart1);
418
6.39k
  dst = (dst + vstart2 * dst_bc->uv_stride + hstart2);
419
420
526k
  for (row = vstart1; row < vend1; ++row) {
421
520k
    memcpy(dst, src, (hend1 - hstart1));
422
520k
    src += src_bc->uv_stride;
423
520k
    dst += dst_bc->uv_stride;
424
520k
  }
425
6.39k
}
426
427
void aom_yv12_partial_coloc_copy_v_c(const YV12_BUFFER_CONFIG *src_bc,
428
                                     YV12_BUFFER_CONFIG *dst_bc, int hstart,
429
9.78k
                                     int hend, int vstart, int vend) {
430
9.78k
  aom_yv12_partial_copy_v_c(src_bc, hstart, hend, vstart, vend, dst_bc, hstart,
431
9.78k
                            vstart);
432
9.78k
}
433
434
int aom_yv12_realloc_with_new_border_c(YV12_BUFFER_CONFIG *ybf, int new_border,
435
                                       int byte_alignment, bool alloc_pyramid,
436
0
                                       int num_planes) {
437
0
  if (ybf) {
438
0
    if (new_border == ybf->border) return 0;
439
0
    YV12_BUFFER_CONFIG new_buf;
440
0
    memset(&new_buf, 0, sizeof(new_buf));
441
0
    const int error = aom_alloc_frame_buffer(
442
0
        &new_buf, ybf->y_crop_width, ybf->y_crop_height, ybf->subsampling_x,
443
0
        ybf->subsampling_y, ybf->flags & YV12_FLAG_HIGHBITDEPTH, new_border,
444
0
        byte_alignment, alloc_pyramid, 0);
445
0
    if (error) return error;
446
    // Copy image buffer
447
0
    aom_yv12_copy_frame(ybf, &new_buf, num_planes);
448
449
    // Now free the old buffer and replace with the new
450
0
    aom_free_frame_buffer(ybf);
451
0
    *ybf = new_buf;
452
0
    return 0;
453
0
  }
454
0
  return -2;
455
0
}