Index: content/browser/renderer_host/media/video_capture_controller.cc |
=================================================================== |
--- content/browser/renderer_host/media/video_capture_controller.cc (revision 151718) |
+++ content/browser/renderer_host/media/video_capture_controller.cc (working copy) |
@@ -67,7 +67,9 @@ |
VideoCaptureController::VideoCaptureController( |
media_stream::VideoCaptureManager* video_capture_manager) |
- : frame_info_available_(false), |
+ : chopped_width_(0), |
+ chopped_height_(0), |
+ frame_info_available_(false), |
video_capture_manager_(video_capture_manager), |
device_in_use_(false), |
state_(video_capture::kStopped) { |
@@ -277,10 +279,12 @@ |
case media::VideoCaptureCapability::kColorUnknown: // Color format not set. |
break; |
case media::VideoCaptureCapability::kI420: { |
+ DCHECK(!chopped_width_ && !chopped_height_); |
memcpy(target, data, (frame_info_.width * frame_info_.height * 3) / 2); |
break; |
} |
case media::VideoCaptureCapability::kYV12: { |
+ DCHECK(!chopped_width_ && !chopped_height_); |
const uint8* ptr = data; |
memcpy(yplane, ptr, (frame_info_.width * frame_info_.height)); |
ptr += frame_info_.width * frame_info_.height; |
@@ -290,11 +294,13 @@ |
break; |
} |
case media::VideoCaptureCapability::kNV21: { |
+ DCHECK(!chopped_width_ && !chopped_height_); |
media::ConvertNV21ToYUV(data, yplane, uplane, vplane, frame_info_.width, |
frame_info_.height); |
break; |
} |
case media::VideoCaptureCapability::kYUY2: { |
+ DCHECK(!chopped_width_ && !chopped_height_); |
media::ConvertYUY2ToYUV(data, yplane, uplane, vplane, frame_info_.width, |
frame_info_.height); |
break; |
@@ -303,11 +309,11 @@ |
int ystride = frame_info_.width; |
int uvstride = frame_info_.width / 2; |
#if defined(OS_WIN) // RGB on Windows start at the bottom line. |
- int rgb_stride = -3 * frame_info_.width; |
- const uint8* rgb_src = data + 3 * frame_info_.width * |
- (frame_info_.height -1); |
+ int rgb_stride = -3 * (frame_info_.width + chopped_width_); |
+ const uint8* rgb_src = data + 3 * (frame_info_.width + chopped_width_) * |
+ (frame_info_.height -1 + chopped_height_); |
#else |
- int rgb_stride = 3 * frame_info_.width; |
+ int rgb_stride = 3 * (frame_info_.width + chopped_width_); |
const uint8* rgb_src = data; |
#endif |
media::ConvertRGB24ToYUV(rgb_src, yplane, uplane, vplane, |
@@ -317,7 +323,8 @@ |
} |
case media::VideoCaptureCapability::kARGB: { |
media::ConvertRGB32ToYUV(data, yplane, uplane, vplane, frame_info_.width, |
- frame_info_.height, frame_info_.width * 4, |
+ frame_info_.height, |
+ (frame_info_.width + chopped_width_) * 4, |
frame_info_.width, frame_info_.width / 2); |
break; |
} |
@@ -341,10 +348,22 @@ |
void VideoCaptureController::OnFrameInfo( |
const media::VideoCaptureCapability& info) { |
frame_info_= info; |
+ // Handle cases when |info| has odd numbers for width/height. |
+ if (info.width & 1) { |
+ --frame_info_.width; |
+ chopped_width_ = 1; |
+ } else { |
+ chopped_width_ = 0; |
+ } |
+ if (info.height & 1) { |
+ --frame_info_.height; |
+ chopped_height_ = 1; |
+ } else { |
+ chopped_height_ = 0; |
+ } |
BrowserThread::PostTask(BrowserThread::IO, |
FROM_HERE, |
- base::Bind(&VideoCaptureController::DoFrameInfoOnIOThread, |
- this, info)); |
+ base::Bind(&VideoCaptureController::DoFrameInfoOnIOThread, this)); |
} |
VideoCaptureController::~VideoCaptureController() { |
@@ -389,14 +408,17 @@ |
} |
} |
-void VideoCaptureController::DoFrameInfoOnIOThread( |
- const media::VideoCaptureCapability& info) { |
+void VideoCaptureController::DoFrameInfoOnIOThread() { |
DCHECK(BrowserThread::CurrentlyOn(BrowserThread::IO)); |
DCHECK(owned_dibs_.empty()) |
<< "Device is restarted without releasing shared memory."; |
+ // Allocate memory only when device has been started. |
+ if (state_ != video_capture::kStarted) |
+ return; |
+ |
bool frames_created = true; |
- const size_t needed_size = (info.width * info.height * 3) / 2; |
+ const size_t needed_size = (frame_info_.width * frame_info_.height * 3) / 2; |
{ |
base::AutoLock lock(lock_); |
for (size_t i = 1; i <= kNoOfDIBS; ++i) { |