mirror of
https://github.com/mollyim/webrtc.git
synced 2025-05-12 21:30:45 +01:00
254 lines
8.8 KiB
C++
254 lines
8.8 KiB
C++
/*
|
|
* 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/rtp_vp8_ref_finder.h"
|
|
|
|
#include <utility>
|
|
|
|
#include "rtc_base/logging.h"
|
|
|
|
namespace webrtc {
|
|
|
|
RtpFrameReferenceFinder::ReturnVector RtpVp8RefFinder::ManageFrame(
|
|
std::unique_ptr<RtpFrameObject> frame) {
|
|
const RTPVideoHeaderVP8& codec_header = absl::get<RTPVideoHeaderVP8>(
|
|
frame->GetRtpVideoHeader().video_type_header);
|
|
|
|
if (codec_header.temporalIdx != kNoTemporalIdx)
|
|
frame->SetTemporalIndex(codec_header.temporalIdx);
|
|
|
|
int64_t unwrapped_tl0 = tl0_unwrapper_.Unwrap(codec_header.tl0PicIdx & 0xFF);
|
|
FrameDecision decision =
|
|
ManageFrameInternal(frame.get(), codec_header, unwrapped_tl0);
|
|
|
|
RtpFrameReferenceFinder::ReturnVector res;
|
|
switch (decision) {
|
|
case kStash:
|
|
if (stashed_frames_.size() > kMaxStashedFrames) {
|
|
stashed_frames_.pop_back();
|
|
}
|
|
stashed_frames_.push_front(
|
|
{.unwrapped_tl0 = unwrapped_tl0, .frame = std::move(frame)});
|
|
return res;
|
|
case kHandOff:
|
|
res.push_back(std::move(frame));
|
|
RetryStashedFrames(res);
|
|
return res;
|
|
case kDrop:
|
|
return res;
|
|
}
|
|
|
|
return res;
|
|
}
|
|
|
|
RtpVp8RefFinder::FrameDecision RtpVp8RefFinder::ManageFrameInternal(
|
|
RtpFrameObject* frame,
|
|
const RTPVideoHeaderVP8& codec_header,
|
|
int64_t unwrapped_tl0) {
|
|
// Protect against corrupted packets with arbitrary large temporal idx.
|
|
if (codec_header.temporalIdx >= kMaxTemporalLayers)
|
|
return kDrop;
|
|
|
|
frame->SetSpatialIndex(0);
|
|
frame->SetId(codec_header.pictureId & 0x7FFF);
|
|
|
|
if (last_picture_id_ == -1)
|
|
last_picture_id_ = frame->Id();
|
|
|
|
// Clean up info about not yet received frames that are too old.
|
|
uint16_t old_picture_id =
|
|
Subtract<kFrameIdLength>(frame->Id(), kMaxNotYetReceivedFrames);
|
|
auto clean_frames_to = not_yet_received_frames_.lower_bound(old_picture_id);
|
|
not_yet_received_frames_.erase(not_yet_received_frames_.begin(),
|
|
clean_frames_to);
|
|
// Avoid re-adding picture ids that were just erased.
|
|
if (AheadOf<uint16_t, kFrameIdLength>(old_picture_id, last_picture_id_)) {
|
|
last_picture_id_ = old_picture_id;
|
|
}
|
|
// Find if there has been a gap in fully received frames and save the picture
|
|
// id of those frames in `not_yet_received_frames_`.
|
|
if (AheadOf<uint16_t, kFrameIdLength>(frame->Id(), last_picture_id_)) {
|
|
do {
|
|
last_picture_id_ = Add<kFrameIdLength>(last_picture_id_, 1);
|
|
not_yet_received_frames_.insert(last_picture_id_);
|
|
} while (last_picture_id_ != frame->Id());
|
|
}
|
|
|
|
// Clean up info for base layers that are too old.
|
|
int64_t old_tl0_pic_idx = unwrapped_tl0 - kMaxLayerInfo;
|
|
auto clean_layer_info_to = layer_info_.lower_bound(old_tl0_pic_idx);
|
|
layer_info_.erase(layer_info_.begin(), clean_layer_info_to);
|
|
|
|
if (frame->frame_type() == VideoFrameType::kVideoFrameKey) {
|
|
if (codec_header.temporalIdx != 0) {
|
|
return kDrop;
|
|
}
|
|
frame->num_references = 0;
|
|
layer_info_[unwrapped_tl0].fill(-1);
|
|
UpdateLayerInfoVp8(frame, unwrapped_tl0, codec_header.temporalIdx);
|
|
return kHandOff;
|
|
}
|
|
|
|
auto layer_info_it = layer_info_.find(
|
|
codec_header.temporalIdx == 0 ? unwrapped_tl0 - 1 : unwrapped_tl0);
|
|
|
|
// If we don't have the base layer frame yet, stash this frame.
|
|
if (layer_info_it == layer_info_.end())
|
|
return kStash;
|
|
|
|
// A non keyframe base layer frame has been received, copy the layer info
|
|
// from the previous base layer frame and set a reference to the previous
|
|
// base layer frame.
|
|
if (codec_header.temporalIdx == 0) {
|
|
layer_info_it =
|
|
layer_info_.emplace(unwrapped_tl0, layer_info_it->second).first;
|
|
frame->num_references = 1;
|
|
int64_t last_pid_on_layer = layer_info_it->second[0];
|
|
|
|
// Is this an old frame that has already been used to update the state? If
|
|
// so, drop it.
|
|
if (AheadOrAt<uint16_t, kFrameIdLength>(last_pid_on_layer, frame->Id())) {
|
|
return kDrop;
|
|
}
|
|
|
|
frame->references[0] = last_pid_on_layer;
|
|
UpdateLayerInfoVp8(frame, unwrapped_tl0, codec_header.temporalIdx);
|
|
return kHandOff;
|
|
}
|
|
|
|
// Layer sync frame, this frame only references its base layer frame.
|
|
if (codec_header.layerSync) {
|
|
frame->num_references = 1;
|
|
int64_t last_pid_on_layer = layer_info_it->second[codec_header.temporalIdx];
|
|
|
|
// Is this an old frame that has already been used to update the state? If
|
|
// so, drop it.
|
|
if (last_pid_on_layer != -1 &&
|
|
AheadOrAt<uint16_t, kFrameIdLength>(last_pid_on_layer, frame->Id())) {
|
|
return kDrop;
|
|
}
|
|
|
|
frame->references[0] = layer_info_it->second[0];
|
|
UpdateLayerInfoVp8(frame, unwrapped_tl0, codec_header.temporalIdx);
|
|
return kHandOff;
|
|
}
|
|
|
|
// Find all references for this frame.
|
|
frame->num_references = 0;
|
|
for (uint8_t layer = 0; layer <= codec_header.temporalIdx; ++layer) {
|
|
// If we have not yet received a previous frame on this temporal layer,
|
|
// stash this frame.
|
|
if (layer_info_it->second[layer] == -1)
|
|
return kStash;
|
|
|
|
// If the last frame on this layer is ahead of this frame it means that
|
|
// a layer sync frame has been received after this frame for the same
|
|
// base layer frame, drop this frame.
|
|
if (AheadOf<uint16_t, kFrameIdLength>(layer_info_it->second[layer],
|
|
frame->Id())) {
|
|
return kDrop;
|
|
}
|
|
|
|
// If we have not yet received a frame between this frame and the referenced
|
|
// frame then we have to wait for that frame to be completed first.
|
|
auto not_received_frame_it =
|
|
not_yet_received_frames_.upper_bound(layer_info_it->second[layer]);
|
|
if (not_received_frame_it != not_yet_received_frames_.end() &&
|
|
AheadOf<uint16_t, kFrameIdLength>(frame->Id(),
|
|
*not_received_frame_it)) {
|
|
return kStash;
|
|
}
|
|
|
|
if (!(AheadOf<uint16_t, kFrameIdLength>(frame->Id(),
|
|
layer_info_it->second[layer]))) {
|
|
RTC_LOG(LS_INFO) << "Frame with picture id " << frame->Id()
|
|
<< " and packet range [" << frame->first_seq_num()
|
|
<< ", " << frame->last_seq_num()
|
|
<< "] already received, "
|
|
" dropping frame.";
|
|
return kDrop;
|
|
}
|
|
|
|
++frame->num_references;
|
|
frame->references[layer] = layer_info_it->second[layer];
|
|
}
|
|
|
|
UpdateLayerInfoVp8(frame, unwrapped_tl0, codec_header.temporalIdx);
|
|
return kHandOff;
|
|
}
|
|
|
|
void RtpVp8RefFinder::UpdateLayerInfoVp8(RtpFrameObject* frame,
|
|
int64_t unwrapped_tl0,
|
|
uint8_t temporal_idx) {
|
|
auto layer_info_it = layer_info_.find(unwrapped_tl0);
|
|
|
|
// Update this layer info and newer.
|
|
while (layer_info_it != layer_info_.end()) {
|
|
if (layer_info_it->second[temporal_idx] != -1 &&
|
|
AheadOf<uint16_t, kFrameIdLength>(layer_info_it->second[temporal_idx],
|
|
frame->Id())) {
|
|
// The frame was not newer, then no subsequent layer info have to be
|
|
// update.
|
|
break;
|
|
}
|
|
|
|
layer_info_it->second[temporal_idx] = frame->Id();
|
|
++unwrapped_tl0;
|
|
layer_info_it = layer_info_.find(unwrapped_tl0);
|
|
}
|
|
not_yet_received_frames_.erase(frame->Id());
|
|
|
|
UnwrapPictureIds(frame);
|
|
}
|
|
|
|
void RtpVp8RefFinder::RetryStashedFrames(
|
|
RtpFrameReferenceFinder::ReturnVector& res) {
|
|
bool complete_frame = false;
|
|
do {
|
|
complete_frame = false;
|
|
for (auto it = stashed_frames_.begin(); it != stashed_frames_.end();) {
|
|
const RTPVideoHeaderVP8& codec_header = absl::get<RTPVideoHeaderVP8>(
|
|
it->frame->GetRtpVideoHeader().video_type_header);
|
|
FrameDecision decision =
|
|
ManageFrameInternal(it->frame.get(), codec_header, it->unwrapped_tl0);
|
|
|
|
switch (decision) {
|
|
case kStash:
|
|
++it;
|
|
break;
|
|
case kHandOff:
|
|
complete_frame = true;
|
|
res.push_back(std::move(it->frame));
|
|
[[fallthrough]];
|
|
case kDrop:
|
|
it = stashed_frames_.erase(it);
|
|
}
|
|
}
|
|
} while (complete_frame);
|
|
}
|
|
|
|
void RtpVp8RefFinder::UnwrapPictureIds(RtpFrameObject* frame) {
|
|
for (size_t i = 0; i < frame->num_references; ++i)
|
|
frame->references[i] = unwrapper_.Unwrap(frame->references[i]);
|
|
frame->SetId(unwrapper_.Unwrap(frame->Id()));
|
|
}
|
|
|
|
void RtpVp8RefFinder::ClearTo(uint16_t seq_num) {
|
|
auto it = stashed_frames_.begin();
|
|
while (it != stashed_frames_.end()) {
|
|
if (AheadOf<uint16_t>(seq_num, it->frame->first_seq_num())) {
|
|
it = stashed_frames_.erase(it);
|
|
} else {
|
|
++it;
|
|
}
|
|
}
|
|
}
|
|
|
|
} // namespace webrtc
|