Deliver original decoded resolution unless output resolution is explicitly configured via ChangeResolution(). Bug: none Change-Id: I1d2a47fa564010202762062d7ac483ad3c4effde Reviewed-on: https://webrtc-review.googlesource.com/c/src/+/375340 Reviewed-by: Erik Språng <sprang@webrtc.org> Commit-Queue: Sergey Silkin <ssilkin@webrtc.org> Cr-Commit-Position: refs/heads/main@{#43902}
101 lines
3.7 KiB
C++
101 lines
3.7 KiB
C++
/*
|
|
* Copyright (c) 2019 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.
|
|
*/
|
|
|
|
#ifndef TEST_TESTSUPPORT_IVF_VIDEO_FRAME_GENERATOR_H_
|
|
#define TEST_TESTSUPPORT_IVF_VIDEO_FRAME_GENERATOR_H_
|
|
|
|
#include <memory>
|
|
#include <optional>
|
|
#include <string>
|
|
|
|
#include "absl/strings/string_view.h"
|
|
#include "api/environment/environment.h"
|
|
#include "api/sequence_checker.h"
|
|
#include "api/test/frame_generator_interface.h"
|
|
#include "api/video/video_codec_type.h"
|
|
#include "api/video/video_frame.h"
|
|
#include "api/video_codecs/video_decoder.h"
|
|
#include "modules/video_coding/utility/ivf_file_reader.h"
|
|
#include "rtc_base/event.h"
|
|
#include "rtc_base/synchronization/mutex.h"
|
|
|
|
namespace webrtc {
|
|
namespace test {
|
|
|
|
// All methods except constructor must be used from the same thread.
|
|
class IvfVideoFrameGenerator : public FrameGeneratorInterface {
|
|
public:
|
|
// Allow to specify a `fps_hint` in case the fps of the video is known.
|
|
IvfVideoFrameGenerator(const Environment& env,
|
|
absl::string_view file_name,
|
|
std::optional<int> fps_hint);
|
|
~IvfVideoFrameGenerator() override;
|
|
|
|
VideoFrameData NextFrame() override;
|
|
void SkipNextFrame() override;
|
|
void ChangeResolution(size_t width, size_t height) override;
|
|
Resolution GetResolution() const override;
|
|
|
|
std::optional<int> fps() const override { return fps_hint_; }
|
|
|
|
private:
|
|
class DecodedCallback : public DecodedImageCallback {
|
|
public:
|
|
explicit DecodedCallback(IvfVideoFrameGenerator* reader)
|
|
: reader_(reader) {}
|
|
|
|
int32_t Decoded(VideoFrame& decoded_image) override;
|
|
int32_t Decoded(VideoFrame& decoded_image, int64_t decode_time_ms) override;
|
|
void Decoded(VideoFrame& decoded_image,
|
|
std::optional<int32_t> decode_time_ms,
|
|
std::optional<uint8_t> qp) override;
|
|
|
|
private:
|
|
IvfVideoFrameGenerator* const reader_;
|
|
};
|
|
|
|
void OnFrameDecoded(const VideoFrame& decoded_frame);
|
|
|
|
DecodedCallback callback_;
|
|
std::unique_ptr<IvfFileReader> file_reader_;
|
|
std::unique_ptr<VideoDecoder> video_decoder_;
|
|
|
|
// Resolution of IVF. Initially readed from IVF header and then set to
|
|
// resolution of decoded frame.
|
|
Resolution original_resolution_;
|
|
// Resolution of output frames. When set, the decoded frames scaled to
|
|
// `output_resolution_`. Otherwise the decoded resolution, which may vary from
|
|
// frame to frame, is preserved.
|
|
std::optional<Resolution> output_resolution_;
|
|
std::optional<int> fps_hint_;
|
|
|
|
// This lock is used to ensure that all API method will be called
|
|
// sequentially. It is required because we need to ensure that generator
|
|
// won't be destroyed while it is reading the next frame on another thread,
|
|
// because it will cause SIGSEGV when decoder callback will be invoked.
|
|
//
|
|
// FrameGenerator is injected into PeerConnection via some scoped_ref object
|
|
// and it can happen that the last pointer will be destroyed on the different
|
|
// thread comparing to the one from which frames were read.
|
|
Mutex lock_;
|
|
// This lock is used to sync between sending and receiving frame from decoder.
|
|
// We can't reuse `lock_` because then generator can be destroyed between
|
|
// frame was sent to decoder and decoder callback was invoked.
|
|
Mutex frame_decode_lock_;
|
|
|
|
rtc::Event next_frame_decoded_;
|
|
std::optional<VideoFrame> next_frame_ RTC_GUARDED_BY(frame_decode_lock_);
|
|
};
|
|
|
|
} // namespace test
|
|
} // namespace webrtc
|
|
|
|
#endif // TEST_TESTSUPPORT_IVF_VIDEO_FRAME_GENERATOR_H_
|