| /* |
| * 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<webrtc::I420Buffer>& 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<uint8_t*>(buffer_->DataY()); |
| } |
| uint8_t* MutableDataU() override { |
| RTC_DCHECK(IsMutable()); |
| return const_cast<uint8_t*>(buffer_->DataU()); |
| } |
| uint8_t* MutableDataV() override { |
| RTC_DCHECK(IsMutable()); |
| return const_cast<uint8_t*>(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<VideoFrameBuffer> NativeToI420Buffer() override { |
| RTC_NOTREACHED(); |
| return nullptr; |
| } |
| |
| friend class rtc::RefCountedObject<PooledI420Buffer>; |
| rtc::scoped_refptr<webrtc::I420Buffer> 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<VideoFrameBuffer> 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<I420Buffer>& 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<PooledI420Buffer>(buffer); |
| } |
| // Allocate new buffer. |
| rtc::scoped_refptr<I420Buffer> buffer = new rtc::RefCountedObject<I420Buffer>( |
| width, height); |
| if (zero_initialize_) |
| buffer->InitializeData(); |
| buffers_.push_back(buffer); |
| return new rtc::RefCountedObject<PooledI420Buffer>(buffers_.back()); |
| } |
| |
| } // namespace webrtc |