blob: 6de5081b947c1ba9ca3d4b337f8144cd4be52887 [file] [log] [blame]
/*
* Copyright (c) 2020 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 "modules/video_coding/frame_dependencies_calculator.h"
#include <stdint.h>
#include <iterator>
#include <set>
#include "absl/algorithm/container.h"
#include "absl/container/inlined_vector.h"
#include "api/array_view.h"
#include "api/video/video_frame_type.h"
#include "rtc_base/checks.h"
#include "rtc_base/logging.h"
namespace webrtc {
absl::InlinedVector<int64_t, 5> FrameDependenciesCalculator::FromBuffersUsage(
VideoFrameType frame_type,
int64_t frame_id,
rtc::ArrayView<const CodecBufferUsage> buffers_usage) {
absl::InlinedVector<int64_t, 5> dependencies;
RTC_DCHECK_GT(buffers_usage.size(), 0);
for (const CodecBufferUsage& buffer_usage : buffers_usage) {
RTC_CHECK_GE(buffer_usage.id, 0);
if (buffers_.size() <= static_cast<size_t>(buffer_usage.id)) {
buffers_.resize(buffer_usage.id + 1);
}
}
std::set<int64_t> direct_depenendencies;
std::set<int64_t> indirect_depenendencies;
if (frame_type == VideoFrameType::kVideoFrameDelta) {
for (const CodecBufferUsage& buffer_usage : buffers_usage) {
if (!buffer_usage.referenced) {
continue;
}
const BufferUsage& buffer = buffers_[buffer_usage.id];
if (buffer.frame_id == absl::nullopt) {
RTC_LOG(LS_ERROR) << "Odd configuration: frame " << frame_id
<< " references buffer #" << buffer_usage.id
<< " that was never updated.";
continue;
}
direct_depenendencies.insert(*buffer.frame_id);
indirect_depenendencies.insert(buffer.dependencies.begin(),
buffer.dependencies.end());
}
// Reduce references: if frame #3 depends on frame #2 and #1, and frame #2
// depends on frame #1, then frame #3 needs to depend just on frame #2.
// Though this set diff removes only 1 level of indirection, it seems
// enough for all currently used structures.
absl::c_set_difference(direct_depenendencies, indirect_depenendencies,
std::back_inserter(dependencies));
}
// Update buffers.
for (const CodecBufferUsage& buffer_usage : buffers_usage) {
if (!buffer_usage.updated) {
continue;
}
BufferUsage& buffer = buffers_[buffer_usage.id];
buffer.frame_id = frame_id;
buffer.dependencies.assign(direct_depenendencies.begin(),
direct_depenendencies.end());
}
return dependencies;
}
} // namespace webrtc