// Copyright (c) 2006-2008 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 "chrome/plugin/webplugin_delegate_stub.h" #include "base/command_line.h" #include "base/time.h" #include "base/gfx/bitmap_header.h" #include "base/gfx/platform_device_win.h" #include "chrome/common/chrome_switches.h" #include "chrome/common/gfx/emf.h" #include "chrome/common/plugin_messages.h" #include "chrome/common/win_util.h" #include "chrome/plugin/npobject_stub.h" #include "chrome/plugin/plugin_channel.h" #include "chrome/plugin/plugin_thread.h" #include "chrome/plugin/webplugin_proxy.h" #include "third_party/npapi/bindings/npapi.h" #include "third_party/npapi/bindings/npruntime.h" #include "webkit/glue/plugins/webplugin_delegate_impl.h" #include "webkit/glue/webcursor.h" class FinishDestructionTask : public Task { public: FinishDestructionTask(WebPluginDelegate* delegate, WebPlugin* webplugin) : delegate_(delegate), webplugin_(webplugin) { } void Run() { // WebPlugin must outlive WebPluginDelegate. if (delegate_) delegate_->PluginDestroyed(); delete webplugin_; } private: WebPluginDelegate* delegate_; WebPlugin* webplugin_; }; WebPluginDelegateStub::WebPluginDelegateStub( const std::string& mime_type, int instance_id, PluginChannel* channel) : mime_type_(mime_type), instance_id_(instance_id), channel_(channel), delegate_(NULL), webplugin_(NULL) { DCHECK(channel); } WebPluginDelegateStub::~WebPluginDelegateStub() { if (channel_->in_send()) { // The delegate or an npobject is in the callstack, so don't delete it // right away. MessageLoop::current()->PostTask(FROM_HERE, new FinishDestructionTask(delegate_, webplugin_)); } else { // Safe to delete right away. if (delegate_) delegate_->PluginDestroyed(); delete webplugin_; } } void WebPluginDelegateStub::OnMessageReceived(const IPC::Message& msg) { // A plugin can execute a script to delete itself in any of its NPP methods. // Hold an extra reference to ourself so that if this does occur and we're // handling a sync message, we don't crash when attempting to send a reply. AddRef(); IPC_BEGIN_MESSAGE_MAP(WebPluginDelegateStub, msg) IPC_MESSAGE_HANDLER(PluginMsg_Init, OnInit) IPC_MESSAGE_HANDLER(PluginMsg_WillSendRequest, OnWillSendRequest) IPC_MESSAGE_HANDLER(PluginMsg_DidReceiveResponse, OnDidReceiveResponse) IPC_MESSAGE_HANDLER(PluginMsg_DidReceiveData, OnDidReceiveData) IPC_MESSAGE_HANDLER(PluginMsg_DidFinishLoading, OnDidFinishLoading) IPC_MESSAGE_HANDLER(PluginMsg_DidFail, OnDidFail) IPC_MESSAGE_HANDLER(PluginMsg_DidFinishLoadWithReason, OnDidFinishLoadWithReason) IPC_MESSAGE_HANDLER(PluginMsg_SetFocus, OnSetFocus) IPC_MESSAGE_HANDLER(PluginMsg_HandleEvent, OnHandleEvent) IPC_MESSAGE_HANDLER(PluginMsg_Paint, OnPaint) IPC_MESSAGE_HANDLER(PluginMsg_Print, OnPrint) IPC_MESSAGE_HANDLER(PluginMsg_PaintIntoSharedMemory, OnPaintIntoSharedMemory) IPC_MESSAGE_HANDLER(PluginMsg_GetPluginScriptableObject, OnGetPluginScriptableObject) IPC_MESSAGE_HANDLER(PluginMsg_UpdateGeometry, OnUpdateGeometry) IPC_MESSAGE_HANDLER(PluginMsg_SendJavaScriptStream, OnSendJavaScriptStream) IPC_MESSAGE_HANDLER(PluginMsg_DidReceiveManualResponse, OnDidReceiveManualResponse) IPC_MESSAGE_HANDLER(PluginMsg_DidReceiveManualData, OnDidReceiveManualData) IPC_MESSAGE_HANDLER(PluginMsg_DidFinishManualLoading, OnDidFinishManualLoading) IPC_MESSAGE_HANDLER(PluginMsg_DidManualLoadFail, OnDidManualLoadFail) IPC_MESSAGE_HANDLER(PluginMsg_InstallMissingPlugin, OnInstallMissingPlugin) IPC_MESSAGE_HANDLER(PluginMsg_HandleURLRequestReply, OnHandleURLRequestReply) IPC_MESSAGE_HANDLER(PluginMsg_URLRequestRouted, OnURLRequestRouted) IPC_MESSAGE_UNHANDLED_ERROR() IPC_END_MESSAGE_MAP() Release(); } bool WebPluginDelegateStub::Send(IPC::Message* msg) { return channel_->Send(msg); } void WebPluginDelegateStub::OnInit(const PluginMsg_Init_Params& params, bool* result) { *result = false; int argc = static_cast(params.arg_names.size()); if (argc != static_cast(params.arg_values.size())) { NOTREACHED(); return; } char **argn = new char*[argc]; char **argv = new char*[argc]; for (int i = 0; i < argc; ++i) { argn[i] = const_cast(params.arg_names[i].c_str()); argv[i] = const_cast(params.arg_values[i].c_str()); } CommandLine command_line; std::wstring path = command_line.GetSwitchValue(switches::kPluginPath); delegate_ = WebPluginDelegateImpl::Create( path, mime_type_, params.containing_window); if (delegate_) { webplugin_ = new WebPluginProxy( channel_, instance_id_, delegate_, params.modal_dialog_event); *result = delegate_->Initialize( params.url, argn, argv, argc, webplugin_, params.load_manually); } delete[] argn; delete[] argv; } void WebPluginDelegateStub::OnWillSendRequest(int id, const GURL& url) { WebPluginResourceClient* client = webplugin_->GetResourceClient(id); if (!client) return; client->WillSendRequest(url); } void WebPluginDelegateStub::OnDidReceiveResponse( const PluginMsg_DidReceiveResponseParams& params, bool* cancel) { *cancel = false; WebPluginResourceClient* client = webplugin_->GetResourceClient(params.id); if (!client) return; client->DidReceiveResponse(params.mime_type, params.headers, params.expected_length, params.last_modified, cancel); } void WebPluginDelegateStub::OnDidReceiveData(int id, const std::vector& buffer) { WebPluginResourceClient* client = webplugin_->GetResourceClient(id); if (!client) return; client->DidReceiveData(&buffer.front(), static_cast(buffer.size())); } void WebPluginDelegateStub::OnDidFinishLoading(int id) { WebPluginResourceClient* client = webplugin_->GetResourceClient(id); if (!client) return; client->DidFinishLoading(); } void WebPluginDelegateStub::OnDidFail(int id) { WebPluginResourceClient* client = webplugin_->GetResourceClient(id); if (!client) return; client->DidFail(); } void WebPluginDelegateStub::OnDidFinishLoadWithReason(int reason) { delegate_->DidFinishLoadWithReason(reason); } void WebPluginDelegateStub::OnSetFocus() { delegate_->SetFocus(); } void WebPluginDelegateStub::OnHandleEvent(const NPEvent& event, bool* handled, WebCursor* cursor) { *handled = delegate_->HandleEvent(const_cast(&event), cursor); } void WebPluginDelegateStub::OnPaint(const PluginMsg_Paint_Params& params) { // Convert the shared memory handle to a handle that works in our process, // and then use that to create an HDC. win_util::ScopedHandle shared_section(win_util::GetSectionFromProcess( params.shared_memory, channel_->renderer_handle(), false)); if (shared_section == NULL) { NOTREACHED(); return; } void* data = NULL; HDC screen_dc = GetDC(NULL); BITMAPINFOHEADER bitmap_header; gfx::CreateBitmapHeader(params.size.width(), params.size.height(), &bitmap_header); win_util::ScopedBitmap hbitmap(CreateDIBSection( screen_dc, reinterpret_cast(&bitmap_header), DIB_RGB_COLORS, &data, shared_section, 0)); ReleaseDC(NULL, screen_dc); if (hbitmap == NULL) { NOTREACHED(); return; } win_util::ScopedHDC hdc(CreateCompatibleDC(NULL)); if (hdc == NULL) { NOTREACHED(); return; } gfx::PlatformDeviceWin::InitializeDC(hdc); SelectObject(hdc, hbitmap); SetWorldTransform(hdc, ¶ms.xf); win_util::ScopedHRGN hrgn(CreateRectRgnIndirect(¶ms.clip_rect.ToRECT())); SelectClipRgn(hdc, hrgn); webplugin_->WillPaint(); delegate_->Paint(hdc, params.damaged_rect); } void WebPluginDelegateStub::OnPrint(PluginMsg_PrintResponse_Params* params) { gfx::Emf emf; if (!emf.CreateDc(NULL, NULL)) { NOTREACHED(); return; } HDC hdc = emf.hdc(); gfx::PlatformDeviceWin::InitializeDC(hdc); delegate_->Print(hdc); if (!emf.CloseDc()) { NOTREACHED(); return; } size_t size = emf.GetDataSize(); DCHECK(size); params->size = size; SharedMemory shared_buf; CreateSharedBuffer(size, &shared_buf, ¶ms->shared_memory); // Retrieve a copy of the data. bool success = emf.GetData(shared_buf.memory(), size); DCHECK(success); } void WebPluginDelegateStub::OnPaintIntoSharedMemory( const PluginMsg_Paint_Params& params, SharedMemoryHandle* emf_buffer, size_t* bytes) { *emf_buffer = NULL; *bytes = 0; gfx::Emf emf; if (!emf.CreateDc(NULL, NULL)) { NOTREACHED(); return; } HDC hdc = emf.hdc(); gfx::PlatformDeviceWin::InitializeDC(hdc); if (delegate_->windowless()) { WindowlessPaint(hdc, params); } else { WindowedPaint(hdc, params.damaged_rect); } // Need to send back the data as shared memory. if (!emf.CloseDc()) { NOTREACHED(); return; } size_t size = emf.GetDataSize(); DCHECK(size); *bytes = size; SharedMemory shared_buf; CreateSharedBuffer(size, &shared_buf, emf_buffer); // Retrieve a copy of the data. bool success = emf.GetData(shared_buf.memory(), size); DCHECK(success); } void WebPluginDelegateStub::WindowedPaint(HDC hdc, const gfx::Rect& window_rect) { // Use the NPAPI print() function to render the plugin. delegate_->Print(hdc); } void WebPluginDelegateStub::WindowlessPaint( HDC hdc, const PluginMsg_Paint_Params& params) { void* data = NULL; HDC screen_dc = GetDC(NULL); BITMAPINFOHEADER bitmap_header; gfx::CreateBitmapHeader(params.size.width(), params.size.height(), &bitmap_header); win_util::ScopedBitmap hbitmap(CreateDIBSection( screen_dc, reinterpret_cast(&bitmap_header), DIB_RGB_COLORS, &data, NULL, 0)); ReleaseDC(NULL, screen_dc); if (hbitmap == NULL) { NOTREACHED(); return; } SelectObject(hdc, hbitmap); // Apply transform and clipping. SetWorldTransform(hdc, ¶ms.xf); win_util::ScopedHRGN hrgn(CreateRectRgnIndirect(¶ms.clip_rect.ToRECT())); SelectClipRgn(hdc, hrgn); webplugin_->WillPaint(); delegate_->Paint(hdc, params.damaged_rect); } void WebPluginDelegateStub::OnUpdateGeometry(const gfx::Rect& window_rect, const gfx::Rect& clip_rect, bool visible) { delegate_->UpdateGeometry(window_rect, clip_rect, visible); } void WebPluginDelegateStub::OnGetPluginScriptableObject(int* route_id, void** npobject_ptr) { NPObject* object = delegate_->GetPluginScriptableObject(); if (!object) { *route_id = MSG_ROUTING_NONE; return; } *route_id = channel_->GenerateRouteID(); *npobject_ptr = object; // The stub will delete itself when the proxy tells it that it's released, or // otherwise when the channel is closed. NPObjectStub* stub = new NPObjectStub(object, channel_.get(), *route_id); // Release ref added by GetPluginScriptableObject (our stub holds its own). NPN_ReleaseObject(object); } void WebPluginDelegateStub::OnSendJavaScriptStream(const std::string& url, const std::wstring& result, bool success, bool notify_needed, int notify_data) { delegate_->SendJavaScriptStream(url, result, success, notify_needed, notify_data); } void WebPluginDelegateStub::OnDidReceiveManualResponse( const std::string& url, const PluginMsg_DidReceiveResponseParams& params) { delegate_->DidReceiveManualResponse(url, params.mime_type, params.headers, params.expected_length, params.last_modified); } void WebPluginDelegateStub::OnDidReceiveManualData( const std::vector& buffer) { delegate_->DidReceiveManualData(&buffer.front(), static_cast(buffer.size())); } void WebPluginDelegateStub::OnDidFinishManualLoading() { delegate_->DidFinishManualLoading(); } void WebPluginDelegateStub::OnDidManualLoadFail() { delegate_->DidManualLoadFail(); } void WebPluginDelegateStub::OnInstallMissingPlugin() { delegate_->InstallMissingPlugin(); } void WebPluginDelegateStub::CreateSharedBuffer( size_t size, SharedMemory* shared_buf, SharedMemoryHandle* remote_handle) { if (!shared_buf->Create(std::wstring(), false, false, size)) { NOTREACHED(); return; } if (!shared_buf->Map(size)) { NOTREACHED(); shared_buf->Close(); return; } BOOL result = DuplicateHandle(GetCurrentProcess(), shared_buf->handle(), channel_->renderer_handle(), remote_handle, 0, FALSE, DUPLICATE_SAME_ACCESS); DCHECK_NE(result, 0); // If the calling function's shared_buf is on the stack, its destructor will // close the shared memory buffer handle. This is fine since we already // duplicated the handle to the renderer process so it will stay "alive". } void WebPluginDelegateStub::OnHandleURLRequestReply( const PluginMsg_URLRequestReply_Params& params) { WebPluginResourceClient* resource_client = delegate_->CreateResourceClient(params.resource_id, params.url, params.notify_needed, params.notify_data); webplugin_->OnResourceCreated(params.resource_id, resource_client); } void WebPluginDelegateStub::OnURLRequestRouted(const std::string& url, bool notify_needed, HANDLE notify_data) { delegate_->URLRequestRouted(url, notify_needed, notify_data); }