summaryrefslogtreecommitdiffstats
path: root/content
diff options
context:
space:
mode:
authorronghuawu@chromium.org <ronghuawu@chromium.org@0039d316-1c4b-4281-b951-d872f2087c98>2014-02-20 00:02:38 +0000
committerronghuawu@chromium.org <ronghuawu@chromium.org@0039d316-1c4b-4281-b951-d872f2087c98>2014-02-20 00:02:38 +0000
commit8854cdda2dceab4d1659ada3f07da54bc1eaf5fc (patch)
tree2fbaad2e383e22860078ee9b4ffc34f054dc6099 /content
parent479c25578c68c62a54230f66cb29b2d3cd8ddbf2 (diff)
downloadchromium_src-8854cdda2dceab4d1659ada3f07da54bc1eaf5fc.zip
chromium_src-8854cdda2dceab4d1659ada3f07da54bc1eaf5fc.tar.gz
chromium_src-8854cdda2dceab4d1659ada3f07da54bc1eaf5fc.tar.bz2
The cricket::VideoCapturer no longer makes a copy of the frame from the camera. As a result we can't just do a shallow copy as the underlying buffer maybe invalid after the function returns. Instead we now have to make a copy in video_source_handler before return.
TEST=updated unit test BUG=344957 R=perkj@chromium.org, wjia@chromium.org Review URL: https://codereview.chromium.org/172843002 git-svn-id: svn://svn.chromium.org/chrome/trunk/src@252129 0039d316-1c4b-4281-b951-d872f2087c98
Diffstat (limited to 'content')
-rw-r--r--content/renderer/media/video_source_handler.cc8
-rw-r--r--content/renderer/media/video_source_handler_unittest.cc60
2 files changed, 51 insertions, 17 deletions
diff --git a/content/renderer/media/video_source_handler.cc b/content/renderer/media/video_source_handler.cc
index 5adb90a..036fa90 100644
--- a/content/renderer/media/video_source_handler.cc
+++ b/content/renderer/media/video_source_handler.cc
@@ -36,9 +36,11 @@ class PpFrameReceiver : public cricket::VideoRenderer {
virtual bool RenderFrame(const cricket::VideoFrame* frame) OVERRIDE {
base::AutoLock auto_lock(lock_);
if (reader_) {
- // Make a shallow copy of the frame as the |reader_| may need to queue it.
- // Both frames will share a single reference-counted frame buffer.
- reader_->GotFrame(frame->Copy());
+ // |frame| will be invalid after this function is returned. So keep a copy
+ // before return.
+ cricket::VideoFrame* copied_frame = frame->Copy();
+ copied_frame->MakeExclusive();
+ reader_->GotFrame(copied_frame);
}
return true;
}
diff --git a/content/renderer/media/video_source_handler_unittest.cc b/content/renderer/media/video_source_handler_unittest.cc
index 5841274..5c1bbaa 100644
--- a/content/renderer/media/video_source_handler_unittest.cc
+++ b/content/renderer/media/video_source_handler_unittest.cc
@@ -14,6 +14,7 @@
#include "testing/gtest/include/gtest/gtest.h"
#include "third_party/WebKit/public/platform/WebMediaStreamTrack.h"
#include "third_party/WebKit/public/platform/WebString.h"
+#include "third_party/libjingle/source/talk/media/base/videocapturer.h"
#include "third_party/libjingle/source/talk/media/base/videorenderer.h"
#include "third_party/libjingle/source/talk/media/webrtc/webrtcvideoframe.h"
@@ -59,27 +60,58 @@ TEST_F(VideoSourceHandlerTest, OpenClose) {
// Unknow url will return false.
EXPECT_FALSE(handler_->Open(kUnknownStreamUrl, &reader));
EXPECT_TRUE(handler_->Open(kTestStreamUrl, &reader));
- cricket::WebRtcVideoFrame test_frame;
- int width = 640;
- int height = 360;
+
+ size_t width = 640;
+ size_t height = 360;
int64 et = 123456;
int64 ts = 789012;
- test_frame.InitToBlack(width, height, 1, 1, et, ts);
+ size_t size = VideoFrame::SizeOf(width, height);
+ std::vector<uint8_t> test_buffer(size);
+ for (size_t i = 0; i < size; ++i) {
+ test_buffer[i] = (i & 0xFF);
+ }
+
+ // A new frame is captured.
+ cricket::CapturedFrame captured_frame;
+ captured_frame.width = width;
+ captured_frame.height = height;
+ captured_frame.fourcc = cricket::FOURCC_I420;
+ // cricket::CapturedFrame time is in nanoseconds.
+ captured_frame.elapsed_time = et;
+ captured_frame.time_stamp = ts;
+ captured_frame.data = &test_buffer[0];
+ captured_frame.data_size = size;
+ captured_frame.pixel_height = 1;
+ captured_frame.pixel_width = 1;
+
+ // The frame is delivered to VideoSourceHandler as cricket::VideoFrame.
+ cricket::WebRtcVideoFrame i420_frame;
+ EXPECT_TRUE(i420_frame.Alias(&captured_frame, width, height));
cricket::VideoRenderer* receiver = handler_->GetReceiver(&reader);
ASSERT(receiver != NULL);
- receiver->RenderFrame(&test_frame);
+ receiver->RenderFrame(&i420_frame);
+ // Compare |frame| to |captured_frame|.
const VideoFrame* frame = reader.last_frame();
ASSERT_TRUE(frame != NULL);
-
- // Compare |frame| to |test_frame|.
- EXPECT_EQ(test_frame.GetWidth(), frame->GetWidth());
- EXPECT_EQ(test_frame.GetHeight(), frame->GetHeight());
- EXPECT_EQ(test_frame.GetElapsedTime(), frame->GetElapsedTime());
- EXPECT_EQ(test_frame.GetTimeStamp(), frame->GetTimeStamp());
- EXPECT_EQ(test_frame.GetYPlane(), frame->GetYPlane());
- EXPECT_EQ(test_frame.GetUPlane(), frame->GetUPlane());
- EXPECT_EQ(test_frame.GetVPlane(), frame->GetVPlane());
+ EXPECT_EQ(width, frame->GetWidth());
+ EXPECT_EQ(height, frame->GetHeight());
+ EXPECT_EQ(et, frame->GetElapsedTime());
+ EXPECT_EQ(ts, frame->GetTimeStamp());
+ std::vector<uint8_t> tmp_buffer1(size);
+ EXPECT_EQ(size, frame->CopyToBuffer(&tmp_buffer1[0], size));
+ EXPECT_TRUE(std::equal(test_buffer.begin(), test_buffer.end(),
+ tmp_buffer1.begin()));
+
+ // Invalid the original frame
+ memset(&test_buffer[0], 0, size);
+ test_buffer.clear();
+
+ // We should still have the same |frame|.
+ std::vector<uint8_t> tmp_buffer2(size);
+ EXPECT_EQ(size, frame->CopyToBuffer(&tmp_buffer2[0], size));
+ EXPECT_TRUE(std::equal(tmp_buffer1.begin(), tmp_buffer1.end(),
+ tmp_buffer2.begin()));
EXPECT_FALSE(handler_->Close(NULL));
EXPECT_TRUE(handler_->Close(&reader));