|  | /* | 
|  | * Copyright (c) 2021, Alliance for Open Media. All rights reserved | 
|  | * | 
|  | * This source code is subject to the terms of the BSD 3-Clause Clear License | 
|  | * and the Alliance for Open Media Patent License 1.0. If the BSD 3-Clause Clear | 
|  | * License was not distributed with this source code in the LICENSE file, you | 
|  | * can obtain it at aomedia.org/license/software-license/bsd-3-c-c/.  If the | 
|  | * Alliance for Open Media Patent License 1.0 was not distributed with this | 
|  | * source code in the PATENTS file, you can obtain it at | 
|  | * aomedia.org/license/patent-license/. | 
|  | */ | 
|  |  | 
|  | #include <assert.h> | 
|  |  | 
|  | #include "av1/common/frame_buffers.h" | 
|  | #include "aom_mem/aom_mem.h" | 
|  |  | 
|  | int av1_alloc_internal_frame_buffers(InternalFrameBufferList *list) { | 
|  | assert(list != NULL); | 
|  | av1_free_internal_frame_buffers(list); | 
|  |  | 
|  | // When all reference frames at frame buffers are output at the same time | 
|  | // and these frames use film grain synthesis, the total number of required | 
|  | // frame buffers is (total references numbers + current frame) * 2 + working | 
|  | // buffers for multh-threads | 
|  | list->num_internal_frame_buffers = | 
|  | (AOM_MAXIMUM_REF_BUFFERS + 1) * 2 + AOM_MAXIMUM_WORK_BUFFERS; | 
|  | list->int_fb = (InternalFrameBuffer *)aom_calloc( | 
|  | list->num_internal_frame_buffers, sizeof(*list->int_fb)); | 
|  | if (list->int_fb == NULL) { | 
|  | list->num_internal_frame_buffers = 0; | 
|  | return 1; | 
|  | } | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | void av1_free_internal_frame_buffers(InternalFrameBufferList *list) { | 
|  | int i; | 
|  |  | 
|  | assert(list != NULL); | 
|  |  | 
|  | for (i = 0; i < list->num_internal_frame_buffers; ++i) { | 
|  | aom_free(list->int_fb[i].data); | 
|  | list->int_fb[i].data = NULL; | 
|  | } | 
|  | aom_free(list->int_fb); | 
|  | list->int_fb = NULL; | 
|  | list->num_internal_frame_buffers = 0; | 
|  | } | 
|  |  | 
|  | void av1_zero_unused_internal_frame_buffers(InternalFrameBufferList *list) { | 
|  | int i; | 
|  |  | 
|  | assert(list != NULL); | 
|  |  | 
|  | for (i = 0; i < list->num_internal_frame_buffers; ++i) { | 
|  | if (list->int_fb[i].data && !list->int_fb[i].in_use) | 
|  | memset(list->int_fb[i].data, 0, list->int_fb[i].size); | 
|  | } | 
|  | } | 
|  |  | 
|  | int av1_get_frame_buffer(void *cb_priv, size_t min_size, | 
|  | aom_codec_frame_buffer_t *fb) { | 
|  | int i; | 
|  | InternalFrameBufferList *const int_fb_list = | 
|  | (InternalFrameBufferList *)cb_priv; | 
|  | if (int_fb_list == NULL) return -1; | 
|  |  | 
|  | // Find a free frame buffer. | 
|  | for (i = 0; i < int_fb_list->num_internal_frame_buffers; ++i) { | 
|  | if (!int_fb_list->int_fb[i].in_use) break; | 
|  | } | 
|  |  | 
|  | if (i == int_fb_list->num_internal_frame_buffers) return -1; | 
|  |  | 
|  | if (int_fb_list->int_fb[i].size < min_size) { | 
|  | aom_free(int_fb_list->int_fb[i].data); | 
|  | // The data must be zeroed to fix a valgrind error from the C loop filter | 
|  | // due to access uninitialized memory in frame border. It could be | 
|  | // skipped if border were totally removed. | 
|  | int_fb_list->int_fb[i].data = (uint8_t *)aom_calloc(1, min_size); | 
|  | if (!int_fb_list->int_fb[i].data) { | 
|  | int_fb_list->int_fb[i].size = 0; | 
|  | return -1; | 
|  | } | 
|  | int_fb_list->int_fb[i].size = min_size; | 
|  | } | 
|  |  | 
|  | fb->data = int_fb_list->int_fb[i].data; | 
|  | fb->size = int_fb_list->int_fb[i].size; | 
|  | int_fb_list->int_fb[i].in_use = 1; | 
|  |  | 
|  | // Set the frame buffer's private data to point at the internal frame buffer. | 
|  | fb->priv = &int_fb_list->int_fb[i]; | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | int av1_release_frame_buffer(void *cb_priv, aom_codec_frame_buffer_t *fb) { | 
|  | InternalFrameBuffer *const int_fb = (InternalFrameBuffer *)fb->priv; | 
|  | (void)cb_priv; | 
|  | if (int_fb) int_fb->in_use = 0; | 
|  | return 0; | 
|  | } |