blob: 966c3ee02aee362125449ecd44f6a2b7ce105ca2 [file] [log] [blame]
// Copyright 2017 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/capture/video/video_capture_system_impl.h"
#include <utility>
#include "base/bind.h"
#include "base/bind_helpers.h"
#include "build/build_config.h"
#include "media/base/bind_to_current_loop.h"
#if defined(OS_CHROMEOS)
#include "media/capture/video/chromeos/public/cros_features.h"
#include "media/capture/video/chromeos/video_capture_device_factory_chromeos.h"
#endif // defined(OS_CHROMEOS)
namespace {
// Compares two VideoCaptureFormat by checking smallest frame_size area, then
// by width, and then by _largest_ frame_rate. Used to order a
// VideoCaptureFormats vector so that the first entry for a given resolution has
// the largest frame rate.
bool IsCaptureFormatSmaller(const media::VideoCaptureFormat& format1,
const media::VideoCaptureFormat& format2) {
DCHECK(format1.frame_size.GetCheckedArea().IsValid());
DCHECK(format2.frame_size.GetCheckedArea().IsValid());
if (format1.frame_size.GetCheckedArea().ValueOrDefault(0) ==
format2.frame_size.GetCheckedArea().ValueOrDefault(0)) {
if (format1.frame_size.width() == format2.frame_size.width()) {
return format1.frame_rate > format2.frame_rate;
}
return format1.frame_size.width() > format2.frame_size.width();
}
return format1.frame_size.GetCheckedArea().ValueOrDefault(0) <
format2.frame_size.GetCheckedArea().ValueOrDefault(0);
}
bool IsCaptureFormatEqual(const media::VideoCaptureFormat& format1,
const media::VideoCaptureFormat& format2) {
return format1.frame_size == format2.frame_size &&
format1.frame_rate == format2.frame_rate &&
format1.pixel_format == format2.pixel_format;
}
// This function receives a list of capture formats, sets all of them to I420
// (while keeping Y16 as is), and then removes duplicates.
void ConsolidateCaptureFormats(media::VideoCaptureFormats* formats) {
if (formats->empty())
return;
// Mark all formats as I420, since this is what the renderer side will get
// anyhow: the actual pixel format is decided at the device level.
// Don't do this for Y16 format as it is handled separatelly.
for (auto& format : *formats) {
if (format.pixel_format != media::PIXEL_FORMAT_Y16)
format.pixel_format = media::PIXEL_FORMAT_I420;
}
std::sort(formats->begin(), formats->end(), IsCaptureFormatSmaller);
// Remove duplicates
auto last =
std::unique(formats->begin(), formats->end(), IsCaptureFormatEqual);
formats->erase(last, formats->end());
}
} // anonymous namespace
namespace media {
VideoCaptureSystemImpl::VideoCaptureSystemImpl(
std::unique_ptr<VideoCaptureDeviceFactory> factory)
: factory_(std::move(factory)) {
thread_checker_.DetachFromThread();
}
VideoCaptureSystemImpl::~VideoCaptureSystemImpl() = default;
void VideoCaptureSystemImpl::GetDeviceInfosAsync(
DeviceInfoCallback result_callback) {
DCHECK(thread_checker_.CalledOnValidThread());
device_enum_request_queue_.push_back(std::move(result_callback));
if (device_enum_request_queue_.size() == 1) {
ProcessDeviceInfoRequest();
}
}
void VideoCaptureSystemImpl::ProcessDeviceInfoRequest() {
auto request = device_enum_request_queue_.begin();
if (request == device_enum_request_queue_.end()) {
return;
}
std::unique_ptr<VideoCaptureDeviceDescriptors> descriptors =
std::make_unique<VideoCaptureDeviceDescriptors>();
factory_->GetDeviceDescriptors(descriptors.get());
#if defined(OS_WIN)
factory_->GetCameraLocationsAsync(
std::move(descriptors),
base::BindOnce(&VideoCaptureSystemImpl::DeviceInfosReady,
base::Unretained(this)));
#else
DeviceInfosReady(std::move(descriptors));
#endif
}
// Creates a VideoCaptureDevice object. Returns NULL if something goes wrong.
std::unique_ptr<VideoCaptureDevice> VideoCaptureSystemImpl::CreateDevice(
const std::string& device_id) {
DCHECK(thread_checker_.CalledOnValidThread());
const VideoCaptureDeviceInfo* device_info = LookupDeviceInfoFromId(device_id);
if (!device_info)
return nullptr;
return factory_->CreateDevice(device_info->descriptor);
}
const VideoCaptureDeviceInfo* VideoCaptureSystemImpl::LookupDeviceInfoFromId(
const std::string& device_id) {
DCHECK(thread_checker_.CalledOnValidThread());
auto iter =
std::find_if(devices_info_cache_.begin(), devices_info_cache_.end(),
[&device_id](const VideoCaptureDeviceInfo& device_info) {
return device_info.descriptor.device_id == device_id;
});
if (iter == devices_info_cache_.end())
return nullptr;
return &(*iter);
}
void VideoCaptureSystemImpl::DeviceInfosReady(
std::unique_ptr<VideoCaptureDeviceDescriptors> descriptors) {
DCHECK(thread_checker_.CalledOnValidThread());
DCHECK(!device_enum_request_queue_.empty());
// For devices for which we already have an entry in |devices_info_cache_|,
// we do not want to query the |factory_| for supported formats again. We
// simply copy them from |devices_info_cache_|.
std::vector<VideoCaptureDeviceInfo> new_devices_info_cache;
new_devices_info_cache.reserve(descriptors->size());
for (const auto& descriptor : *descriptors) {
if (auto* cached_info = LookupDeviceInfoFromId(descriptor.device_id)) {
new_devices_info_cache.push_back(*cached_info);
} else {
// Query for supported formats in order to create the entry.
VideoCaptureDeviceInfo device_info(descriptor);
factory_->GetSupportedFormats(descriptor, &device_info.supported_formats);
ConsolidateCaptureFormats(&device_info.supported_formats);
new_devices_info_cache.push_back(device_info);
}
}
devices_info_cache_.swap(new_devices_info_cache);
auto request_cb = std::move(device_enum_request_queue_.front());
device_enum_request_queue_.pop_front();
// If |request_cb| was the last callback in |device_enum_request_queue_|,
// |this| may be out of scope after running it. We need to be careful to
// not touch the state of |this| after running the callback in this case.
if (device_enum_request_queue_.empty()) {
std::move(request_cb).Run(devices_info_cache_);
return;
}
std::move(request_cb).Run(devices_info_cache_);
ProcessDeviceInfoRequest();
}
#if defined(OS_CHROMEOS)
void VideoCaptureSystemImpl::BindCrosImageCaptureRequest(
cros::mojom::CrosImageCaptureRequest request) {
CHECK(factory_);
if (media::ShouldUseCrosCameraService()) {
static_cast<VideoCaptureDeviceFactoryChromeOS*>(factory_.get())
->BindCrosImageCaptureRequest(std::move(request));
}
}
#endif // defined(OS_CHROMEOS)
} // namespace media