From 02ed57bf9d12a959d5ec139b3fc49170d16b5f30 Mon Sep 17 00:00:00 2001 From: "perkj@webrtc.org" Date: Fri, 27 Feb 2015 19:10:05 +0000 Subject: [PATCH] Fix CVO in androidvideocapturer. This add bool apply_rotation to WebrtcVideoFrame::Init and removes the need for WebrtcVideoFrame::SetRotation. BUG=4145 R=guoweis@webrtc.org, pthatcher@webrtc.org Review URL: https://webrtc-codereview.appspot.com/40759004 Cr-Commit-Position: refs/heads/master@{#8536} git-svn-id: http://webrtc.googlecode.com/svn/trunk@8536 4adac7df-926f-26a2-2b94-8c16560cd09d --- talk/app/webrtc/androidvideocapturer.cc | 2 +- talk/media/base/videoadapter_unittest.cc | 2 +- talk/media/base/videoframe.h | 7 ++- talk/media/base/videoframe_unittest.h | 3 +- talk/media/webrtc/webrtcvideoframe.cc | 28 ++++++++---- talk/media/webrtc/webrtcvideoframe.h | 11 +++-- .../media/webrtc/webrtcvideoframe_unittest.cc | 43 ++++++++++++++----- 7 files changed, 64 insertions(+), 32 deletions(-) diff --git a/talk/app/webrtc/androidvideocapturer.cc b/talk/app/webrtc/androidvideocapturer.cc index 48b822b27..01619aff5 100644 --- a/talk/app/webrtc/androidvideocapturer.cc +++ b/talk/app/webrtc/androidvideocapturer.cc @@ -85,7 +85,7 @@ class AndroidVideoCapturer::FrameFactory : public cricket::VideoFrameFactory { // Check that captured_frame is actually our frame. DCHECK(captured_frame == &captured_frame_); scoped_ptr frame(new WebRtcVideoFrame()); - frame->Init(captured_frame, dst_width, dst_height); + frame->Init(captured_frame, dst_width, dst_height, apply_rotation_); return frame.release(); } diff --git a/talk/media/base/videoadapter_unittest.cc b/talk/media/base/videoadapter_unittest.cc index 76b1502df..32bf9c3e6 100755 --- a/talk/media/base/videoadapter_unittest.cc +++ b/talk/media/base/videoadapter_unittest.cc @@ -100,7 +100,7 @@ class VideoAdapterTest : public testing::Test { const CapturedFrame* captured_frame) { WebRtcVideoFrame temp_i420; EXPECT_TRUE(temp_i420.Init(captured_frame, - captured_frame->width, abs(captured_frame->height))); + captured_frame->width, abs(captured_frame->height), true)); VideoFrame* out_frame = NULL; rtc::CritScope lock(&crit_); EXPECT_TRUE(video_adapter_->AdaptFrame(&temp_i420, &out_frame)); diff --git a/talk/media/base/videoframe.h b/talk/media/base/videoframe.h index 56621ad5a..7d097fb07 100644 --- a/talk/media/base/videoframe.h +++ b/talk/media/base/videoframe.h @@ -62,7 +62,8 @@ class VideoFrame { size_t pixel_height, int64_t elapsed_time, int64_t time_stamp, - webrtc::VideoRotation rotation) { + webrtc::VideoRotation rotation, + bool apply_rotation) { return false; } @@ -80,7 +81,7 @@ class VideoFrame { int rotation) { return Reset(fourcc, w, h, dw, dh, sample, sample_size, pixel_width, pixel_height, elapsed_time, time_stamp, - static_cast(rotation)); + static_cast(rotation), true); } // Basic accessors. @@ -123,8 +124,6 @@ class VideoFrame { virtual webrtc::VideoRotation GetVideoRotation() const { return webrtc::kVideoRotation_0; } - // TODO(guoweis): Remove the skeleton implementation once chrome is updated. - virtual void SetRotation(webrtc::VideoRotation rotation) {} // Make a shallow copy of the frame. The frame buffer itself is not copied. // Both the current and new VideoFrame will share a single reference-counted diff --git a/talk/media/base/videoframe_unittest.h b/talk/media/base/videoframe_unittest.h index 17f83c6e8..628b95214 100644 --- a/talk/media/base/videoframe_unittest.h +++ b/talk/media/base/videoframe_unittest.h @@ -1428,7 +1428,8 @@ class VideoFrameTest : public testing::Test { EXPECT_TRUE(IsEqual(frame1, frame2, 0)); EXPECT_TRUE(frame1.Reset(cricket::FOURCC_I420, kWidth, kHeight, kWidth, kHeight, reinterpret_cast(ms->GetBuffer()), - data_size, 1, 1, 0, 0, webrtc::kVideoRotation_0)); + data_size, 1, 1, 0, 0, webrtc::kVideoRotation_0, + true)); EXPECT_FALSE(IsBlack(frame1)); EXPECT_FALSE(IsEqual(frame1, frame2, 0)); } diff --git a/talk/media/webrtc/webrtcvideoframe.cc b/talk/media/webrtc/webrtcvideoframe.cc index 9764f31af..af424f56d 100644 --- a/talk/media/webrtc/webrtcvideoframe.cc +++ b/talk/media/webrtc/webrtcvideoframe.cc @@ -112,7 +112,8 @@ const webrtc::VideoFrame* WebRtcVideoFrame::FrameBuffer::frame() const { } WebRtcVideoFrame::WebRtcVideoFrame() - : video_buffer_(new RefCountedBuffer()) {} + : video_buffer_(new RefCountedBuffer()), + rotation_(webrtc::kVideoRotation_0) {} WebRtcVideoFrame::~WebRtcVideoFrame() {} @@ -129,14 +130,18 @@ bool WebRtcVideoFrame::Init(uint32 format, int64_t time_stamp, webrtc::VideoRotation rotation) { return Reset(format, w, h, dw, dh, sample, sample_size, pixel_width, - pixel_height, elapsed_time, time_stamp, rotation); + pixel_height, elapsed_time, time_stamp, rotation, + true /*apply_rotation*/); } -bool WebRtcVideoFrame::Init(const CapturedFrame* frame, int dw, int dh) { +bool WebRtcVideoFrame::Init(const CapturedFrame* frame, int dw, int dh, + bool apply_rotation) { return Reset(frame->fourcc, frame->width, frame->height, dw, dh, static_cast(frame->data), frame->data_size, frame->pixel_width, frame->pixel_height, frame->elapsed_time, - frame->time_stamp, frame->GetRotation()); + frame->time_stamp, + frame->GetRotation(), + apply_rotation); } bool WebRtcVideoFrame::Alias(const CapturedFrame* frame, @@ -148,7 +153,7 @@ bool WebRtcVideoFrame::Alias(const CapturedFrame* frame, frame->GetRotation() != webrtc::kVideoRotation_0) || frame->width != dw || frame->height != dh) { // TODO(fbarchard): Enable aliasing of more formats. - return Init(frame, dw, dh); + return Init(frame, dw, dh, apply_rotation); } else { Alias(static_cast(frame->data), frame->data_size, frame->width, frame->height, frame->pixel_width, frame->pixel_height, @@ -302,7 +307,8 @@ bool WebRtcVideoFrame::Reset(uint32 format, size_t pixel_height, int64_t elapsed_time, int64_t time_stamp, - webrtc::VideoRotation rotation) { + webrtc::VideoRotation rotation, + bool apply_rotation) { if (!Validate(format, w, h, sample, sample_size)) { return false; } @@ -319,7 +325,8 @@ bool WebRtcVideoFrame::Reset(uint32 format, // TODO(fbarchard): Support lazy allocation. int new_width = dw; int new_height = dh; - if (rotation == 90 || rotation == 270) { // If rotated swap width, height. + // If rotated swap width, height. + if (apply_rotation && (rotation == 90 || rotation == 270)) { new_width = dh; new_height = dw; } @@ -350,7 +357,9 @@ bool WebRtcVideoFrame::Reset(uint32 format, int v_stride = GetVPitch(); int r = libyuv::ConvertToI420( sample, sample_size, y, y_stride, u, u_stride, v, v_stride, horiz_crop, - vert_crop, w, h, dw, idh, static_cast(rotation), + vert_crop, w, h, dw, idh, + static_cast( + apply_rotation ? rotation : webrtc::kVideoRotation_0), format); if (r) { LOG(LS_ERROR) << "Error parsing format: " << GetFourccName(format) @@ -407,7 +416,8 @@ bool WebRtcVideoRenderFrame::Reset(uint32 fourcc, size_t pixel_height, int64_t elapsed_time, int64_t time_stamp, - webrtc::VideoRotation rotation) { + webrtc::VideoRotation rotation, + bool apply_rotation) { UNIMPLEMENTED; return false; } diff --git a/talk/media/webrtc/webrtcvideoframe.h b/talk/media/webrtc/webrtcvideoframe.h index 4de12b224..6d6e5310e 100644 --- a/talk/media/webrtc/webrtcvideoframe.h +++ b/talk/media/webrtc/webrtcvideoframe.h @@ -65,7 +65,7 @@ class WebRtcVideoFrame : public VideoFrame { int64_t time_stamp, webrtc::VideoRotation rotation); - bool Init(const CapturedFrame* frame, int dw, int dh); + bool Init(const CapturedFrame* frame, int dw, int dh, bool apply_rotation); void InitToEmptyBuffer(int w, int h, size_t pixel_width, size_t pixel_height, int64_t elapsed_time, int64_t time_stamp); @@ -107,7 +107,8 @@ class WebRtcVideoFrame : public VideoFrame { size_t pixel_height, int64_t elapsed_time, int64_t time_stamp, - webrtc::VideoRotation rotation); + webrtc::VideoRotation rotation, + bool apply_rotation); virtual size_t GetWidth() const; virtual size_t GetHeight() const; @@ -132,9 +133,6 @@ class WebRtcVideoFrame : public VideoFrame { virtual void SetTimeStamp(int64_t time_stamp) { time_stamp_ = time_stamp; } virtual webrtc::VideoRotation GetVideoRotation() const { return rotation_; } - virtual void SetRotation(webrtc::VideoRotation rotation) { - rotation_ = rotation; - } virtual VideoFrame* Copy() const; virtual bool MakeExclusive(); @@ -194,7 +192,8 @@ class WebRtcVideoRenderFrame : public VideoFrame { size_t pixel_height, int64_t elapsed_time, int64_t time_stamp, - webrtc::VideoRotation rotation) OVERRIDE; + webrtc::VideoRotation rotation, + bool apply_rotation) OVERRIDE; virtual size_t GetWidth() const OVERRIDE; virtual size_t GetHeight() const OVERRIDE; virtual const uint8* GetYPlane() const OVERRIDE; diff --git a/talk/media/webrtc/webrtcvideoframe_unittest.cc b/talk/media/webrtc/webrtcvideoframe_unittest.cc index 656183c31..9ddad13e3 100644 --- a/talk/media/webrtc/webrtcvideoframe_unittest.cc +++ b/talk/media/webrtc/webrtcvideoframe_unittest.cc @@ -33,7 +33,9 @@ class WebRtcVideoFrameTest : public VideoFrameTest { WebRtcVideoFrameTest() { } - void TestInit(int cropped_width, int cropped_height) { + void TestInit(int cropped_width, int cropped_height, + webrtc::VideoRotation frame_rotation, + bool apply_rotation) { const int frame_width = 1920; const int frame_height = 1080; @@ -44,7 +46,7 @@ class WebRtcVideoFrameTest : public VideoFrameTest { captured_frame.pixel_height = 1; captured_frame.elapsed_time = 1234; captured_frame.time_stamp = 5678; - captured_frame.rotation = webrtc::kVideoRotation_0; + captured_frame.rotation = frame_rotation; captured_frame.width = frame_width; captured_frame.height = frame_height; captured_frame.data_size = (frame_width * frame_height) + @@ -55,17 +57,30 @@ class WebRtcVideoFrameTest : public VideoFrameTest { // Create the new frame from the CapturedFrame. cricket::WebRtcVideoFrame frame; - EXPECT_TRUE(frame.Init(&captured_frame, cropped_width, cropped_height)); + EXPECT_TRUE( + frame.Init(&captured_frame, cropped_width, cropped_height, + apply_rotation)); // Verify the new frame. EXPECT_EQ(1u, frame.GetPixelWidth()); EXPECT_EQ(1u, frame.GetPixelHeight()); EXPECT_EQ(1234, frame.GetElapsedTime()); EXPECT_EQ(5678, frame.GetTimeStamp()); - EXPECT_EQ(webrtc::kVideoRotation_0, frame.GetRotation()); + if (apply_rotation) + EXPECT_EQ(webrtc::kVideoRotation_0, frame.GetRotation()); + else + EXPECT_EQ(frame_rotation, frame.GetRotation()); // The size of the new frame should have been cropped to multiple of 4. - EXPECT_EQ(static_cast(cropped_width & ~3), frame.GetWidth()); - EXPECT_EQ(static_cast(cropped_height & ~3), frame.GetHeight()); + // If |apply_rotation| and the frame rotation is 90 or 270, width and + // height are flipped. + if (apply_rotation && (frame_rotation == webrtc::kVideoRotation_90 + || frame_rotation == webrtc::kVideoRotation_270)) { + EXPECT_EQ(static_cast(cropped_width & ~3), frame.GetHeight()); + EXPECT_EQ(static_cast(cropped_height & ~3), frame.GetWidth() ); + } else { + EXPECT_EQ(static_cast(cropped_width & ~3), frame.GetWidth()); + EXPECT_EQ(static_cast(cropped_height & ~3), frame.GetHeight()); + } } }; @@ -274,17 +289,25 @@ TEST_F(WebRtcVideoFrameTest, Alias) { // Tests the Init function with different cropped size. TEST_F(WebRtcVideoFrameTest, InitEvenSize) { - TestInit(640, 360); + TestInit(640, 360, webrtc::kVideoRotation_0, true); } TEST_F(WebRtcVideoFrameTest, InitOddWidth) { - TestInit(601, 480); + TestInit(601, 480, webrtc::kVideoRotation_0, true); } TEST_F(WebRtcVideoFrameTest, InitOddHeight) { - TestInit(360, 765); + TestInit(360, 765, webrtc::kVideoRotation_0, true); } TEST_F(WebRtcVideoFrameTest, InitOddWidthHeight) { - TestInit(355, 1021); + TestInit(355, 1021, webrtc::kVideoRotation_0, true); +} + +TEST_F(WebRtcVideoFrameTest, InitRotated90ApplyRotation) { + TestInit(640, 360, webrtc::kVideoRotation_90, true); +} + +TEST_F(WebRtcVideoFrameTest, InitRotated90DontApplyRotation) { + TestInit(640, 360, webrtc::kVideoRotation_90, false); }