mirror of
https://github.com/mollyim/webrtc.git
synced 2025-05-17 15:47:53 +01:00

This change adds support for emitting encoded frames for recording when the decoder can't easily read out encoded width and height as is the case for AV1 streams, in which case the information is buried in OBUs. Downstream project relies on resolution information being present for key frames. With the change, VideoReceiveStream2 infers the resolution from decoded frames, and supplies it in the RecordableEncodedFrames. Bug: chromium:1191972 Change-Id: I07beda6526206c80a732976e8e19d3581489b8fe Reviewed-on: https://webrtc-review.googlesource.com/c/src/+/214126 Reviewed-by: Harald Alvestrand <hta@webrtc.org> Reviewed-by: Philip Eliasson <philipel@webrtc.org> Reviewed-by: Erik Språng <sprang@webrtc.org> Commit-Queue: Markus Handell <handellm@webrtc.org> Cr-Commit-Position: refs/heads/master@{#33662}
128 lines
3.9 KiB
C++
128 lines
3.9 KiB
C++
/*
|
|
* Copyright (c) 2013 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 "test/fake_decoder.h"
|
|
|
|
#include <string.h>
|
|
|
|
#include <memory>
|
|
|
|
#include "api/scoped_refptr.h"
|
|
#include "api/video/i420_buffer.h"
|
|
#include "api/video/video_frame.h"
|
|
#include "api/video/video_frame_buffer.h"
|
|
#include "api/video/video_rotation.h"
|
|
#include "modules/video_coding/include/video_error_codes.h"
|
|
#include "rtc_base/checks.h"
|
|
#include "rtc_base/task_queue.h"
|
|
#include "rtc_base/time_utils.h"
|
|
|
|
namespace webrtc {
|
|
namespace test {
|
|
|
|
FakeDecoder::FakeDecoder() : FakeDecoder(nullptr) {}
|
|
|
|
FakeDecoder::FakeDecoder(TaskQueueFactory* task_queue_factory)
|
|
: callback_(nullptr),
|
|
width_(kDefaultWidth),
|
|
height_(kDefaultHeight),
|
|
task_queue_factory_(task_queue_factory),
|
|
decode_delay_ms_(0) {}
|
|
|
|
int32_t FakeDecoder::InitDecode(const VideoCodec* config,
|
|
int32_t number_of_cores) {
|
|
return WEBRTC_VIDEO_CODEC_OK;
|
|
}
|
|
|
|
int32_t FakeDecoder::Decode(const EncodedImage& input,
|
|
bool missing_frames,
|
|
int64_t render_time_ms) {
|
|
if (input._encodedWidth > 0 && input._encodedHeight > 0) {
|
|
width_ = input._encodedWidth;
|
|
height_ = input._encodedHeight;
|
|
}
|
|
|
|
rtc::scoped_refptr<I420Buffer> buffer = I420Buffer::Create(width_, height_);
|
|
I420Buffer::SetBlack(buffer);
|
|
VideoFrame frame = VideoFrame::Builder()
|
|
.set_video_frame_buffer(buffer)
|
|
.set_rotation(webrtc::kVideoRotation_0)
|
|
.set_timestamp_ms(render_time_ms)
|
|
.build();
|
|
frame.set_timestamp(input.Timestamp());
|
|
frame.set_ntp_time_ms(input.ntp_time_ms_);
|
|
|
|
if (decode_delay_ms_ == 0 || !task_queue_) {
|
|
callback_->Decoded(frame);
|
|
} else {
|
|
task_queue_->PostDelayedTask(
|
|
[frame, this]() {
|
|
VideoFrame copy = frame;
|
|
callback_->Decoded(copy);
|
|
},
|
|
decode_delay_ms_);
|
|
}
|
|
|
|
return WEBRTC_VIDEO_CODEC_OK;
|
|
}
|
|
|
|
void FakeDecoder::SetDelayedDecoding(int decode_delay_ms) {
|
|
RTC_CHECK(task_queue_factory_);
|
|
if (!task_queue_) {
|
|
task_queue_ =
|
|
std::make_unique<rtc::TaskQueue>(task_queue_factory_->CreateTaskQueue(
|
|
"fake_decoder", TaskQueueFactory::Priority::NORMAL));
|
|
}
|
|
decode_delay_ms_ = decode_delay_ms;
|
|
}
|
|
|
|
int32_t FakeDecoder::RegisterDecodeCompleteCallback(
|
|
DecodedImageCallback* callback) {
|
|
callback_ = callback;
|
|
return WEBRTC_VIDEO_CODEC_OK;
|
|
}
|
|
|
|
int32_t FakeDecoder::Release() {
|
|
return WEBRTC_VIDEO_CODEC_OK;
|
|
}
|
|
|
|
const char* FakeDecoder::kImplementationName = "fake_decoder";
|
|
VideoDecoder::DecoderInfo FakeDecoder::GetDecoderInfo() const {
|
|
DecoderInfo info;
|
|
info.implementation_name = kImplementationName;
|
|
info.is_hardware_accelerated = false;
|
|
return info;
|
|
}
|
|
const char* FakeDecoder::ImplementationName() const {
|
|
return kImplementationName;
|
|
}
|
|
|
|
int32_t FakeH264Decoder::Decode(const EncodedImage& input,
|
|
bool missing_frames,
|
|
int64_t render_time_ms) {
|
|
uint8_t value = 0;
|
|
for (size_t i = 0; i < input.size(); ++i) {
|
|
uint8_t kStartCode[] = {0, 0, 0, 1};
|
|
if (i < input.size() - sizeof(kStartCode) &&
|
|
!memcmp(&input.data()[i], kStartCode, sizeof(kStartCode))) {
|
|
i += sizeof(kStartCode) + 1; // Skip start code and NAL header.
|
|
}
|
|
if (input.data()[i] != value) {
|
|
RTC_CHECK_EQ(value, input.data()[i])
|
|
<< "Bitstream mismatch between sender and receiver.";
|
|
return -1;
|
|
}
|
|
++value;
|
|
}
|
|
return FakeDecoder::Decode(input, missing_frames, render_time_ms);
|
|
}
|
|
|
|
} // namespace test
|
|
} // namespace webrtc
|