blob: 7134c119865e9108f08c3572cca1582254e7dde1 [file] [log] [blame]
// Copyright 2017 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "mojo/public/cpp/bindings/associated_receiver.h"
#include <stdint.h>
#include <memory>
#include <string_view>
#include <utility>
#include "base/check.h"
#include "base/containers/span.h"
#include "base/functional/bind.h"
#include "base/task/sequenced_task_runner.h"
#include "mojo/public/cpp/bindings/lib/multiplex_router.h"
#include "mojo/public/cpp/bindings/lib/task_runner_helper.h"
#include "mojo/public/cpp/system/message_pipe.h"
namespace mojo {
namespace internal {
AssociatedReceiverBase::AssociatedReceiverBase() = default;
void AssociatedReceiverBase::SetFilter(std::unique_ptr<MessageFilter> filter) {
DCHECK(endpoint_client_);
endpoint_client_->SetFilter(std::move(filter));
}
void AssociatedReceiverBase::reset() {
endpoint_client_.reset();
}
void AssociatedReceiverBase::ResetWithReason(uint32_t custom_reason,
std::string_view description) {
// TODO(dcheng): This should unconditionally assert that there is an endpoint
// client.
if (endpoint_client_)
endpoint_client_->CloseWithReason(custom_reason, description);
reset();
}
void AssociatedReceiverBase::set_disconnect_handler(
base::OnceClosure error_handler) {
DCHECK(is_bound());
endpoint_client_->set_connection_error_handler(std::move(error_handler));
}
void AssociatedReceiverBase::set_disconnect_with_reason_handler(
ConnectionErrorWithReasonCallback error_handler) {
DCHECK(is_bound());
endpoint_client_->set_connection_error_with_reason_handler(
std::move(error_handler));
}
void AssociatedReceiverBase::reset_on_disconnect() {
DCHECK(is_bound());
set_disconnect_handler(
base::BindOnce(&AssociatedReceiverBase::reset, base::Unretained(this)));
}
void AssociatedReceiverBase::FlushForTesting() {
endpoint_client_->FlushForTesting(); // IN-TEST
}
AssociatedReceiverBase::~AssociatedReceiverBase() = default;
void AssociatedReceiverBase::BindImpl(
ScopedInterfaceEndpointHandle handle,
MessageReceiverWithResponderStatus* receiver,
std::unique_ptr<MessageReceiver> payload_validator,
base::span<const uint32_t> sync_method_ordinals,
scoped_refptr<base::SequencedTaskRunner> runner,
uint32_t interface_version,
const char* interface_name,
MessageToMethodInfoCallback method_info_callback,
MessageToMethodNameCallback method_name_callback) {
DCHECK(handle.is_valid());
endpoint_client_ = std::make_unique<InterfaceEndpointClient>(
std::move(handle), receiver, std::move(payload_validator),
sync_method_ordinals,
internal::GetTaskRunnerToUseFromUserProvidedTaskRunner(std::move(runner)),
interface_version, interface_name, method_info_callback,
method_name_callback);
}
} // namespace internal
void AssociateWithDisconnectedPipe(ScopedInterfaceEndpointHandle handle) {
MessagePipe pipe;
scoped_refptr<internal::MultiplexRouter> router =
internal::MultiplexRouter::CreateAndStartReceiving(
std::move(pipe.handle0), internal::MultiplexRouter::MULTI_INTERFACE,
false, base::SequencedTaskRunner::GetCurrentDefault());
router->AssociateInterface(std::move(handle));
}
} // namespace mojo