blob: 28f7b7899486d6feea09de3c7360817b174a6a78 [file] [log] [blame]
// 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 "ppapi/proxy/ppp_instance_proxy.h"
#include <algorithm>
#include "ppapi/c/pp_var.h"
#include "ppapi/c/ppb_core.h"
#include "ppapi/c/ppp_instance.h"
#include "ppapi/c/private/ppb_flash_fullscreen.h"
#include "ppapi/proxy/host_dispatcher.h"
#include "ppapi/proxy/plugin_dispatcher.h"
#include "ppapi/proxy/plugin_resource_tracker.h"
#include "ppapi/proxy/ppapi_messages.h"
#include "ppapi/proxy/ppb_url_loader_proxy.h"
#include "ppapi/shared_impl/ppapi_globals.h"
namespace ppapi {
namespace proxy {
namespace {
PP_Bool DidCreate(PP_Instance instance,
uint32_t argc,
const char* argn[],
const char* argv[]) {
std::vector<std::string> argn_vect;
std::vector<std::string> argv_vect;
for (uint32_t i = 0; i < argc; i++) {
argn_vect.push_back(std::string(argn[i]));
argv_vect.push_back(std::string(argv[i]));
}
PP_Bool result = PP_FALSE;
HostDispatcher::GetForInstance(instance)->Send(
new PpapiMsg_PPPInstance_DidCreate(API_ID_PPP_INSTANCE, instance,
argn_vect, argv_vect, &result));
return result;
}
void DidDestroy(PP_Instance instance) {
HostDispatcher::GetForInstance(instance)->Send(
new PpapiMsg_PPPInstance_DidDestroy(API_ID_PPP_INSTANCE, instance));
}
void DidChangeView(PP_Instance instance,
const PP_Rect* position,
const PP_Rect* clip) {
HostDispatcher* dispatcher = HostDispatcher::GetForInstance(instance);
const PPB_FlashFullscreen* fullscreen_interface =
static_cast<const PPB_FlashFullscreen*>(
dispatcher->local_get_interface()(PPB_FLASHFULLSCREEN_INTERFACE));
DCHECK(fullscreen_interface);
const PPB_FlashFullscreen* flash_fullscreen_interface =
static_cast<const PPB_FlashFullscreen*>(
dispatcher->local_get_interface()(PPB_FLASHFULLSCREEN_INTERFACE));
DCHECK(flash_fullscreen_interface);
PP_Bool fullscreen = fullscreen_interface->IsFullscreen(instance);
PP_Bool flash_fullscreen =
flash_fullscreen_interface->IsFullscreen(instance);
dispatcher->Send(
new PpapiMsg_PPPInstance_DidChangeView(API_ID_PPP_INSTANCE,
instance, *position, *clip,
fullscreen,
flash_fullscreen));
}
void DidChangeFocus(PP_Instance instance, PP_Bool has_focus) {
HostDispatcher::GetForInstance(instance)->Send(
new PpapiMsg_PPPInstance_DidChangeFocus(API_ID_PPP_INSTANCE,
instance, has_focus));
}
PP_Bool HandleDocumentLoad(PP_Instance instance,
PP_Resource url_loader) {
PP_Bool result = PP_FALSE;
HostDispatcher* dispatcher = HostDispatcher::GetForInstance(instance);
// Set up the URLLoader for proxying.
PPB_URLLoader_Proxy* url_loader_proxy = static_cast<PPB_URLLoader_Proxy*>(
dispatcher->GetInterfaceProxy(API_ID_PPB_URL_LOADER));
url_loader_proxy->PrepareURLLoaderForSendingToPlugin(url_loader);
// PluginResourceTracker in the plugin process assumes that resources that it
// tracks have been addrefed on behalf of the plugin at the renderer side. So
// we explicitly do it for |url_loader| here.
//
// Please also see comments in PPP_Instance_Proxy::OnMsgHandleDocumentLoad()
// about releasing of this extra reference.
const PPB_Core* core = reinterpret_cast<const PPB_Core*>(
dispatcher->local_get_interface()(PPB_CORE_INTERFACE));
if (!core) {
NOTREACHED();
return PP_FALSE;
}
core->AddRefResource(url_loader);
HostResource serialized_loader;
serialized_loader.SetHostResource(instance, url_loader);
dispatcher->Send(new PpapiMsg_PPPInstance_HandleDocumentLoad(
API_ID_PPP_INSTANCE, instance, serialized_loader, &result));
return result;
}
static const PPP_Instance_1_0 instance_interface_1_0 = {
&DidCreate,
&DidDestroy,
&DidChangeView,
&DidChangeFocus,
&HandleDocumentLoad
};
InterfaceProxy* CreateInstanceProxy(Dispatcher* dispatcher) {
return new PPP_Instance_Proxy(dispatcher);
}
} // namespace
PPP_Instance_Proxy::PPP_Instance_Proxy(Dispatcher* dispatcher)
: InterfaceProxy(dispatcher) {
if (dispatcher->IsPlugin()) {
combined_interface_.reset(
new PPP_Instance_Combined(*static_cast<const PPP_Instance_1_0*>(
dispatcher->local_get_interface()(PPP_INSTANCE_INTERFACE_1_0))));
}
}
PPP_Instance_Proxy::~PPP_Instance_Proxy() {
}
// static
const InterfaceProxy::Info* PPP_Instance_Proxy::GetInfo1_0() {
static const Info info = {
&instance_interface_1_0,
PPP_INSTANCE_INTERFACE_1_0,
API_ID_PPP_INSTANCE,
false,
&CreateInstanceProxy
};
return &info;
}
bool PPP_Instance_Proxy::OnMessageReceived(const IPC::Message& msg) {
bool handled = true;
IPC_BEGIN_MESSAGE_MAP(PPP_Instance_Proxy, msg)
IPC_MESSAGE_HANDLER(PpapiMsg_PPPInstance_DidCreate,
OnMsgDidCreate)
IPC_MESSAGE_HANDLER(PpapiMsg_PPPInstance_DidDestroy,
OnMsgDidDestroy)
IPC_MESSAGE_HANDLER(PpapiMsg_PPPInstance_DidChangeView,
OnMsgDidChangeView)
IPC_MESSAGE_HANDLER(PpapiMsg_PPPInstance_DidChangeFocus,
OnMsgDidChangeFocus)
IPC_MESSAGE_HANDLER(PpapiMsg_PPPInstance_HandleDocumentLoad,
OnMsgHandleDocumentLoad)
IPC_MESSAGE_UNHANDLED(handled = false)
IPC_END_MESSAGE_MAP()
return handled;
}
void PPP_Instance_Proxy::OnMsgDidCreate(
PP_Instance instance,
const std::vector<std::string>& argn,
const std::vector<std::string>& argv,
PP_Bool* result) {
*result = PP_FALSE;
if (argn.size() != argv.size())
return;
// Set up the routing associating this new instance with the dispatcher we
// just got the message from. This must be done before calling into the
// plugin so it can in turn call PPAPI functions.
PluginDispatcher* plugin_dispatcher =
static_cast<PluginDispatcher*>(dispatcher());
plugin_dispatcher->DidCreateInstance(instance);
PpapiGlobals::Get()->GetResourceTracker()->DidCreateInstance(instance);
// Make sure the arrays always have at least one element so we can take the
// address below.
std::vector<const char*> argn_array(
std::max(static_cast<size_t>(1), argn.size()));
std::vector<const char*> argv_array(
std::max(static_cast<size_t>(1), argn.size()));
for (size_t i = 0; i < argn.size(); i++) {
argn_array[i] = argn[i].c_str();
argv_array[i] = argv[i].c_str();
}
DCHECK(combined_interface_.get());
*result = combined_interface_->DidCreate(instance,
static_cast<uint32_t>(argn.size()),
&argn_array[0], &argv_array[0]);
}
void PPP_Instance_Proxy::OnMsgDidDestroy(PP_Instance instance) {
combined_interface_->DidDestroy(instance);
PpapiGlobals::Get()->GetResourceTracker()->DidDeleteInstance(instance);
static_cast<PluginDispatcher*>(dispatcher())->DidDestroyInstance(instance);
}
void PPP_Instance_Proxy::OnMsgDidChangeView(PP_Instance instance,
const PP_Rect& position,
const PP_Rect& clip,
PP_Bool fullscreen,
PP_Bool flash_fullscreen) {
PluginDispatcher* dispatcher = PluginDispatcher::GetForInstance(instance);
if (!dispatcher)
return;
InstanceData* data = dispatcher->GetInstanceData(instance);
if (!data)
return;
data->position = position;
data->fullscreen = fullscreen;
data->flash_fullscreen = flash_fullscreen;
combined_interface_->DidChangeView(instance, &position, &clip);
}
void PPP_Instance_Proxy::OnMsgDidChangeFocus(PP_Instance instance,
PP_Bool has_focus) {
combined_interface_->DidChangeFocus(instance, has_focus);
}
void PPP_Instance_Proxy::OnMsgHandleDocumentLoad(PP_Instance instance,
const HostResource& url_loader,
PP_Bool* result) {
PP_Resource plugin_loader =
PPB_URLLoader_Proxy::TrackPluginResource(url_loader);
*result = combined_interface_->HandleDocumentLoad(instance, plugin_loader);
// This balances the one reference that TrackPluginResource() initialized it
// with. The plugin will normally take an additional reference which will keep
// the resource alive in the plugin (and the one reference in the renderer
// representing all plugin references).
// Once all references at the plugin side are released, the renderer side will
// be notified and release the reference added in HandleDocumentLoad() above.
PpapiGlobals::Get()->GetResourceTracker()->ReleaseResource(plugin_loader);
}
} // namespace proxy
} // namespace ppapi