Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(70)

Side by Side Diff: content/browser/renderer_host/video_capture_controller.cc

Issue 7002027: VideoCaptureHost (Closed) Base URL: http://src.chromium.org/svn/trunk/src/
Patch Set: '' Created 9 years, 7 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View unified diff | Download patch | Annotate | Revision Log
OLDNEW
(Empty)
1 // Copyright (c) 2011 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
4
5 #include "content/browser/renderer_host/video_capture_controller.h"
6
7 #include "base/logging.h"
8 #include "base/stl_util-inl.h"
9 #include "content/browser/browser_thread.h"
10 #include "content/browser/media_stream/video_capture_manager.h"
11 #include "media/base/yuv_convert.h"
12
13 // The number of TransportDIBs VideoCaptureController allocate.
14 static const unsigned int kNoOfDIBS = 3;
scherkus (not reviewing) 2011/05/23 05:05:09 size_t ?
Per K 2011/05/23 12:05:47 Done.
15
16 VideoCaptureController::VideoCaptureController(
17 ControllerId id,
18 VideoCaptureController::EventHandler* event_handler)
19 : report_ready_to_delete_(false),
20 event_handler_(*event_handler),
21 id_(id) {}
22
23 VideoCaptureController::~VideoCaptureController() {
24 // Delete all TransportDIBs
scherkus (not reviewing) 2011/05/23 05:05:09 comments end w/ periods
Per K 2011/05/23 12:05:47 Done.
25 STLDeleteContainerPairSecondPointers(owned_dibs_.begin(),
26 owned_dibs_.end());
27 }
28
29 void VideoCaptureController::StartCapture(
30 const media::VideoCaptureParams& params) {
31 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::IO));
32
33 params_ = params;
34 media_stream::VideoCaptureManager* manager =
35 media_stream::VideoCaptureManager::Get();
36 // Order the manager to start the actual capture.
37 manager->Start(params, this);
38 }
39
40 void VideoCaptureController::StopCapture(Task* stopped_task) {
41 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::IO));
42
43 media_stream::VideoCaptureManager* manager =
44 media_stream::VideoCaptureManager::Get();
45 manager->Stop(params_.session_id,
46 NewRunnableMethod(this,
47 &VideoCaptureController::OnDeviceStopped,
48 stopped_task));
49 }
50
51 void VideoCaptureController::ReturnTransportDIB(TransportDIB::Handle handle) {
52 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::IO));
53 // Check if we own this handle
scherkus (not reviewing) 2011/05/23 05:05:09 indent
Per K 2011/05/23 12:05:47 Done.
54 CHECK(owned_dibs_.find(handle) != owned_dibs_.end());
55
56 bool ready_to_delete;
57 lock_.Acquire();
scherkus (not reviewing) 2011/05/23 05:05:09 nit: use AutoLock with a scope { AutoLock l(loc
Per K 2011/05/23 12:05:47 Done.
58 free_dibs_.push_back(handle);
59 ready_to_delete = (free_dibs_.size() == owned_dibs_.size());
60 lock_.Release();
61
62 if (report_ready_to_delete_ && ready_to_delete) {
63 event_handler_.OnReadyToDelete(id_);
64 }
65 }
66
67 ///////////////////////////////////////////////////////////////////////////////
68 // Implements VideoCaptureDevice::EventHandler.
69 void VideoCaptureController::OnIncomingCapturedFrame(const uint8* data,
scherkus (not reviewing) 2011/05/23 05:05:09 which thread is this executing on? we're doing co
Per K 2011/05/23 12:05:47 This will be done by the thread running the captur
70 int length,
71 base::Time timestamp) {
72 TransportDIB::Handle handle;
73 // Check if there is a TransportDIB to fill.
74 bool buffer_exist = false;
75 lock_.Acquire();
scherkus (not reviewing) 2011/05/23 05:05:09 nit: use AutoLock with a scope { AutoLock l(loc
Per K 2011/05/23 12:05:47 Done.
76 if (!report_ready_to_delete_ && free_dibs_.size() > 0) {
77 handle = free_dibs_.back();
78 free_dibs_.pop_back();
79 buffer_exist = true;
80 }
81 lock_.Release();
82
83 if (!buffer_exist) {
84 return;
85 }
86
87 TransportDIB* dib = TransportDIB::Map(handle);
88 uint8* target = static_cast<uint8*> (dib->memory());
scherkus (not reviewing) 2011/05/23 05:05:09 no space between > and (
Per K 2011/05/23 12:05:47 Done.
89 CHECK(dib->size() >= (unsigned int) (frame_info_.width*frame_info_.height*3) /
scherkus (not reviewing) 2011/05/23 05:05:09 static_cast<>
Per K 2011/05/23 12:05:47 Done.
90 2);
91
92 // Do color conversion from the camera format to I420.
93 switch (frame_info_.color) {
94 case media::VideoCaptureDevice::kColorUnknown: // Color format not set
95 break;
96 case media::VideoCaptureDevice::kI420: {
97 memcpy(target, data, (frame_info_.width * frame_info_.height * 3) / 2);
98 break;
99 }
100 case media::VideoCaptureDevice::kYUY2: {
101 uint8* yplane = target;
102 uint8* uplane = target + frame_info_.width * frame_info_.height;
scherkus (not reviewing) 2011/05/23 05:05:09 nit: remove extra space after +
Per K 2011/05/23 12:05:47 Done.
103 uint8* vplane = uplane + (frame_info_.width * frame_info_.height) / 4;
scherkus (not reviewing) 2011/05/23 05:05:09 nit: remove extra space after +
Per K 2011/05/23 12:05:47 Done.
104 media::ConvertYUY2ToYUV(data, yplane, uplane, vplane, frame_info_.width,
105 frame_info_.height);
106 break;
107 }
108 case media::VideoCaptureDevice::kRGB24: {
109 #if defined(OS_WIN) // RGB on Windows start at the bottom line.
110 uint8* yplane = target;
111 uint8* uplane = target + frame_info_.width * frame_info_.height;
112 uint8* vplane = uplane + (frame_info_.width * frame_info_.height) / 4;
113 int ystride = frame_info_.width;
114 int uvstride = frame_info_.width / 2;
115 int rgb_stride = - 3 * frame_info_.width;
116 const uint8* rgb_src = data + 3 * frame_info_.width *
117 (frame_info_.height -1);
118 #else
119 uint8* yplane = target;
120 uint8* uplane = target + frame_info_.width * frame_info_.height;
scherkus (not reviewing) 2011/05/23 05:05:09 nit: remove extra space after +
Per K 2011/05/23 12:05:47 Done.
121 uint8* vplane = uplane + (frame_info_.width * frame_info_.height) / 4;
scherkus (not reviewing) 2011/05/23 05:05:09 nit: remove extra space after +
Per K 2011/05/23 12:05:47 Done.
122 int ystride = frame_info_.width;
123 int uvstride = frame_info_.width / 2;
124 int rgb_stride = 3 * frame_info_.width;
125 const uint8* rgb_src = data;
126 #endif
127 media::ConvertRGB24ToYUV(rgb_src, yplane, uplane, vplane,
128 frame_info_.width, frame_info_.height,
129 rgb_stride, ystride, uvstride);
130 break;
131 }
132 case media::VideoCaptureDevice::kARGB: {
133 uint8* yplane = target;
134 uint8* uplane = target + frame_info_.width * frame_info_.height;
scherkus (not reviewing) 2011/05/23 05:05:09 nit: remove extra space after +
Per K 2011/05/23 12:05:47 Done.
135 uint8* vplane = uplane + (frame_info_.width * frame_info_.height) / 4;
scherkus (not reviewing) 2011/05/23 05:05:09 nit: remove extra space after +
Per K 2011/05/23 12:05:47 Done.
136 media::ConvertRGB32ToYUV(data, yplane, uplane, vplane, frame_info_.width,
137 frame_info_.height, frame_info_.width * 4,
138 frame_info_.width, frame_info_.width / 2);
139 break;
140 }
141 default:
142 NOTREACHED();
143 }
144
145 event_handler_.OnBufferReady(id_, handle, timestamp);
146 }
147
148 void VideoCaptureController::OnError() {
149 event_handler_.OnError(id_);
150 }
151
152 void VideoCaptureController::OnFrameInfo(
153 const media::VideoCaptureDevice::Capability& info) {
154 DCHECK(owned_dibs_.empty());
155 const size_t needed_size = (info.width * info.height * 3) / 2;
156 for (unsigned int i = 0; i < kNoOfDIBS; ++i) {
157 TransportDIB* dib = TransportDIB::Create(needed_size, i);
158 if (!dib) {
159 break;
160 }
161 // Lock needed since the buffers are used in OnIncomingFrame
162 // and we need to use it there in order to avoid memcpy of complete frames.
163 lock_.Acquire();
scherkus (not reviewing) 2011/05/23 05:05:09 nit: use AutoLock with a scope { AutoLock l(loc
Per K 2011/05/23 12:05:47 Done.
164 owned_dibs_.insert(std::make_pair(dib->handle(), dib));
165 free_dibs_.push_back(dib->handle());
166 lock_.Release();
167 }
168 frame_info_= info;
169
170 // Check that all Dibs where created.
171 if (owned_dibs_.size() != kNoOfDIBS) {
scherkus (not reviewing) 2011/05/23 05:05:09 warning: accessing owned_dibs_ outside of lock
Per K 2011/05/23 12:05:47 Done.
172 event_handler_.OnError(id_);
173 }
174 event_handler_.OnFrameInfo(id_, info.width, info.height,
175 info.frame_rate);
176 }
177
178 ///////////////////////////////////////////////////////////////////////////////
179 // Called by VideoCaptureManager when a device have been stopped.
180 // This will report to the event handler that this object is ready to be deleted
181 // if all DIBS have been returned.
182 void VideoCaptureController::OnDeviceStopped(Task* stopped_task) {
183 bool ready_to_delete_now;
184 lock_.Acquire();
scherkus (not reviewing) 2011/05/23 05:05:09 nit: use AutoLock with a scope { AutoLock l(loc
Per K 2011/05/23 12:05:47 Done.
185 // Set flag to indicate we need to report when all DIBs have been returned.
186 report_ready_to_delete_ = true;
187 ready_to_delete_now = (free_dibs_.size() == owned_dibs_.size());
188 lock_.Release();
189 if (ready_to_delete_now) {
190 event_handler_.OnReadyToDelete(id_);
191 }
192 if (stopped_task) {
scherkus (not reviewing) 2011/05/23 05:05:09 who would call OnDeviceStopped() w/o a task? it l
Per K 2011/05/23 12:05:47 VideoCaptureController::StopCapture called from V
193 stopped_task->Run();
194 delete stopped_task;
195 }
196 }
OLDNEW

Powered by Google App Engine
This is Rietveld 408576698