Coverage Report

Created: 2026-05-16 06:27

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
17.2k
                         int v_end) {
26
17.2k
  assert(src != NULL);
27
17.2k
  int i;
28
17.2k
  const int linesize = extend_left + extend_right + width;
29
17.2k
  assert(linesize <= src_stride);
30
31
  /* copy the left and right most columns out */
32
17.2k
  uint8_t *src_ptr1 = src + v_start * src_stride;
33
17.2k
  uint8_t *src_ptr2 = src + v_start * src_stride + width - 1;
34
17.2k
  uint8_t *dst_ptr1 = src + v_start * src_stride - extend_left;
35
17.2k
  uint8_t *dst_ptr2 = src_ptr2 + 1;
36
37
1.33M
  for (i = v_start; i < v_end; ++i) {
38
1.31M
    memset(dst_ptr1, src_ptr1[0], extend_left);
39
1.31M
    memset(dst_ptr2, src_ptr2[0], extend_right);
40
1.31M
    src_ptr1 += src_stride;
41
1.31M
    src_ptr2 += src_stride;
42
1.31M
    dst_ptr1 += src_stride;
43
1.31M
    dst_ptr2 += src_stride;
44
1.31M
  }
45
46
  /* Now copy the top and bottom lines into each line of the respective
47
   * borders
48
   */
49
17.2k
  src_ptr1 = src - extend_left;
50
17.2k
  dst_ptr1 = src_ptr1 + src_stride * -extend_top;
51
52
3.84M
  for (i = 0; i < extend_top; ++i) {
53
3.82M
    memcpy(dst_ptr1, src_ptr1, linesize);
54
3.82M
    dst_ptr1 += src_stride;
55
3.82M
  }
56
57
17.2k
  src_ptr2 = src_ptr1 + src_stride * (height - 1);
58
17.2k
  dst_ptr2 = src_ptr2;
59
60
3.91M
  for (i = 0; i < extend_bottom; ++i) {
61
3.89M
    dst_ptr2 += src_stride;
62
3.89M
    memcpy(dst_ptr2, src_ptr2, linesize);
63
3.89M
  }
64
17.2k
}
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
12.8k
                              int v_end) {
71
12.8k
  int i;
72
12.8k
  const int linesize = extend_left + extend_right + width;
73
12.8k
  assert(linesize <= src_stride);
74
12.8k
  uint16_t *src = CONVERT_TO_SHORTPTR(src8);
75
76
  /* copy the left and right most columns out */
77
12.8k
  uint16_t *src_ptr1 = src + v_start * src_stride;
78
12.8k
  uint16_t *src_ptr2 = src + v_start * src_stride + width - 1;
79
12.8k
  uint16_t *dst_ptr1 = src + v_start * src_stride - extend_left;
80
12.8k
  uint16_t *dst_ptr2 = src_ptr2 + 1;
81
82
2.82M
  for (i = v_start; i < v_end; ++i) {
83
2.81M
    aom_memset16(dst_ptr1, src_ptr1[0], extend_left);
84
2.81M
    aom_memset16(dst_ptr2, src_ptr2[0], extend_right);
85
2.81M
    src_ptr1 += src_stride;
86
2.81M
    src_ptr2 += src_stride;
87
2.81M
    dst_ptr1 += src_stride;
88
2.81M
    dst_ptr2 += src_stride;
89
2.81M
  }
90
91
  /* Now copy the top and bottom lines into each line of the respective
92
   * borders
93
   */
94
12.8k
  src_ptr1 = src - extend_left;
95
12.8k
  dst_ptr1 = src_ptr1 + src_stride * -extend_top;
96
97
3.66M
  for (i = 0; i < extend_top; ++i) {
98
3.65M
    memcpy(dst_ptr1, src_ptr1, linesize * sizeof(uint16_t));
99
3.65M
    dst_ptr1 += src_stride;
100
3.65M
  }
101
102
12.8k
  src_ptr2 = src_ptr1 + src_stride * (height - 1);
103
12.8k
  dst_ptr2 = src_ptr2;
104
105
3.69M
  for (i = 0; i < extend_bottom; ++i) {
106
3.68M
    dst_ptr2 += src_stride;
107
3.68M
    memcpy(dst_ptr2, src_ptr2, linesize * sizeof(uint16_t));
108
3.68M
  }
109
12.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
17.4k
                         const int num_planes) {
149
17.4k
  const int ss_x = ybf->subsampling_x;
150
17.4k
  const int ss_y = ybf->subsampling_y;
151
152
17.4k
  assert(ybf->y_height - ybf->y_crop_height < 16);
153
17.4k
  assert(ybf->y_width - ybf->y_crop_width < 16);
154
17.4k
  assert(ybf->y_height - ybf->y_crop_height >= 0);
155
17.4k
  assert(ybf->y_width - ybf->y_crop_width >= 0);
156
157
17.4k
#if CONFIG_AV1_HIGHBITDEPTH
158
17.4k
  if (ybf->flags & YV12_FLAG_HIGHBITDEPTH) {
159
24.1k
    for (int plane = 0; plane < num_planes; ++plane) {
160
12.8k
      const int is_uv = plane > 0;
161
12.8k
      const int top = ext_size >> (is_uv ? ss_y : 0);
162
12.8k
      const int left = ext_size >> (is_uv ? ss_x : 0);
163
12.8k
      const int bottom = top + ybf->heights[is_uv] - ybf->crop_heights[is_uv];
164
12.8k
      const int right = left + ybf->widths[is_uv] - ybf->crop_widths[is_uv];
165
12.8k
      extend_plane_high(ybf->buffers[plane], ybf->strides[is_uv],
166
12.8k
                        ybf->crop_widths[is_uv], ybf->crop_heights[is_uv], top,
167
12.8k
                        left, bottom, right, 0, ybf->crop_heights[is_uv]);
168
12.8k
    }
169
11.3k
    return;
170
11.3k
  }
171
6.16k
#endif
172
173
23.4k
  for (int plane = 0; plane < num_planes; ++plane) {
174
17.2k
    const int is_uv = plane > 0;
175
17.2k
    const int top = ext_size >> (is_uv ? ss_y : 0);
176
17.2k
    const int left = ext_size >> (is_uv ? ss_x : 0);
177
17.2k
    const int bottom = top + ybf->heights[is_uv] - ybf->crop_heights[is_uv];
178
17.2k
    const int right = left + ybf->widths[is_uv] - ybf->crop_widths[is_uv];
179
17.2k
    extend_plane(ybf->buffers[plane], ybf->strides[is_uv],
180
17.2k
                 ybf->crop_widths[is_uv], ybf->crop_heights[is_uv], top, left,
181
17.2k
                 bottom, right, 0, ybf->crop_heights[is_uv]);
182
17.2k
  }
183
6.16k
}
184
185
17.4k
void aom_extend_frame_borders_c(YV12_BUFFER_CONFIG *ybf, const int num_planes) {
186
17.4k
  extend_frame(ybf, ybf->border, num_planes);
187
17.4k
}
188
189
#if CONFIG_AV1_HIGHBITDEPTH
190
1.42M
static void memcpy_short_addr(uint8_t *dst8, const uint8_t *src8, int num) {
191
1.42M
  uint16_t *dst = CONVERT_TO_SHORTPTR(dst8);
192
1.42M
  uint16_t *src = CONVERT_TO_SHORTPTR(src8);
193
1.42M
  memcpy(dst, src, num * sizeof(uint16_t));
194
1.42M
}
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
8.73k
                           YV12_BUFFER_CONFIG *dst_bc, const int num_planes) {
202
8.73k
  assert(src_bc->y_width == dst_bc->y_width);
203
8.73k
  assert(src_bc->y_height == dst_bc->y_height);
204
205
8.73k
#if CONFIG_AV1_HIGHBITDEPTH
206
8.73k
  assert((src_bc->flags & YV12_FLAG_HIGHBITDEPTH) ==
207
8.73k
         (dst_bc->flags & YV12_FLAG_HIGHBITDEPTH));
208
209
8.73k
  if (src_bc->flags & YV12_FLAG_HIGHBITDEPTH) {
210
12.0k
    for (int plane = 0; plane < num_planes; ++plane) {
211
6.42k
      const uint8_t *plane_src = src_bc->buffers[plane];
212
6.42k
      uint8_t *plane_dst = dst_bc->buffers[plane];
213
6.42k
      const int is_uv = plane > 0;
214
215
1.42M
      for (int row = 0; row < src_bc->heights[is_uv]; ++row) {
216
1.42M
        memcpy_short_addr(plane_dst, plane_src, src_bc->widths[is_uv]);
217
1.42M
        plane_src += src_bc->strides[is_uv];
218
1.42M
        plane_dst += dst_bc->strides[is_uv];
219
1.42M
      }
220
6.42k
    }
221
5.65k
    aom_extend_frame_borders_c(dst_bc, num_planes);
222
5.65k
    return;
223
5.65k
  }
224
3.08k
#endif
225
11.7k
  for (int plane = 0; plane < num_planes; ++plane) {
226
8.64k
    const uint8_t *plane_src = src_bc->buffers[plane];
227
8.64k
    uint8_t *plane_dst = dst_bc->buffers[plane];
228
8.64k
    const int is_uv = plane > 0;
229
230
704k
    for (int row = 0; row < src_bc->heights[is_uv]; ++row) {
231
696k
      memcpy(plane_dst, plane_src, src_bc->widths[is_uv]);
232
696k
      plane_src += src_bc->strides[is_uv];
233
696k
      plane_dst += dst_bc->strides[is_uv];
234
696k
    }
235
8.64k
  }
236
3.08k
  aom_extend_frame_borders_c(dst_bc, num_planes);
237
3.08k
}
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
19.4k
                               int vstart2) {
323
19.4k
  int row;
324
19.4k
  const uint8_t *src = src_ybc->y_buffer;
325
19.4k
  uint8_t *dst = dst_ybc->y_buffer;
326
19.4k
#if CONFIG_AV1_HIGHBITDEPTH
327
19.4k
  if (src_ybc->flags & YV12_FLAG_HIGHBITDEPTH) {
328
11.3k
    const uint16_t *src16 =
329
11.3k
        CONVERT_TO_SHORTPTR(src + vstart1 * src_ybc->y_stride + hstart1);
330
11.3k
    uint16_t *dst16 =
331
11.3k
        CONVERT_TO_SHORTPTR(dst + vstart2 * dst_ybc->y_stride + hstart2);
332
333
1.31M
    for (row = vstart1; row < vend1; ++row) {
334
1.30M
      memcpy(dst16, src16, (hend1 - hstart1) * sizeof(uint16_t));
335
1.30M
      src16 += src_ybc->y_stride;
336
1.30M
      dst16 += dst_ybc->y_stride;
337
1.30M
    }
338
11.3k
    return;
339
11.3k
  }
340
8.13k
#endif
341
8.13k
  src = (src + vstart1 * src_ybc->y_stride + hstart1);
342
8.13k
  dst = (dst + vstart2 * dst_ybc->y_stride + hstart2);
343
344
963k
  for (row = vstart1; row < vend1; ++row) {
345
955k
    memcpy(dst, src, (hend1 - hstart1));
346
955k
    src += src_ybc->y_stride;
347
955k
    dst += dst_ybc->y_stride;
348
955k
  }
349
8.13k
}
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
19.4k
                                     int hend, int vstart, int vend) {
354
19.4k
  aom_yv12_partial_copy_y_c(src_ybc, hstart, hend, vstart, vend, dst_ybc,
355
19.4k
                            hstart, vstart);
356
19.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
15.4k
                               int vstart2) {
362
15.4k
  int row;
363
15.4k
  const uint8_t *src = src_bc->u_buffer;
364
15.4k
  uint8_t *dst = dst_bc->u_buffer;
365
15.4k
#if CONFIG_AV1_HIGHBITDEPTH
366
15.4k
  if (src_bc->flags & YV12_FLAG_HIGHBITDEPTH) {
367
7.56k
    const uint16_t *src16 =
368
7.56k
        CONVERT_TO_SHORTPTR(src + vstart1 * src_bc->uv_stride + hstart1);
369
7.56k
    uint16_t *dst16 =
370
7.56k
        CONVERT_TO_SHORTPTR(dst + vstart2 * dst_bc->uv_stride + hstart2);
371
481k
    for (row = vstart1; row < vend1; ++row) {
372
474k
      memcpy(dst16, src16, (hend1 - hstart1) * sizeof(uint16_t));
373
474k
      src16 += src_bc->uv_stride;
374
474k
      dst16 += dst_bc->uv_stride;
375
474k
    }
376
7.56k
    return;
377
7.56k
  }
378
7.90k
#endif
379
7.90k
  src = (src + vstart1 * src_bc->uv_stride + hstart1);
380
7.90k
  dst = (dst + vstart2 * dst_bc->uv_stride + hstart2);
381
382
538k
  for (row = vstart1; row < vend1; ++row) {
383
530k
    memcpy(dst, src, (hend1 - hstart1));
384
530k
    src += src_bc->uv_stride;
385
530k
    dst += dst_bc->uv_stride;
386
530k
  }
387
7.90k
}
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
15.4k
                                     int hend, int vstart, int vend) {
392
15.4k
  aom_yv12_partial_copy_u_c(src_bc, hstart, hend, vstart, vend, dst_bc, hstart,
393
15.4k
                            vstart);
394
15.4k
}
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
8.56k
                               int vstart2) {
400
8.56k
  int row;
401
8.56k
  const uint8_t *src = src_bc->v_buffer;
402
8.56k
  uint8_t *dst = dst_bc->v_buffer;
403
8.56k
#if CONFIG_AV1_HIGHBITDEPTH
404
8.56k
  if (src_bc->flags & YV12_FLAG_HIGHBITDEPTH) {
405
4.32k
    const uint16_t *src16 =
406
4.32k
        CONVERT_TO_SHORTPTR(src + vstart1 * src_bc->uv_stride + hstart1);
407
4.32k
    uint16_t *dst16 =
408
4.32k
        CONVERT_TO_SHORTPTR(dst + vstart2 * dst_bc->uv_stride + hstart2);
409
332k
    for (row = vstart1; row < vend1; ++row) {
410
328k
      memcpy(dst16, src16, (hend1 - hstart1) * sizeof(uint16_t));
411
328k
      src16 += src_bc->uv_stride;
412
328k
      dst16 += dst_bc->uv_stride;
413
328k
    }
414
4.32k
    return;
415
4.32k
  }
416
4.24k
#endif
417
4.24k
  src = (src + vstart1 * src_bc->uv_stride + hstart1);
418
4.24k
  dst = (dst + vstart2 * dst_bc->uv_stride + hstart2);
419
420
282k
  for (row = vstart1; row < vend1; ++row) {
421
278k
    memcpy(dst, src, (hend1 - hstart1));
422
278k
    src += src_bc->uv_stride;
423
278k
    dst += dst_bc->uv_stride;
424
278k
  }
425
4.24k
}
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
8.57k
                                     int hend, int vstart, int vend) {
430
8.57k
  aom_yv12_partial_copy_v_c(src_bc, hstart, hend, vstart, vend, dst_bc, hstart,
431
8.57k
                            vstart);
432
8.57k
}
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
}