| Index: mojo/public/cpp/bindings/lib/interface_ptr_state.h
|
| diff --git a/mojo/public/cpp/bindings/lib/interface_ptr_state.h b/mojo/public/cpp/bindings/lib/interface_ptr_state.h
|
| new file mode 100644
|
| index 0000000000000000000000000000000000000000..cb38dceb8f7f335e4f3dc7c56a2e470883a6020f
|
| --- /dev/null
|
| +++ b/mojo/public/cpp/bindings/lib/interface_ptr_state.h
|
| @@ -0,0 +1,347 @@
|
| +// Copyright 2014 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.
|
| +
|
| +#ifndef MOJO_PUBLIC_CPP_BINDINGS_LIB_INTERFACE_PTR_STATE_H_
|
| +#define MOJO_PUBLIC_CPP_BINDINGS_LIB_INTERFACE_PTR_STATE_H_
|
| +
|
| +#include <algorithm> // For |std::swap()|.
|
| +
|
| +#include "base/logging.h"
|
| +#include "base/macros.h"
|
| +#include "base/memory/ref_counted.h"
|
| +#include "base/memory/scoped_ptr.h"
|
| +#include "mojo/public/cpp/bindings/associated_group.h"
|
| +#include "mojo/public/cpp/bindings/callback.h"
|
| +#include "mojo/public/cpp/bindings/interface_ptr_info.h"
|
| +#include "mojo/public/cpp/bindings/lib/control_message_proxy.h"
|
| +#include "mojo/public/cpp/bindings/lib/filter_chain.h"
|
| +#include "mojo/public/cpp/bindings/lib/interface_endpoint_client.h"
|
| +#include "mojo/public/cpp/bindings/lib/interface_id.h"
|
| +#include "mojo/public/cpp/bindings/lib/message_header_validator.h"
|
| +#include "mojo/public/cpp/bindings/lib/multiplex_router.h"
|
| +#include "mojo/public/cpp/bindings/lib/router.h"
|
| +#include "mojo/public/cpp/bindings/lib/scoped_interface_endpoint_handle.h"
|
| +
|
| +struct MojoAsyncWaiter;
|
| +
|
| +namespace mojo {
|
| +namespace internal {
|
| +
|
| +template <typename Interface, bool use_multiplex_router>
|
| +class InterfacePtrState;
|
| +
|
| +// Uses a single-threaded, dedicated router. If |Interface| doesn't have any
|
| +// methods to pass associated interface pointers or requests, there won't be
|
| +// multiple interfaces running on the underlying message pipe. In that case, we
|
| +// can use this specialization to reduce cost.
|
| +template <typename Interface>
|
| +class InterfacePtrState<Interface, false> {
|
| + public:
|
| + InterfacePtrState()
|
| + : proxy_(nullptr), router_(nullptr), waiter_(nullptr), version_(0u) {}
|
| +
|
| + ~InterfacePtrState() {
|
| + // Destruction order matters here. We delete |proxy_| first, even though
|
| + // |router_| may have a reference to it, so that destructors for any request
|
| + // callbacks still pending can interact with the InterfacePtr.
|
| + delete proxy_;
|
| + delete router_;
|
| + }
|
| +
|
| + Interface* instance() {
|
| + ConfigureProxyIfNecessary();
|
| +
|
| + // This will be null if the object is not bound.
|
| + return proxy_;
|
| + }
|
| +
|
| + uint32_t version() const { return version_; }
|
| +
|
| + void QueryVersion(const Callback<void(uint32_t)>& callback) {
|
| + ConfigureProxyIfNecessary();
|
| +
|
| + // It is safe to capture |this| because the callback won't be run after this
|
| + // object goes away.
|
| + auto callback_wrapper = [this, callback](uint32_t version) {
|
| + this->version_ = version;
|
| + callback.Run(version);
|
| + };
|
| +
|
| + // Do a static cast in case the interface contains methods with the same
|
| + // name.
|
| + static_cast<ControlMessageProxy*>(proxy_)->QueryVersion(callback_wrapper);
|
| + }
|
| +
|
| + void RequireVersion(uint32_t version) {
|
| + ConfigureProxyIfNecessary();
|
| +
|
| + if (version <= version_)
|
| + return;
|
| +
|
| + version_ = version;
|
| + // Do a static cast in case the interface contains methods with the same
|
| + // name.
|
| + static_cast<ControlMessageProxy*>(proxy_)->RequireVersion(version);
|
| + }
|
| +
|
| + void Swap(InterfacePtrState* other) {
|
| + using std::swap;
|
| + swap(other->proxy_, proxy_);
|
| + swap(other->router_, router_);
|
| + handle_.swap(other->handle_);
|
| + swap(other->waiter_, waiter_);
|
| + swap(other->version_, version_);
|
| + }
|
| +
|
| + void Bind(InterfacePtrInfo<Interface> info, const MojoAsyncWaiter* waiter) {
|
| + DCHECK(!proxy_);
|
| + DCHECK(!router_);
|
| + DCHECK(!handle_.is_valid());
|
| + DCHECK(!waiter_);
|
| + DCHECK_EQ(0u, version_);
|
| + DCHECK(info.is_valid());
|
| +
|
| + handle_ = info.PassHandle();
|
| + waiter_ = waiter;
|
| + version_ = info.version();
|
| + }
|
| +
|
| + bool WaitForIncomingResponse() {
|
| + ConfigureProxyIfNecessary();
|
| +
|
| + DCHECK(router_);
|
| + return router_->WaitForIncomingMessage(MOJO_DEADLINE_INDEFINITE);
|
| + }
|
| +
|
| + // After this method is called, the object is in an invalid state and
|
| + // shouldn't be reused.
|
| + InterfacePtrInfo<Interface> PassInterface() {
|
| + return InterfacePtrInfo<Interface>(
|
| + router_ ? router_->PassMessagePipe() : handle_.Pass(), version_);
|
| + }
|
| +
|
| + bool is_bound() const { return handle_.is_valid() || router_; }
|
| +
|
| + bool encountered_error() const {
|
| + return router_ ? router_->encountered_error() : false;
|
| + }
|
| +
|
| + void set_connection_error_handler(const Closure& error_handler) {
|
| + ConfigureProxyIfNecessary();
|
| +
|
| + DCHECK(router_);
|
| + router_->set_connection_error_handler(error_handler);
|
| + }
|
| +
|
| + // Returns true if bound and awaiting a response to a message.
|
| + bool has_pending_callbacks() const {
|
| + return router_ && router_->has_pending_responders();
|
| + }
|
| +
|
| + AssociatedGroup* associated_group() { return nullptr; }
|
| +
|
| + void EnableTestingMode() {
|
| + ConfigureProxyIfNecessary();
|
| + router_->EnableTestingMode();
|
| + }
|
| +
|
| + private:
|
| + using Proxy = typename Interface::Proxy_;
|
| +
|
| + void ConfigureProxyIfNecessary() {
|
| + // The proxy has been configured.
|
| + if (proxy_) {
|
| + DCHECK(router_);
|
| + return;
|
| + }
|
| + // The object hasn't been bound.
|
| + if (!waiter_) {
|
| + DCHECK(!handle_.is_valid());
|
| + return;
|
| + }
|
| +
|
| + FilterChain filters;
|
| + filters.Append<MessageHeaderValidator>();
|
| + filters.Append<typename Interface::ResponseValidator_>();
|
| +
|
| + router_ = new Router(handle_.Pass(), filters.Pass(), waiter_);
|
| + waiter_ = nullptr;
|
| +
|
| + proxy_ = new Proxy(router_);
|
| + }
|
| +
|
| + Proxy* proxy_;
|
| + Router* router_;
|
| +
|
| + // |proxy_| and |router_| are not initialized until read/write with the
|
| + // message pipe handle is needed. |handle_| and |waiter_| are valid between
|
| + // the Bind() call and the initialization of |proxy_| and |router_|.
|
| + ScopedMessagePipeHandle handle_;
|
| + const MojoAsyncWaiter* waiter_;
|
| +
|
| + uint32_t version_;
|
| +
|
| + DISALLOW_COPY_AND_ASSIGN(InterfacePtrState);
|
| +};
|
| +
|
| +// Uses a multiplexing router. If |Interface| has methods to pass associated
|
| +// interface pointers or requests, this specialization should be used.
|
| +template <typename Interface>
|
| +class InterfacePtrState<Interface, true> {
|
| + public:
|
| + InterfacePtrState() : waiter_(nullptr), version_(0u) {}
|
| +
|
| + ~InterfacePtrState() {
|
| + endpoint_client_.reset();
|
| + proxy_.reset();
|
| + if (router_)
|
| + router_->CloseMessagePipe();
|
| + }
|
| +
|
| + Interface* instance() {
|
| + ConfigureProxyIfNecessary();
|
| +
|
| + // This will be null if the object is not bound.
|
| + return proxy_.get();
|
| + }
|
| +
|
| + uint32_t version() const { return version_; }
|
| +
|
| + void QueryVersion(const Callback<void(uint32_t)>& callback) {
|
| + ConfigureProxyIfNecessary();
|
| +
|
| + // It is safe to capture |this| because the callback won't be run after this
|
| + // object goes away.
|
| + auto callback_wrapper = [this, callback](uint32_t version) {
|
| + this->version_ = version;
|
| + callback.Run(version);
|
| + };
|
| +
|
| + // Do a static cast in case the interface contains methods with the same
|
| + // name.
|
| + static_cast<ControlMessageProxy*>(proxy_.get())
|
| + ->QueryVersion(callback_wrapper);
|
| + }
|
| +
|
| + void RequireVersion(uint32_t version) {
|
| + ConfigureProxyIfNecessary();
|
| +
|
| + if (version <= version_)
|
| + return;
|
| +
|
| + version_ = version;
|
| + // Do a static cast in case the interface contains methods with the same
|
| + // name.
|
| + static_cast<ControlMessageProxy*>(proxy_.get())->RequireVersion(version);
|
| + }
|
| +
|
| + void Swap(InterfacePtrState* other) {
|
| + using std::swap;
|
| + swap(other->router_, router_);
|
| + swap(other->endpoint_client_, endpoint_client_);
|
| + swap(other->proxy_, proxy_);
|
| + handle_.swap(other->handle_);
|
| + swap(other->waiter_, waiter_);
|
| + swap(other->version_, version_);
|
| + }
|
| +
|
| + void Bind(InterfacePtrInfo<Interface> info, const MojoAsyncWaiter* waiter) {
|
| + DCHECK(!router_);
|
| + DCHECK(!endpoint_client_);
|
| + DCHECK(!proxy_);
|
| + DCHECK(!handle_.is_valid());
|
| + DCHECK(!waiter_);
|
| + DCHECK_EQ(0u, version_);
|
| + DCHECK(info.is_valid());
|
| +
|
| + handle_ = info.PassHandle();
|
| + waiter_ = waiter;
|
| + version_ = info.version();
|
| + }
|
| +
|
| + // After this method is called, the object is in an invalid state and
|
| + // shouldn't be reused.
|
| + InterfacePtrInfo<Interface> PassInterface() {
|
| + endpoint_client_.reset();
|
| + proxy_.reset();
|
| + return InterfacePtrInfo<Interface>(
|
| + router_ ? router_->PassMessagePipe() : handle_.Pass(), version_);
|
| + }
|
| +
|
| + bool is_bound() const { return handle_.is_valid() || endpoint_client_; }
|
| +
|
| + bool encountered_error() const {
|
| + return endpoint_client_ ? endpoint_client_->encountered_error() : false;
|
| + }
|
| +
|
| + void set_connection_error_handler(const Closure& error_handler) {
|
| + ConfigureProxyIfNecessary();
|
| +
|
| + DCHECK(endpoint_client_);
|
| + endpoint_client_->set_connection_error_handler(error_handler);
|
| + }
|
| +
|
| + // Returns true if bound and awaiting a response to a message.
|
| + bool has_pending_callbacks() const {
|
| + return endpoint_client_ && endpoint_client_->has_pending_responders();
|
| + }
|
| +
|
| + AssociatedGroup* associated_group() {
|
| + ConfigureProxyIfNecessary();
|
| + return endpoint_client_->associated_group();
|
| + }
|
| +
|
| + void EnableTestingMode() {
|
| + ConfigureProxyIfNecessary();
|
| + router_->EnableTestingMode();
|
| + }
|
| +
|
| + // Intentionally not defined:
|
| + // bool WaitForIncomingResponse();
|
| +
|
| + private:
|
| + using Proxy = typename Interface::Proxy_;
|
| +
|
| + void ConfigureProxyIfNecessary() {
|
| + // The proxy has been configured.
|
| + if (proxy_) {
|
| + DCHECK(router_);
|
| + DCHECK(endpoint_client_);
|
| + return;
|
| + }
|
| + // The object hasn't been bound.
|
| + if (!waiter_) {
|
| + DCHECK(!handle_.is_valid());
|
| + return;
|
| + }
|
| +
|
| + router_ = new MultiplexRouter(true, handle_.Pass(), waiter_);
|
| + endpoint_client_.reset(new InterfaceEndpointClient(
|
| + router_->CreateLocalEndpointHandle(kMasterInterfaceId), nullptr,
|
| + make_scoped_ptr(new typename Interface::ResponseValidator_())));
|
| + proxy_.reset(new Proxy(endpoint_client_.get(), endpoint_client_->router()));
|
| +
|
| + waiter_ = nullptr;
|
| + }
|
| +
|
| + scoped_refptr<MultiplexRouter> router_;
|
| +
|
| + scoped_ptr<InterfaceEndpointClient> endpoint_client_;
|
| + scoped_ptr<Proxy> proxy_;
|
| +
|
| + // |router_| (as well as other members above) is not initialized until
|
| + // read/write with the message pipe handle is needed. |handle_| and |waiter_|
|
| + // are valid between the Bind() call and the initialization of |router_|.
|
| + ScopedMessagePipeHandle handle_;
|
| + const MojoAsyncWaiter* waiter_;
|
| +
|
| + uint32_t version_;
|
| +
|
| + DISALLOW_COPY_AND_ASSIGN(InterfacePtrState);
|
| +};
|
| +
|
| +} // namespace internal
|
| +} // namespace mojo
|
| +
|
| +#endif // MOJO_PUBLIC_CPP_BINDINGS_LIB_INTERFACE_PTR_STATE_H_
|
|
|