|  | // Copyright (c) 2012 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 "ppapi/proxy/ppp_messaging_proxy.h" | 
|  |  | 
|  | #include <algorithm> | 
|  |  | 
|  | #include "ppapi/c/ppp_messaging.h" | 
|  | #include "ppapi/proxy/host_dispatcher.h" | 
|  | #include "ppapi/proxy/message_handler.h" | 
|  | #include "ppapi/proxy/plugin_dispatcher.h" | 
|  | #include "ppapi/proxy/plugin_resource_tracker.h" | 
|  | #include "ppapi/proxy/plugin_var_tracker.h" | 
|  | #include "ppapi/proxy/ppapi_messages.h" | 
|  | #include "ppapi/proxy/serialized_var.h" | 
|  | #include "ppapi/shared_impl/ppapi_globals.h" | 
|  | #include "ppapi/shared_impl/proxy_lock.h" | 
|  | #include "ppapi/shared_impl/scoped_pp_var.h" | 
|  | #include "ppapi/shared_impl/var_tracker.h" | 
|  |  | 
|  | namespace ppapi { | 
|  | namespace proxy { | 
|  |  | 
|  | namespace { | 
|  |  | 
|  | MessageHandler* GetMessageHandler(Dispatcher* dispatcher, | 
|  | PP_Instance instance) { | 
|  | if (!dispatcher || !dispatcher->IsPlugin()) { | 
|  | NOTREACHED(); | 
|  | return NULL; | 
|  | } | 
|  | PluginDispatcher* plugin_dispatcher = | 
|  | static_cast<PluginDispatcher*>(dispatcher); | 
|  | InstanceData* instance_data = plugin_dispatcher->GetInstanceData(instance); | 
|  | if (!instance_data) | 
|  | return NULL; | 
|  |  | 
|  | return instance_data->message_handler.get(); | 
|  | } | 
|  |  | 
|  | void ResetMessageHandler(Dispatcher* dispatcher, PP_Instance instance) { | 
|  | if (!dispatcher || !dispatcher->IsPlugin()) { | 
|  | NOTREACHED(); | 
|  | return; | 
|  | } | 
|  | PluginDispatcher* plugin_dispatcher = | 
|  | static_cast<PluginDispatcher*>(dispatcher); | 
|  | InstanceData* instance_data = plugin_dispatcher->GetInstanceData(instance); | 
|  | if (!instance_data) | 
|  | return; | 
|  |  | 
|  | instance_data->message_handler.reset(); | 
|  | } | 
|  |  | 
|  | }  // namespace | 
|  |  | 
|  | PPP_Messaging_Proxy::PPP_Messaging_Proxy(Dispatcher* dispatcher) | 
|  | : InterfaceProxy(dispatcher), | 
|  | ppp_messaging_impl_(NULL) { | 
|  | if (dispatcher->IsPlugin()) { | 
|  | ppp_messaging_impl_ = static_cast<const PPP_Messaging*>( | 
|  | dispatcher->local_get_interface()(PPP_MESSAGING_INTERFACE)); | 
|  | } | 
|  | } | 
|  |  | 
|  | PPP_Messaging_Proxy::~PPP_Messaging_Proxy() { | 
|  | } | 
|  |  | 
|  | bool PPP_Messaging_Proxy::OnMessageReceived(const IPC::Message& msg) { | 
|  | if (!dispatcher()->IsPlugin()) | 
|  | return false; | 
|  |  | 
|  | bool handled = true; | 
|  | IPC_BEGIN_MESSAGE_MAP(PPP_Messaging_Proxy, msg) | 
|  | IPC_MESSAGE_HANDLER(PpapiMsg_PPPMessaging_HandleMessage, | 
|  | OnMsgHandleMessage) | 
|  | IPC_MESSAGE_HANDLER_DELAY_REPLY( | 
|  | PpapiMsg_PPPMessageHandler_HandleBlockingMessage, | 
|  | OnMsgHandleBlockingMessage) | 
|  | IPC_MESSAGE_UNHANDLED(handled = false) | 
|  | IPC_END_MESSAGE_MAP() | 
|  | return handled; | 
|  | } | 
|  |  | 
|  | void PPP_Messaging_Proxy::OnMsgHandleMessage( | 
|  | PP_Instance instance, SerializedVarReceiveInput message_data) { | 
|  | PP_Var received_var(message_data.GetForInstance(dispatcher(), instance)); | 
|  | MessageHandler* message_handler = GetMessageHandler(dispatcher(), instance); | 
|  | if (message_handler) { | 
|  | if (message_handler->LoopIsValid()) { | 
|  | message_handler->HandleMessage(ScopedPPVar(received_var)); | 
|  | return; | 
|  | } else { | 
|  | // If the MessageHandler's loop has been quit, then we should treat it as | 
|  | // though it has been unregistered and start sending messages to the | 
|  | // default handler. This might mean the plugin has lost messages, but | 
|  | // there's not really anything sane we can do about it. They should have | 
|  | // used UnregisterMessageHandler. | 
|  | ResetMessageHandler(dispatcher(), instance); | 
|  | } | 
|  | } | 
|  | // If we reach this point, then there's no message handler registered, so | 
|  | // we send to the default PPP_Messaging one for the instance. | 
|  |  | 
|  | // SerializedVarReceiveInput will decrement the reference count, but we want | 
|  | // to give the recipient a reference in the legacy API. | 
|  | PpapiGlobals::Get()->GetVarTracker()->AddRefVar(received_var); | 
|  | CallWhileUnlocked(ppp_messaging_impl_->HandleMessage, | 
|  | instance, | 
|  | received_var); | 
|  | } | 
|  |  | 
|  | void PPP_Messaging_Proxy::OnMsgHandleBlockingMessage( | 
|  | PP_Instance instance, | 
|  | SerializedVarReceiveInput message_data, | 
|  | IPC::Message* reply_msg) { | 
|  | ScopedPPVar received_var(message_data.GetForInstance(dispatcher(), instance)); | 
|  | MessageHandler* message_handler = GetMessageHandler(dispatcher(), instance); | 
|  | if (message_handler) { | 
|  | if (message_handler->LoopIsValid()) { | 
|  | message_handler->HandleBlockingMessage( | 
|  | received_var, scoped_ptr<IPC::Message>(reply_msg)); | 
|  | return; | 
|  | } else { | 
|  | // If the MessageHandler's loop has been quit, then we should treat it as | 
|  | // though it has been unregistered. Also see the note for PostMessage. | 
|  | ResetMessageHandler(dispatcher(), instance); | 
|  | } | 
|  | } | 
|  | // We have no handler, but we still need to respond to unblock the renderer | 
|  | // and inform the JavaScript caller. | 
|  | PpapiMsg_PPPMessageHandler_HandleBlockingMessage::WriteReplyParams( | 
|  | reply_msg, | 
|  | SerializedVarReturnValue::Convert(dispatcher(), PP_MakeUndefined()), | 
|  | false /* was_handled */); | 
|  | dispatcher()->Send(reply_msg); | 
|  | } | 
|  |  | 
|  |  | 
|  | }  // namespace proxy | 
|  | }  // namespace ppapi |