/* * libjingle * Copyright 2012 Google Inc. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions are met: * * 1. Redistributions of source code must retain the above copyright notice, * this list of conditions and the following disclaimer. * 2. Redistributions in binary form must reproduce the above copyright notice, * this list of conditions and the following disclaimer in the documentation * and/or other materials provided with the distribution. * 3. The name of the author may not be used to endorse or promote products * derived from this software without specific prior written permission. * * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR IMPLIED * WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF * MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO * EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; * OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, * WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR * OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF * ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. */ #include #include "talk/app/webrtc/remotevideocapturer.h" #include "talk/app/webrtc/test/fakevideotrackrenderer.h" #include "talk/app/webrtc/videosource.h" #include "talk/app/webrtc/videotrack.h" #include "talk/media/base/fakemediaengine.h" #include "talk/media/devices/fakedevicemanager.h" #include "talk/media/webrtc/webrtcvideoframe.h" #include "talk/session/media/channelmanager.h" #include "webrtc/base/gunit.h" #include "webrtc/base/scoped_ptr.h" using webrtc::FakeVideoTrackRenderer; using webrtc::VideoSource; using webrtc::VideoTrack; using webrtc::VideoTrackInterface; namespace { class WebRtcVideoTestFrame : public cricket::WebRtcVideoFrame { public: using cricket::WebRtcVideoFrame::SetRotation; }; } // namespace class VideoTrackTest : public testing::Test { public: VideoTrackTest() { static const char kVideoTrackId[] = "track_id"; channel_manager_.reset(new cricket::ChannelManager( new cricket::FakeMediaEngine(), new cricket::FakeDeviceManager(), rtc::Thread::Current())); EXPECT_TRUE(channel_manager_->Init()); video_track_ = VideoTrack::Create( kVideoTrackId, VideoSource::Create(channel_manager_.get(), new webrtc::RemoteVideoCapturer(), NULL)); } protected: rtc::scoped_ptr channel_manager_; rtc::scoped_refptr video_track_; }; // Test adding renderers to a video track and render to them by providing // frames to the source. TEST_F(VideoTrackTest, RenderVideo) { // FakeVideoTrackRenderer register itself to |video_track_| rtc::scoped_ptr renderer_1( new FakeVideoTrackRenderer(video_track_.get())); cricket::VideoRenderer* renderer_input = video_track_->GetSource()->FrameInput(); ASSERT_FALSE(renderer_input == NULL); cricket::WebRtcVideoFrame frame; frame.InitToBlack(123, 123, 1, 1, 0, 0); renderer_input->RenderFrame(&frame); EXPECT_EQ(1, renderer_1->num_rendered_frames()); EXPECT_EQ(123, renderer_1->width()); EXPECT_EQ(123, renderer_1->height()); // FakeVideoTrackRenderer register itself to |video_track_| rtc::scoped_ptr renderer_2( new FakeVideoTrackRenderer(video_track_.get())); renderer_input->RenderFrame(&frame); EXPECT_EQ(123, renderer_1->width()); EXPECT_EQ(123, renderer_1->height()); EXPECT_EQ(123, renderer_2->width()); EXPECT_EQ(123, renderer_2->height()); EXPECT_EQ(2, renderer_1->num_rendered_frames()); EXPECT_EQ(1, renderer_2->num_rendered_frames()); video_track_->RemoveRenderer(renderer_1.get()); renderer_input->RenderFrame(&frame); EXPECT_EQ(2, renderer_1->num_rendered_frames()); EXPECT_EQ(2, renderer_2->num_rendered_frames()); } // Test adding renderers which support and don't support rotation and receive // the right frame. TEST_F(VideoTrackTest, RenderVideoWithPendingRotation) { const size_t kWidth = 800; const size_t kHeight = 400; // Add a renderer which supports rotation. rtc::scoped_ptr rotating_renderer( new FakeVideoTrackRenderer(video_track_.get(), true)); cricket::VideoRenderer* renderer_input = video_track_->GetSource()->FrameInput(); ASSERT_FALSE(renderer_input == NULL); // Create a frame with rotation 90 degree. WebRtcVideoTestFrame frame; frame.InitToBlack(kWidth, kHeight, 1, 1, 0, 0); frame.SetRotation(webrtc::kVideoRotation_90); // rotating_renderer should see the frame unrotated. renderer_input->RenderFrame(&frame); EXPECT_EQ(1, rotating_renderer->num_rendered_frames()); EXPECT_EQ(kWidth, rotating_renderer->width()); EXPECT_EQ(kHeight, rotating_renderer->height()); EXPECT_EQ(&frame, rotating_renderer->last_frame()); // Add 2nd renderer which doesn't support rotation. rtc::scoped_ptr non_rotating_renderer( new FakeVideoTrackRenderer(video_track_.get(), false)); // Render the same 90 degree frame. renderer_input->RenderFrame(&frame); // rotating_renderer should see the same frame. EXPECT_EQ(kWidth, rotating_renderer->width()); EXPECT_EQ(kHeight, rotating_renderer->height()); EXPECT_EQ(&frame, rotating_renderer->last_frame()); // non_rotating_renderer should see the frame rotated. EXPECT_EQ(kHeight, non_rotating_renderer->width()); EXPECT_EQ(kWidth, non_rotating_renderer->height()); EXPECT_NE(&frame, non_rotating_renderer->last_frame()); // Render the same 90 degree frame the 3rd time. renderer_input->RenderFrame(&frame); // Now render a frame without rotation. frame.SetRotation(webrtc::kVideoRotation_0); renderer_input->RenderFrame(&frame); // rotating_renderer should still only have 1 setsize. EXPECT_EQ(kWidth, rotating_renderer->width()); EXPECT_EQ(kHeight, rotating_renderer->height()); EXPECT_EQ(&frame, rotating_renderer->last_frame()); // render_2 should have a new size but should have the same frame. EXPECT_EQ(kWidth, non_rotating_renderer->width()); EXPECT_EQ(kHeight, non_rotating_renderer->height()); EXPECT_EQ(&frame, non_rotating_renderer->last_frame()); }