blob: d13da6a172a90454dc5ea7a48f6afa21cf55e163 [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
Mirko Bonadei92ea95e2017-09-15 06:47:31 +020011#include "common_video/include/i420_buffer_pool.h"
magjed@webrtc.org73d763e2015-03-17 11:40:45 +000012
Yves Gerey988cc082018-10-23 12:03:01 +020013#include <limits>
14
Mirko Bonadei92ea95e2017-09-15 06:47:31 +020015#include "rtc_base/checks.h"
magjed@webrtc.org73d763e2015-03-17 11:40:45 +000016
magjed@webrtc.org73d763e2015-03-17 11:40:45 +000017namespace webrtc {
18
Paulina Hensmana680a6a2018-04-05 11:42:24 +020019I420BufferPool::I420BufferPool() : I420BufferPool(false) {}
20I420BufferPool::I420BufferPool(bool zero_initialize)
21 : I420BufferPool(zero_initialize, std::numeric_limits<size_t>::max()) {}
Per00983572016-11-04 08:57:26 +010022I420BufferPool::I420BufferPool(bool zero_initialize,
23 size_t max_number_of_buffers)
24 : zero_initialize_(zero_initialize),
25 max_number_of_buffers_(max_number_of_buffers) {}
Paulina Hensmana680a6a2018-04-05 11:42:24 +020026I420BufferPool::~I420BufferPool() = default;
pbos@webrtc.orga3209a22015-03-20 13:35:56 +000027
28void I420BufferPool::Release() {
pbos@webrtc.orga3209a22015-03-20 13:35:56 +000029 buffers_.clear();
magjed@webrtc.org73d763e2015-03-17 11:40:45 +000030}
31
Artem Titov8525a802019-12-16 11:41:57 +010032bool I420BufferPool::Resize(size_t max_number_of_buffers) {
33 RTC_DCHECK_RUNS_SERIALIZED(&race_checker_);
34 size_t used_buffers_count = 0;
35 for (const rtc::scoped_refptr<PooledI420Buffer>& buffer : buffers_) {
36 // If the buffer is in use, the ref count will be >= 2, one from the list we
37 // are looping over and one from the application. If the ref count is 1,
38 // then the list we are looping over holds the only reference and it's safe
39 // to reuse.
40 if (!buffer->HasOneRef()) {
41 used_buffers_count++;
42 }
43 }
44 if (used_buffers_count > max_number_of_buffers) {
45 return false;
46 }
47 max_number_of_buffers_ = max_number_of_buffers;
48
49 size_t buffers_to_purge = buffers_.size() - max_number_of_buffers_;
50 auto iter = buffers_.begin();
51 while (iter != buffers_.end() && buffers_to_purge > 0) {
52 if ((*iter)->HasOneRef()) {
53 iter = buffers_.erase(iter);
54 buffers_to_purge--;
55 } else {
56 ++iter;
57 }
58 }
59 return true;
60}
61
nissed591e3f2016-05-26 06:49:55 -070062rtc::scoped_refptr<I420Buffer> I420BufferPool::CreateBuffer(int width,
63 int height) {
Noah Richards408a3c62019-04-01 10:14:47 -070064 // Default stride_y is width, default uv stride is width / 2 (rounding up).
65 return CreateBuffer(width, height, width, (width + 1) / 2, (width + 1) / 2);
66}
67
68rtc::scoped_refptr<I420Buffer> I420BufferPool::CreateBuffer(int width,
69 int height,
70 int stride_y,
71 int stride_u,
72 int stride_v) {
Magnus Jedvert7fd92862016-09-17 11:27:35 +020073 RTC_DCHECK_RUNS_SERIALIZED(&race_checker_);
magjed@webrtc.org73d763e2015-03-17 11:40:45 +000074 // Release buffers with wrong resolution.
75 for (auto it = buffers_.begin(); it != buffers_.end();) {
Noah Richards408a3c62019-04-01 10:14:47 -070076 const auto& buffer = *it;
77 if (buffer->width() != width || buffer->height() != height ||
78 buffer->StrideY() != stride_y || buffer->StrideU() != stride_u ||
79 buffer->StrideV() != stride_v) {
magjed@webrtc.org73d763e2015-03-17 11:40:45 +000080 it = buffers_.erase(it);
Noah Richards408a3c62019-04-01 10:14:47 -070081 } else {
magjed@webrtc.org73d763e2015-03-17 11:40:45 +000082 ++it;
Noah Richards408a3c62019-04-01 10:14:47 -070083 }
magjed@webrtc.org73d763e2015-03-17 11:40:45 +000084 }
85 // Look for a free buffer.
nissed591e3f2016-05-26 06:49:55 -070086 for (const rtc::scoped_refptr<PooledI420Buffer>& buffer : buffers_) {
87 // If the buffer is in use, the ref count will be >= 2, one from the list we
88 // are looping over and one from the application. If the ref count is 1,
89 // then the list we are looping over holds the only reference and it's safe
90 // to reuse.
91 if (buffer->HasOneRef())
92 return buffer;
magjed@webrtc.org73d763e2015-03-17 11:40:45 +000093 }
Per00983572016-11-04 08:57:26 +010094
95 if (buffers_.size() >= max_number_of_buffers_)
96 return nullptr;
magjed@webrtc.org73d763e2015-03-17 11:40:45 +000097 // Allocate new buffer.
nissed591e3f2016-05-26 06:49:55 -070098 rtc::scoped_refptr<PooledI420Buffer> buffer =
Noah Richards408a3c62019-04-01 10:14:47 -070099 new PooledI420Buffer(width, height, stride_y, stride_u, stride_v);
hbos900f9752016-02-05 08:08:34 -0800100 if (zero_initialize_)
101 buffer->InitializeData();
102 buffers_.push_back(buffer);
nissed591e3f2016-05-26 06:49:55 -0700103 return buffer;
magjed@webrtc.org73d763e2015-03-17 11:40:45 +0000104}
105
106} // namespace webrtc