blob: 2006dd3d542d64ab5a8d858ff45761b7c60f57f9 [file] [log] [blame]
/*
* Copyright (C) 2008 Apple Inc. All Rights Reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
* 1. Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* 2. Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution.
*
* THIS SOFTWARE IS PROVIDED BY APPLE COMPUTER, INC. ``AS IS'' AND ANY
* EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
* PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE COMPUTER, INC. OR
* CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
* EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
* PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
* PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY
* OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*
*/
#include "third_party/blink/renderer/core/messaging/message_port.h"
#include <memory>
#include "mojo/public/cpp/base/big_buffer_mojom_traits.h"
#include "third_party/blink/public/platform/task_type.h"
#include "third_party/blink/public/platform/web_string.h"
#include "third_party/blink/renderer/bindings/core/v8/serialization/post_message_helper.h"
#include "third_party/blink/renderer/core/events/message_event.h"
#include "third_party/blink/renderer/core/execution_context/execution_context.h"
#include "third_party/blink/renderer/core/frame/local_dom_window.h"
#include "third_party/blink/renderer/core/frame/user_activation.h"
#include "third_party/blink/renderer/core/inspector/thread_debugger.h"
#include "third_party/blink/renderer/core/messaging/blink_transferable_message_mojom_traits.h"
#include "third_party/blink/renderer/core/messaging/post_message_options.h"
#include "third_party/blink/renderer/core/workers/worker_global_scope.h"
#include "third_party/blink/renderer/platform/bindings/exception_state.h"
#include "third_party/blink/renderer/platform/bindings/script_state.h"
#include "third_party/blink/renderer/platform/instrumentation/use_counter.h"
#include "third_party/blink/renderer/platform/wtf/cross_thread_functional.h"
#include "third_party/blink/renderer/platform/wtf/text/atomic_string.h"
namespace blink {
// TODO(altimin): Remove these after per-task mojo dispatching.
// The maximum number of MessageEvents to dispatch from one task.
constexpr int kMaximumMessagesPerTask = 200;
// The threshold to stop processing new tasks.
constexpr base::TimeDelta kYieldThreshold =
base::TimeDelta::FromMilliseconds(50);
MessagePort::MessagePort(ExecutionContext& execution_context)
: ContextLifecycleObserver(&execution_context),
task_runner_(execution_context.GetTaskRunner(TaskType::kPostedMessage)) {}
MessagePort::~MessagePort() {
DCHECK(!started_ || !IsEntangled());
}
void MessagePort::postMessage(ScriptState* script_state,
const ScriptValue& message,
HeapVector<ScriptValue>& transfer,
ExceptionState& exception_state) {
PostMessageOptions* options = PostMessageOptions::Create();
if (!transfer.IsEmpty())
options->setTransfer(transfer);
postMessage(script_state, message, options, exception_state);
}
void MessagePort::postMessage(ScriptState* script_state,
const ScriptValue& message,
const PostMessageOptions* options,
ExceptionState& exception_state) {
if (!IsEntangled())
return;
DCHECK(GetExecutionContext());
DCHECK(!IsNeutered());
BlinkTransferableMessage msg;
Transferables transferables;
msg.message = PostMessageHelper::SerializeMessageByMove(
script_state->GetIsolate(), message, options, transferables,
exception_state);
if (exception_state.HadException())
return;
DCHECK(msg.message);
// Make sure we aren't connected to any of the passed-in ports.
for (unsigned i = 0; i < transferables.message_ports.size(); ++i) {
if (transferables.message_ports[i] == this) {
exception_state.ThrowDOMException(
DOMExceptionCode::kDataCloneError,
"Port at index " + String::Number(i) + " contains the source port.");
return;
}
}
msg.ports = MessagePort::DisentanglePorts(
ExecutionContext::From(script_state), transferables.message_ports,
exception_state);
if (exception_state.HadException())
return;
msg.user_activation = PostMessageHelper::CreateUserActivationSnapshot(
GetExecutionContext(), options);
msg.sender_origin =
GetExecutionContext()->GetSecurityOrigin()->IsolatedCopy();
ThreadDebugger* debugger = ThreadDebugger::From(script_state->GetIsolate());
if (debugger)
msg.sender_stack_trace_id = debugger->StoreCurrentStackTrace("postMessage");
if (msg.message->IsLockedToAgentCluster()) {
msg.locked_agent_cluster_id = GetExecutionContext()->GetAgentClusterID();
} else {
msg.locked_agent_cluster_id = base::nullopt;
}
mojo::Message mojo_message =
mojom::blink::TransferableMessage::WrapAsMessage(std::move(msg));
connector_->Accept(&mojo_message);
}
MessagePortChannel MessagePort::Disentangle() {
DCHECK(!IsNeutered());
auto result = MessagePortChannel(connector_->PassMessagePipe());
connector_ = nullptr;
return result;
}
void MessagePort::start() {
// Do nothing if we've been cloned or closed.
if (!IsEntangled())
return;
DCHECK(GetExecutionContext());
if (started_)
return;
started_ = true;
connector_->ResumeIncomingMethodCallProcessing();
}
void MessagePort::close() {
if (closed_)
return;
// A closed port should not be neutered, so rather than merely disconnecting
// from the mojo message pipe, also entangle with a new dangling message pipe.
if (!IsNeutered()) {
connector_ = nullptr;
Entangle(mojo::MessagePipe().handle0);
}
closed_ = true;
}
void MessagePort::Entangle(mojo::ScopedMessagePipeHandle handle) {
// Only invoked to set our initial entanglement.
DCHECK(handle.is_valid());
DCHECK(!connector_);
DCHECK(GetExecutionContext());
connector_ = std::make_unique<mojo::Connector>(
std::move(handle), mojo::Connector::SINGLE_THREADED_SEND, task_runner_);
connector_->PauseIncomingMethodCallProcessing();
connector_->set_incoming_receiver(this);
connector_->set_connection_error_handler(
WTF::Bind(&MessagePort::close, WrapWeakPersistent(this)));
}
void MessagePort::Entangle(MessagePortChannel channel) {
Entangle(channel.ReleaseHandle());
}
const AtomicString& MessagePort::InterfaceName() const {
return event_target_names::kMessagePort;
}
bool MessagePort::HasPendingActivity() const {
// The spec says that entangled message ports should always be treated as if
// they have a strong reference.
// We'll also stipulate that the queue needs to be open (if the app drops its
// reference to the port before start()-ing it, then it's not really entangled
// as it's unreachable).
return started_ && IsEntangled();
}
Vector<MessagePortChannel> MessagePort::DisentanglePorts(
ExecutionContext* context,
const MessagePortArray& ports,
ExceptionState& exception_state) {
if (!ports.size())
return Vector<MessagePortChannel>();
HeapHashSet<Member<MessagePort>> visited;
bool has_closed_ports = false;
// Walk the incoming array - if there are any duplicate ports, or null ports
// or cloned ports, throw an error (per section 8.3.3 of the HTML5 spec).
for (unsigned i = 0; i < ports.size(); ++i) {
MessagePort* port = ports[i];
if (!port || port->IsNeutered() || visited.Contains(port)) {
String type;
if (!port)
type = "null";
else if (port->IsNeutered())
type = "already neutered";
else
type = "a duplicate";
exception_state.ThrowDOMException(
DOMExceptionCode::kDataCloneError,
"Port at index " + String::Number(i) + " is " + type + ".");
return Vector<MessagePortChannel>();
}
if (port->closed_)
has_closed_ports = true;
visited.insert(port);
}
UseCounter::Count(context, WebFeature::kMessagePortsTransferred);
if (has_closed_ports)
UseCounter::Count(context, WebFeature::kMessagePortTransferClosedPort);
// Passed-in ports passed validity checks, so we can disentangle them.
Vector<MessagePortChannel> channels;
channels.ReserveInitialCapacity(ports.size());
for (unsigned i = 0; i < ports.size(); ++i)
channels.push_back(ports[i]->Disentangle());
return channels;
}
MessagePortArray* MessagePort::EntanglePorts(
ExecutionContext& context,
Vector<MessagePortChannel> channels) {
return EntanglePorts(context,
WebVector<MessagePortChannel>(std::move(channels)));
}
MessagePortArray* MessagePort::EntanglePorts(
ExecutionContext& context,
WebVector<MessagePortChannel> channels) {
// https://html.spec.whatwg.org/C/#message-ports
// |ports| should be an empty array, not null even when there is no ports.
wtf_size_t count = SafeCast<wtf_size_t>(channels.size());
MessagePortArray* port_array = MakeGarbageCollected<MessagePortArray>(count);
for (wtf_size_t i = 0; i < count; ++i) {
auto* port = MakeGarbageCollected<MessagePort>(context);
port->Entangle(std::move(channels[i]));
(*port_array)[i] = port;
}
return port_array;
}
::MojoHandle MessagePort::EntangledHandleForTesting() const {
return connector_->handle().value();
}
void MessagePort::Trace(blink::Visitor* visitor) {
ContextLifecycleObserver::Trace(visitor);
EventTargetWithInlineData::Trace(visitor);
}
bool MessagePort::Accept(mojo::Message* mojo_message) {
TRACE_EVENT0("blink", "MessagePort::Accept");
// Connector repeatedly calls Accept as long as any messages are available. To
// avoid completely starving the event loop and give some time for other tasks
// the connector is temporarily paused after |kMaximumMessagesPerTask| have
// been received without other tasks having had a chance to run (in particular
// the ResetMessageCount task posted here).
// TODO(altimin): Remove this after per-task mojo dispatching lands[1].
// [1] https://chromium-review.googlesource.com/c/chromium/src/+/1145692
if (messages_in_current_task_ == 0) {
task_runner_->PostTask(FROM_HERE, WTF::Bind(&MessagePort::ResetMessageCount,
WrapWeakPersistent(this)));
}
if (ShouldYieldAfterNewMessage()) {
connector_->PauseIncomingMethodCallProcessing();
}
BlinkTransferableMessage message;
if (!mojom::blink::TransferableMessage::DeserializeFromMessage(
std::move(*mojo_message), &message)) {
return false;
}
// WorkerGlobalScope::close() in Worker onmessage handler should prevent
// the next message from dispatching.
if (auto* scope = DynamicTo<WorkerGlobalScope>(GetExecutionContext())) {
if (scope->IsClosing())
return true;
}
Event* evt = CreateMessageEvent(message);
v8::Isolate* isolate = GetExecutionContext()->GetIsolate();
ThreadDebugger* debugger = ThreadDebugger::From(isolate);
if (debugger)
debugger->ExternalAsyncTaskStarted(message.sender_stack_trace_id);
DispatchEvent(*evt);
if (debugger)
debugger->ExternalAsyncTaskFinished(message.sender_stack_trace_id);
return true;
}
Event* MessagePort::CreateMessageEvent(BlinkTransferableMessage& message) {
ExecutionContext* context = GetExecutionContext();
// Dispatch a messageerror event when the target is a remote origin that is
// not allowed to access the message's data.
if (message.message->IsOriginCheckRequired()) {
const SecurityOrigin* target_origin = context->GetSecurityOrigin();
if (!message.sender_origin ||
!message.sender_origin->IsSameSchemeHostPort(target_origin)) {
return MessageEvent::CreateError();
}
}
if (message.locked_agent_cluster_id) {
if (!context->IsSameAgentCluster(*message.locked_agent_cluster_id)) {
UseCounter::Count(
context,
WebFeature::kMessageEventSharedArrayBufferDifferentAgentCluster);
return MessageEvent::CreateError();
}
const SecurityOrigin* target_origin = context->GetSecurityOrigin();
if (!message.sender_origin ||
!message.sender_origin->IsSameSchemeHostPort(target_origin)) {
UseCounter::Count(
context, WebFeature::kMessageEventSharedArrayBufferSameAgentCluster);
} else {
UseCounter::Count(context,
WebFeature::kMessageEventSharedArrayBufferSameOrigin);
}
}
MessagePortArray* ports = MessagePort::EntanglePorts(
*GetExecutionContext(), std::move(message.ports));
UserActivation* user_activation = nullptr;
if (message.user_activation) {
user_activation = MakeGarbageCollected<UserActivation>(
message.user_activation->has_been_active,
message.user_activation->was_active);
}
return MessageEvent::Create(ports, std::move(message.message),
user_activation);
}
void MessagePort::ResetMessageCount() {
DCHECK_GT(messages_in_current_task_, 0);
messages_in_current_task_ = 0;
task_start_time_ = base::nullopt;
// No-op if not paused already.
if (connector_)
connector_->ResumeIncomingMethodCallProcessing();
}
bool MessagePort::ShouldYieldAfterNewMessage() {
++messages_in_current_task_;
if (messages_in_current_task_ > kMaximumMessagesPerTask)
return true;
base::TimeTicks now = base::TimeTicks::Now();
if (!task_start_time_)
task_start_time_ = now;
return now - task_start_time_.value() > kYieldThreshold;
}
} // namespace blink