|  | /* | 
|  | *  Copyright (c) 2015 The WebRTC project authors. All Rights Reserved. | 
|  | * | 
|  | *  Use of this source code is governed by a BSD-style license | 
|  | *  that can be found in the LICENSE file in the root of the source | 
|  | *  tree. An additional intellectual property rights grant can be found | 
|  | *  in the file PATENTS.  All contributing project authors may | 
|  | *  be found in the AUTHORS file in the root of the source tree. | 
|  | * | 
|  | */ | 
|  |  | 
|  | #ifdef RTC_ENABLE_VP9 | 
|  |  | 
|  | #include "modules/video_coding/codecs/vp9/vp9_frame_buffer_pool.h" | 
|  |  | 
|  | #include "rtc_base/checks.h" | 
|  | #include "rtc_base/logging.h" | 
|  | #include "vpx/vpx_codec.h" | 
|  | #include "vpx/vpx_decoder.h" | 
|  | #include "vpx/vpx_frame_buffer.h" | 
|  |  | 
|  | namespace webrtc { | 
|  |  | 
|  | uint8_t* Vp9FrameBufferPool::Vp9FrameBuffer::GetData() { | 
|  | return data_.data<uint8_t>(); | 
|  | } | 
|  |  | 
|  | size_t Vp9FrameBufferPool::Vp9FrameBuffer::GetDataSize() const { | 
|  | return data_.size(); | 
|  | } | 
|  |  | 
|  | void Vp9FrameBufferPool::Vp9FrameBuffer::SetSize(size_t size) { | 
|  | data_.SetSize(size); | 
|  | } | 
|  |  | 
|  | bool Vp9FrameBufferPool::InitializeVpxUsePool( | 
|  | vpx_codec_ctx* vpx_codec_context) { | 
|  | RTC_DCHECK(vpx_codec_context); | 
|  | // Tell libvpx to use this pool. | 
|  | if (vpx_codec_set_frame_buffer_functions( | 
|  | // In which context to use these callback functions. | 
|  | vpx_codec_context, | 
|  | // Called by libvpx when it needs another frame buffer. | 
|  | &Vp9FrameBufferPool::VpxGetFrameBuffer, | 
|  | // Called by libvpx when it no longer uses a frame buffer. | 
|  | &Vp9FrameBufferPool::VpxReleaseFrameBuffer, | 
|  | // `this` will be passed as `user_priv` to VpxGetFrameBuffer. | 
|  | this)) { | 
|  | // Failed to configure libvpx to use Vp9FrameBufferPool. | 
|  | return false; | 
|  | } | 
|  | return true; | 
|  | } | 
|  |  | 
|  | rtc::scoped_refptr<Vp9FrameBufferPool::Vp9FrameBuffer> | 
|  | Vp9FrameBufferPool::GetFrameBuffer(size_t min_size) { | 
|  | RTC_DCHECK_GT(min_size, 0); | 
|  | rtc::scoped_refptr<Vp9FrameBuffer> available_buffer = nullptr; | 
|  | { | 
|  | MutexLock lock(&buffers_lock_); | 
|  | // Do we have a buffer we can recycle? | 
|  | for (const auto& buffer : allocated_buffers_) { | 
|  | if (buffer->HasOneRef()) { | 
|  | available_buffer = buffer; | 
|  | break; | 
|  | } | 
|  | } | 
|  | // Otherwise create one. | 
|  | if (available_buffer == nullptr) { | 
|  | available_buffer = new Vp9FrameBuffer(); | 
|  | allocated_buffers_.push_back(available_buffer); | 
|  | if (allocated_buffers_.size() > max_num_buffers_) { | 
|  | RTC_LOG(LS_WARNING) | 
|  | << allocated_buffers_.size() | 
|  | << " Vp9FrameBuffers have been " | 
|  | "allocated by a Vp9FrameBufferPool (exceeding what is " | 
|  | "considered reasonable, " | 
|  | << max_num_buffers_ << ")."; | 
|  |  | 
|  | // TODO(phoglund): this limit is being hit in tests since Oct 5 2016. | 
|  | // See https://bugs.chromium.org/p/webrtc/issues/detail?id=6484. | 
|  | // RTC_DCHECK_NOTREACHED(); | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | available_buffer->SetSize(min_size); | 
|  | return available_buffer; | 
|  | } | 
|  |  | 
|  | int Vp9FrameBufferPool::GetNumBuffersInUse() const { | 
|  | int num_buffers_in_use = 0; | 
|  | MutexLock lock(&buffers_lock_); | 
|  | for (const auto& buffer : allocated_buffers_) { | 
|  | if (!buffer->HasOneRef()) | 
|  | ++num_buffers_in_use; | 
|  | } | 
|  | return num_buffers_in_use; | 
|  | } | 
|  |  | 
|  | bool Vp9FrameBufferPool::Resize(size_t max_number_of_buffers) { | 
|  | MutexLock lock(&buffers_lock_); | 
|  | size_t used_buffers_count = 0; | 
|  | for (const auto& buffer : allocated_buffers_) { | 
|  | // If the buffer is in use, the ref count will be >= 2, one from the list we | 
|  | // are looping over and one from the application. If the ref count is 1, | 
|  | // then the list we are looping over holds the only reference and it's safe | 
|  | // to reuse. | 
|  | if (!buffer->HasOneRef()) { | 
|  | used_buffers_count++; | 
|  | } | 
|  | } | 
|  | if (used_buffers_count > max_number_of_buffers) { | 
|  | return false; | 
|  | } | 
|  | max_num_buffers_ = max_number_of_buffers; | 
|  |  | 
|  | size_t buffers_to_purge = allocated_buffers_.size() - max_num_buffers_; | 
|  | auto iter = allocated_buffers_.begin(); | 
|  | while (iter != allocated_buffers_.end() && buffers_to_purge > 0) { | 
|  | if ((*iter)->HasOneRef()) { | 
|  | iter = allocated_buffers_.erase(iter); | 
|  | buffers_to_purge--; | 
|  | } else { | 
|  | ++iter; | 
|  | } | 
|  | } | 
|  | return true; | 
|  | } | 
|  |  | 
|  | void Vp9FrameBufferPool::ClearPool() { | 
|  | MutexLock lock(&buffers_lock_); | 
|  | allocated_buffers_.clear(); | 
|  | } | 
|  |  | 
|  | // static | 
|  | int32_t Vp9FrameBufferPool::VpxGetFrameBuffer(void* user_priv, | 
|  | size_t min_size, | 
|  | vpx_codec_frame_buffer* fb) { | 
|  | RTC_DCHECK(user_priv); | 
|  | RTC_DCHECK(fb); | 
|  |  | 
|  | #ifdef FUZZING_BUILD_MODE_UNSAFE_FOR_PRODUCTION | 
|  | // Limit size of 8k YUV highdef frame | 
|  | size_t size_limit = 7680 * 4320 * 3 / 2 * 2; | 
|  | if (min_size > size_limit) | 
|  | return -1; | 
|  | #endif | 
|  |  | 
|  | Vp9FrameBufferPool* pool = static_cast<Vp9FrameBufferPool*>(user_priv); | 
|  |  | 
|  | rtc::scoped_refptr<Vp9FrameBuffer> buffer = pool->GetFrameBuffer(min_size); | 
|  | fb->data = buffer->GetData(); | 
|  | fb->size = buffer->GetDataSize(); | 
|  | // Store Vp9FrameBuffer* in `priv` for use in VpxReleaseFrameBuffer. | 
|  | // This also makes vpx_codec_get_frame return images with their `fb_priv` set | 
|  | // to `buffer` which is important for external reference counting. | 
|  | // Release from refptr so that the buffer's `ref_count_` remains 1 when | 
|  | // `buffer` goes out of scope. | 
|  | fb->priv = static_cast<void*>(buffer.release()); | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | // static | 
|  | int32_t Vp9FrameBufferPool::VpxReleaseFrameBuffer(void* user_priv, | 
|  | vpx_codec_frame_buffer* fb) { | 
|  | RTC_DCHECK(user_priv); | 
|  | RTC_DCHECK(fb); | 
|  | Vp9FrameBuffer* buffer = static_cast<Vp9FrameBuffer*>(fb->priv); | 
|  | if (buffer != nullptr) { | 
|  | buffer->Release(); | 
|  | // When libvpx fails to decode and you continue to try to decode (and fail) | 
|  | // libvpx can for some reason try to release the same buffer multiple times. | 
|  | // Setting `priv` to null protects against trying to Release multiple times. | 
|  | fb->priv = nullptr; | 
|  | } | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | }  // namespace webrtc | 
|  |  | 
|  | #endif  // RTC_ENABLE_VP9 |