/* * Copyright (c) 2011 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 MEDIA_BASE_FAKE_VIDEO_RENDERER_H_ #define MEDIA_BASE_FAKE_VIDEO_RENDERER_H_ #include #include "api/scoped_refptr.h" #include "api/video/video_frame.h" #include "api/video/video_frame_buffer.h" #include "api/video/video_rotation.h" #include "api/video/video_sink_interface.h" #include "rtc_base/event.h" #include "rtc_base/synchronization/mutex.h" namespace cricket { // Faked video renderer that has a callback for actions on rendering. class FakeVideoRenderer : public rtc::VideoSinkInterface { public: FakeVideoRenderer(); void OnFrame(const webrtc::VideoFrame& frame) override; int errors() const { return errors_; } int width() const { webrtc::MutexLock lock(&mutex_); return width_; } int height() const { webrtc::MutexLock lock(&mutex_); return height_; } webrtc::VideoRotation rotation() const { webrtc::MutexLock lock(&mutex_); return rotation_; } int64_t timestamp_us() const { webrtc::MutexLock lock(&mutex_); return timestamp_us_; } int num_rendered_frames() const { webrtc::MutexLock lock(&mutex_); return num_rendered_frames_; } bool black_frame() const { webrtc::MutexLock lock(&mutex_); return black_frame_; } int64_t ntp_time_ms() const { webrtc::MutexLock lock(&mutex_); return ntp_timestamp_ms_; } absl::optional color_space() const { webrtc::MutexLock lock(&mutex_); return color_space_; } webrtc::RtpPacketInfos packet_infos() const { webrtc::MutexLock lock(&mutex_); return packet_infos_; } bool WaitForRenderedFrame(int64_t timeout_ms); private: static bool CheckFrameColorYuv(uint8_t y_min, uint8_t y_max, uint8_t u_min, uint8_t u_max, uint8_t v_min, uint8_t v_max, const webrtc::VideoFrame* frame) { if (!frame || !frame->video_frame_buffer()) { return false; } rtc::scoped_refptr i420_buffer = frame->video_frame_buffer()->ToI420(); // Y int y_width = frame->width(); int y_height = frame->height(); const uint8_t* y_plane = i420_buffer->DataY(); const uint8_t* y_pos = y_plane; int32_t y_pitch = i420_buffer->StrideY(); for (int i = 0; i < y_height; ++i) { for (int j = 0; j < y_width; ++j) { uint8_t y_value = *(y_pos + j); if (y_value < y_min || y_value > y_max) { return false; } } y_pos += y_pitch; } // U and V int chroma_width = i420_buffer->ChromaWidth(); int chroma_height = i420_buffer->ChromaHeight(); const uint8_t* u_plane = i420_buffer->DataU(); const uint8_t* v_plane = i420_buffer->DataV(); const uint8_t* u_pos = u_plane; const uint8_t* v_pos = v_plane; int32_t u_pitch = i420_buffer->StrideU(); int32_t v_pitch = i420_buffer->StrideV(); for (int i = 0; i < chroma_height; ++i) { for (int j = 0; j < chroma_width; ++j) { uint8_t u_value = *(u_pos + j); if (u_value < u_min || u_value > u_max) { return false; } uint8_t v_value = *(v_pos + j); if (v_value < v_min || v_value > v_max) { return false; } } u_pos += u_pitch; v_pos += v_pitch; } return true; } int errors_ = 0; int width_ = 0; int height_ = 0; webrtc::VideoRotation rotation_ = webrtc::kVideoRotation_0; int64_t timestamp_us_ = 0; int num_rendered_frames_ = 0; int64_t ntp_timestamp_ms_ = 0; bool black_frame_ = false; mutable webrtc::Mutex mutex_; rtc::Event frame_rendered_event_; absl::optional color_space_; webrtc::RtpPacketInfos packet_infos_; }; } // namespace cricket #endif // MEDIA_BASE_FAKE_VIDEO_RENDERER_H_