/* * 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. */ #include "common_video/include/i420_buffer_pool.h" #include #include "rtc_base/checks.h" namespace webrtc { I420BufferPool::I420BufferPool() : I420BufferPool(false) {} I420BufferPool::I420BufferPool(bool zero_initialize) : I420BufferPool(zero_initialize, std::numeric_limits::max()) {} I420BufferPool::I420BufferPool(bool zero_initialize, size_t max_number_of_buffers) : zero_initialize_(zero_initialize), max_number_of_buffers_(max_number_of_buffers) {} I420BufferPool::~I420BufferPool() = default; void I420BufferPool::Release() { buffers_.clear(); } bool I420BufferPool::Resize(size_t max_number_of_buffers) { RTC_DCHECK_RUNS_SERIALIZED(&race_checker_); size_t used_buffers_count = 0; for (const rtc::scoped_refptr& buffer : 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_number_of_buffers_ = max_number_of_buffers; size_t buffers_to_purge = buffers_.size() - max_number_of_buffers_; auto iter = buffers_.begin(); while (iter != buffers_.end() && buffers_to_purge > 0) { if ((*iter)->HasOneRef()) { iter = buffers_.erase(iter); buffers_to_purge--; } else { ++iter; } } return true; } rtc::scoped_refptr I420BufferPool::CreateBuffer(int width, int height) { // Default stride_y is width, default uv stride is width / 2 (rounding up). return CreateBuffer(width, height, width, (width + 1) / 2, (width + 1) / 2); } rtc::scoped_refptr I420BufferPool::CreateBuffer(int width, int height, int stride_y, int stride_u, int stride_v) { RTC_DCHECK_RUNS_SERIALIZED(&race_checker_); // Release buffers with wrong resolution. for (auto it = buffers_.begin(); it != buffers_.end();) { const auto& buffer = *it; if (buffer->width() != width || buffer->height() != height || buffer->StrideY() != stride_y || buffer->StrideU() != stride_u || buffer->StrideV() != stride_v) { it = buffers_.erase(it); } else { ++it; } } // Look for a free buffer. for (const rtc::scoped_refptr& buffer : 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()) return buffer; } if (buffers_.size() >= max_number_of_buffers_) return nullptr; // Allocate new buffer. rtc::scoped_refptr buffer = new PooledI420Buffer(width, height, stride_y, stride_u, stride_v); if (zero_initialize_) buffer->InitializeData(); buffers_.push_back(buffer); return buffer; } } // namespace webrtc