blob: 98daec99f66fcf067ae75ad8998940c0f882f037 [file] [log] [blame]
magjed@webrtc.org73d763e2015-03-17 11:40:45 +00001/*
2 * Copyright (c) 2015 The WebRTC project authors. All Rights Reserved.
3 *
4 * Use of this source code is governed by a BSD-style license
5 * that can be found in the LICENSE file in the root of the source
6 * tree. An additional intellectual property rights grant can be found
7 * in the file PATENTS. All contributing project authors may
8 * be found in the AUTHORS file in the root of the source tree.
9 */
10
kjellander6f8ce062015-11-16 13:52:24 -080011#include "webrtc/common_video/include/i420_buffer_pool.h"
magjed@webrtc.org73d763e2015-03-17 11:40:45 +000012
13#include "webrtc/base/checks.h"
14
15namespace {
16
17// One extra indirection is needed to make |HasOneRef| work.
18class PooledI420Buffer : public webrtc::VideoFrameBuffer {
19 public:
20 explicit PooledI420Buffer(
21 const rtc::scoped_refptr<webrtc::I420Buffer>& buffer)
22 : buffer_(buffer) {}
23
24 private:
25 ~PooledI420Buffer() override {}
26
27 int width() const override { return buffer_->width(); }
28 int height() const override { return buffer_->height(); }
29 const uint8_t* data(webrtc::PlaneType type) const override {
Magnus Jedvert3318f982015-08-26 16:06:21 +020030 return buffer_->data(type);
magjed@webrtc.org73d763e2015-03-17 11:40:45 +000031 }
Magnus Jedvert3318f982015-08-26 16:06:21 +020032 uint8_t* MutableData(webrtc::PlaneType type) override {
33 // Make the HasOneRef() check here instead of in |buffer_|, because the pool
34 // also has a reference to |buffer_|.
henrikg91d6ede2015-09-17 00:24:34 -070035 RTC_DCHECK(HasOneRef());
Magnus Jedvert3318f982015-08-26 16:06:21 +020036 return const_cast<uint8_t*>(buffer_->data(type));
magjed@webrtc.org73d763e2015-03-17 11:40:45 +000037 }
38 int stride(webrtc::PlaneType type) const override {
39 return buffer_->stride(type);
40 }
Per9b3f56e2015-04-09 13:44:16 +020041 void* native_handle() const override { return nullptr; }
magjed@webrtc.org73d763e2015-03-17 11:40:45 +000042
Peter Boströmeb66e802015-06-05 11:08:03 +020043 rtc::scoped_refptr<VideoFrameBuffer> NativeToI420Buffer() override {
44 RTC_NOTREACHED();
45 return nullptr;
46 }
47
magjed@webrtc.org73d763e2015-03-17 11:40:45 +000048 friend class rtc::RefCountedObject<PooledI420Buffer>;
49 rtc::scoped_refptr<webrtc::I420Buffer> buffer_;
50};
51
52} // namespace
53
54namespace webrtc {
55
56I420BufferPool::I420BufferPool() {
pbos@webrtc.orga3209a22015-03-20 13:35:56 +000057 Release();
58}
59
60void I420BufferPool::Release() {
magjed@webrtc.org73d763e2015-03-17 11:40:45 +000061 thread_checker_.DetachFromThread();
pbos@webrtc.orga3209a22015-03-20 13:35:56 +000062 buffers_.clear();
magjed@webrtc.org73d763e2015-03-17 11:40:45 +000063}
64
65rtc::scoped_refptr<VideoFrameBuffer> I420BufferPool::CreateBuffer(int width,
66 int height) {
henrikg91d6ede2015-09-17 00:24:34 -070067 RTC_DCHECK(thread_checker_.CalledOnValidThread());
magjed@webrtc.org73d763e2015-03-17 11:40:45 +000068 // Release buffers with wrong resolution.
69 for (auto it = buffers_.begin(); it != buffers_.end();) {
70 if ((*it)->width() != width || (*it)->height() != height)
71 it = buffers_.erase(it);
72 else
73 ++it;
74 }
75 // Look for a free buffer.
76 for (const rtc::scoped_refptr<I420Buffer>& buffer : buffers_) {
77 // If the buffer is in use, the ref count will be 2, one from the list we
78 // are looping over and one from a PooledI420Buffer returned from
79 // CreateBuffer that has not been released yet. If the ref count is 1
80 // (HasOneRef), then the list we are looping over holds the only reference
81 // and it's safe to reuse.
82 if (buffer->HasOneRef())
83 return new rtc::RefCountedObject<PooledI420Buffer>(buffer);
84 }
85 // Allocate new buffer.
86 buffers_.push_back(new rtc::RefCountedObject<I420Buffer>(width, height));
87 return new rtc::RefCountedObject<PooledI420Buffer>(buffers_.back());
88}
89
90} // namespace webrtc