Rotation is best done when rendered in GPU, added the shader code which rotates the frame. For renderers which don't support rotation, the rotation will be done before sending down the frame to render. By default, assume renderer can't do rotation. Tested with peerconnection_client on windows, AppRTCDemo on Mac. BUG=4145 R=glaznev@webrtc.org, pthatcher@webrtc.org Committed: https://code.google.com/p/webrtc/source/detail?r=8660 Committed: https://code.google.com/p/webrtc/source/detail?r=8661 Review URL: https://webrtc-codereview.appspot.com/43569004 Cr-Commit-Position: refs/heads/master@{#8705} git-svn-id: http://webrtc.googlecode.com/svn/trunk@8705 4adac7df-926f-26a2-2b94-8c16560cd09d
175 lines
6.4 KiB
C++
175 lines
6.4 KiB
C++
/*
|
|
* 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 <string>
|
|
|
|
#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<cricket::ChannelManager> channel_manager_;
|
|
rtc::scoped_refptr<VideoTrackInterface> 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<FakeVideoTrackRenderer> 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<FakeVideoTrackRenderer> 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<FakeVideoTrackRenderer> 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<FakeVideoTrackRenderer> 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());
|
|
}
|