blob: cd413613cd2d6de1c901040c65fe67a21abf9cf0 [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 "services/video_capture/virtual_device_enabled_device_factory.h"
#include "base/bind.h"
#include "base/logging.h"
#include "media/capture/video/video_capture_device_info.h"
#include "services/video_capture/device_factory.h"
#include "services/video_capture/shared_memory_virtual_device_mojo_adapter.h"
#include "services/video_capture/texture_virtual_device_mojo_adapter.h"
namespace video_capture {
class VirtualDeviceEnabledDeviceFactory::VirtualDeviceEntry {
public:
VirtualDeviceEntry(
const media::VideoCaptureDeviceInfo& device_info,
std::unique_ptr<SharedMemoryVirtualDeviceMojoAdapter> device,
std::unique_ptr<mojo::Binding<mojom::SharedMemoryVirtualDevice>>
producer_binding)
: device_info_(device_info),
device_type_(DeviceType::kSharedMemory),
shared_memory_device_(std::move(device)),
shared_memory_producer_binding_(std::move(producer_binding)) {}
VirtualDeviceEntry(const media::VideoCaptureDeviceInfo& device_info,
std::unique_ptr<TextureVirtualDeviceMojoAdapter> device,
std::unique_ptr<mojo::Binding<mojom::TextureVirtualDevice>>
producer_binding)
: device_info_(device_info),
device_type_(DeviceType::kTexture),
texture_device_(std::move(device)),
texture_producer_binding_(std::move(producer_binding)) {}
VirtualDeviceEntry(VirtualDeviceEntry&& other) = default;
VirtualDeviceEntry& operator=(VirtualDeviceEntry&& other) = default;
bool HasConsumerBinding() { return consumer_binding_ != nullptr; }
void EstablishConsumerBinding(mojom::DeviceRequest device_request,
base::OnceClosure connection_error_handler) {
switch (device_type_) {
case DeviceType::kSharedMemory:
consumer_binding_ = std::make_unique<mojo::Binding<mojom::Device>>(
shared_memory_device_.get(), std::move(device_request));
break;
case DeviceType::kTexture:
consumer_binding_ = std::make_unique<mojo::Binding<mojom::Device>>(
texture_device_.get(), std::move(device_request));
break;
}
consumer_binding_->set_connection_error_handler(
std::move(connection_error_handler));
}
void ResetConsumerBinding() { consumer_binding_.reset(); }
void StopDevice() {
if (shared_memory_device_)
shared_memory_device_->Stop();
else
texture_device_->Stop();
}
media::VideoCaptureDeviceInfo device_info() const { return device_info_; }
private:
enum class DeviceType { kSharedMemory, kTexture };
media::VideoCaptureDeviceInfo device_info_;
DeviceType device_type_;
// Only valid for |device_type_ == kSharedMemory|
std::unique_ptr<SharedMemoryVirtualDeviceMojoAdapter> shared_memory_device_;
std::unique_ptr<mojo::Binding<mojom::SharedMemoryVirtualDevice>>
shared_memory_producer_binding_;
// Only valid for |device_type_ == kTexture|
std::unique_ptr<TextureVirtualDeviceMojoAdapter> texture_device_;
std::unique_ptr<mojo::Binding<mojom::TextureVirtualDevice>>
texture_producer_binding_;
std::unique_ptr<mojo::Binding<mojom::Device>> consumer_binding_;
};
VirtualDeviceEnabledDeviceFactory::VirtualDeviceEnabledDeviceFactory(
std::unique_ptr<DeviceFactory> device_factory)
: device_factory_(std::move(device_factory)), weak_factory_(this) {}
VirtualDeviceEnabledDeviceFactory::~VirtualDeviceEnabledDeviceFactory() =
default;
void VirtualDeviceEnabledDeviceFactory::GetDeviceInfos(
GetDeviceInfosCallback callback) {
device_factory_->GetDeviceInfos(
base::BindOnce(&VirtualDeviceEnabledDeviceFactory::OnGetDeviceInfos,
weak_factory_.GetWeakPtr(), base::Passed(&callback)));
}
void VirtualDeviceEnabledDeviceFactory::CreateDevice(
const std::string& device_id,
mojom::DeviceRequest device_request,
CreateDeviceCallback callback) {
auto virtual_device_iter = virtual_devices_by_id_.find(device_id);
if (virtual_device_iter != virtual_devices_by_id_.end()) {
VirtualDeviceEntry& device_entry = virtual_device_iter->second;
if (device_entry.HasConsumerBinding()) {
// The requested virtual device is already used by another client.
// Revoke the access for the current client, then bind to the new request.
device_entry.ResetConsumerBinding();
device_entry.StopDevice();
}
device_entry.EstablishConsumerBinding(
std::move(device_request),
base::Bind(&VirtualDeviceEnabledDeviceFactory::
OnVirtualDeviceConsumerConnectionErrorOrClose,
base::Unretained(this), device_id));
std::move(callback).Run(mojom::DeviceAccessResultCode::SUCCESS);
return;
}
device_factory_->CreateDevice(device_id, std::move(device_request),
std::move(callback));
}
void VirtualDeviceEnabledDeviceFactory::AddSharedMemoryVirtualDevice(
const media::VideoCaptureDeviceInfo& device_info,
mojom::ProducerPtr producer,
bool send_buffer_handles_to_producer_as_raw_file_descriptors,
mojom::SharedMemoryVirtualDeviceRequest virtual_device_request) {
auto device_id = device_info.descriptor.device_id;
auto virtual_device_iter = virtual_devices_by_id_.find(device_id);
if (virtual_device_iter != virtual_devices_by_id_.end()) {
// Revoke the access for the current producer and consumer by
// removing it from the list.
virtual_devices_by_id_.erase(virtual_device_iter);
}
producer.set_connection_error_handler(
base::Bind(&VirtualDeviceEnabledDeviceFactory::
OnVirtualDeviceProducerConnectionErrorOrClose,
base::Unretained(this), device_id));
auto device = std::make_unique<SharedMemoryVirtualDeviceMojoAdapter>(
std::move(producer),
send_buffer_handles_to_producer_as_raw_file_descriptors);
auto producer_binding =
std::make_unique<mojo::Binding<mojom::SharedMemoryVirtualDevice>>(
device.get(), std::move(virtual_device_request));
producer_binding->set_connection_error_handler(
base::Bind(&VirtualDeviceEnabledDeviceFactory::
OnVirtualDeviceProducerConnectionErrorOrClose,
base::Unretained(this), device_id));
VirtualDeviceEntry device_entry(device_info, std::move(device),
std::move(producer_binding));
virtual_devices_by_id_.insert(
std::make_pair(device_id, std::move(device_entry)));
EmitDevicesChangedEvent();
}
void VirtualDeviceEnabledDeviceFactory::AddTextureVirtualDevice(
const media::VideoCaptureDeviceInfo& device_info,
mojom::TextureVirtualDeviceRequest virtual_device_request) {
auto device_id = device_info.descriptor.device_id;
auto virtual_device_iter = virtual_devices_by_id_.find(device_id);
if (virtual_device_iter != virtual_devices_by_id_.end()) {
// Revoke the access for the current producer and consumer by
// removing it from the list.
virtual_devices_by_id_.erase(virtual_device_iter);
}
auto device = std::make_unique<TextureVirtualDeviceMojoAdapter>();
auto producer_binding =
std::make_unique<mojo::Binding<mojom::TextureVirtualDevice>>(
device.get(), std::move(virtual_device_request));
producer_binding->set_connection_error_handler(
base::BindOnce(&VirtualDeviceEnabledDeviceFactory::
OnVirtualDeviceProducerConnectionErrorOrClose,
base::Unretained(this), device_id));
VirtualDeviceEntry device_entry(device_info, std::move(device),
std::move(producer_binding));
virtual_devices_by_id_.insert(
std::make_pair(device_id, std::move(device_entry)));
EmitDevicesChangedEvent();
}
void VirtualDeviceEnabledDeviceFactory::RegisterVirtualDevicesChangedObserver(
mojom::DevicesChangedObserverPtr observer,
bool raise_event_if_virtual_devices_already_present) {
observer.set_connection_error_handler(base::BindOnce(
&VirtualDeviceEnabledDeviceFactory::OnDevicesChangedObserverDisconnected,
weak_factory_.GetWeakPtr(), observer.get()));
if (!virtual_devices_by_id_.empty() &&
raise_event_if_virtual_devices_already_present) {
observer->OnDevicesChanged();
}
devices_changed_observers_.push_back(std::move(observer));
}
void VirtualDeviceEnabledDeviceFactory::OnGetDeviceInfos(
GetDeviceInfosCallback callback,
const std::vector<media::VideoCaptureDeviceInfo>& device_infos) {
std::vector<media::VideoCaptureDeviceInfo> all_device_infos;
for (const auto& device_entry : virtual_devices_by_id_) {
all_device_infos.push_back(device_entry.second.device_info());
}
all_device_infos.insert(std::end(all_device_infos), std::begin(device_infos),
std::end(device_infos));
std::move(callback).Run(all_device_infos);
}
void VirtualDeviceEnabledDeviceFactory::
OnVirtualDeviceProducerConnectionErrorOrClose(
const std::string& device_id) {
virtual_devices_by_id_.at(device_id).StopDevice();
virtual_devices_by_id_.erase(device_id);
EmitDevicesChangedEvent();
}
void VirtualDeviceEnabledDeviceFactory::
OnVirtualDeviceConsumerConnectionErrorOrClose(
const std::string& device_id) {
virtual_devices_by_id_.at(device_id).StopDevice();
}
void VirtualDeviceEnabledDeviceFactory::EmitDevicesChangedEvent() {
for (auto& observer : devices_changed_observers_)
observer->OnDevicesChanged();
}
void VirtualDeviceEnabledDeviceFactory::OnDevicesChangedObserverDisconnected(
mojom::DevicesChangedObserverPtr::Proxy* observer) {
auto iter = std::find_if(
devices_changed_observers_.begin(), devices_changed_observers_.end(),
[observer](const mojom::DevicesChangedObserverPtr& entry) {
return entry.get() == observer;
});
if (iter == devices_changed_observers_.end()) {
DCHECK(false);
return;
}
devices_changed_observers_.erase(iter);
}
#if defined(OS_CHROMEOS)
void VirtualDeviceEnabledDeviceFactory::BindCrosImageCaptureRequest(
cros::mojom::CrosImageCaptureRequest request) {
CHECK(device_factory_);
device_factory_->BindCrosImageCaptureRequest(std::move(request));
}
#endif // defined(OS_CHROMEOS)
} // namespace video_capture