| // 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 "content/renderer/webplugin_delegate_proxy.h" |
| |
| #if defined(TOOLKIT_GTK) |
| #include <gtk/gtk.h> |
| #elif defined(USE_X11) |
| #include <cairo/cairo.h> |
| #endif |
| |
| #include <algorithm> |
| |
| #include "base/auto_reset.h" |
| #include "base/basictypes.h" |
| #include "base/command_line.h" |
| #include "base/file_util.h" |
| #include "base/logging.h" |
| #include "base/memory/ref_counted.h" |
| #include "base/memory/scoped_ptr.h" |
| #include "base/process.h" |
| #include "base/string_split.h" |
| #include "base/string_util.h" |
| #include "base/utf_string_conversions.h" |
| #include "base/version.h" |
| #include "content/common/child_process.h" |
| #include "content/common/npobject_proxy.h" |
| #include "content/common/npobject_stub.h" |
| #include "content/common/npobject_util.h" |
| #include "content/common/plugin_messages.h" |
| #include "content/common/view_messages.h" |
| #include "content/public/renderer/content_renderer_client.h" |
| #include "content/renderer/plugin_channel_host.h" |
| #include "content/renderer/render_thread_impl.h" |
| #include "content/renderer/render_view_impl.h" |
| #include "ipc/ipc_channel_handle.h" |
| #include "net/base/mime_util.h" |
| #include "skia/ext/platform_canvas.h" |
| #include "third_party/WebKit/Source/WebKit/chromium/public/WebBindings.h" |
| #include "third_party/WebKit/Source/WebKit/chromium/public/WebCursorInfo.h" |
| #include "third_party/WebKit/Source/WebKit/chromium/public/WebDocument.h" |
| #include "third_party/WebKit/Source/WebKit/chromium/public/WebFrame.h" |
| #include "third_party/WebKit/Source/WebKit/chromium/public/WebView.h" |
| #include "third_party/WebKit/Source/WebKit/chromium/public/platform/WebDragData.h" |
| #include "third_party/WebKit/Source/WebKit/chromium/public/platform/WebString.h" |
| #include "ui/gfx/blit.h" |
| #include "ui/gfx/canvas.h" |
| #include "ui/gfx/native_widget_types.h" |
| #include "ui/gfx/size.h" |
| #include "webkit/glue/webkit_glue.h" |
| #include "webkit/plugins/npapi/plugin_group.h" |
| #include "webkit/plugins/npapi/webplugin.h" |
| #include "webkit/plugins/plugin_constants.h" |
| #include "webkit/plugins/sad_plugin.h" |
| |
| #if defined(OS_POSIX) |
| #include "ipc/ipc_channel_posix.h" |
| #endif |
| |
| #if defined(OS_MACOSX) |
| #include "base/mac/mac_util.h" |
| #endif |
| |
| #if defined(OS_WIN) |
| #include "content/public/common/sandbox_init.h" |
| #endif |
| |
| using WebKit::WebBindings; |
| using WebKit::WebCursorInfo; |
| using WebKit::WebDragData; |
| using WebKit::WebInputEvent; |
| using WebKit::WebString; |
| using WebKit::WebView; |
| |
| // Proxy for WebPluginResourceClient. The object owns itself after creation, |
| // deleting itself after its callback has been called. |
| class ResourceClientProxy : public webkit::npapi::WebPluginResourceClient { |
| public: |
| ResourceClientProxy(PluginChannelHost* channel, int instance_id) |
| : channel_(channel), instance_id_(instance_id), resource_id_(0), |
| multibyte_response_expected_(false) { |
| } |
| |
| ~ResourceClientProxy() { |
| } |
| |
| void Initialize(unsigned long resource_id, const GURL& url, int notify_id) { |
| resource_id_ = resource_id; |
| channel_->Send(new PluginMsg_HandleURLRequestReply( |
| instance_id_, resource_id, url, notify_id)); |
| } |
| |
| void InitializeForSeekableStream(unsigned long resource_id, |
| int range_request_id) { |
| resource_id_ = resource_id; |
| multibyte_response_expected_ = true; |
| channel_->Send(new PluginMsg_HTTPRangeRequestReply( |
| instance_id_, resource_id, range_request_id)); |
| } |
| |
| // PluginResourceClient implementation: |
| void WillSendRequest(const GURL& url, int http_status_code) { |
| DCHECK(channel_ != NULL); |
| channel_->Send(new PluginMsg_WillSendRequest(instance_id_, resource_id_, |
| url, http_status_code)); |
| } |
| |
| void DidReceiveResponse(const std::string& mime_type, |
| const std::string& headers, |
| uint32 expected_length, |
| uint32 last_modified, |
| bool request_is_seekable) { |
| DCHECK(channel_ != NULL); |
| PluginMsg_DidReceiveResponseParams params; |
| params.id = resource_id_; |
| params.mime_type = mime_type; |
| params.headers = headers; |
| params.expected_length = expected_length; |
| params.last_modified = last_modified; |
| params.request_is_seekable = request_is_seekable; |
| // Grab a reference on the underlying channel so it does not get |
| // deleted from under us. |
| scoped_refptr<PluginChannelHost> channel_ref(channel_); |
| channel_->Send(new PluginMsg_DidReceiveResponse(instance_id_, params)); |
| } |
| |
| void DidReceiveData(const char* buffer, int length, int data_offset) { |
| DCHECK(channel_ != NULL); |
| DCHECK_GT(length, 0); |
| std::vector<char> data; |
| data.resize(static_cast<size_t>(length)); |
| memcpy(&data.front(), buffer, length); |
| // Grab a reference on the underlying channel so it does not get |
| // deleted from under us. |
| scoped_refptr<PluginChannelHost> channel_ref(channel_); |
| channel_->Send(new PluginMsg_DidReceiveData(instance_id_, resource_id_, |
| data, data_offset)); |
| } |
| |
| void DidFinishLoading() { |
| DCHECK(channel_ != NULL); |
| channel_->Send(new PluginMsg_DidFinishLoading(instance_id_, resource_id_)); |
| channel_ = NULL; |
| MessageLoop::current()->DeleteSoon(FROM_HERE, this); |
| } |
| |
| void DidFail() { |
| DCHECK(channel_ != NULL); |
| channel_->Send(new PluginMsg_DidFail(instance_id_, resource_id_)); |
| channel_ = NULL; |
| MessageLoop::current()->DeleteSoon(FROM_HERE, this); |
| } |
| |
| bool IsMultiByteResponseExpected() { |
| return multibyte_response_expected_; |
| } |
| |
| int ResourceId() { |
| return resource_id_; |
| } |
| |
| private: |
| scoped_refptr<PluginChannelHost> channel_; |
| int instance_id_; |
| unsigned long resource_id_; |
| // Set to true if the response expected is a multibyte response. |
| // For e.g. response for a HTTP byte range request. |
| bool multibyte_response_expected_; |
| }; |
| |
| WebPluginDelegateProxy::WebPluginDelegateProxy( |
| const std::string& mime_type, |
| const base::WeakPtr<RenderViewImpl>& render_view) |
| : render_view_(render_view), |
| plugin_(NULL), |
| uses_shared_bitmaps_(false), |
| #if defined(OS_MACOSX) |
| uses_compositor_(false), |
| #endif |
| window_(gfx::kNullPluginWindow), |
| mime_type_(mime_type), |
| instance_id_(MSG_ROUTING_NONE), |
| npobject_(NULL), |
| sad_plugin_(NULL), |
| invalidate_pending_(false), |
| transparent_(false), |
| front_buffer_index_(0), |
| page_url_(render_view_->webview()->mainFrame()->document().url()) { |
| } |
| |
| WebPluginDelegateProxy::~WebPluginDelegateProxy() { |
| if (npobject_) |
| WebBindings::releaseObject(npobject_); |
| } |
| |
| WebPluginDelegateProxy::SharedBitmap::SharedBitmap() {} |
| |
| WebPluginDelegateProxy::SharedBitmap::~SharedBitmap() {} |
| |
| void WebPluginDelegateProxy::PluginDestroyed() { |
| #if defined(OS_MACOSX) || defined(OS_WIN) |
| // Ensure that the renderer doesn't think the plugin still has focus. |
| if (render_view_) |
| render_view_->PluginFocusChanged(false, instance_id_); |
| #endif |
| |
| if (window_) |
| WillDestroyWindow(); |
| |
| if (render_view_) |
| render_view_->UnregisterPluginDelegate(this); |
| |
| if (channel_host_) { |
| Send(new PluginMsg_DestroyInstance(instance_id_)); |
| |
| // Must remove the route after sending the destroy message, since |
| // RemoveRoute can lead to all the outstanding NPObjects being told the |
| // channel went away if this was the last instance. |
| channel_host_->RemoveRoute(instance_id_); |
| |
| // Release the channel host now. If we are is the last reference to the |
| // channel, this avoids a race where this renderer asks a new connection to |
| // the same plugin between now and the time 'this' is actually deleted. |
| // Destroying the channel host is what releases the channel name -> FD |
| // association on POSIX, and if we ask for a new connection before it is |
| // released, the plugin will give us a new FD, and we'll assert when trying |
| // to associate it with the channel name. |
| channel_host_ = NULL; |
| } |
| |
| if (window_script_object_) { |
| // Release the window script object, if the plugin didn't already. |
| // If we don't do this then it will linger until the last plugin instance is |
| // destroyed. In the meantime, though, the frame that it refers to may have |
| // been destroyed by WebKit, at which point WebKit will forcibly deallocate |
| // the window script object. The window script object stub is unique to the |
| // plugin instance, so this won't affect other instances. |
| window_script_object_->DeleteSoon(); |
| } |
| |
| plugin_ = NULL; |
| |
| MessageLoop::current()->DeleteSoon(FROM_HERE, this); |
| } |
| |
| // Returns true if the given Silverlight 'background' value corresponds to |
| // one that should make the plugin transparent. See: |
| // http://msdn.microsoft.com/en-us/library/cc838148(VS.95).aspx |
| // for possible values. |
| static bool SilverlightColorIsTransparent(const std::string& color) { |
| if (StartsWithASCII(color, "#", false)) { |
| // If it's #ARGB or #AARRGGBB check the alpha; if not it's an RGB form and |
| // it's not transparent. |
| if ((color.length() == 5 && !StartsWithASCII(color, "#F", false)) || |
| (color.length() == 9 && !StartsWithASCII(color, "#FF", false))) |
| return true; |
| } else if (StartsWithASCII(color, "sc#", false)) { |
| // It's either sc#A,R,G,B or sc#R,G,B; if the former, check the alpha. |
| if (color.length() < 4) |
| return false; |
| std::string value_string = color.substr(3, std::string::npos); |
| std::vector<std::string> components; |
| base::SplitString(value_string, ',', &components); |
| if (components.size() == 4 && !StartsWithASCII(components[0], "1", false)) |
| return true; |
| } else if (LowerCaseEqualsASCII(color, "transparent")) { |
| return true; |
| } |
| // Anything else is a named, opaque color or an RGB form with no alpha. |
| return false; |
| } |
| |
| bool WebPluginDelegateProxy::Initialize( |
| const GURL& url, |
| const std::vector<std::string>& arg_names, |
| const std::vector<std::string>& arg_values, |
| webkit::npapi::WebPlugin* plugin, |
| bool load_manually) { |
| #if defined(OS_MACOSX) |
| // TODO(shess): Debugging for http://crbug.com/97285 . See comment |
| // in plugin_channel_host.cc. |
| scoped_ptr<AutoReset<bool> > track_nested_removes(new AutoReset<bool>( |
| PluginChannelHost::GetRemoveTrackingFlag(), true)); |
| #endif |
| |
| IPC::ChannelHandle channel_handle; |
| if (!RenderThreadImpl::current()->Send(new ViewHostMsg_OpenChannelToPlugin( |
| render_view_->routing_id(), url, page_url_, mime_type_, |
| &channel_handle, &info_))) { |
| return false; |
| } |
| |
| if (channel_handle.name.empty()) { |
| // We got an invalid handle. Either the plugin couldn't be found (which |
| // shouldn't happen, since if we got here the plugin should exist) or the |
| // plugin crashed on initialization. |
| if (!info_.path.empty()) { |
| render_view_->PluginCrashed(info_.path); |
| LOG(ERROR) << "Plug-in crashed on start"; |
| |
| // Return true so that the plugin widget is created and we can paint the |
| // crashed plugin there. |
| return true; |
| } |
| LOG(ERROR) << "Plug-in couldn't be found"; |
| return false; |
| } |
| |
| scoped_refptr<PluginChannelHost> channel_host( |
| PluginChannelHost::GetPluginChannelHost( |
| channel_handle, ChildProcess::current()->io_message_loop_proxy())); |
| if (!channel_host.get()) { |
| LOG(ERROR) << "Couldn't get PluginChannelHost"; |
| return false; |
| } |
| #if defined(OS_MACOSX) |
| track_nested_removes.reset(); |
| #endif |
| |
| int instance_id; |
| bool result = channel_host->Send(new PluginMsg_CreateInstance( |
| mime_type_, &instance_id)); |
| if (!result) { |
| LOG(ERROR) << "Couldn't send PluginMsg_CreateInstance"; |
| return false; |
| } |
| |
| channel_host_ = channel_host; |
| instance_id_ = instance_id; |
| |
| channel_host_->AddRoute(instance_id_, this, NULL); |
| |
| // Now tell the PluginInstance in the plugin process to initialize. |
| PluginMsg_Init_Params params; |
| params.containing_window = render_view_->host_window(); |
| params.url = url; |
| params.page_url = page_url_; |
| params.arg_names = arg_names; |
| params.arg_values = arg_values; |
| params.host_render_view_routing_id = render_view_->routing_id(); |
| |
| bool flash = |
| LowerCaseEqualsASCII(mime_type_, "application/x-shockwave-flash"); |
| bool silverlight = |
| StartsWithASCII(mime_type_, "application/x-silverlight", false); |
| for (size_t i = 0; i < arg_names.size(); ++i) { |
| if ((flash && LowerCaseEqualsASCII(arg_names[i], "wmode") && |
| LowerCaseEqualsASCII(arg_values[i], "transparent")) || |
| (silverlight && LowerCaseEqualsASCII(arg_names[i], "background") && |
| SilverlightColorIsTransparent(arg_values[i]))) { |
| transparent_ = true; |
| } |
| } |
| params.load_manually = load_manually; |
| |
| plugin_ = plugin; |
| |
| result = false; |
| IPC::Message* msg = new PluginMsg_Init(instance_id_, params, &result); |
| Send(msg); |
| |
| if (!result) |
| LOG(ERROR) << "PluginMsg_Init returned false"; |
| |
| render_view_->RegisterPluginDelegate(this); |
| |
| return result; |
| } |
| |
| bool WebPluginDelegateProxy::Send(IPC::Message* msg) { |
| if (!channel_host_) { |
| DLOG(WARNING) << "dropping message because channel host is null"; |
| delete msg; |
| return false; |
| } |
| |
| return channel_host_->Send(msg); |
| } |
| |
| void WebPluginDelegateProxy::SendJavaScriptStream(const GURL& url, |
| const std::string& result, |
| bool success, |
| int notify_id) { |
| Send(new PluginMsg_SendJavaScriptStream( |
| instance_id_, url, result, success, notify_id)); |
| } |
| |
| void WebPluginDelegateProxy::DidReceiveManualResponse( |
| const GURL& url, const std::string& mime_type, |
| const std::string& headers, uint32 expected_length, |
| uint32 last_modified) { |
| PluginMsg_DidReceiveResponseParams params; |
| params.id = 0; |
| params.mime_type = mime_type; |
| params.headers = headers; |
| params.expected_length = expected_length; |
| params.last_modified = last_modified; |
| Send(new PluginMsg_DidReceiveManualResponse(instance_id_, url, params)); |
| } |
| |
| void WebPluginDelegateProxy::DidReceiveManualData(const char* buffer, |
| int length) { |
| DCHECK_GT(length, 0); |
| std::vector<char> data; |
| data.resize(static_cast<size_t>(length)); |
| memcpy(&data.front(), buffer, length); |
| Send(new PluginMsg_DidReceiveManualData(instance_id_, data)); |
| } |
| |
| void WebPluginDelegateProxy::DidFinishManualLoading() { |
| Send(new PluginMsg_DidFinishManualLoading(instance_id_)); |
| } |
| |
| void WebPluginDelegateProxy::DidManualLoadFail() { |
| Send(new PluginMsg_DidManualLoadFail(instance_id_)); |
| } |
| |
| bool WebPluginDelegateProxy::OnMessageReceived(const IPC::Message& msg) { |
| content::GetContentClient()->SetActiveURL(page_url_); |
| |
| bool handled = true; |
| IPC_BEGIN_MESSAGE_MAP(WebPluginDelegateProxy, msg) |
| IPC_MESSAGE_HANDLER(PluginHostMsg_SetWindow, OnSetWindow) |
| #if defined(OS_WIN) |
| IPC_MESSAGE_HANDLER(PluginHostMsg_SetWindowlessPumpEvent, |
| OnSetWindowlessPumpEvent) |
| IPC_MESSAGE_HANDLER(PluginHostMsg_NotifyIMEStatus, |
| OnNotifyIMEStatus) |
| #endif |
| IPC_MESSAGE_HANDLER(PluginHostMsg_CancelResource, OnCancelResource) |
| IPC_MESSAGE_HANDLER(PluginHostMsg_InvalidateRect, OnInvalidateRect) |
| IPC_MESSAGE_HANDLER(PluginHostMsg_GetWindowScriptNPObject, |
| OnGetWindowScriptNPObject) |
| IPC_MESSAGE_HANDLER(PluginHostMsg_GetPluginElement, |
| OnGetPluginElement) |
| IPC_MESSAGE_HANDLER(PluginHostMsg_ResolveProxy, OnResolveProxy) |
| IPC_MESSAGE_HANDLER(PluginHostMsg_SetCookie, OnSetCookie) |
| IPC_MESSAGE_HANDLER(PluginHostMsg_GetCookies, OnGetCookies) |
| IPC_MESSAGE_HANDLER(PluginHostMsg_URLRequest, OnHandleURLRequest) |
| IPC_MESSAGE_HANDLER(PluginHostMsg_CancelDocumentLoad, OnCancelDocumentLoad) |
| IPC_MESSAGE_HANDLER(PluginHostMsg_InitiateHTTPRangeRequest, |
| OnInitiateHTTPRangeRequest) |
| IPC_MESSAGE_HANDLER(PluginHostMsg_DeferResourceLoading, |
| OnDeferResourceLoading) |
| |
| #if defined(OS_MACOSX) |
| IPC_MESSAGE_HANDLER(PluginHostMsg_FocusChanged, |
| OnFocusChanged); |
| IPC_MESSAGE_HANDLER(PluginHostMsg_StartIme, |
| OnStartIme); |
| IPC_MESSAGE_HANDLER(PluginHostMsg_BindFakePluginWindowHandle, |
| OnBindFakePluginWindowHandle); |
| // Used only on 10.6 and later. |
| IPC_MESSAGE_HANDLER(PluginHostMsg_AcceleratedSurfaceSetIOSurface, |
| OnAcceleratedSurfaceSetIOSurface) |
| // Used on 10.5 and earlier. |
| IPC_MESSAGE_HANDLER(PluginHostMsg_AcceleratedSurfaceSetTransportDIB, |
| OnAcceleratedSurfaceSetTransportDIB) |
| IPC_MESSAGE_HANDLER(PluginHostMsg_AllocTransportDIB, |
| OnAcceleratedSurfaceAllocTransportDIB) |
| IPC_MESSAGE_HANDLER(PluginHostMsg_FreeTransportDIB, |
| OnAcceleratedSurfaceFreeTransportDIB) |
| IPC_MESSAGE_HANDLER(PluginHostMsg_AcceleratedSurfaceBuffersSwapped, |
| OnAcceleratedSurfaceBuffersSwapped) |
| // Used only on 10.6 and later. |
| IPC_MESSAGE_HANDLER(PluginHostMsg_AcceleratedPluginEnabledRendering, |
| OnAcceleratedPluginEnabledRendering) |
| IPC_MESSAGE_HANDLER(PluginHostMsg_AcceleratedPluginAllocatedIOSurface, |
| OnAcceleratedPluginAllocatedIOSurface) |
| IPC_MESSAGE_HANDLER(PluginHostMsg_AcceleratedPluginSwappedIOSurface, |
| OnAcceleratedPluginSwappedIOSurface) |
| #endif |
| IPC_MESSAGE_HANDLER(PluginHostMsg_URLRedirectResponse, |
| OnURLRedirectResponse) |
| IPC_MESSAGE_UNHANDLED(handled = false) |
| IPC_END_MESSAGE_MAP() |
| DCHECK(handled); |
| return handled; |
| } |
| |
| void WebPluginDelegateProxy::OnChannelError() { |
| if (plugin_) { |
| if (window_) { |
| // The actual WebPluginDelegate never got a chance to tell the WebPlugin |
| // its window was going away. Do it on its behalf. |
| WillDestroyWindow(); |
| } |
| plugin_->Invalidate(); |
| } |
| if (!channel_host_->expecting_shutdown()) |
| render_view_->PluginCrashed(info_.path); |
| |
| #if defined(OS_MACOSX) || defined(OS_WIN) |
| // Ensure that the renderer doesn't think the plugin still has focus. |
| if (render_view_) |
| render_view_->PluginFocusChanged(false, instance_id_); |
| #endif |
| } |
| |
| static void CopyTransportDIBHandleForMessage( |
| const TransportDIB::Handle& handle_in, |
| TransportDIB::Handle* handle_out, |
| base::ProcessId peer_pid) { |
| #if defined(OS_MACOSX) |
| // On Mac, TransportDIB::Handle is typedef'ed to FileDescriptor, and |
| // FileDescriptor message fields needs to remain valid until the message is |
| // sent or else the sendmsg() call will fail. |
| if ((handle_out->fd = HANDLE_EINTR(dup(handle_in.fd))) < 0) { |
| PLOG(ERROR) << "dup()"; |
| return; |
| } |
| handle_out->auto_close = true; |
| #elif defined(OS_WIN) |
| // On Windows we need to duplicate the handle for the plugin process. |
| *handle_out = NULL; |
| content::BrokerDuplicateHandle(handle_in, peer_pid, handle_out, |
| FILE_MAP_READ | FILE_MAP_WRITE, 0); |
| DCHECK(*handle_out != NULL); |
| #else |
| // Don't need to do anything special for other platforms. |
| *handle_out = handle_in; |
| #endif |
| } |
| |
| void WebPluginDelegateProxy::SendUpdateGeometry( |
| bool bitmaps_changed) { |
| PluginMsg_UpdateGeometry_Param param; |
| param.window_rect = plugin_rect_; |
| param.clip_rect = clip_rect_; |
| param.windowless_buffer0 = TransportDIB::DefaultHandleValue(); |
| param.windowless_buffer1 = TransportDIB::DefaultHandleValue(); |
| param.windowless_buffer_index = back_buffer_index(); |
| param.background_buffer = TransportDIB::DefaultHandleValue(); |
| param.transparent = transparent_; |
| |
| #if defined(OS_POSIX) |
| // If we're using POSIX mmap'd TransportDIBs, sending the handle across |
| // IPC establishes a new mapping rather than just sending a window ID, |
| // so only do so if we've actually changed the shared memory bitmaps. |
| if (bitmaps_changed) |
| #endif |
| { |
| if (transport_stores_[0].dib.get()) |
| CopyTransportDIBHandleForMessage(transport_stores_[0].dib->handle(), |
| ¶m.windowless_buffer0, |
| channel_host_->peer_pid()); |
| |
| if (transport_stores_[1].dib.get()) |
| CopyTransportDIBHandleForMessage(transport_stores_[1].dib->handle(), |
| ¶m.windowless_buffer1, |
| channel_host_->peer_pid()); |
| |
| if (background_store_.dib.get()) |
| CopyTransportDIBHandleForMessage(background_store_.dib->handle(), |
| ¶m.background_buffer, |
| channel_host_->peer_pid()); |
| } |
| |
| IPC::Message* msg; |
| #if defined(OS_WIN) |
| if (UseSynchronousGeometryUpdates()) { |
| msg = new PluginMsg_UpdateGeometrySync(instance_id_, param); |
| } else // NOLINT |
| #endif |
| { |
| msg = new PluginMsg_UpdateGeometry(instance_id_, param); |
| msg->set_unblock(true); |
| } |
| |
| Send(msg); |
| } |
| |
| void WebPluginDelegateProxy::UpdateGeometry(const gfx::Rect& window_rect, |
| const gfx::Rect& clip_rect) { |
| // window_rect becomes either a window in native windowing system |
| // coords, or a backing buffer. In either case things will go bad |
| // if the rectangle is very large. |
| if (window_rect.width() < 0 || window_rect.width() > kMaxPluginSideLength || |
| window_rect.height() < 0 || window_rect.height() > kMaxPluginSideLength || |
| // We know this won't overflow due to above checks. |
| static_cast<uint32>(window_rect.width()) * |
| static_cast<uint32>(window_rect.height()) > kMaxPluginSize) { |
| return; |
| } |
| |
| plugin_rect_ = window_rect; |
| clip_rect_ = clip_rect; |
| |
| bool bitmaps_changed = false; |
| |
| if (uses_shared_bitmaps_) { |
| if (!front_buffer_canvas() || |
| (window_rect.width() != front_buffer_canvas()->getDevice()->width() || |
| window_rect.height() != front_buffer_canvas()->getDevice()->height())) |
| { |
| bitmaps_changed = true; |
| |
| bool needs_background_store = transparent_; |
| #if defined(OS_MACOSX) |
| // We don't support transparency under QuickDraw, and CoreGraphics |
| // preserves transparency information (and does the compositing itself) |
| // so plugins don't need access to the page background. |
| needs_background_store = false; |
| #endif |
| |
| // Create a shared memory section that the plugin paints into |
| // asynchronously. |
| ResetWindowlessBitmaps(); |
| if (!window_rect.IsEmpty()) { |
| if (!CreateSharedBitmap(&transport_stores_[0].dib, |
| &transport_stores_[0].canvas) || |
| !CreateSharedBitmap(&transport_stores_[1].dib, |
| &transport_stores_[1].canvas) || |
| (needs_background_store && |
| !CreateSharedBitmap(&background_store_.dib, |
| &background_store_.canvas))) { |
| DCHECK(false); |
| ResetWindowlessBitmaps(); |
| return; |
| } |
| } |
| } |
| } |
| |
| SendUpdateGeometry(bitmaps_changed); |
| } |
| |
| void WebPluginDelegateProxy::ResetWindowlessBitmaps() { |
| transport_stores_[0].dib.reset(); |
| transport_stores_[1].dib.reset(); |
| background_store_.dib.reset(); |
| |
| transport_stores_[0].canvas.reset(); |
| transport_stores_[1].canvas.reset(); |
| background_store_.canvas.reset(); |
| transport_store_painted_ = gfx::Rect(); |
| front_buffer_diff_ = gfx::Rect(); |
| } |
| |
| static size_t BitmapSizeForPluginRect(const gfx::Rect& plugin_rect) { |
| const size_t stride = |
| skia::PlatformCanvas::StrideForWidth(plugin_rect.width()); |
| return stride * plugin_rect.height(); |
| } |
| |
| #if !defined(OS_WIN) |
| bool WebPluginDelegateProxy::CreateLocalBitmap( |
| std::vector<uint8>* memory, |
| scoped_ptr<skia::PlatformCanvas>* canvas) { |
| const size_t size = BitmapSizeForPluginRect(plugin_rect_); |
| memory->resize(size); |
| if (memory->size() != size) |
| return false; |
| canvas->reset(new skia::PlatformCanvas( |
| plugin_rect_.width(), plugin_rect_.height(), true, &((*memory)[0]))); |
| return true; |
| } |
| #endif |
| |
| bool WebPluginDelegateProxy::CreateSharedBitmap( |
| scoped_ptr<TransportDIB>* memory, |
| scoped_ptr<skia::PlatformCanvas>* canvas) { |
| const size_t size = BitmapSizeForPluginRect(plugin_rect_); |
| #if defined(OS_POSIX) && !defined(OS_MACOSX) |
| memory->reset(TransportDIB::Create(size, 0)); |
| if (!memory->get()) |
| return false; |
| #endif |
| #if defined(OS_MACOSX) |
| TransportDIB::Handle handle; |
| IPC::Message* msg = new ViewHostMsg_AllocTransportDIB(size, false, &handle); |
| if (!RenderThreadImpl::current()->Send(msg)) |
| return false; |
| if (handle.fd < 0) |
| return false; |
| memory->reset(TransportDIB::Map(handle)); |
| #else |
| static uint32 sequence_number = 0; |
| memory->reset(TransportDIB::Create(size, sequence_number++)); |
| #endif |
| canvas->reset((*memory)->GetPlatformCanvas(plugin_rect_.width(), |
| plugin_rect_.height())); |
| return !!canvas->get(); |
| } |
| |
| #if defined(OS_MACOSX) |
| // Flips |rect| vertically within an enclosing rect with height |height|. |
| // Intended for converting rects between flipped and non-flipped contexts. |
| static void FlipRectVerticallyWithHeight(gfx::Rect* rect, int height) { |
| rect->set_y(height - rect->bottom()); |
| } |
| #endif |
| |
| void WebPluginDelegateProxy::Paint(WebKit::WebCanvas* canvas, |
| const gfx::Rect& damaged_rect) { |
| // Limit the damaged rectangle to whatever is contained inside the plugin |
| // rectangle, as that's the rectangle that we'll actually draw. |
| gfx::Rect rect = damaged_rect.Intersect(plugin_rect_); |
| |
| // If the plugin is no longer connected (channel crashed) draw a crashed |
| // plugin bitmap |
| if (!channel_host_ || !channel_host_->channel_valid()) { |
| PaintSadPlugin(canvas, rect); |
| return; |
| } |
| |
| if (!uses_shared_bitmaps_) |
| return; |
| |
| // We got a paint before the plugin's coordinates, so there's no buffer to |
| // copy from. |
| if (!front_buffer_canvas()) |
| return; |
| |
| // We're using the native OS APIs from here on out. |
| if (!skia::SupportsPlatformPaint(canvas)) { |
| // TODO(alokp): Implement this path. |
| // This block will only get hit with --enable-accelerated-drawing flag. |
| // With accelerated canvas, we do not have a bitmap that can be provided |
| // to the plugin for compositing. We may have to implement a solution |
| // described in crbug.com/12586. |
| DLOG(WARNING) << "Could not paint plugin"; |
| return; |
| } |
| skia::ScopedPlatformPaint scoped_platform_paint(canvas); |
| gfx::NativeDrawingContext context = |
| scoped_platform_paint.GetPlatformSurface(); |
| |
| gfx::Rect offset_rect = rect; |
| offset_rect.Offset(-plugin_rect_.x(), -plugin_rect_.y()); |
| gfx::Rect canvas_rect = offset_rect; |
| #if defined(OS_MACOSX) |
| // The canvases are flipped relative to the context, so flip the rect too. |
| FlipRectVerticallyWithHeight(&canvas_rect, plugin_rect_.height()); |
| #endif |
| |
| bool background_changed = false; |
| if (background_store_.canvas.get() && BackgroundChanged(context, rect)) { |
| background_changed = true; |
| BlitContextToCanvas(background_store_.canvas.get(), canvas_rect, |
| context, rect.origin()); |
| } |
| |
| // transport_store_painted_ is really a bounding box, so in principle this |
| // check could falsely indicate that we don't need to paint offset_rect, but |
| // in practice it works fine. |
| if (background_changed || |
| !transport_store_painted_.Contains(offset_rect)) { |
| Send(new PluginMsg_Paint(instance_id_, offset_rect)); |
| // Since the plugin is not blocked on the renderer in this context, there is |
| // a chance that it will begin repainting the back-buffer before we complete |
| // capturing the data. Buffer flipping would increase that risk because |
| // geometry update is asynchronous, so we don't want to use buffer flipping |
| // here. |
| UpdateFrontBuffer(offset_rect, false); |
| } |
| |
| #if defined(OS_MACOSX) |
| // The canvases are flipped relative to the context, so flip the context's |
| // coordinate space so that the blit unflips the content. |
| CGContextSaveGState(context); |
| CGContextScaleCTM(context, 1, -1); |
| rect.set_y(-rect.bottom()); |
| #endif |
| BlitCanvasToContext(context, |
| rect, |
| front_buffer_canvas(), |
| offset_rect.origin()); |
| #if defined(OS_MACOSX) |
| CGContextRestoreGState(context); |
| #endif |
| |
| if (invalidate_pending_) { |
| // Only send the PaintAck message if this paint is in response to an |
| // invalidate from the plugin, since this message acts as an access token |
| // to ensure only one process is using the transport dib at a time. |
| invalidate_pending_ = false; |
| Send(new PluginMsg_DidPaint(instance_id_)); |
| } |
| } |
| |
| bool WebPluginDelegateProxy::BackgroundChanged( |
| gfx::NativeDrawingContext context, |
| const gfx::Rect& rect) { |
| #if defined(OS_ANDROID) |
| NOTIMPLEMENTED(); |
| #else |
| #if defined(OS_WIN) |
| HBITMAP hbitmap = static_cast<HBITMAP>(GetCurrentObject(context, OBJ_BITMAP)); |
| if (hbitmap == NULL) { |
| NOTREACHED(); |
| return true; |
| } |
| |
| BITMAP bitmap = { 0 }; |
| int result = GetObject(hbitmap, sizeof(bitmap), &bitmap); |
| if (!result) { |
| NOTREACHED(); |
| return true; |
| } |
| |
| XFORM xf; |
| if (!GetWorldTransform(context, &xf)) { |
| NOTREACHED(); |
| return true; |
| } |
| |
| // The damaged rect that we're given can be larger than the bitmap, so |
| // intersect their rects first. |
| gfx::Rect bitmap_rect(static_cast<int>(-xf.eDx), static_cast<int>(-xf.eDy), |
| bitmap.bmWidth, bitmap.bmHeight); |
| gfx::Rect check_rect = rect.Intersect(bitmap_rect); |
| int row_byte_size = check_rect.width() * (bitmap.bmBitsPixel / 8); |
| for (int y = check_rect.y(); y < check_rect.bottom(); y++) { |
| char* hdc_row_start = static_cast<char*>(bitmap.bmBits) + |
| (y + static_cast<int>(xf.eDy)) * bitmap.bmWidthBytes + |
| (check_rect.x() + static_cast<int>(xf.eDx)) * (bitmap.bmBitsPixel / 8); |
| |
| // getAddr32 doesn't use the translation units, so we have to subtract |
| // the plugin origin from the coordinates. |
| uint32_t* canvas_row_start = |
| background_store_.canvas->getDevice()->accessBitmap(true).getAddr32( |
| check_rect.x() - plugin_rect_.x(), y - plugin_rect_.y()); |
| if (memcmp(hdc_row_start, canvas_row_start, row_byte_size) != 0) |
| return true; |
| } |
| #else |
| #if defined(OS_MACOSX) |
| // If there is a translation on the content area context, we need to account |
| // for it; the context may be a subset of the full content area with a |
| // transform that makes the coordinates work out. |
| CGAffineTransform transform = CGContextGetCTM(context); |
| bool flipped = fabs(transform.d + 1) < 0.0001; |
| CGFloat context_offset_x = -transform.tx; |
| CGFloat context_offset_y = flipped ? transform.ty - |
| CGBitmapContextGetHeight(context) |
| : -transform.ty; |
| gfx::Rect full_content_rect(context_offset_x, context_offset_y, |
| CGBitmapContextGetWidth(context), |
| CGBitmapContextGetHeight(context)); |
| #else |
| cairo_surface_t* page_surface = cairo_get_target(context); |
| DCHECK_EQ(cairo_surface_get_type(page_surface), CAIRO_SURFACE_TYPE_IMAGE); |
| DCHECK_EQ(cairo_image_surface_get_format(page_surface), CAIRO_FORMAT_ARGB32); |
| |
| // Transform context coordinates into surface coordinates. |
| double page_x_double = rect.x(); |
| double page_y_double = rect.y(); |
| cairo_user_to_device(context, &page_x_double, &page_y_double); |
| gfx::Rect full_content_rect(0, 0, |
| cairo_image_surface_get_width(page_surface), |
| cairo_image_surface_get_height(page_surface)); |
| #endif |
| // According to comments in the Windows code, the damage rect that we're given |
| // may project outside the image, so intersect their rects. |
| gfx::Rect content_rect = rect.Intersect(full_content_rect); |
| |
| #if defined(OS_MACOSX) |
| const unsigned char* page_bytes = static_cast<const unsigned char*>( |
| CGBitmapContextGetData(context)); |
| int page_stride = CGBitmapContextGetBytesPerRow(context); |
| int page_start_x = content_rect.x() - context_offset_x; |
| int page_start_y = content_rect.y() - context_offset_y; |
| |
| skia::ScopedPlatformPaint scoped_platform_paint( |
| background_store_.canvas.get()); |
| CGContextRef bg_context = scoped_platform_paint.GetPlatformSurface(); |
| |
| DCHECK_EQ(CGBitmapContextGetBitsPerPixel(context), |
| CGBitmapContextGetBitsPerPixel(bg_context)); |
| const unsigned char* bg_bytes = static_cast<const unsigned char*>( |
| CGBitmapContextGetData(bg_context)); |
| int full_bg_width = CGBitmapContextGetWidth(bg_context); |
| int full_bg_height = CGBitmapContextGetHeight(bg_context); |
| int bg_stride = CGBitmapContextGetBytesPerRow(bg_context); |
| int bg_last_row = CGBitmapContextGetHeight(bg_context) - 1; |
| |
| int bytes_per_pixel = CGBitmapContextGetBitsPerPixel(context) / 8; |
| #else |
| cairo_surface_flush(page_surface); |
| const unsigned char* page_bytes = cairo_image_surface_get_data(page_surface); |
| int page_stride = cairo_image_surface_get_stride(page_surface); |
| int page_start_x = static_cast<int>(page_x_double); |
| int page_start_y = static_cast<int>(page_y_double); |
| |
| skia::ScopedPlatformPaint scoped_platform_paint( |
| background_store_.canvas.get()); |
| cairo_surface_t* bg_surface =cairo_get_target( |
| scoped_platform_paint.GetPlatformSurface()); |
| DCHECK_EQ(cairo_surface_get_type(bg_surface), CAIRO_SURFACE_TYPE_IMAGE); |
| DCHECK_EQ(cairo_image_surface_get_format(bg_surface), CAIRO_FORMAT_ARGB32); |
| cairo_surface_flush(bg_surface); |
| const unsigned char* bg_bytes = cairo_image_surface_get_data(bg_surface); |
| int full_bg_width = cairo_image_surface_get_width(bg_surface); |
| int full_bg_height = cairo_image_surface_get_height(bg_surface); |
| int bg_stride = cairo_image_surface_get_stride(bg_surface); |
| |
| int bytes_per_pixel = 4; // ARGB32 = 4 bytes per pixel. |
| #endif |
| |
| int damage_width = content_rect.width(); |
| int damage_height = content_rect.height(); |
| |
| int bg_start_x = rect.x() - plugin_rect_.x(); |
| int bg_start_y = rect.y() - plugin_rect_.y(); |
| // The damage rect is supposed to have been intersected with the plugin rect; |
| // double-check, since if it hasn't we'll walk off the end of the buffer. |
| DCHECK_LE(bg_start_x + damage_width, full_bg_width); |
| DCHECK_LE(bg_start_y + damage_height, full_bg_height); |
| |
| int bg_x_byte_offset = bg_start_x * bytes_per_pixel; |
| int page_x_byte_offset = page_start_x * bytes_per_pixel; |
| for (int row = 0; row < damage_height; ++row) { |
| int page_offset = page_stride * (page_start_y + row) + page_x_byte_offset; |
| int bg_y = bg_start_y + row; |
| #if defined(OS_MACOSX) |
| // The background buffer is upside down relative to the content. |
| bg_y = bg_last_row - bg_y; |
| #endif |
| int bg_offset = bg_stride * bg_y + bg_x_byte_offset; |
| if (memcmp(page_bytes + page_offset, |
| bg_bytes + bg_offset, |
| damage_width * bytes_per_pixel) != 0) |
| return true; |
| } |
| #endif |
| #endif // OS_ANDROID |
| |
| return false; |
| } |
| |
| NPObject* WebPluginDelegateProxy::GetPluginScriptableObject() { |
| if (npobject_) |
| return WebBindings::retainObject(npobject_); |
| |
| int route_id = MSG_ROUTING_NONE; |
| Send(new PluginMsg_GetPluginScriptableObject(instance_id_, &route_id)); |
| if (route_id == MSG_ROUTING_NONE) |
| return NULL; |
| |
| npobject_ = NPObjectProxy::Create( |
| channel_host_.get(), route_id, 0, page_url_); |
| |
| return WebBindings::retainObject(npobject_); |
| } |
| |
| bool WebPluginDelegateProxy::GetFormValue(string16* value) { |
| bool success = false; |
| Send(new PluginMsg_GetFormValue(instance_id_, value, &success)); |
| return success; |
| } |
| |
| void WebPluginDelegateProxy::DidFinishLoadWithReason( |
| const GURL& url, NPReason reason, int notify_id) { |
| Send(new PluginMsg_DidFinishLoadWithReason( |
| instance_id_, url, reason, notify_id)); |
| } |
| |
| void WebPluginDelegateProxy::SetFocus(bool focused) { |
| Send(new PluginMsg_SetFocus(instance_id_, focused)); |
| #if defined(OS_WIN) |
| if (render_view_) |
| render_view_->PluginFocusChanged(focused, instance_id_); |
| #endif |
| } |
| |
| bool WebPluginDelegateProxy::HandleInputEvent( |
| const WebInputEvent& event, |
| WebCursorInfo* cursor_info) { |
| bool handled; |
| WebCursor cursor; |
| // A windowless plugin can enter a modal loop in the context of a |
| // NPP_HandleEvent call, in which case we need to pump messages to |
| // the plugin. We pass of the corresponding event handle to the |
| // plugin process, which is set if the plugin does enter a modal loop. |
| IPC::SyncMessage* message = new PluginMsg_HandleInputEvent( |
| instance_id_, &event, &handled, &cursor); |
| message->set_pump_messages_event(modal_loop_pump_messages_event_.get()); |
| Send(message); |
| cursor.GetCursorInfo(cursor_info); |
| return handled; |
| } |
| |
| int WebPluginDelegateProxy::GetProcessId() { |
| return channel_host_->peer_pid(); |
| } |
| |
| void WebPluginDelegateProxy::SetContentAreaFocus(bool has_focus) { |
| IPC::Message* msg = new PluginMsg_SetContentAreaFocus(instance_id_, |
| has_focus); |
| // Make sure focus events are delivered in the right order relative to |
| // sync messages they might interact with (Paint, HandleEvent, etc.). |
| msg->set_unblock(true); |
| Send(msg); |
| } |
| |
| #if defined(OS_WIN) |
| void WebPluginDelegateProxy::ImeCompositionUpdated( |
| const string16& text, |
| const std::vector<int>& clauses, |
| const std::vector<int>& target, |
| int cursor_position, |
| int plugin_id) { |
| // Dispatch the raw IME data if this plug-in is the focused one. |
| if (instance_id_ != plugin_id) |
| return; |
| |
| IPC::Message* msg = new PluginMsg_ImeCompositionUpdated(instance_id_, |
| text, clauses, target, cursor_position); |
| msg->set_unblock(true); |
| Send(msg); |
| } |
| |
| void WebPluginDelegateProxy::ImeCompositionCompleted(const string16& text, |
| int plugin_id) { |
| // Dispatch the IME text if this plug-in is the focused one. |
| if (instance_id_ != plugin_id) |
| return; |
| |
| IPC::Message* msg = new PluginMsg_ImeCompositionCompleted(instance_id_, text); |
| msg->set_unblock(true); |
| Send(msg); |
| } |
| #endif |
| |
| #if defined(OS_MACOSX) |
| void WebPluginDelegateProxy::SetWindowFocus(bool window_has_focus) { |
| IPC::Message* msg = new PluginMsg_SetWindowFocus(instance_id_, |
| window_has_focus); |
| // Make sure focus events are delivered in the right order relative to |
| // sync messages they might interact with (Paint, HandleEvent, etc.). |
| msg->set_unblock(true); |
| Send(msg); |
| } |
| |
| void WebPluginDelegateProxy::SetContainerVisibility(bool is_visible) { |
| IPC::Message* msg; |
| if (is_visible) { |
| gfx::Rect window_frame = render_view_->rootWindowRect(); |
| gfx::Rect view_frame = render_view_->windowRect(); |
| WebKit::WebView* webview = render_view_->webview(); |
| msg = new PluginMsg_ContainerShown(instance_id_, window_frame, view_frame, |
| webview && webview->isActive()); |
| } else { |
| msg = new PluginMsg_ContainerHidden(instance_id_); |
| } |
| // Make sure visibility events are delivered in the right order relative to |
| // sync messages they might interact with (Paint, HandleEvent, etc.). |
| msg->set_unblock(true); |
| Send(msg); |
| } |
| |
| void WebPluginDelegateProxy::WindowFrameChanged(gfx::Rect window_frame, |
| gfx::Rect view_frame) { |
| IPC::Message* msg = new PluginMsg_WindowFrameChanged(instance_id_, |
| window_frame, |
| view_frame); |
| // Make sure frame events are delivered in the right order relative to |
| // sync messages they might interact with (e.g., HandleEvent). |
| msg->set_unblock(true); |
| Send(msg); |
| } |
| void WebPluginDelegateProxy::ImeCompositionCompleted(const string16& text, |
| int plugin_id) { |
| // If the message isn't intended for this plugin, there's nothing to do. |
| if (instance_id_ != plugin_id) |
| return; |
| |
| IPC::Message* msg = new PluginMsg_ImeCompositionCompleted(instance_id_, |
| text); |
| // Order relative to other key events is important. |
| msg->set_unblock(true); |
| Send(msg); |
| } |
| #endif // OS_MACOSX |
| |
| void WebPluginDelegateProxy::OnSetWindow(gfx::PluginWindowHandle window) { |
| #if defined(OS_MACOSX) |
| uses_shared_bitmaps_ = !window && !uses_compositor_; |
| #else |
| uses_shared_bitmaps_ = !window; |
| #endif |
| window_ = window; |
| if (plugin_) |
| plugin_->SetWindow(window); |
| } |
| |
| void WebPluginDelegateProxy::WillDestroyWindow() { |
| DCHECK(window_); |
| plugin_->WillDestroyWindow(window_); |
| #if defined(OS_MACOSX) |
| if (window_) { |
| // This is actually a "fake" window handle only for the GPU |
| // plugin. Deallocate it on the browser side. |
| if (render_view_) |
| render_view_->DestroyFakePluginWindowHandle(window_); |
| } |
| #endif |
| window_ = gfx::kNullPluginWindow; |
| } |
| |
| #if defined(OS_WIN) |
| void WebPluginDelegateProxy::OnSetWindowlessPumpEvent( |
| HANDLE modal_loop_pump_messages_event) { |
| DCHECK(modal_loop_pump_messages_event_ == NULL); |
| |
| // Bug 25583: this can be null because some "virus scanners" block the |
| // DuplicateHandle call in the plugin process. |
| if (!modal_loop_pump_messages_event) |
| return; |
| |
| modal_loop_pump_messages_event_.reset( |
| new base::WaitableEvent(modal_loop_pump_messages_event)); |
| } |
| |
| void WebPluginDelegateProxy::OnNotifyIMEStatus(int input_type, |
| const gfx::Rect& caret_rect) { |
| if (!render_view_) |
| return; |
| |
| render_view_->Send(new ViewHostMsg_TextInputStateChanged( |
| render_view_->routing_id(), |
| static_cast<ui::TextInputType>(input_type), |
| true)); |
| |
| render_view_->Send(new ViewHostMsg_SelectionBoundsChanged( |
| render_view_->routing_id(), caret_rect, caret_rect)); |
| } |
| #endif |
| |
| void WebPluginDelegateProxy::OnCancelResource(int id) { |
| if (plugin_) |
| plugin_->CancelResource(id); |
| } |
| |
| void WebPluginDelegateProxy::OnInvalidateRect(const gfx::Rect& rect, |
| bool allow_buffer_flipping) { |
| if (!plugin_) |
| return; |
| |
| // Clip the invalidation rect to the plugin bounds; the plugin may have been |
| // resized since the invalidate message was sent. |
| const gfx::Rect clipped_rect(rect.Intersect(gfx::Rect(plugin_rect_.size()))); |
| |
| invalidate_pending_ = true; |
| // The plugin is blocked on the renderer because the invalidate message it has |
| // sent us is synchronous, so we can use buffer flipping here if the caller |
| // allows it. |
| UpdateFrontBuffer(clipped_rect, allow_buffer_flipping); |
| plugin_->InvalidateRect(clipped_rect); |
| } |
| |
| void WebPluginDelegateProxy::OnGetWindowScriptNPObject( |
| int route_id, bool* success) { |
| *success = false; |
| NPObject* npobject = NULL; |
| if (plugin_) |
| npobject = plugin_->GetWindowScriptNPObject(); |
| |
| if (!npobject) |
| return; |
| |
| // The stub will delete itself when the proxy tells it that it's released, or |
| // otherwise when the channel is closed. |
| window_script_object_ = (new NPObjectStub( |
| npobject, channel_host_.get(), route_id, 0, page_url_))->AsWeakPtr(); |
| *success = true; |
| } |
| |
| void WebPluginDelegateProxy::OnResolveProxy(const GURL& url, |
| bool* result, |
| std::string* proxy_list) { |
| *result = false; |
| RenderThreadImpl::current()->Send( |
| new ViewHostMsg_ResolveProxy(url, result, proxy_list)); |
| } |
| |
| void WebPluginDelegateProxy::OnGetPluginElement(int route_id, bool* success) { |
| *success = false; |
| NPObject* npobject = NULL; |
| if (plugin_) |
| npobject = plugin_->GetPluginElement(); |
| if (!npobject) |
| return; |
| |
| // The stub will delete itself when the proxy tells it that it's released, or |
| // otherwise when the channel is closed. |
| new NPObjectStub( |
| npobject, channel_host_.get(), route_id, 0, page_url_); |
| *success = true; |
| } |
| |
| void WebPluginDelegateProxy::OnSetCookie(const GURL& url, |
| const GURL& first_party_for_cookies, |
| const std::string& cookie) { |
| if (plugin_) |
| plugin_->SetCookie(url, first_party_for_cookies, cookie); |
| } |
| |
| void WebPluginDelegateProxy::OnGetCookies(const GURL& url, |
| const GURL& first_party_for_cookies, |
| std::string* cookies) { |
| DCHECK(cookies); |
| if (plugin_) |
| *cookies = plugin_->GetCookies(url, first_party_for_cookies); |
| } |
| |
| void WebPluginDelegateProxy::PaintSadPlugin(WebKit::WebCanvas* native_context, |
| const gfx::Rect& rect) { |
| // Lazily load the sad plugin image. |
| if (!sad_plugin_) |
| sad_plugin_ = content::GetContentClient()->renderer()->GetSadPluginBitmap(); |
| if (sad_plugin_) |
| webkit::PaintSadPlugin(native_context, plugin_rect_, *sad_plugin_); |
| } |
| |
| void WebPluginDelegateProxy::CopyFromBackBufferToFrontBuffer( |
| const gfx::Rect& rect) { |
| #if defined(OS_MACOSX) |
| // Blitting the bits directly is much faster than going through CG, and since |
| // the goal is just to move the raw pixels between two bitmaps with the same |
| // pixel format (no compositing, color correction, etc.), it's safe. |
| const size_t stride = |
| skia::PlatformCanvas::StrideForWidth(plugin_rect_.width()); |
| const size_t chunk_size = 4 * rect.width(); |
| DCHECK(back_buffer_dib() != NULL); |
| uint8* source_data = static_cast<uint8*>(back_buffer_dib()->memory()) + |
| rect.y() * stride + 4 * rect.x(); |
| DCHECK(front_buffer_dib() != NULL); |
| uint8* target_data = static_cast<uint8*>(front_buffer_dib()->memory()) + |
| rect.y() * stride + 4 * rect.x(); |
| for (int row = 0; row < rect.height(); ++row) { |
| memcpy(target_data, source_data, chunk_size); |
| source_data += stride; |
| target_data += stride; |
| } |
| #else |
| BlitCanvasToCanvas(front_buffer_canvas(), |
| rect, |
| back_buffer_canvas(), |
| rect.origin()); |
| #endif |
| } |
| |
| void WebPluginDelegateProxy::UpdateFrontBuffer( |
| const gfx::Rect& rect, |
| bool allow_buffer_flipping) { |
| if (!front_buffer_canvas()) { |
| return; |
| } |
| |
| #if defined(OS_WIN) |
| // If SendUpdateGeometry() would block on the plugin process then we don't |
| // want to use buffer flipping at all since it would add extra locking. |
| // (Alternatively we could probably safely use async updates for buffer |
| // flipping all the time since the size is not changing.) |
| if (UseSynchronousGeometryUpdates()) { |
| allow_buffer_flipping = false; |
| } |
| #endif |
| |
| // Plugin has just painted "rect" into the back-buffer, so the front-buffer |
| // no longer holds the latest content for that rectangle. |
| front_buffer_diff_ = front_buffer_diff_.Subtract(rect); |
| if (allow_buffer_flipping && front_buffer_diff_.IsEmpty()) { |
| // Back-buffer contains the latest content for all areas; simply flip |
| // the buffers. |
| front_buffer_index_ = back_buffer_index(); |
| SendUpdateGeometry(false); |
| // The front-buffer now holds newer content for this region than the |
| // back-buffer. |
| front_buffer_diff_ = rect; |
| } else { |
| // Back-buffer contains the latest content for "rect" but the front-buffer |
| // contains the latest content for some other areas (or buffer flipping not |
| // allowed); fall back to copying the data. |
| CopyFromBackBufferToFrontBuffer(rect); |
| } |
| transport_store_painted_ = transport_store_painted_.Union(rect); |
| } |
| |
| void WebPluginDelegateProxy::OnHandleURLRequest( |
| const PluginHostMsg_URLRequest_Params& params) { |
| const char* data = NULL; |
| if (params.buffer.size()) |
| data = ¶ms.buffer[0]; |
| |
| const char* target = NULL; |
| if (params.target.length()) |
| target = params.target.c_str(); |
| |
| plugin_->HandleURLRequest( |
| params.url.c_str(), params.method.c_str(), target, data, |
| static_cast<unsigned int>(params.buffer.size()), params.notify_id, |
| params.popups_allowed, params.notify_redirects); |
| } |
| |
| webkit::npapi::WebPluginResourceClient* |
| WebPluginDelegateProxy::CreateResourceClient( |
| unsigned long resource_id, const GURL& url, int notify_id) { |
| if (!channel_host_) |
| return NULL; |
| |
| ResourceClientProxy* proxy = new ResourceClientProxy(channel_host_, |
| instance_id_); |
| proxy->Initialize(resource_id, url, notify_id); |
| return proxy; |
| } |
| |
| webkit::npapi::WebPluginResourceClient* |
| WebPluginDelegateProxy::CreateSeekableResourceClient( |
| unsigned long resource_id, int range_request_id) { |
| if (!channel_host_) |
| return NULL; |
| |
| ResourceClientProxy* proxy = new ResourceClientProxy(channel_host_, |
| instance_id_); |
| proxy->InitializeForSeekableStream(resource_id, range_request_id); |
| return proxy; |
| } |
| |
| #if defined(OS_MACOSX) |
| void WebPluginDelegateProxy::OnFocusChanged(bool focused) { |
| if (render_view_) |
| render_view_->PluginFocusChanged(focused, instance_id_); |
| } |
| |
| void WebPluginDelegateProxy::OnStartIme() { |
| if (render_view_) |
| render_view_->StartPluginIme(); |
| } |
| |
| void WebPluginDelegateProxy::OnBindFakePluginWindowHandle(bool opaque) { |
| BindFakePluginWindowHandle(opaque); |
| } |
| |
| // Synthesize a fake window handle for the plug-in to identify the instance |
| // to the browser, allowing mapping to a surface for hardware acceleration |
| // of plug-in content. The browser generates the handle which is then set on |
| // the plug-in. Returns true if it successfully sets the window handle on the |
| // plug-in. |
| bool WebPluginDelegateProxy::BindFakePluginWindowHandle(bool opaque) { |
| gfx::PluginWindowHandle fake_window = gfx::kNullPluginWindow; |
| if (render_view_) |
| fake_window = render_view_->AllocateFakePluginWindowHandle(opaque, false); |
| // If we aren't running on 10.6, this allocation will fail. |
| if (!fake_window) |
| return false; |
| OnSetWindow(fake_window); |
| if (!Send(new PluginMsg_SetFakeAcceleratedSurfaceWindowHandle(instance_id_, |
| fake_window))) { |
| return false; |
| } |
| |
| // Since this isn't a real window, it doesn't get initial size and location |
| // information the way a real windowed plugin would, so we need to feed it its |
| // starting geometry. |
| webkit::npapi::WebPluginGeometry geom; |
| geom.window = fake_window; |
| geom.window_rect = plugin_rect_; |
| geom.clip_rect = clip_rect_; |
| geom.rects_valid = true; |
| geom.visible = true; |
| render_view_->DidMovePlugin(geom); |
| // Invalidate the plugin region to ensure that the move event actually gets |
| // dispatched (for a plugin on an otherwise static page). |
| render_view_->didInvalidateRect(WebKit::WebRect(plugin_rect_.x(), |
| plugin_rect_.y(), |
| plugin_rect_.width(), |
| plugin_rect_.height())); |
| |
| return true; |
| } |
| #endif |
| |
| gfx::PluginWindowHandle WebPluginDelegateProxy::GetPluginWindowHandle() { |
| return window_; |
| } |
| |
| void WebPluginDelegateProxy::OnCancelDocumentLoad() { |
| plugin_->CancelDocumentLoad(); |
| } |
| |
| void WebPluginDelegateProxy::OnInitiateHTTPRangeRequest( |
| const std::string& url, |
| const std::string& range_info, |
| int range_request_id) { |
| plugin_->InitiateHTTPRangeRequest( |
| url.c_str(), range_info.c_str(), range_request_id); |
| } |
| |
| void WebPluginDelegateProxy::OnDeferResourceLoading(unsigned long resource_id, |
| bool defer) { |
| plugin_->SetDeferResourceLoading(resource_id, defer); |
| } |
| |
| #if defined(OS_MACOSX) |
| void WebPluginDelegateProxy::OnAcceleratedSurfaceSetIOSurface( |
| gfx::PluginWindowHandle window, |
| int32 width, |
| int32 height, |
| uint64 io_surface_identifier) { |
| if (render_view_) |
| render_view_->AcceleratedSurfaceSetIOSurface(window, width, height, |
| io_surface_identifier); |
| } |
| |
| void WebPluginDelegateProxy::OnAcceleratedSurfaceSetTransportDIB( |
| gfx::PluginWindowHandle window, |
| int32 width, |
| int32 height, |
| TransportDIB::Handle transport_dib) { |
| if (render_view_) |
| render_view_->AcceleratedSurfaceSetTransportDIB(window, width, height, |
| transport_dib); |
| } |
| |
| void WebPluginDelegateProxy::OnAcceleratedSurfaceAllocTransportDIB( |
| size_t size, |
| TransportDIB::Handle* dib_handle) { |
| if (render_view_) |
| *dib_handle = render_view_->AcceleratedSurfaceAllocTransportDIB(size); |
| else |
| *dib_handle = TransportDIB::DefaultHandleValue(); |
| } |
| |
| void WebPluginDelegateProxy::OnAcceleratedSurfaceFreeTransportDIB( |
| TransportDIB::Id dib_id) { |
| if (render_view_) |
| render_view_->AcceleratedSurfaceFreeTransportDIB(dib_id); |
| } |
| |
| void WebPluginDelegateProxy::OnAcceleratedSurfaceBuffersSwapped( |
| gfx::PluginWindowHandle window, uint64 surface_handle) { |
| if (render_view_) |
| render_view_->AcceleratedSurfaceBuffersSwapped(window, surface_handle); |
| } |
| |
| void WebPluginDelegateProxy::OnAcceleratedPluginEnabledRendering() { |
| uses_compositor_ = true; |
| OnSetWindow(gfx::kNullPluginWindow); |
| } |
| |
| void WebPluginDelegateProxy::OnAcceleratedPluginAllocatedIOSurface( |
| int32 width, |
| int32 height, |
| uint32 surface_id) { |
| if (plugin_) |
| plugin_->AcceleratedPluginAllocatedIOSurface(width, height, surface_id); |
| } |
| |
| void WebPluginDelegateProxy::OnAcceleratedPluginSwappedIOSurface() { |
| if (plugin_) |
| plugin_->AcceleratedPluginSwappedIOSurface(); |
| } |
| #endif |
| |
| #if defined(OS_WIN) |
| bool WebPluginDelegateProxy::UseSynchronousGeometryUpdates() { |
| // Need to update geometry synchronously with WMP, otherwise if a site |
| // scripts the plugin to start playing while it's in the middle of handling |
| // an update geometry message, videos don't play. See urls in bug 20260. |
| if (info_.name.find(ASCIIToUTF16("Windows Media Player")) != string16::npos) |
| return true; |
| |
| // The move networks plugin needs to be informed of geometry updates |
| // synchronously. |
| std::vector<webkit::WebPluginMimeType>::iterator index; |
| for (index = info_.mime_types.begin(); index != info_.mime_types.end(); |
| index++) { |
| if (index->mime_type == "application/x-vnd.moveplayer.qm" || |
| index->mime_type == "application/x-vnd.moveplay2.qm" || |
| index->mime_type == "application/x-vnd.movenetworks.qm" || |
| index->mime_type == "application/x-vnd.mnplayer.qm") { |
| return true; |
| } |
| } |
| return false; |
| } |
| #endif |
| |
| void WebPluginDelegateProxy::OnURLRedirectResponse(bool allow, |
| int resource_id) { |
| if (!plugin_) |
| return; |
| |
| plugin_->URLRedirectResponse(allow, resource_id); |
| } |