blob: baea7b1de87e0de2fda67ee18f598f6b9a5e7ef8 [file] [log] [blame]
// Copyright 2015 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 "third_party/blink/renderer/modules/mediastream/media_devices.h"
#include "services/service_manager/public/cpp/interface_provider.h"
#include "third_party/blink/public/platform/platform.h"
#include "third_party/blink/public/platform/task_type.h"
#include "third_party/blink/renderer/bindings/core/v8/script_promise.h"
#include "third_party/blink/renderer/bindings/core/v8/script_promise_resolver.h"
#include "third_party/blink/renderer/core/dom/document.h"
#include "third_party/blink/renderer/core/dom/dom_exception.h"
#include "third_party/blink/renderer/core/dom/events/event.h"
#include "third_party/blink/renderer/core/execution_context/execution_context.h"
#include "third_party/blink/renderer/core/frame/local_frame.h"
#include "third_party/blink/renderer/modules/mediastream/input_device_info.h"
#include "third_party/blink/renderer/modules/mediastream/media_error_state.h"
#include "third_party/blink/renderer/modules/mediastream/media_stream.h"
#include "third_party/blink/renderer/modules/mediastream/media_stream_constraints.h"
#include "third_party/blink/renderer/modules/mediastream/media_track_supported_constraints.h"
#include "third_party/blink/renderer/modules/mediastream/navigator_media_stream.h"
#include "third_party/blink/renderer/modules/mediastream/user_media_controller.h"
#include "third_party/blink/renderer/platform/bindings/script_state.h"
#include "third_party/blink/renderer/platform/wtf/functional.h"
using blink::mojom::blink::MediaDeviceType;
namespace blink {
namespace {
class PromiseResolverCallbacks final : public UserMediaRequest::Callbacks {
public:
static PromiseResolverCallbacks* Create(ScriptPromiseResolver* resolver) {
return new PromiseResolverCallbacks(resolver);
}
~PromiseResolverCallbacks() override = default;
void OnSuccess(ScriptWrappable* callback_this_value,
MediaStream* stream) override {
resolver_->Resolve(stream);
}
void OnError(ScriptWrappable* callback_this_value,
DOMExceptionOrOverconstrainedError error) override {
resolver_->Reject(error);
}
void Trace(blink::Visitor* visitor) override {
visitor->Trace(resolver_);
UserMediaRequest::Callbacks::Trace(visitor);
}
private:
explicit PromiseResolverCallbacks(ScriptPromiseResolver* resolver)
: resolver_(resolver) {}
Member<ScriptPromiseResolver> resolver_;
};
} // namespace
MediaDevices* MediaDevices::Create(ExecutionContext* context) {
MediaDevices* media_devices = new MediaDevices(context);
media_devices->PauseIfNeeded();
return media_devices;
}
MediaDevices::MediaDevices(ExecutionContext* context)
: PausableObject(context),
stopped_(false),
dispatch_scheduled_event_runner_(
context ? AsyncMethodRunner<MediaDevices>::Create(
this,
&MediaDevices::DispatchScheduledEvent,
context->GetTaskRunner(TaskType::kMediaElementEvent))
: nullptr),
binding_(this) {}
MediaDevices::~MediaDevices() = default;
ScriptPromise MediaDevices::enumerateDevices(ScriptState* script_state) {
Platform::Current()->UpdateWebRTCAPICount(WebRTCAPIName::kEnumerateDevices);
LocalFrame* frame =
To<Document>(ExecutionContext::From(script_state))->GetFrame();
if (!frame) {
return ScriptPromise::RejectWithDOMException(
script_state, DOMException::Create(DOMExceptionCode::kNotSupportedError,
"Current frame is detached."));
}
ScriptPromiseResolver* resolver = ScriptPromiseResolver::Create(script_state);
ScriptPromise promise = resolver->Promise();
requests_.insert(resolver);
GetDispatcherHost(frame)->EnumerateDevices(
true /* audio input */, true /* video input */, true /* audio output */,
true /* request_video_input_capabilities */,
WTF::Bind(&MediaDevices::DevicesEnumerated, WrapPersistent(this),
WrapPersistent(resolver)));
return promise;
}
MediaTrackSupportedConstraints* MediaDevices::getSupportedConstraints() const {
return MediaTrackSupportedConstraints::Create();
}
ScriptPromise MediaDevices::getUserMedia(ScriptState* script_state,
const MediaStreamConstraints* options,
ExceptionState& exception_state) {
return SendUserMediaRequest(script_state,
WebUserMediaRequest::MediaType::kUserMedia,
options, exception_state);
}
ScriptPromise MediaDevices::SendUserMediaRequest(
ScriptState* script_state,
WebUserMediaRequest::MediaType media_type,
const MediaStreamConstraints* options,
ExceptionState& exception_state) {
ScriptPromiseResolver* resolver = ScriptPromiseResolver::Create(script_state);
PromiseResolverCallbacks* callbacks =
PromiseResolverCallbacks::Create(resolver);
Document* document = To<Document>(ExecutionContext::From(script_state));
UserMediaController* user_media =
UserMediaController::From(document->GetFrame());
if (!user_media)
return ScriptPromise::RejectWithDOMException(
script_state,
DOMException::Create(DOMExceptionCode::kNotSupportedError,
"No media device controller available; is this a "
"detached window?"));
MediaErrorState error_state;
UserMediaRequest* request = UserMediaRequest::Create(
document, user_media, media_type, options, callbacks, error_state);
if (!request) {
DCHECK(error_state.HadException());
if (error_state.CanGenerateException()) {
error_state.RaiseException(exception_state);
return ScriptPromise();
}
ScriptPromise rejected_promise = resolver->Promise();
resolver->Reject(error_state.CreateError());
return rejected_promise;
}
String error_message;
if (!request->IsSecureContextUse(error_message)) {
return ScriptPromise::RejectWithDOMException(
script_state, DOMException::Create(DOMExceptionCode::kNotSupportedError,
error_message));
}
auto promise = resolver->Promise();
request->Start();
return promise;
}
ScriptPromise MediaDevices::getDisplayMedia(
ScriptState* script_state,
const MediaStreamConstraints* options,
ExceptionState& exception_state) {
return SendUserMediaRequest(script_state,
WebUserMediaRequest::MediaType::kDisplayMedia,
options, exception_state);
}
const AtomicString& MediaDevices::InterfaceName() const {
return event_target_names::kMediaDevices;
}
ExecutionContext* MediaDevices::GetExecutionContext() const {
return PausableObject::GetExecutionContext();
}
void MediaDevices::RemoveAllEventListeners() {
EventTargetWithInlineData::RemoveAllEventListeners();
DCHECK(!HasEventListeners());
StopObserving();
}
void MediaDevices::AddedEventListener(
const AtomicString& event_type,
RegisteredEventListener& registered_listener) {
EventTargetWithInlineData::AddedEventListener(event_type,
registered_listener);
StartObserving();
}
void MediaDevices::RemovedEventListener(
const AtomicString& event_type,
const RegisteredEventListener& registered_listener) {
EventTargetWithInlineData::RemovedEventListener(event_type,
registered_listener);
if (!HasEventListeners())
StopObserving();
}
bool MediaDevices::HasPendingActivity() const {
DCHECK(stopped_ || binding_.is_bound() == HasEventListeners());
return binding_.is_bound();
}
void MediaDevices::ContextDestroyed(ExecutionContext*) {
if (stopped_)
return;
stopped_ = true;
StopObserving();
requests_.clear();
dispatcher_host_.reset();
}
void MediaDevices::Pause() {
DCHECK(dispatch_scheduled_event_runner_);
dispatch_scheduled_event_runner_->Pause();
}
void MediaDevices::Unpause() {
DCHECK(dispatch_scheduled_event_runner_);
dispatch_scheduled_event_runner_->Unpause();
}
void MediaDevices::OnDevicesChanged(
MediaDeviceType type,
Vector<mojom::blink::MediaDeviceInfoPtr> device_infos) {
Document* document = To<Document>(GetExecutionContext());
DCHECK(document);
if (RuntimeEnabledFeatures::OnDeviceChangeEnabled())
ScheduleDispatchEvent(Event::Create(event_type_names::kDevicechange));
if (device_change_test_callback_)
std::move(device_change_test_callback_).Run();
}
void MediaDevices::ScheduleDispatchEvent(Event* event) {
scheduled_events_.push_back(event);
DCHECK(dispatch_scheduled_event_runner_);
dispatch_scheduled_event_runner_->RunAsync();
}
void MediaDevices::DispatchScheduledEvent() {
HeapVector<Member<Event>> events;
events.swap(scheduled_events_);
for (const auto& event : events)
DispatchEvent(*event);
}
void MediaDevices::StartObserving() {
if (binding_.is_bound() || stopped_)
return;
Document* document = To<Document>(GetExecutionContext());
if (!document || !document->GetFrame())
return;
mojom::blink::MediaDevicesListenerPtr listener;
binding_.Bind(mojo::MakeRequest(&listener));
GetDispatcherHost(document->GetFrame())
->AddMediaDevicesListener(true /* audio input */, true /* video input */,
true /* audio output */, std::move(listener));
}
void MediaDevices::StopObserving() {
if (!binding_.is_bound())
return;
binding_.Close();
}
void MediaDevices::Dispose() {
StopObserving();
}
void MediaDevices::DevicesEnumerated(
ScriptPromiseResolver* resolver,
Vector<Vector<mojom::blink::MediaDeviceInfoPtr>> enumeration,
Vector<mojom::blink::VideoInputDeviceCapabilitiesPtr>
video_input_capabilities) {
if (!requests_.Contains(resolver))
return;
requests_.erase(resolver);
if (!resolver->GetExecutionContext() ||
resolver->GetExecutionContext()->IsContextDestroyed()) {
return;
}
DCHECK_EQ(static_cast<wtf_size_t>(MediaDeviceType::NUM_MEDIA_DEVICE_TYPES),
enumeration.size());
if (!video_input_capabilities.IsEmpty()) {
DCHECK_EQ(
enumeration[static_cast<wtf_size_t>(MediaDeviceType::MEDIA_VIDEO_INPUT)]
.size(),
video_input_capabilities.size());
}
MediaDeviceInfoVector media_devices;
for (wtf_size_t i = 0;
i < static_cast<wtf_size_t>(MediaDeviceType::NUM_MEDIA_DEVICE_TYPES);
++i) {
for (wtf_size_t j = 0; j < enumeration[i].size(); ++j) {
MediaDeviceType device_type = static_cast<MediaDeviceType>(i);
mojom::blink::MediaDeviceInfoPtr device_info =
std::move(enumeration[i][j]);
if (device_type == MediaDeviceType::MEDIA_AUDIO_INPUT ||
device_type == MediaDeviceType::MEDIA_VIDEO_INPUT) {
InputDeviceInfo* input_device_info =
InputDeviceInfo::Create(device_info->device_id, device_info->label,
device_info->group_id, device_type);
if (device_type == MediaDeviceType::MEDIA_VIDEO_INPUT &&
!video_input_capabilities.IsEmpty()) {
input_device_info->SetVideoInputCapabilities(
std::move(video_input_capabilities[j]));
}
media_devices.push_back(input_device_info);
} else {
media_devices.push_back(
MediaDeviceInfo::Create(device_info->device_id, device_info->label,
device_info->group_id, device_type));
}
}
}
if (enumerate_devices_test_callback_)
std::move(enumerate_devices_test_callback_).Run(media_devices);
resolver->Resolve(media_devices);
}
void MediaDevices::OnDispatcherHostConnectionError() {
for (ScriptPromiseResolver* resolver : requests_) {
resolver->Reject(DOMException::Create(DOMExceptionCode::kAbortError,
"enumerateDevices() failed."));
}
requests_.clear();
dispatcher_host_.reset();
if (connection_error_test_callback_)
std::move(connection_error_test_callback_).Run();
}
const mojom::blink::MediaDevicesDispatcherHostPtr&
MediaDevices::GetDispatcherHost(LocalFrame* frame) {
if (!dispatcher_host_) {
frame->GetInterfaceProvider().GetInterface(
mojo::MakeRequest(&dispatcher_host_));
dispatcher_host_.set_connection_error_handler(
WTF::Bind(&MediaDevices::OnDispatcherHostConnectionError,
WrapWeakPersistent(this)));
}
return dispatcher_host_;
}
void MediaDevices::SetDispatcherHostForTesting(
mojom::blink::MediaDevicesDispatcherHostPtr dispatcher_host) {
dispatcher_host_ = std::move(dispatcher_host);
dispatcher_host_.set_connection_error_handler(
WTF::Bind(&MediaDevices::OnDispatcherHostConnectionError,
WrapWeakPersistent(this)));
}
void MediaDevices::Trace(blink::Visitor* visitor) {
visitor->Trace(dispatch_scheduled_event_runner_);
visitor->Trace(scheduled_events_);
visitor->Trace(requests_);
EventTargetWithInlineData::Trace(visitor);
PausableObject::Trace(visitor);
}
} // namespace blink