// 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