Index: media/video/capture/mac/video_capture_device_factory_mac.mm |
diff --git a/media/video/capture/mac/video_capture_device_mac.mm b/media/video/capture/mac/video_capture_device_factory_mac.mm |
similarity index 22% |
copy from media/video/capture/mac/video_capture_device_mac.mm |
copy to media/video/capture/mac/video_capture_device_factory_mac.mm |
index bc3d54153c5bd355e01d6f18181546f8832d01e1..0e24203340eeeb4bff80ac236f6b6851583f2e3f 100644 |
--- a/media/video/capture/mac/video_capture_device_mac.mm |
+++ b/media/video/capture/mac/video_capture_device_factory_mac.mm |
@@ -1,27 +1,16 @@ |
-// Copyright (c) 2012 The Chromium Authors. All rights reserved. |
+// Copyright 2014 The Chromium Authors. All rights reserved. |
// Use of this source code is governed by a BSD-style license that can be |
// found in the LICENSE file. |
-#include "media/video/capture/mac/video_capture_device_mac.h" |
+#include "media/video/capture/mac/video_capture_device_factory_mac.h" |
-#include "base/bind.h" |
-#include "base/location.h" |
-#include "base/logging.h" |
-#include "base/message_loop/message_loop_proxy.h" |
-#include "base/time/time.h" |
#import "media/video/capture/mac/avfoundation_glue.h" |
-#import "media/video/capture/mac/platform_video_capturing_mac.h" |
+#include "media/video/capture/mac/video_capture_device_mac.h" |
#import "media/video/capture/mac/video_capture_device_avfoundation_mac.h" |
#import "media/video/capture/mac/video_capture_device_qtkit_mac.h" |
namespace media { |
-const int kMinFrameRate = 1; |
-const int kMaxFrameRate = 30; |
- |
-// In device identifiers, the USB VID and PID are stored in 4 bytes each. |
-const size_t kVidPidSize = 4; |
- |
// Some devices are not correctly supported in AVFoundation, f.i. Blackmagic, |
// see http://crbug.com/347371. The devices are identified by USB Vendor ID and |
// by a characteristic substring of the name, usually the vendor's name. |
@@ -30,47 +19,41 @@ const struct NameAndVid { |
const char* name; |
} kBlacklistedCameras[] = { { "a82c", "Blackmagic" } }; |
-const struct Resolution { |
- const int width; |
- const int height; |
-} kQVGA = { 320, 240 }, |
- kVGA = { 640, 480 }, |
- kHD = { 1280, 720 }; |
+// In device identifiers, the USB VID and PID are stored in 4 bytes each. |
+const size_t kVidPidSize = 4; |
-const struct Resolution* const kWellSupportedResolutions[] = { |
- &kQVGA, |
- &kVGA, |
- &kHD, |
-}; |
+VideoCaptureDeviceFactoryMac::VideoCaptureDeviceFactoryMac() { |
+ thread_checker_.DetachFromThread(); |
+} |
-// Rescaling the image to fix the pixel aspect ratio runs the risk of making |
-// the aspect ratio worse, if QTKit selects a new source mode with a different |
-// shape. This constant ensures that we don't take this risk if the current |
-// aspect ratio is tolerable. |
-const float kMaxPixelAspectRatio = 1.15; |
+scoped_ptr<VideoCaptureDevice> VideoCaptureDeviceFactoryMac::Create( |
+ const VideoCaptureDevice::Name& device_name) { |
+ DCHECK(thread_checker_.CalledOnValidThread()); |
+ DCHECK_NE(device_name.capture_api_type(), |
+ VideoCaptureDevice::Name::API_TYPE_UNKNOWN); |
-// TODO(ronghuawu): Replace this with CapabilityList::GetBestMatchedCapability. |
-void GetBestMatchSupportedResolution(int* width, int* height) { |
- int min_diff = kint32max; |
- int matched_width = *width; |
- int matched_height = *height; |
- int desired_res_area = *width * *height; |
- for (size_t i = 0; i < arraysize(kWellSupportedResolutions); ++i) { |
- int area = kWellSupportedResolutions[i]->width * |
- kWellSupportedResolutions[i]->height; |
- int diff = std::abs(desired_res_area - area); |
- if (diff < min_diff) { |
- min_diff = diff; |
- matched_width = kWellSupportedResolutions[i]->width; |
- matched_height = kWellSupportedResolutions[i]->height; |
- } |
+ VideoCaptureDevice::Names device_names; |
+ GetDeviceNames(&device_names); |
+ VideoCaptureDevice::Names::iterator it = device_names.begin(); |
+ for (; it != device_names.end(); ++it) { |
+ if (it->id() == device_name.id()) |
+ break; |
+ } |
+ if (it == device_names.end()) |
+ return scoped_ptr<VideoCaptureDevice>(); |
+ |
+ scoped_ptr<VideoCaptureDeviceMac> capture_device( |
+ new VideoCaptureDeviceMac(device_name)); |
+ if (!capture_device->Init(device_name.capture_api_type())) { |
+ LOG(ERROR) << "Could not initialize VideoCaptureDevice."; |
+ capture_device.reset(); |
} |
- *width = matched_width; |
- *height = matched_height; |
+ return scoped_ptr<VideoCaptureDevice>(capture_device.Pass()); |
} |
-//static |
-void VideoCaptureDevice::GetDeviceNames(Names* device_names) { |
+void VideoCaptureDeviceFactoryMac::GetDeviceNames( |
+ VideoCaptureDevice::Names* const device_names) { |
+ DCHECK(thread_checker_.CalledOnValidThread()); |
// Loop through all available devices and add to |device_names|. |
NSDictionary* capture_devices; |
if (AVFoundationGlue::IsAVFoundationSupported()) { |
@@ -81,8 +64,9 @@ void VideoCaptureDevice::GetDeviceNames(Names* device_names) { |
// Enumerate all devices found by AVFoundation, translate the info for each |
// to class Name and add it to |device_names|. |
for (NSString* key in capture_devices) { |
- Name name([[capture_devices valueForKey:key] UTF8String], |
- [key UTF8String], Name::AVFOUNDATION); |
+ VideoCaptureDevice::Name name( |
+ [[capture_devices valueForKey:key] UTF8String], |
+ [key UTF8String], VideoCaptureDevice::Name::AVFOUNDATION); |
device_names->push_back(name); |
// Extract the device's Vendor ID and compare to all blacklisted ones. |
device_vid = name.GetModel().substr(0, kVidPidSize); |
@@ -105,8 +89,9 @@ void VideoCaptureDevice::GetDeviceNames(Names* device_names) { |
if ([device_name rangeOfString:@(kBlacklistedCameras[i].name) |
options:NSCaseInsensitiveSearch].length != 0) { |
DVLOG(1) << "Enumerated blacklisted " << [device_name UTF8String]; |
- Name name("QTKit " + std::string([device_name UTF8String]), |
- [key UTF8String], Name::QTKIT); |
+ VideoCaptureDevice::Name name( |
+ "QTKit " + std::string([device_name UTF8String]), |
+ [key UTF8String], VideoCaptureDevice::Name::QTKIT); |
device_names->push_back(name); |
} |
} |
@@ -116,271 +101,25 @@ void VideoCaptureDevice::GetDeviceNames(Names* device_names) { |
DVLOG(1) << "Enumerating video capture devices using QTKit"; |
capture_devices = [VideoCaptureDeviceQTKit deviceNames]; |
for (NSString* key in capture_devices) { |
- Name name([[capture_devices valueForKey:key] UTF8String], |
- [key UTF8String], Name::QTKIT); |
+ VideoCaptureDevice::Name name( |
+ [[capture_devices valueForKey:key] UTF8String], |
+ [key UTF8String], VideoCaptureDevice::Name::QTKIT); |
device_names->push_back(name); |
} |
} |
} |
-// static |
-void VideoCaptureDevice::GetDeviceSupportedFormats(const Name& device, |
- VideoCaptureFormats* formats) { |
- if (device.capture_api_type() == Name::AVFOUNDATION) { |
+void VideoCaptureDeviceFactoryMac::GetDeviceSupportedFormats( |
+ const VideoCaptureDevice::Name& device, |
+ VideoCaptureFormats* supported_formats) { |
+ DCHECK(thread_checker_.CalledOnValidThread()); |
+ if (device.capture_api_type() == VideoCaptureDevice::Name::AVFOUNDATION) { |
DVLOG(1) << "Enumerating video capture capabilities, AVFoundation"; |
[VideoCaptureDeviceAVFoundation getDevice:device |
- supportedFormats:formats]; |
+ supportedFormats:supported_formats]; |
} else { |
NOTIMPLEMENTED(); |
} |
} |
-const std::string VideoCaptureDevice::Name::GetModel() const { |
- // Both PID and VID are 4 characters. |
- if (unique_id_.size() < 2 * kVidPidSize) { |
- return ""; |
- } |
- |
- // The last characters of device id is a concatenation of VID and then PID. |
- const size_t vid_location = unique_id_.size() - 2 * kVidPidSize; |
- std::string id_vendor = unique_id_.substr(vid_location, kVidPidSize); |
- const size_t pid_location = unique_id_.size() - kVidPidSize; |
- std::string id_product = unique_id_.substr(pid_location, kVidPidSize); |
- |
- return id_vendor + ":" + id_product; |
-} |
- |
-VideoCaptureDevice* VideoCaptureDevice::Create(const Name& device_name) { |
- VideoCaptureDeviceMac* capture_device = |
- new VideoCaptureDeviceMac(device_name); |
- if (!capture_device->Init()) { |
- LOG(ERROR) << "Could not initialize VideoCaptureDevice."; |
- delete capture_device; |
- capture_device = NULL; |
- } |
- return capture_device; |
-} |
- |
-VideoCaptureDeviceMac::VideoCaptureDeviceMac(const Name& device_name) |
- : device_name_(device_name), |
- tried_to_square_pixels_(false), |
- task_runner_(base::MessageLoopProxy::current()), |
- state_(kNotInitialized), |
- capture_device_(nil), |
- weak_factory_(this) { |
- final_resolution_selected_ = AVFoundationGlue::IsAVFoundationSupported(); |
-} |
- |
-VideoCaptureDeviceMac::~VideoCaptureDeviceMac() { |
- DCHECK(task_runner_->BelongsToCurrentThread()); |
- [capture_device_ release]; |
-} |
- |
-void VideoCaptureDeviceMac::AllocateAndStart( |
- const VideoCaptureParams& params, |
- scoped_ptr<VideoCaptureDevice::Client> client) { |
- DCHECK(task_runner_->BelongsToCurrentThread()); |
- if (state_ != kIdle) { |
- return; |
- } |
- int width = params.requested_format.frame_size.width(); |
- int height = params.requested_format.frame_size.height(); |
- int frame_rate = params.requested_format.frame_rate; |
- |
- // QTKit API can scale captured frame to any size requested, which would lead |
- // to undesired aspect ratio changes. Try to open the camera with a known |
- // supported format and let the client crop/pad the captured frames. |
- if (!AVFoundationGlue::IsAVFoundationSupported()) |
- GetBestMatchSupportedResolution(&width, &height); |
- |
- client_ = client.Pass(); |
- NSString* deviceId = |
- [NSString stringWithUTF8String:device_name_.id().c_str()]; |
- |
- [capture_device_ setFrameReceiver:this]; |
- |
- if (![capture_device_ setCaptureDevice:deviceId]) { |
- SetErrorState("Could not open capture device."); |
- return; |
- } |
- if (frame_rate < kMinFrameRate) |
- frame_rate = kMinFrameRate; |
- else if (frame_rate > kMaxFrameRate) |
- frame_rate = kMaxFrameRate; |
- |
- capture_format_.frame_size.SetSize(width, height); |
- capture_format_.frame_rate = frame_rate; |
- capture_format_.pixel_format = PIXEL_FORMAT_UYVY; |
- |
- // QTKit: Set the capture resolution only if this is VGA or smaller, otherwise |
- // leave it unconfigured and start capturing: QTKit will produce frames at the |
- // native resolution, allowing us to identify cameras whose native resolution |
- // is too low for HD. This additional information comes at a cost in startup |
- // latency, because the webcam will need to be reopened if its default |
- // resolution is not HD or VGA. |
- // AVfoundation is configured for all resolutions. |
- if (AVFoundationGlue::IsAVFoundationSupported() || width <= kVGA.width || |
- height <= kVGA.height) { |
- if (!UpdateCaptureResolution()) |
- return; |
- } |
- if (![capture_device_ startCapture]) { |
- SetErrorState("Could not start capture device."); |
- return; |
- } |
- |
- state_ = kCapturing; |
-} |
- |
-void VideoCaptureDeviceMac::StopAndDeAllocate() { |
- DCHECK(task_runner_->BelongsToCurrentThread()); |
- DCHECK(state_ == kCapturing || state_ == kError) << state_; |
- [capture_device_ stopCapture]; |
- |
- [capture_device_ setCaptureDevice:nil]; |
- [capture_device_ setFrameReceiver:nil]; |
- client_.reset(); |
- state_ = kIdle; |
- tried_to_square_pixels_ = false; |
-} |
- |
-bool VideoCaptureDeviceMac::Init() { |
- DCHECK(task_runner_->BelongsToCurrentThread()); |
- DCHECK_EQ(state_, kNotInitialized); |
- |
- // TODO(mcasas): The following check might not be necessary; if the device has |
- // disappeared after enumeration and before coming here, opening would just |
- // fail but not necessarily produce a crash. |
- Names device_names; |
- GetDeviceNames(&device_names); |
- Names::iterator it = device_names.begin(); |
- for (; it != device_names.end(); ++it) { |
- if (it->id() == device_name_.id()) |
- break; |
- } |
- if (it == device_names.end()) |
- return false; |
- |
- DCHECK_NE(it->capture_api_type(), Name::API_TYPE_UNKNOWN); |
- if (it->capture_api_type() == Name::AVFOUNDATION) { |
- capture_device_ = |
- [[VideoCaptureDeviceAVFoundation alloc] initWithFrameReceiver:this]; |
- } else { |
- capture_device_ = |
- [[VideoCaptureDeviceQTKit alloc] initWithFrameReceiver:this]; |
- } |
- |
- if (!capture_device_) |
- return false; |
- |
- state_ = kIdle; |
- return true; |
-} |
- |
-void VideoCaptureDeviceMac::ReceiveFrame( |
- const uint8* video_frame, |
- int video_frame_length, |
- const VideoCaptureFormat& frame_format, |
- int aspect_numerator, |
- int aspect_denominator) { |
- // This method is safe to call from a device capture thread, i.e. any thread |
- // controlled by QTKit/AVFoundation. |
- if (!final_resolution_selected_) { |
- DCHECK(!AVFoundationGlue::IsAVFoundationSupported()); |
- if (capture_format_.frame_size.width() > kVGA.width || |
- capture_format_.frame_size.height() > kVGA.height) { |
- // We are requesting HD. Make sure that the picture is good, otherwise |
- // drop down to VGA. |
- bool change_to_vga = false; |
- if (frame_format.frame_size.width() < |
- capture_format_.frame_size.width() || |
- frame_format.frame_size.height() < |
- capture_format_.frame_size.height()) { |
- // These are the default capture settings, not yet configured to match |
- // |capture_format_|. |
- DCHECK(frame_format.frame_rate == 0); |
- DVLOG(1) << "Switching to VGA because the default resolution is " << |
- frame_format.frame_size.ToString(); |
- change_to_vga = true; |
- } |
- |
- if (capture_format_.frame_size == frame_format.frame_size && |
- aspect_numerator != aspect_denominator) { |
- DVLOG(1) << "Switching to VGA because HD has nonsquare pixel " << |
- "aspect ratio " << aspect_numerator << ":" << aspect_denominator; |
- change_to_vga = true; |
- } |
- |
- if (change_to_vga) |
- capture_format_.frame_size.SetSize(kVGA.width, kVGA.height); |
- } |
- |
- if (capture_format_.frame_size == frame_format.frame_size && |
- !tried_to_square_pixels_ && |
- (aspect_numerator > kMaxPixelAspectRatio * aspect_denominator || |
- aspect_denominator > kMaxPixelAspectRatio * aspect_numerator)) { |
- // The requested size results in non-square PAR. Shrink the frame to 1:1 |
- // PAR (assuming QTKit selects the same input mode, which is not |
- // guaranteed). |
- int new_width = capture_format_.frame_size.width(); |
- int new_height = capture_format_.frame_size.height(); |
- if (aspect_numerator < aspect_denominator) |
- new_width = (new_width * aspect_numerator) / aspect_denominator; |
- else |
- new_height = (new_height * aspect_denominator) / aspect_numerator; |
- capture_format_.frame_size.SetSize(new_width, new_height); |
- tried_to_square_pixels_ = true; |
- } |
- |
- if (capture_format_.frame_size == frame_format.frame_size) { |
- final_resolution_selected_ = true; |
- } else { |
- UpdateCaptureResolution(); |
- // Let the resolution update sink through QTKit and wait for next frame. |
- return; |
- } |
- } |
- |
- // QTKit capture source can change resolution if someone else reconfigures the |
- // camera, and that is fine: http://crbug.com/353620. In AVFoundation, this |
- // should not happen, it should resize internally. |
- if (!AVFoundationGlue::IsAVFoundationSupported()) { |
- capture_format_.frame_size = frame_format.frame_size; |
- } else if (capture_format_.frame_size != frame_format.frame_size) { |
- ReceiveError("Captured resolution " + frame_format.frame_size.ToString() + |
- ", and expected " + capture_format_.frame_size.ToString()); |
- return; |
- } |
- |
- client_->OnIncomingCapturedData(video_frame, |
- video_frame_length, |
- capture_format_, |
- 0, |
- base::TimeTicks::Now()); |
-} |
- |
-void VideoCaptureDeviceMac::ReceiveError(const std::string& reason) { |
- task_runner_->PostTask(FROM_HERE, |
- base::Bind(&VideoCaptureDeviceMac::SetErrorState, |
- weak_factory_.GetWeakPtr(), |
- reason)); |
-} |
- |
-void VideoCaptureDeviceMac::SetErrorState(const std::string& reason) { |
- DCHECK(task_runner_->BelongsToCurrentThread()); |
- DLOG(ERROR) << reason; |
- state_ = kError; |
- client_->OnError(reason); |
-} |
- |
-bool VideoCaptureDeviceMac::UpdateCaptureResolution() { |
- if (![capture_device_ setCaptureHeight:capture_format_.frame_size.height() |
- width:capture_format_.frame_size.width() |
- frameRate:capture_format_.frame_rate]) { |
- ReceiveError("Could not configure capture device."); |
- return false; |
- } |
- return true; |
-} |
- |
-} // namespace media |
+} // namespace media |