blob: 529866d5f8dd47d5f54ea5942819d8f109e95b89 [file] [log] [blame]
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001/*
2 * libjingle
jlmiller@webrtc.org5f93d0a2015-01-20 21:36:13 +00003 * Copyright 2012 Google Inc.
henrike@webrtc.org28e20752013-07-10 00:45:36 +00004 *
5 * Redistribution and use in source and binary forms, with or without
6 * modification, are permitted provided that the following conditions are met:
7 *
8 * 1. Redistributions of source code must retain the above copyright notice,
9 * this list of conditions and the following disclaimer.
10 * 2. Redistributions in binary form must reproduce the above copyright notice,
11 * this list of conditions and the following disclaimer in the documentation
12 * and/or other materials provided with the distribution.
13 * 3. The name of the author may not be used to endorse or promote products
14 * derived from this software without specific prior written permission.
15 *
16 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR IMPLIED
17 * WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
18 * MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO
19 * EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
20 * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
21 * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS;
22 * OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
23 * WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR
24 * OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF
25 * ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
26 */
27
28#include <string>
29
wu@webrtc.org967bfff2013-09-19 05:49:50 +000030#include "talk/app/webrtc/remotevideocapturer.h"
buildbot@webrtc.orga09a9992014-08-13 17:26:08 +000031#include "talk/app/webrtc/test/fakevideotrackrenderer.h"
wu@webrtc.org967bfff2013-09-19 05:49:50 +000032#include "talk/app/webrtc/videosource.h"
henrike@webrtc.org28e20752013-07-10 00:45:36 +000033#include "talk/app/webrtc/videotrack.h"
wu@webrtc.org967bfff2013-09-19 05:49:50 +000034#include "talk/media/base/fakemediaengine.h"
35#include "talk/media/devices/fakedevicemanager.h"
henrike@webrtc.org28e20752013-07-10 00:45:36 +000036#include "talk/media/webrtc/webrtcvideoframe.h"
wu@webrtc.org967bfff2013-09-19 05:49:50 +000037#include "talk/session/media/channelmanager.h"
buildbot@webrtc.orga09a9992014-08-13 17:26:08 +000038#include "webrtc/base/gunit.h"
39#include "webrtc/base/scoped_ptr.h"
henrike@webrtc.org28e20752013-07-10 00:45:36 +000040
41using webrtc::FakeVideoTrackRenderer;
wu@webrtc.org967bfff2013-09-19 05:49:50 +000042using webrtc::VideoSource;
henrike@webrtc.org28e20752013-07-10 00:45:36 +000043using webrtc::VideoTrack;
44using webrtc::VideoTrackInterface;
45
guoweis@webrtc.org00c509a2015-03-12 21:37:26 +000046namespace {
47
48class WebRtcVideoTestFrame : public cricket::WebRtcVideoFrame {
49 public:
50 using cricket::WebRtcVideoFrame::SetRotation;
51};
52
53} // namespace
54
55class VideoTrackTest : public testing::Test {
56 public:
57 VideoTrackTest() {
58 static const char kVideoTrackId[] = "track_id";
59
60 channel_manager_.reset(new cricket::ChannelManager(
61 new cricket::FakeMediaEngine(), new cricket::FakeDeviceManager(),
62 rtc::Thread::Current()));
63 EXPECT_TRUE(channel_manager_->Init());
64 video_track_ = VideoTrack::Create(
65 kVideoTrackId,
66 VideoSource::Create(channel_manager_.get(),
67 new webrtc::RemoteVideoCapturer(), NULL));
68 }
69
70 protected:
71 rtc::scoped_ptr<cricket::ChannelManager> channel_manager_;
72 rtc::scoped_refptr<VideoTrackInterface> video_track_;
73};
74
henrike@webrtc.org28e20752013-07-10 00:45:36 +000075// Test adding renderers to a video track and render to them by providing
wu@webrtc.org967bfff2013-09-19 05:49:50 +000076// frames to the source.
guoweis@webrtc.org00c509a2015-03-12 21:37:26 +000077TEST_F(VideoTrackTest, RenderVideo) {
78 // FakeVideoTrackRenderer register itself to |video_track_|
guoweis@webrtc.orgf9a75d92015-03-10 06:37:00 +000079 rtc::scoped_ptr<FakeVideoTrackRenderer> renderer_1(
guoweis@webrtc.org00c509a2015-03-12 21:37:26 +000080 new FakeVideoTrackRenderer(video_track_.get()));
guoweis@webrtc.orgf9a75d92015-03-10 06:37:00 +000081
guoweis@webrtc.org00c509a2015-03-12 21:37:26 +000082 cricket::VideoRenderer* renderer_input =
83 video_track_->GetSource()->FrameInput();
84 ASSERT_FALSE(renderer_input == NULL);
henrike@webrtc.org28e20752013-07-10 00:45:36 +000085
henrike@webrtc.org28e20752013-07-10 00:45:36 +000086 cricket::WebRtcVideoFrame frame;
87 frame.InitToBlack(123, 123, 1, 1, 0, 0);
guoweis@webrtc.org00c509a2015-03-12 21:37:26 +000088 renderer_input->RenderFrame(&frame);
henrike@webrtc.org28e20752013-07-10 00:45:36 +000089 EXPECT_EQ(1, renderer_1->num_rendered_frames());
90
wu@webrtc.org967bfff2013-09-19 05:49:50 +000091 EXPECT_EQ(123, renderer_1->width());
92 EXPECT_EQ(123, renderer_1->height());
93
guoweis@webrtc.org00c509a2015-03-12 21:37:26 +000094 // FakeVideoTrackRenderer register itself to |video_track_|
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +000095 rtc::scoped_ptr<FakeVideoTrackRenderer> renderer_2(
guoweis@webrtc.org00c509a2015-03-12 21:37:26 +000096 new FakeVideoTrackRenderer(video_track_.get()));
henrike@webrtc.org28e20752013-07-10 00:45:36 +000097
guoweis@webrtc.org00c509a2015-03-12 21:37:26 +000098 renderer_input->RenderFrame(&frame);
henrike@webrtc.org28e20752013-07-10 00:45:36 +000099
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000100 EXPECT_EQ(123, renderer_1->width());
101 EXPECT_EQ(123, renderer_1->height());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000102 EXPECT_EQ(123, renderer_2->width());
103 EXPECT_EQ(123, renderer_2->height());
104
105 EXPECT_EQ(2, renderer_1->num_rendered_frames());
106 EXPECT_EQ(1, renderer_2->num_rendered_frames());
107
guoweis@webrtc.org00c509a2015-03-12 21:37:26 +0000108 video_track_->RemoveRenderer(renderer_1.get());
109 renderer_input->RenderFrame(&frame);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000110
111 EXPECT_EQ(2, renderer_1->num_rendered_frames());
112 EXPECT_EQ(2, renderer_2->num_rendered_frames());
113}
guoweis@webrtc.org00c509a2015-03-12 21:37:26 +0000114
115// Test adding renderers which support and don't support rotation and receive
116// the right frame.
117TEST_F(VideoTrackTest, RenderVideoWithPendingRotation) {
118 const size_t kWidth = 800;
119 const size_t kHeight = 400;
120
121 // Add a renderer which supports rotation.
122 rtc::scoped_ptr<FakeVideoTrackRenderer> rotating_renderer(
123 new FakeVideoTrackRenderer(video_track_.get(), true));
124
125 cricket::VideoRenderer* renderer_input =
126 video_track_->GetSource()->FrameInput();
127 ASSERT_FALSE(renderer_input == NULL);
128
129 // Create a frame with rotation 90 degree.
130 WebRtcVideoTestFrame frame;
131 frame.InitToBlack(kWidth, kHeight, 1, 1, 0, 0);
132 frame.SetRotation(webrtc::kVideoRotation_90);
133
134 // rotating_renderer should see the frame unrotated.
135 renderer_input->RenderFrame(&frame);
136 EXPECT_EQ(1, rotating_renderer->num_rendered_frames());
137 EXPECT_EQ(kWidth, rotating_renderer->width());
138 EXPECT_EQ(kHeight, rotating_renderer->height());
139 EXPECT_EQ(&frame, rotating_renderer->last_frame());
140
141 // Add 2nd renderer which doesn't support rotation.
142 rtc::scoped_ptr<FakeVideoTrackRenderer> non_rotating_renderer(
143 new FakeVideoTrackRenderer(video_track_.get(), false));
144
145 // Render the same 90 degree frame.
146 renderer_input->RenderFrame(&frame);
147
148 // rotating_renderer should see the same frame.
149 EXPECT_EQ(kWidth, rotating_renderer->width());
150 EXPECT_EQ(kHeight, rotating_renderer->height());
151 EXPECT_EQ(&frame, rotating_renderer->last_frame());
152
153 // non_rotating_renderer should see the frame rotated.
154 EXPECT_EQ(kHeight, non_rotating_renderer->width());
155 EXPECT_EQ(kWidth, non_rotating_renderer->height());
156 EXPECT_NE(&frame, non_rotating_renderer->last_frame());
157
158 // Render the same 90 degree frame the 3rd time.
159 renderer_input->RenderFrame(&frame);
160
161 // Now render a frame without rotation.
162 frame.SetRotation(webrtc::kVideoRotation_0);
163 renderer_input->RenderFrame(&frame);
164
165 // rotating_renderer should still only have 1 setsize.
166 EXPECT_EQ(kWidth, rotating_renderer->width());
167 EXPECT_EQ(kHeight, rotating_renderer->height());
168 EXPECT_EQ(&frame, rotating_renderer->last_frame());
169
170 // render_2 should have a new size but should have the same frame.
171 EXPECT_EQ(kWidth, non_rotating_renderer->width());
172 EXPECT_EQ(kHeight, non_rotating_renderer->height());
173 EXPECT_EQ(&frame, non_rotating_renderer->last_frame());
174}