Coverage Report

Created: 2026-04-01 07:49

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