// Copyright (c) 2011 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 "webkit/plugins/ppapi/ppb_broker_impl.h" #include "base/logging.h" #include "webkit/plugins/ppapi/common.h" #include "webkit/plugins/ppapi/plugin_module.h" #include "webkit/plugins/ppapi/resource_helper.h" using ::ppapi::thunk::PPB_Broker_API; namespace webkit { namespace ppapi { namespace { // TODO(ddorwin): Put conversion functions in a common place and/or add an // invalid value to sync_socket.h. int32_t PlatformFileToInt(base::PlatformFile handle) { #if defined(OS_WIN) return static_cast(reinterpret_cast(handle)); #elif defined(OS_POSIX) return handle; #else #error Not implemented. #endif } } // namespace // PPB_Broker_Impl ------------------------------------------------------ PPB_Broker_Impl::PPB_Broker_Impl(PP_Instance instance) : Resource(instance), broker_(NULL), connect_callback_(), pipe_handle_(PlatformFileToInt(base::kInvalidPlatformFileValue)) { } PPB_Broker_Impl::~PPB_Broker_Impl() { if (broker_) { broker_->Disconnect(this); broker_ = NULL; } // The plugin owns the handle. pipe_handle_ = PlatformFileToInt(base::kInvalidPlatformFileValue); } PPB_Broker_API* PPB_Broker_Impl::AsPPB_Broker_API() { return this; } int32_t PPB_Broker_Impl::Connect(PP_CompletionCallback connect_callback) { if (!connect_callback.func) { // Synchronous calls are not supported. return PP_ERROR_BLOCKS_MAIN_THREAD; } // TODO(ddorwin): Return PP_ERROR_FAILED if plugin is in-process. if (broker_) { // May only be called once. return PP_ERROR_FAILED; } PluginInstance* plugin_instance = ResourceHelper::GetPluginInstance(this); if (!plugin_instance) return PP_ERROR_FAILED; // The callback must be populated now in case we are connected to the broker // and BrokerConnected is called before ConnectToPpapiBroker returns. // Because it must be created now, it must be aborted and cleared if // ConnectToPpapiBroker fails. connect_callback_ = new TrackedCompletionCallback( plugin_instance->module()->GetCallbackTracker(), pp_resource(), connect_callback); broker_ = plugin_instance->delegate()->ConnectToPpapiBroker(this); if (!broker_) { scoped_refptr callback; callback.swap(connect_callback_); callback->Abort(); return PP_ERROR_FAILED; } return PP_OK_COMPLETIONPENDING; } int32_t PPB_Broker_Impl::GetHandle(int32_t* handle) { if (pipe_handle_ == PlatformFileToInt(base::kInvalidPlatformFileValue)) return PP_ERROR_FAILED; // Handle not set yet. *handle = pipe_handle_; return PP_OK; } // Transfers ownership of the handle to the plugin. void PPB_Broker_Impl::BrokerConnected(int32_t handle, int32_t result) { DCHECK(pipe_handle_ == PlatformFileToInt(base::kInvalidPlatformFileValue)); DCHECK(result == PP_OK || handle == PlatformFileToInt(base::kInvalidPlatformFileValue)); pipe_handle_ = handle; // Synchronous calls are not supported. DCHECK(connect_callback_.get() && !connect_callback_->completed()); scoped_refptr callback; callback.swap(connect_callback_); callback->Run(result); // Will complete abortively if necessary. } } // namespace ppapi } // namespace webkit