107 lines
3.8 KiB
C++
107 lines
3.8 KiB
C++
|
/*
|
||
|
* 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 <limits>
|
||
|
|
||
|
#include "rtc_base/checks.h"
|
||
|
|
||
|
namespace webrtc {
|
||
|
|
||
|
I420BufferPool::I420BufferPool() : I420BufferPool(false) {}
|
||
|
I420BufferPool::I420BufferPool(bool zero_initialize)
|
||
|
: I420BufferPool(zero_initialize, std::numeric_limits<size_t>::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<PooledI420Buffer>& 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<I420Buffer> 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<I420Buffer> 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<PooledI420Buffer>& 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<PooledI420Buffer> buffer =
|
||
|
new PooledI420Buffer(width, height, stride_y, stride_u, stride_v);
|
||
|
if (zero_initialize_)
|
||
|
buffer->InitializeData();
|
||
|
buffers_.push_back(buffer);
|
||
|
return buffer;
|
||
|
}
|
||
|
|
||
|
} // namespace webrtc
|