blob: 94a93ec14502d05064d4c184abe67db9db4825f3 [file] [log] [blame]
// Copyright 2019 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#ifndef MOJO_PUBLIC_CPP_BINDINGS_PENDING_ASSOCIATED_RECEIVER_H_
#define MOJO_PUBLIC_CPP_BINDINGS_PENDING_ASSOCIATED_RECEIVER_H_
#include <stdint.h>
#include <type_traits>
#include <utility>
#include "base/compiler_specific.h"
#include "base/task/sequenced_task_runner.h"
#include "build/build_config.h"
#include "mojo/public/cpp/bindings/lib/multiplex_router.h"
#include "mojo/public/cpp/bindings/runtime_features.h"
#include "mojo/public/cpp/bindings/scoped_interface_endpoint_handle.h"
#include "mojo/public/cpp/system/message_pipe.h"
namespace mojo {
template <typename T>
class PendingAssociatedRemote;
template <typename T>
struct PendingAssociatedReceiverConverter;
// PendingAssociatedReceiver represents an unbound associated interface
// endpoint that will receive and queue messages. An AssociatedReceiver can
// consume this object to begin receiving method calls from a corresponding
// AssociatedRemote.
template <typename Interface>
class PendingAssociatedReceiver {
public:
PendingAssociatedReceiver() = default;
PendingAssociatedReceiver(PendingAssociatedReceiver&& other)
: handle_(std::move(other.handle_)) {}
explicit PendingAssociatedReceiver(ScopedInterfaceEndpointHandle handle)
: handle_(std::move(handle)) {}
// Move conversion operator for custom receiver types. Only participates in
// overload resolution if a typesafe conversion is supported.
template <
typename T,
std::enable_if_t<std::is_same<
PendingAssociatedReceiver<Interface>,
std::invoke_result_t<decltype(&PendingAssociatedReceiverConverter<
T>::template To<Interface>),
T&&>>::value>* = nullptr>
PendingAssociatedReceiver(T&& other)
: PendingAssociatedReceiver(
PendingAssociatedReceiverConverter<T>::template To<Interface>(
std::move(other))) {}
PendingAssociatedReceiver(const PendingAssociatedReceiver&) = delete;
PendingAssociatedReceiver& operator=(const PendingAssociatedReceiver&) =
delete;
~PendingAssociatedReceiver() = default;
PendingAssociatedReceiver& operator=(PendingAssociatedReceiver&& other) {
handle_ = std::move(other.handle_);
return *this;
}
bool is_valid() const { return handle_.is_valid(); }
explicit operator bool() const { return is_valid(); }
ScopedInterfaceEndpointHandle PassHandle() { return std::move(handle_); }
const ScopedInterfaceEndpointHandle& handle() const { return handle_; }
void set_handle(ScopedInterfaceEndpointHandle handle) {
handle_ = std::move(handle);
}
// Hangs up this endpoint, invalidating the PendingAssociatedReceiver.
void reset() { handle_.reset(); }
// Similar to above but provides additional metadata in case the remote
// endpoint wants details about why this endpoint hung up.
void ResetWithReason(uint32_t custom_reason, const std::string& description) {
handle_.ResetWithReason(custom_reason, description);
}
[[nodiscard]] REINITIALIZES_AFTER_MOVE PendingAssociatedRemote<Interface>
InitWithNewEndpointAndPassRemote();
// Associates this endpoint with a dedicated message pipe. This allows the
// entangled AssociatedReceiver/AssociatedRemote endpoints to be used without
// ever being associated with any other mojom interfaces.
//
// Needless to say, messages sent between the two entangled endpoints will not
// be ordered with respect to any other mojom interfaces. This is generally
// useful for ignoring calls on an associated remote or for binding associated
// endpoints in tests.
void EnableUnassociatedUsage() {
DCHECK(is_valid());
MessagePipe pipe;
scoped_refptr<internal::MultiplexRouter> router0 =
internal::MultiplexRouter::CreateAndStartReceiving(
std::move(pipe.handle0), internal::MultiplexRouter::MULTI_INTERFACE,
false, base::SequencedTaskRunner::GetCurrentDefault());
scoped_refptr<internal::MultiplexRouter> router1 =
internal::MultiplexRouter::CreateAndStartReceiving(
std::move(pipe.handle1), internal::MultiplexRouter::MULTI_INTERFACE,
true, base::SequencedTaskRunner::GetCurrentDefault());
InterfaceId id = router1->AssociateInterface(PassHandle());
set_handle(router0->CreateLocalEndpointHandle(id));
}
private:
ScopedInterfaceEndpointHandle handle_;
};
// Constructs an invalid PendingAssociatedReceiver of any arbitrary interface
// type. Useful as short-hand for a default constructed value.
class COMPONENT_EXPORT(MOJO_CPP_BINDINGS) NullAssociatedReceiver {
public:
template <typename Interface>
operator PendingAssociatedReceiver<Interface>() const {
return PendingAssociatedReceiver<Interface>();
}
};
} // namespace mojo
#include "mojo/public/cpp/bindings/pending_associated_remote.h"
namespace mojo {
template <typename Interface>
PendingAssociatedRemote<Interface>
PendingAssociatedReceiver<Interface>::InitWithNewEndpointAndPassRemote() {
if (!internal::GetRuntimeFeature_ExpectEnabled<Interface>()) {
return PendingAssociatedRemote<Interface>();
}
ScopedInterfaceEndpointHandle remote_handle;
ScopedInterfaceEndpointHandle::CreatePairPendingAssociation(&handle_,
&remote_handle);
return PendingAssociatedRemote<Interface>(std::move(remote_handle), 0u);
}
} // namespace mojo
#endif // MOJO_PUBLIC_CPP_BINDINGS_PENDING_ASSOCIATED_RECEIVER_H_