/* * 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 "webrtc/common_video/include/i420_buffer_pool.h" #include "webrtc/base/checks.h" namespace { // One extra indirection is needed to make |HasOneRef| work. class PooledI420Buffer : public webrtc::VideoFrameBuffer { public: explicit PooledI420Buffer( const rtc::scoped_refptr& buffer) : buffer_(buffer) {} private: ~PooledI420Buffer() override {} int width() const override { return buffer_->width(); } int height() const override { return buffer_->height(); } const uint8_t* DataY() const override { return buffer_->DataY(); } const uint8_t* DataU() const override { return buffer_->DataU(); } const uint8_t* DataV() const override { return buffer_->DataV(); } bool IsMutable() override { return HasOneRef(); } // Make the IsMutable() check here instead of in |buffer_|, because the pool // also has a reference to |buffer_|. uint8_t* MutableDataY() override { RTC_DCHECK(IsMutable()); return const_cast(buffer_->DataY()); } uint8_t* MutableDataU() override { RTC_DCHECK(IsMutable()); return const_cast(buffer_->DataU()); } uint8_t* MutableDataV() override { RTC_DCHECK(IsMutable()); return const_cast(buffer_->DataV()); } int StrideY() const override { return buffer_->StrideY(); } int StrideU() const override { return buffer_->StrideU(); } int StrideV() const override { return buffer_->StrideV(); } void* native_handle() const override { return nullptr; } rtc::scoped_refptr NativeToI420Buffer() override { RTC_NOTREACHED(); return nullptr; } friend class rtc::RefCountedObject; rtc::scoped_refptr buffer_; }; } // namespace namespace webrtc { I420BufferPool::I420BufferPool(bool zero_initialize) : zero_initialize_(zero_initialize) { Release(); } void I420BufferPool::Release() { thread_checker_.DetachFromThread(); buffers_.clear(); } rtc::scoped_refptr I420BufferPool::CreateBuffer(int width, int height) { RTC_DCHECK(thread_checker_.CalledOnValidThread()); // Release buffers with wrong resolution. for (auto it = buffers_.begin(); it != buffers_.end();) { if ((*it)->width() != width || (*it)->height() != height) 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 a PooledI420Buffer returned from // CreateBuffer that has not been released yet. If the ref count is 1 // (HasOneRef), then the list we are looping over holds the only reference // and it's safe to reuse. if (buffer->IsMutable()) return new rtc::RefCountedObject(buffer); } // Allocate new buffer. rtc::scoped_refptr buffer = new rtc::RefCountedObject( width, height); if (zero_initialize_) buffer->InitializeData(); buffers_.push_back(buffer); return new rtc::RefCountedObject(buffers_.back()); } } // namespace webrtc