| /* |
| * Copyright (C) 2006, 2007, 2008 Apple Inc. All rights reserved. |
| * Copyright (C) 2008 Collabora Ltd. All rights reserved. |
| * |
| * Redistribution and use in source and binary forms, with or without |
| * modification, are permitted provided that the following conditions |
| * are met: |
| * 1. Redistributions of source code must retain the above copyright |
| * notice, this list of conditions and the following disclaimer. |
| * 2. Redistributions in binary form must reproduce the above copyright |
| * notice, this list of conditions and the following disclaimer in the |
| * documentation and/or other materials provided with the distribution. |
| * |
| * THIS SOFTWARE IS PROVIDED BY APPLE COMPUTER, INC. ``AS IS'' AND ANY |
| * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE |
| * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR |
| * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE COMPUTER, INC. OR |
| * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, |
| * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, |
| * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR |
| * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY |
| * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT |
| * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE |
| * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. |
| */ |
| |
| #include "config.h" |
| #include "PluginView.h" |
| |
| #include "Document.h" |
| #include "DocumentLoader.h" |
| #include "Element.h" |
| #include "FrameLoader.h" |
| #include "FrameTree.h" |
| #include "Frame.h" |
| #include "FrameView.h" |
| #include "GraphicsContext.h" |
| #include "Image.h" |
| #include "HTMLNames.h" |
| #include "HTMLPlugInElement.h" |
| #include "KeyboardEvent.h" |
| #include "MIMETypeRegistry.h" |
| #include "MouseEvent.h" |
| #include "NotImplemented.h" |
| #include "Page.h" |
| #include "FocusController.h" |
| #include "PlatformMouseEvent.h" |
| #if PLATFORM(WIN_OS) && ENABLE(NETSCAPE_PLUGIN_API) |
| #include "PluginMessageThrottlerWin.h" |
| #endif |
| #include "PluginPackage.h" |
| #include "ScriptController.h" |
| #include "ScriptValue.h" |
| #include "SecurityOrigin.h" |
| #include "PluginDatabase.h" |
| #include "PluginDebug.h" |
| #include "PluginMainThreadScheduler.h" |
| #include "PluginPackage.h" |
| #include "RenderBox.h" |
| #include "RenderObject.h" |
| #include "npruntime_impl.h" |
| #include "Settings.h" |
| #include <wtf/ASCIICType.h> |
| |
| #if defined(ANDROID_PLUGINS) |
| #include "TouchEvent.h" |
| #endif |
| |
| #if USE(JSC) |
| #include "JSDOMWindow.h" |
| #include "JSDOMBinding.h" |
| #include "c_instance.h" |
| #include "runtime_root.h" |
| #include "runtime.h" |
| #include <runtime/JSLock.h> |
| #include <runtime/JSValue.h> |
| |
| using JSC::ExecState; |
| using JSC::JSLock; |
| using JSC::JSObject; |
| using JSC::JSValue; |
| using JSC::UString; |
| #endif |
| |
| using std::min; |
| |
| using namespace WTF; |
| |
| namespace WebCore { |
| |
| using namespace HTMLNames; |
| |
| static int s_callingPlugin; |
| |
| static String scriptStringIfJavaScriptURL(const KURL& url) |
| { |
| if (!protocolIsJavaScript(url)) |
| return String(); |
| |
| // This returns an unescaped string |
| return decodeURLEscapeSequences(url.string().substring(11)); |
| } |
| |
| PluginView* PluginView::s_currentPluginView = 0; |
| |
| void PluginView::popPopupsStateTimerFired(Timer<PluginView>*) |
| { |
| popPopupsEnabledState(); |
| } |
| |
| IntRect PluginView::windowClipRect() const |
| { |
| // Start by clipping to our bounds. |
| IntRect clipRect(m_windowRect); |
| |
| // Take our element and get the clip rect from the enclosing layer and frame view. |
| RenderLayer* layer = m_element->renderer()->enclosingLayer(); |
| FrameView* parentView = m_element->document()->view(); |
| clipRect.intersect(parentView->windowClipRectForLayer(layer, true)); |
| |
| return clipRect; |
| } |
| |
| void PluginView::setFrameRect(const IntRect& rect) |
| { |
| if (m_element->document()->printing()) |
| return; |
| |
| #if defined(ANDROID_PLUGINS) |
| if (rect != frameRect()) { |
| Widget::setFrameRect(rect); |
| setNPWindowRect(rect); // only call when it changes |
| } |
| #else |
| if (rect != frameRect()) |
| Widget::setFrameRect(rect); |
| #endif |
| |
| updatePluginWidget(); |
| |
| #if PLATFORM(WIN_OS) || PLATFORM(SYMBIAN) |
| // On Windows and Symbian, always call plugin to change geometry. |
| setNPWindowRect(rect); |
| #elif defined(XP_UNIX) |
| // On Unix, multiple calls to setNPWindow() in windowed mode causes Flash to crash |
| if (m_mode == NP_FULL || !m_isWindowed) |
| setNPWindowRect(rect); |
| #endif |
| } |
| |
| void PluginView::frameRectsChanged() |
| { |
| updatePluginWidget(); |
| } |
| |
| void PluginView::handleEvent(Event* event) |
| { |
| if (!m_plugin || m_isWindowed) |
| return; |
| |
| if (event->isMouseEvent()) |
| handleMouseEvent(static_cast<MouseEvent*>(event)); |
| else if (event->isKeyboardEvent()) |
| handleKeyboardEvent(static_cast<KeyboardEvent*>(event)); |
| #if defined(ANDROID_PLUGINS) |
| else if (event->isTouchEvent()) |
| handleTouchEvent(static_cast<TouchEvent*>(event)); |
| else if (event->type() == eventNames().DOMFocusOutEvent) |
| handleFocusEvent(false); |
| else if (event->type() == eventNames().DOMFocusInEvent) |
| handleFocusEvent(true); |
| #endif |
| #if defined(Q_WS_X11) && ENABLE(NETSCAPE_PLUGIN_API) |
| else if (event->type() == eventNames().DOMFocusOutEvent) |
| handleFocusOutEvent(); |
| else if (event->type() == eventNames().DOMFocusInEvent) |
| handleFocusInEvent(); |
| #endif |
| } |
| |
| void PluginView::init() |
| { |
| if (m_haveInitialized) |
| return; |
| |
| m_haveInitialized = true; |
| |
| if (!m_plugin) { |
| ASSERT(m_status == PluginStatusCanNotFindPlugin); |
| return; |
| } |
| |
| LOG(Plugins, "PluginView::init(): Initializing plug-in '%s'", m_plugin->name().utf8().data()); |
| |
| if (!m_plugin->load()) { |
| m_plugin = 0; |
| m_status = PluginStatusCanNotLoadPlugin; |
| return; |
| } |
| |
| if (!startOrAddToUnstartedList()) { |
| m_status = PluginStatusCanNotLoadPlugin; |
| return; |
| } |
| |
| m_status = PluginStatusLoadedSuccessfully; |
| } |
| |
| bool PluginView::startOrAddToUnstartedList() |
| { |
| if (!m_parentFrame->page()) |
| return false; |
| |
| if (!m_parentFrame->page()->canStartPlugins()) { |
| m_parentFrame->page()->addUnstartedPlugin(this); |
| m_isWaitingToStart = true; |
| return true; |
| } |
| |
| return start(); |
| } |
| |
| |
| bool PluginView::start() |
| { |
| if (m_isStarted) |
| return false; |
| |
| m_isWaitingToStart = false; |
| |
| PluginMainThreadScheduler::scheduler().registerPlugin(m_instance); |
| |
| ASSERT(m_plugin); |
| ASSERT(m_plugin->pluginFuncs()->newp); |
| |
| NPError npErr; |
| { |
| PluginView::setCurrentPluginView(this); |
| #if USE(JSC) |
| JSC::JSLock::DropAllLocks dropAllLocks(JSC::SilenceAssertionsOnly); |
| #endif |
| setCallingPlugin(true); |
| npErr = m_plugin->pluginFuncs()->newp((NPMIMEType)m_mimeType.utf8().data(), m_instance, m_mode, m_paramCount, m_paramNames, m_paramValues, NULL); |
| setCallingPlugin(false); |
| LOG_NPERROR(npErr); |
| PluginView::setCurrentPluginView(0); |
| } |
| |
| if (npErr != NPERR_NO_ERROR) { |
| m_status = PluginStatusCanNotLoadPlugin; |
| PluginMainThreadScheduler::scheduler().unregisterPlugin(m_instance); |
| return false; |
| } |
| |
| m_isStarted = true; |
| |
| if (!m_url.isEmpty() && !m_loadManually) { |
| FrameLoadRequest frameLoadRequest; |
| frameLoadRequest.resourceRequest().setHTTPMethod("GET"); |
| frameLoadRequest.resourceRequest().setURL(m_url); |
| load(frameLoadRequest, false, 0); |
| } |
| |
| m_status = PluginStatusLoadedSuccessfully; |
| |
| if (!platformStart()) |
| m_status = PluginStatusCanNotLoadPlugin; |
| |
| if (m_status != PluginStatusLoadedSuccessfully) |
| return false; |
| |
| if (parentFrame()->page()) |
| parentFrame()->page()->didStartPlugin(this); |
| |
| return true; |
| } |
| |
| PluginView::~PluginView() |
| { |
| LOG(Plugins, "PluginView::~PluginView()"); |
| |
| removeFromUnstartedListIfNecessary(); |
| |
| stop(); |
| |
| deleteAllValues(m_requests); |
| |
| freeStringArray(m_paramNames, m_paramCount); |
| freeStringArray(m_paramValues, m_paramCount); |
| |
| platformDestroy(); |
| |
| m_parentFrame->script()->cleanupScriptObjectsForPlugin(this); |
| |
| #if PLATFORM(ANDROID) |
| // Since we have no legacy plugins to check, we ignore the quirks check. |
| if (m_plugin) |
| #else |
| if (m_plugin && !(m_plugin->quirks().contains(PluginQuirkDontUnloadPlugin))) |
| #endif |
| m_plugin->unload(); |
| } |
| |
| void PluginView::removeFromUnstartedListIfNecessary() |
| { |
| if (!m_isWaitingToStart) |
| return; |
| |
| if (!m_parentFrame->page()) |
| return; |
| |
| m_parentFrame->page()->removeUnstartedPlugin(this); |
| } |
| |
| void PluginView::stop() |
| { |
| if (!m_isStarted) |
| return; |
| |
| if (parentFrame()->page()) |
| parentFrame()->page()->didStopPlugin(this); |
| |
| LOG(Plugins, "PluginView::stop(): Stopping plug-in '%s'", m_plugin->name().utf8().data()); |
| |
| HashSet<RefPtr<PluginStream> > streams = m_streams; |
| HashSet<RefPtr<PluginStream> >::iterator end = streams.end(); |
| for (HashSet<RefPtr<PluginStream> >::iterator it = streams.begin(); it != end; ++it) { |
| (*it)->stop(); |
| disconnectStream((*it).get()); |
| } |
| |
| ASSERT(m_streams.isEmpty()); |
| |
| m_isStarted = false; |
| #if USE(JSC) |
| JSC::JSLock::DropAllLocks dropAllLocks(JSC::SilenceAssertionsOnly); |
| #endif |
| |
| #if ENABLE(NETSCAPE_PLUGIN_API) |
| #ifdef XP_WIN |
| // Unsubclass the window |
| if (m_isWindowed) { |
| #if PLATFORM(WINCE) |
| WNDPROC currentWndProc = (WNDPROC)GetWindowLong(platformPluginWidget(), GWL_WNDPROC); |
| |
| if (currentWndProc == PluginViewWndProc) |
| SetWindowLong(platformPluginWidget(), GWL_WNDPROC, (LONG)m_pluginWndProc); |
| #else |
| WNDPROC currentWndProc = (WNDPROC)GetWindowLongPtr(platformPluginWidget(), GWLP_WNDPROC); |
| |
| if (currentWndProc == PluginViewWndProc) |
| SetWindowLongPtr(platformPluginWidget(), GWLP_WNDPROC, (LONG)m_pluginWndProc); |
| #endif |
| } |
| #endif // XP_WIN |
| #endif // ENABLE(NETSCAPE_PLUGIN_API) |
| |
| #if !defined(XP_MACOSX) |
| // Clear the window |
| m_npWindow.window = 0; |
| |
| if (m_plugin->pluginFuncs()->setwindow && !m_plugin->quirks().contains(PluginQuirkDontSetNullWindowHandleOnDestroy)) { |
| PluginView::setCurrentPluginView(this); |
| setCallingPlugin(true); |
| m_plugin->pluginFuncs()->setwindow(m_instance, &m_npWindow); |
| setCallingPlugin(false); |
| PluginView::setCurrentPluginView(0); |
| } |
| |
| #ifdef XP_UNIX |
| if (m_isWindowed && m_npWindow.ws_info) |
| delete (NPSetWindowCallbackStruct *)m_npWindow.ws_info; |
| m_npWindow.ws_info = 0; |
| #endif |
| |
| #endif // !defined(XP_MACOSX) |
| |
| PluginMainThreadScheduler::scheduler().unregisterPlugin(m_instance); |
| |
| NPSavedData* savedData = 0; |
| PluginView::setCurrentPluginView(this); |
| setCallingPlugin(true); |
| NPError npErr = m_plugin->pluginFuncs()->destroy(m_instance, &savedData); |
| setCallingPlugin(false); |
| LOG_NPERROR(npErr); |
| PluginView::setCurrentPluginView(0); |
| |
| #if ENABLE(NETSCAPE_PLUGIN_API) |
| if (savedData) { |
| // TODO: Actually save this data instead of just discarding it |
| if (savedData->buf) |
| NPN_MemFree(savedData->buf); |
| NPN_MemFree(savedData); |
| } |
| #endif |
| |
| m_instance->pdata = 0; |
| } |
| |
| void PluginView::setCurrentPluginView(PluginView* pluginView) |
| { |
| s_currentPluginView = pluginView; |
| } |
| |
| PluginView* PluginView::currentPluginView() |
| { |
| return s_currentPluginView; |
| } |
| |
| static char* createUTF8String(const String& str) |
| { |
| CString cstr = str.utf8(); |
| char* result = reinterpret_cast<char*>(fastMalloc(cstr.length() + 1)); |
| |
| strncpy(result, cstr.data(), cstr.length() + 1); |
| |
| return result; |
| } |
| |
| #if USE(JSC) |
| static bool getString(ScriptController* proxy, JSValue result, String& string) |
| { |
| if (!proxy || !result || result.isUndefined()) |
| return false; |
| JSLock lock(JSC::SilenceAssertionsOnly); |
| |
| ExecState* exec = proxy->globalObject(pluginWorld())->globalExec(); |
| UString ustring = result.toString(exec); |
| exec->clearException(); |
| |
| string = ustring; |
| return true; |
| } |
| #endif |
| |
| void PluginView::performRequest(PluginRequest* request) |
| { |
| if (!m_isStarted) |
| return; |
| |
| // don't let a plugin start any loads if it is no longer part of a document that is being |
| // displayed unless the loads are in the same frame as the plugin. |
| const String& targetFrameName = request->frameLoadRequest().frameName(); |
| if (m_parentFrame->loader()->documentLoader() != m_parentFrame->loader()->activeDocumentLoader() && |
| (targetFrameName.isNull() || m_parentFrame->tree()->find(targetFrameName) != m_parentFrame)) |
| return; |
| |
| KURL requestURL = request->frameLoadRequest().resourceRequest().url(); |
| String jsString = scriptStringIfJavaScriptURL(requestURL); |
| |
| if (jsString.isNull()) { |
| // if this is not a targeted request, create a stream for it. otherwise, |
| // just pass it off to the loader |
| if (targetFrameName.isEmpty()) { |
| RefPtr<PluginStream> stream = PluginStream::create(this, m_parentFrame, request->frameLoadRequest().resourceRequest(), request->sendNotification(), request->notifyData(), plugin()->pluginFuncs(), instance(), m_plugin->quirks()); |
| m_streams.add(stream); |
| stream->start(); |
| } else { |
| // If the target frame is our frame, we could destroy the |
| // PluginView, so we protect it. <rdar://problem/6991251> |
| RefPtr<PluginView> protect(this); |
| |
| m_parentFrame->loader()->load(request->frameLoadRequest().resourceRequest(), targetFrameName, false); |
| |
| // FIXME: <rdar://problem/4807469> This should be sent when the document has finished loading |
| if (request->sendNotification()) { |
| PluginView::setCurrentPluginView(this); |
| #if USE(JSC) |
| JSC::JSLock::DropAllLocks dropAllLocks(JSC::SilenceAssertionsOnly); |
| #endif |
| setCallingPlugin(true); |
| m_plugin->pluginFuncs()->urlnotify(m_instance, requestURL.string().utf8().data(), NPRES_DONE, request->notifyData()); |
| setCallingPlugin(false); |
| PluginView::setCurrentPluginView(0); |
| } |
| } |
| return; |
| } |
| |
| // Targeted JavaScript requests are only allowed on the frame that contains the JavaScript plugin |
| // and this has been made sure in ::load. |
| ASSERT(targetFrameName.isEmpty() || m_parentFrame->tree()->find(targetFrameName) == m_parentFrame); |
| |
| #if USE(JSC) |
| // Executing a script can cause the plugin view to be destroyed, so we keep a reference to the parent frame. |
| RefPtr<Frame> parentFrame = m_parentFrame; |
| JSValue result = m_parentFrame->script()->executeScript(jsString, request->shouldAllowPopups()).jsValue(); |
| |
| if (targetFrameName.isNull()) { |
| String resultString; |
| |
| CString cstr; |
| if (getString(parentFrame->script(), result, resultString)) |
| cstr = resultString.utf8(); |
| |
| RefPtr<PluginStream> stream = PluginStream::create(this, m_parentFrame, request->frameLoadRequest().resourceRequest(), request->sendNotification(), request->notifyData(), plugin()->pluginFuncs(), instance(), m_plugin->quirks()); |
| m_streams.add(stream); |
| stream->sendJavaScriptStream(requestURL, cstr); |
| } |
| #endif |
| } |
| |
| void PluginView::requestTimerFired(Timer<PluginView>* timer) |
| { |
| ASSERT(timer == &m_requestTimer); |
| ASSERT(m_requests.size() > 0); |
| ASSERT(!m_isJavaScriptPaused); |
| |
| PluginRequest* request = m_requests[0]; |
| m_requests.remove(0); |
| |
| // Schedule a new request before calling performRequest since the call to |
| // performRequest can cause the plugin view to be deleted. |
| if (m_requests.size() > 0) |
| m_requestTimer.startOneShot(0); |
| |
| performRequest(request); |
| delete request; |
| } |
| |
| void PluginView::scheduleRequest(PluginRequest* request) |
| { |
| m_requests.append(request); |
| |
| if (!m_isJavaScriptPaused) |
| m_requestTimer.startOneShot(0); |
| } |
| |
| NPError PluginView::load(const FrameLoadRequest& frameLoadRequest, bool sendNotification, void* notifyData) |
| { |
| ASSERT(frameLoadRequest.resourceRequest().httpMethod() == "GET" || frameLoadRequest.resourceRequest().httpMethod() == "POST"); |
| |
| KURL url = frameLoadRequest.resourceRequest().url(); |
| |
| if (url.isEmpty()) |
| return NPERR_INVALID_URL; |
| |
| // Don't allow requests to be made when the document loader is stopping all loaders. |
| if (m_parentFrame->loader()->documentLoader()->isStopping()) |
| return NPERR_GENERIC_ERROR; |
| |
| const String& targetFrameName = frameLoadRequest.frameName(); |
| String jsString = scriptStringIfJavaScriptURL(url); |
| |
| if (!jsString.isNull()) { |
| Settings* settings = m_parentFrame->settings(); |
| |
| // Return NPERR_GENERIC_ERROR if JS is disabled. This is what Mozilla does. |
| if (!settings || !settings->isJavaScriptEnabled()) |
| return NPERR_GENERIC_ERROR; |
| |
| // For security reasons, only allow JS requests to be made on the frame that contains the plug-in. |
| if (!targetFrameName.isNull() && m_parentFrame->tree()->find(targetFrameName) != m_parentFrame) |
| return NPERR_INVALID_PARAM; |
| } else if (!SecurityOrigin::canLoad(url, String(), m_parentFrame->document())) |
| return NPERR_GENERIC_ERROR; |
| |
| PluginRequest* request = new PluginRequest(frameLoadRequest, sendNotification, notifyData, arePopupsAllowed()); |
| scheduleRequest(request); |
| |
| return NPERR_NO_ERROR; |
| } |
| |
| static KURL makeURL(const KURL& baseURL, const char* relativeURLString) |
| { |
| String urlString = relativeURLString; |
| |
| // Strip return characters. |
| urlString.replace('\n', ""); |
| urlString.replace('\r', ""); |
| |
| return KURL(baseURL, urlString); |
| } |
| |
| NPError PluginView::getURLNotify(const char* url, const char* target, void* notifyData) |
| { |
| FrameLoadRequest frameLoadRequest; |
| |
| frameLoadRequest.setFrameName(target); |
| frameLoadRequest.resourceRequest().setHTTPMethod("GET"); |
| frameLoadRequest.resourceRequest().setURL(makeURL(m_baseURL, url)); |
| |
| return load(frameLoadRequest, true, notifyData); |
| } |
| |
| NPError PluginView::getURL(const char* url, const char* target) |
| { |
| FrameLoadRequest frameLoadRequest; |
| |
| frameLoadRequest.setFrameName(target); |
| frameLoadRequest.resourceRequest().setHTTPMethod("GET"); |
| frameLoadRequest.resourceRequest().setURL(makeURL(m_baseURL, url)); |
| |
| return load(frameLoadRequest, false, 0); |
| } |
| |
| NPError PluginView::postURLNotify(const char* url, const char* target, uint32 len, const char* buf, NPBool file, void* notifyData) |
| { |
| return handlePost(url, target, len, buf, file, notifyData, true, true); |
| } |
| |
| NPError PluginView::postURL(const char* url, const char* target, uint32 len, const char* buf, NPBool file) |
| { |
| // As documented, only allow headers to be specified via NPP_PostURL when using a file. |
| return handlePost(url, target, len, buf, file, 0, false, file); |
| } |
| |
| NPError PluginView::newStream(NPMIMEType type, const char* target, NPStream** stream) |
| { |
| notImplemented(); |
| // Unsupported |
| return NPERR_GENERIC_ERROR; |
| } |
| |
| int32 PluginView::write(NPStream* stream, int32 len, void* buffer) |
| { |
| notImplemented(); |
| // Unsupported |
| return -1; |
| } |
| |
| NPError PluginView::destroyStream(NPStream* stream, NPReason reason) |
| { |
| if (!stream || PluginStream::ownerForStream(stream) != m_instance) |
| return NPERR_INVALID_INSTANCE_ERROR; |
| |
| PluginStream* browserStream = static_cast<PluginStream*>(stream->ndata); |
| browserStream->cancelAndDestroyStream(reason); |
| |
| return NPERR_NO_ERROR; |
| } |
| |
| void PluginView::status(const char* message) |
| { |
| if (Page* page = m_parentFrame->page()) |
| page->chrome()->setStatusbarText(m_parentFrame, String(message)); |
| } |
| |
| NPError PluginView::setValue(NPPVariable variable, void* value) |
| { |
| LOG(Plugins, "PluginView::setValue(%s): ", prettyNameForNPPVariable(variable, value).data()); |
| |
| switch (variable) { |
| case NPPVpluginWindowBool: |
| m_isWindowed = value; |
| return NPERR_NO_ERROR; |
| case NPPVpluginTransparentBool: |
| m_isTransparent = value; |
| return NPERR_NO_ERROR; |
| #if defined(XP_MACOSX) |
| case NPPVpluginDrawingModel: { |
| // Can only set drawing model inside NPP_New() |
| if (this != currentPluginView()) |
| return NPERR_GENERIC_ERROR; |
| |
| NPDrawingModel newDrawingModel = NPDrawingModel(uintptr_t(value)); |
| switch (newDrawingModel) { |
| case NPDrawingModelCoreGraphics: |
| m_drawingModel = newDrawingModel; |
| return NPERR_NO_ERROR; |
| #ifndef NP_NO_QUICKDRAW |
| case NPDrawingModelQuickDraw: |
| #endif |
| case NPDrawingModelCoreAnimation: |
| default: |
| LOG(Plugins, "Plugin asked for unsupported drawing model: %s", |
| prettyNameForDrawingModel(newDrawingModel)); |
| return NPERR_GENERIC_ERROR; |
| } |
| } |
| |
| case NPPVpluginEventModel: { |
| // Can only set event model inside NPP_New() |
| if (this != currentPluginView()) |
| return NPERR_GENERIC_ERROR; |
| |
| NPEventModel newEventModel = NPEventModel(uintptr_t(value)); |
| switch (newEventModel) { |
| #ifndef NP_NO_CARBON |
| case NPEventModelCarbon: |
| #endif |
| case NPEventModelCocoa: |
| m_eventModel = newEventModel; |
| return NPERR_NO_ERROR; |
| |
| default: |
| LOG(Plugins, "Plugin asked for unsupported event model: %s", |
| prettyNameForEventModel(newEventModel)); |
| return NPERR_GENERIC_ERROR; |
| } |
| } |
| #endif // defined(XP_MACOSX) |
| |
| default: |
| #ifdef PLUGIN_PLATFORM_SETVALUE |
| return platformSetValue(variable, value); |
| #else |
| notImplemented(); |
| return NPERR_GENERIC_ERROR; |
| #endif |
| } |
| } |
| |
| void PluginView::invalidateTimerFired(Timer<PluginView>* timer) |
| { |
| ASSERT(timer == &m_invalidateTimer); |
| |
| for (unsigned i = 0; i < m_invalidRects.size(); i++) |
| invalidateRect(m_invalidRects[i]); |
| m_invalidRects.clear(); |
| } |
| |
| |
| void PluginView::pushPopupsEnabledState(bool state) |
| { |
| m_popupStateStack.append(state); |
| } |
| |
| void PluginView::popPopupsEnabledState() |
| { |
| m_popupStateStack.removeLast(); |
| } |
| |
| bool PluginView::arePopupsAllowed() const |
| { |
| if (!m_popupStateStack.isEmpty()) |
| return m_popupStateStack.last(); |
| |
| return false; |
| } |
| |
| void PluginView::setJavaScriptPaused(bool paused) |
| { |
| if (m_isJavaScriptPaused == paused) |
| return; |
| m_isJavaScriptPaused = paused; |
| |
| if (m_isJavaScriptPaused) |
| m_requestTimer.stop(); |
| else if (!m_requests.isEmpty()) |
| m_requestTimer.startOneShot(0); |
| } |
| |
| |
| #if USE(JSC) |
| PassRefPtr<JSC::Bindings::Instance> PluginView::bindingInstance() |
| { |
| #if ENABLE(NETSCAPE_PLUGIN_API) |
| NPObject* object = 0; |
| |
| if (!m_isStarted || !m_plugin || !m_plugin->pluginFuncs()->getvalue) |
| return 0; |
| |
| // On Windows, calling Java's NPN_GetValue can allow the message loop to |
| // run, allowing loading to take place or JavaScript to run. Protect the |
| // PluginView from destruction. <rdar://problem/6978804> |
| RefPtr<PluginView> protect(this); |
| |
| NPError npErr; |
| { |
| PluginView::setCurrentPluginView(this); |
| JSC::JSLock::DropAllLocks dropAllLocks(JSC::SilenceAssertionsOnly); |
| setCallingPlugin(true); |
| npErr = m_plugin->pluginFuncs()->getvalue(m_instance, NPPVpluginScriptableNPObject, &object); |
| setCallingPlugin(false); |
| PluginView::setCurrentPluginView(0); |
| } |
| |
| if (hasOneRef()) { |
| // The renderer for the PluginView was destroyed during the above call, and |
| // the PluginView will be destroyed when this function returns, so we |
| // return null. |
| return 0; |
| } |
| |
| if (npErr != NPERR_NO_ERROR || !object) |
| return 0; |
| |
| RefPtr<JSC::Bindings::RootObject> root = m_parentFrame->script()->createRootObject(this); |
| RefPtr<JSC::Bindings::Instance> instance = JSC::Bindings::CInstance::create(object, root.release()); |
| |
| _NPN_ReleaseObject(object); |
| |
| return instance.release(); |
| #else |
| return 0; |
| #endif // NETSCAPE_PLUGIN_API |
| } |
| #endif // JSC |
| |
| #if USE(V8) |
| // This is really JS engine independent |
| NPObject* PluginView::getNPObject() { |
| #if ENABLE(NETSCAPE_PLUGIN_API) |
| if (!m_plugin || !m_plugin->pluginFuncs()->getvalue) |
| return 0; |
| |
| NPObject* object = 0; |
| |
| NPError npErr; |
| { |
| PluginView::setCurrentPluginView(this); |
| setCallingPlugin(true); |
| npErr = m_plugin->pluginFuncs()->getvalue(m_instance, NPPVpluginScriptableNPObject, &object); |
| setCallingPlugin(false); |
| PluginView::setCurrentPluginView(0); |
| } |
| |
| if (npErr != NPERR_NO_ERROR || !object) |
| return 0; |
| |
| _NPN_ReleaseObject(object); |
| return object; |
| #else |
| return 0; |
| #endif // NETSCAPE_PLUGIN_API |
| } |
| #endif // V8 |
| |
| void PluginView::disconnectStream(PluginStream* stream) |
| { |
| ASSERT(m_streams.contains(stream)); |
| |
| m_streams.remove(stream); |
| } |
| |
| void PluginView::setParameters(const Vector<String>& paramNames, const Vector<String>& paramValues) |
| { |
| ASSERT(paramNames.size() == paramValues.size()); |
| |
| unsigned size = paramNames.size(); |
| unsigned paramCount = 0; |
| |
| m_paramNames = reinterpret_cast<char**>(fastMalloc(sizeof(char*) * size)); |
| m_paramValues = reinterpret_cast<char**>(fastMalloc(sizeof(char*) * size)); |
| |
| for (unsigned i = 0; i < size; i++) { |
| if (m_plugin->quirks().contains(PluginQuirkRemoveWindowlessVideoParam) && equalIgnoringCase(paramNames[i], "windowlessvideo")) |
| continue; |
| |
| if (paramNames[i] == "pluginspage") |
| m_pluginsPage = paramValues[i]; |
| |
| m_paramNames[paramCount] = createUTF8String(paramNames[i]); |
| m_paramValues[paramCount] = createUTF8String(paramValues[i]); |
| |
| paramCount++; |
| } |
| |
| m_paramCount = paramCount; |
| } |
| |
| PluginView::PluginView(Frame* parentFrame, const IntSize& size, PluginPackage* plugin, Element* element, const KURL& url, const Vector<String>& paramNames, const Vector<String>& paramValues, const String& mimeType, bool loadManually) |
| : m_parentFrame(parentFrame) |
| , m_plugin(plugin) |
| , m_element(element) |
| , m_isStarted(false) |
| , m_url(url) |
| , m_baseURL(m_parentFrame->loader()->completeURL(m_parentFrame->document()->baseURL().string())) |
| , m_status(PluginStatusLoadedSuccessfully) |
| , m_requestTimer(this, &PluginView::requestTimerFired) |
| , m_invalidateTimer(this, &PluginView::invalidateTimerFired) |
| , m_popPopupsStateTimer(this, &PluginView::popPopupsStateTimerFired) |
| , m_paramNames(0) |
| , m_paramValues(0) |
| , m_mimeType(mimeType) |
| #if defined(XP_MACOSX) |
| , m_isWindowed(false) |
| #else |
| , m_isWindowed(true) |
| #endif |
| , m_isTransparent(false) |
| , m_haveInitialized(false) |
| , m_isWaitingToStart(false) |
| #if defined(XP_UNIX) || defined(Q_WS_X11) |
| , m_needsXEmbed(false) |
| #endif |
| #if PLATFORM(WIN_OS) && ENABLE(NETSCAPE_PLUGIN_API) |
| , m_pluginWndProc(0) |
| , m_lastMessage(0) |
| , m_isCallingPluginWndProc(false) |
| , m_wmPrintHDC(0) |
| , m_haveUpdatedPluginWidget(false) |
| #endif |
| #if (PLATFORM(QT) && PLATFORM(WIN_OS)) || defined(XP_MACOSX) |
| , m_window(0) |
| #endif |
| #if defined(XP_MACOSX) |
| , m_drawingModel(NPDrawingModel(-1)) |
| , m_eventModel(NPEventModel(-1)) |
| , m_contextRef(0) |
| , m_fakeWindow(0) |
| #endif |
| #if defined(Q_WS_X11) && ENABLE(NETSCAPE_PLUGIN_API) |
| , m_hasPendingGeometryChange(true) |
| , m_drawable(0) |
| , m_visual(0) |
| , m_colormap(0) |
| , m_pluginDisplay(0) |
| #endif |
| , m_loadManually(loadManually) |
| , m_manualStream(0) |
| , m_isJavaScriptPaused(false) |
| , m_isHalted(false) |
| , m_hasBeenHalted(false) |
| { |
| #if defined(ANDROID_PLUGINS) |
| platformInit(); |
| #endif |
| |
| if (!m_plugin) { |
| m_status = PluginStatusCanNotFindPlugin; |
| return; |
| } |
| |
| m_instance = &m_instanceStruct; |
| m_instance->ndata = this; |
| m_instance->pdata = 0; |
| |
| setParameters(paramNames, paramValues); |
| |
| memset(&m_npWindow, 0, sizeof(m_npWindow)); |
| #if defined(XP_MACOSX) |
| memset(&m_npCgContext, 0, sizeof(m_npCgContext)); |
| #endif |
| |
| m_mode = m_loadManually ? NP_FULL : NP_EMBED; |
| |
| resize(size); |
| } |
| |
| void PluginView::focusPluginElement() |
| { |
| // Focus the plugin |
| if (Page* page = m_parentFrame->page()) |
| page->focusController()->setFocusedFrame(m_parentFrame); |
| m_parentFrame->document()->setFocusedNode(m_element); |
| } |
| |
| void PluginView::didReceiveResponse(const ResourceResponse& response) |
| { |
| if (m_status != PluginStatusLoadedSuccessfully) |
| return; |
| |
| ASSERT(m_loadManually); |
| ASSERT(!m_manualStream); |
| |
| m_manualStream = PluginStream::create(this, m_parentFrame, m_parentFrame->loader()->activeDocumentLoader()->request(), false, 0, plugin()->pluginFuncs(), instance(), m_plugin->quirks()); |
| m_manualStream->setLoadManually(true); |
| |
| m_manualStream->didReceiveResponse(0, response); |
| } |
| |
| void PluginView::didReceiveData(const char* data, int length) |
| { |
| if (m_status != PluginStatusLoadedSuccessfully) |
| return; |
| |
| ASSERT(m_loadManually); |
| ASSERT(m_manualStream); |
| |
| m_manualStream->didReceiveData(0, data, length); |
| } |
| |
| void PluginView::didFinishLoading() |
| { |
| if (m_status != PluginStatusLoadedSuccessfully) |
| return; |
| |
| ASSERT(m_loadManually); |
| ASSERT(m_manualStream); |
| |
| m_manualStream->didFinishLoading(0); |
| } |
| |
| void PluginView::didFail(const ResourceError& error) |
| { |
| if (m_status != PluginStatusLoadedSuccessfully) |
| return; |
| |
| ASSERT(m_loadManually); |
| ASSERT(m_manualStream); |
| |
| m_manualStream->didFail(0, error); |
| } |
| |
| void PluginView::setCallingPlugin(bool b) const |
| { |
| if (!m_plugin->quirks().contains(PluginQuirkHasModalMessageLoop)) |
| return; |
| |
| if (b) |
| ++s_callingPlugin; |
| else |
| --s_callingPlugin; |
| |
| ASSERT(s_callingPlugin >= 0); |
| } |
| |
| bool PluginView::isCallingPlugin() |
| { |
| return s_callingPlugin > 0; |
| } |
| |
| PassRefPtr<PluginView> PluginView::create(Frame* parentFrame, const IntSize& size, Element* element, const KURL& url, const Vector<String>& paramNames, const Vector<String>& paramValues, const String& mimeType, bool loadManually) |
| { |
| // if we fail to find a plugin for this MIME type, findPlugin will search for |
| // a plugin by the file extension and update the MIME type, so pass a mutable String |
| String mimeTypeCopy = mimeType; |
| PluginPackage* plugin = PluginDatabase::installedPlugins()->findPlugin(url, mimeTypeCopy); |
| |
| // No plugin was found, try refreshing the database and searching again |
| if (!plugin && PluginDatabase::installedPlugins()->refresh()) { |
| mimeTypeCopy = mimeType; |
| plugin = PluginDatabase::installedPlugins()->findPlugin(url, mimeTypeCopy); |
| } |
| |
| return adoptRef(new PluginView(parentFrame, size, plugin, element, url, paramNames, paramValues, mimeTypeCopy, loadManually)); |
| } |
| |
| void PluginView::freeStringArray(char** stringArray, int length) |
| { |
| if (!stringArray) |
| return; |
| |
| for (int i = 0; i < length; i++) |
| fastFree(stringArray[i]); |
| |
| fastFree(stringArray); |
| } |
| |
| static inline bool startsWithBlankLine(const Vector<char>& buffer) |
| { |
| return buffer.size() > 0 && buffer[0] == '\n'; |
| } |
| |
| static inline int locationAfterFirstBlankLine(const Vector<char>& buffer) |
| { |
| const char* bytes = buffer.data(); |
| unsigned length = buffer.size(); |
| |
| for (unsigned i = 0; i < length - 4; i++) { |
| // Support for Acrobat. It sends "\n\n". |
| if (bytes[i] == '\n' && bytes[i + 1] == '\n') |
| return i + 2; |
| |
| // Returns the position after 2 CRLF's or 1 CRLF if it is the first line. |
| if (bytes[i] == '\r' && bytes[i + 1] == '\n') { |
| i += 2; |
| if (i == 2) |
| return i; |
| else if (bytes[i] == '\n') |
| // Support for Director. It sends "\r\n\n" (3880387). |
| return i + 1; |
| else if (bytes[i] == '\r' && bytes[i + 1] == '\n') |
| // Support for Flash. It sends "\r\n\r\n" (3758113). |
| return i + 2; |
| } |
| } |
| |
| return -1; |
| } |
| |
| static inline const char* findEOL(const char* bytes, unsigned length) |
| { |
| // According to the HTTP specification EOL is defined as |
| // a CRLF pair. Unfortunately, some servers will use LF |
| // instead. Worse yet, some servers will use a combination |
| // of both (e.g. <header>CRLFLF<body>), so findEOL needs |
| // to be more forgiving. It will now accept CRLF, LF or |
| // CR. |
| // |
| // It returns NULL if EOLF is not found or it will return |
| // a pointer to the first terminating character. |
| for (unsigned i = 0; i < length; i++) { |
| if (bytes[i] == '\n') |
| return bytes + i; |
| if (bytes[i] == '\r') { |
| // Check to see if spanning buffer bounds |
| // (CRLF is across reads). If so, wait for |
| // next read. |
| if (i + 1 == length) |
| break; |
| |
| return bytes + i; |
| } |
| } |
| |
| return 0; |
| } |
| |
| static inline String capitalizeRFC822HeaderFieldName(const String& name) |
| { |
| bool capitalizeCharacter = true; |
| String result; |
| |
| for (unsigned i = 0; i < name.length(); i++) { |
| UChar c; |
| |
| if (capitalizeCharacter && name[i] >= 'a' && name[i] <= 'z') |
| c = toASCIIUpper(name[i]); |
| else if (!capitalizeCharacter && name[i] >= 'A' && name[i] <= 'Z') |
| c = toASCIILower(name[i]); |
| else |
| c = name[i]; |
| |
| if (name[i] == '-') |
| capitalizeCharacter = true; |
| else |
| capitalizeCharacter = false; |
| |
| result.append(c); |
| } |
| |
| return result; |
| } |
| |
| static inline HTTPHeaderMap parseRFC822HeaderFields(const Vector<char>& buffer, unsigned length) |
| { |
| const char* bytes = buffer.data(); |
| const char* eol; |
| String lastKey; |
| HTTPHeaderMap headerFields; |
| |
| // Loop ove rlines until we're past the header, or we can't find any more end-of-lines |
| while ((eol = findEOL(bytes, length))) { |
| const char* line = bytes; |
| int lineLength = eol - bytes; |
| |
| // Move bytes to the character after the terminator as returned by findEOL. |
| bytes = eol + 1; |
| if ((*eol == '\r') && (*bytes == '\n')) |
| bytes++; // Safe since findEOL won't return a spanning CRLF. |
| |
| length -= (bytes - line); |
| if (lineLength == 0) |
| // Blank line; we're at the end of the header |
| break; |
| else if (*line == ' ' || *line == '\t') { |
| // Continuation of the previous header |
| if (lastKey.isNull()) { |
| // malformed header; ignore it and continue |
| continue; |
| } else { |
| // Merge the continuation of the previous header |
| String currentValue = headerFields.get(lastKey); |
| String newValue(line, lineLength); |
| |
| headerFields.set(lastKey, currentValue + newValue); |
| } |
| } else { |
| // Brand new header |
| const char* colon; |
| for (colon = line; *colon != ':' && colon != eol; colon++) { |
| // empty loop |
| } |
| if (colon == eol) |
| // malformed header; ignore it and continue |
| continue; |
| else { |
| lastKey = capitalizeRFC822HeaderFieldName(String(line, colon - line)); |
| String value; |
| |
| for (colon++; colon != eol; colon++) { |
| if (*colon != ' ' && *colon != '\t') |
| break; |
| } |
| if (colon == eol) |
| value = ""; |
| else |
| value = String(colon, eol - colon); |
| |
| String oldValue = headerFields.get(lastKey); |
| if (!oldValue.isNull()) { |
| String tmp = oldValue; |
| tmp += ", "; |
| tmp += value; |
| value = tmp; |
| } |
| |
| headerFields.set(lastKey, value); |
| } |
| } |
| } |
| |
| return headerFields; |
| } |
| |
| NPError PluginView::handlePost(const char* url, const char* target, uint32 len, const char* buf, bool file, void* notifyData, bool sendNotification, bool allowHeaders) |
| { |
| if (!url || !len || !buf) |
| return NPERR_INVALID_PARAM; |
| |
| FrameLoadRequest frameLoadRequest; |
| |
| HTTPHeaderMap headerFields; |
| Vector<char> buffer; |
| |
| if (file) { |
| NPError readResult = handlePostReadFile(buffer, len, buf); |
| if(readResult != NPERR_NO_ERROR) |
| return readResult; |
| } else { |
| buffer.resize(len); |
| memcpy(buffer.data(), buf, len); |
| } |
| |
| const char* postData = buffer.data(); |
| int postDataLength = buffer.size(); |
| |
| if (allowHeaders) { |
| if (startsWithBlankLine(buffer)) { |
| postData++; |
| postDataLength--; |
| } else { |
| int location = locationAfterFirstBlankLine(buffer); |
| if (location != -1) { |
| // If the blank line is somewhere in the middle of the buffer, everything before is the header |
| headerFields = parseRFC822HeaderFields(buffer, location); |
| unsigned dataLength = buffer.size() - location; |
| |
| // Sometimes plugins like to set Content-Length themselves when they post, |
| // but WebFoundation does not like that. So we will remove the header |
| // and instead truncate the data to the requested length. |
| String contentLength = headerFields.get("Content-Length"); |
| |
| if (!contentLength.isNull()) |
| dataLength = min(contentLength.toInt(), (int)dataLength); |
| headerFields.remove("Content-Length"); |
| |
| postData += location; |
| postDataLength = dataLength; |
| } |
| } |
| } |
| |
| frameLoadRequest.resourceRequest().setHTTPMethod("POST"); |
| frameLoadRequest.resourceRequest().setURL(makeURL(m_baseURL, url)); |
| frameLoadRequest.resourceRequest().addHTTPHeaderFields(headerFields); |
| frameLoadRequest.resourceRequest().setHTTPBody(FormData::create(postData, postDataLength)); |
| frameLoadRequest.setFrameName(target); |
| |
| return load(frameLoadRequest, sendNotification, notifyData); |
| } |
| |
| #ifdef PLUGIN_SCHEDULE_TIMER |
| uint32 PluginView::scheduleTimer(NPP instance, uint32 interval, bool repeat, |
| void (*timerFunc)(NPP, uint32 timerID)) |
| { |
| return m_timerList.schedule(instance, interval, repeat, timerFunc); |
| } |
| |
| void PluginView::unscheduleTimer(NPP instance, uint32 timerID) |
| { |
| m_timerList.unschedule(instance, timerID); |
| } |
| #endif |
| |
| void PluginView::invalidateWindowlessPluginRect(const IntRect& rect) |
| { |
| if (!isVisible()) |
| return; |
| |
| if (!m_element->renderer()) |
| return; |
| RenderBox* renderer = toRenderBox(m_element->renderer()); |
| |
| IntRect dirtyRect = rect; |
| dirtyRect.move(renderer->borderLeft() + renderer->paddingLeft(), renderer->borderTop() + renderer->paddingTop()); |
| renderer->repaintRectangle(dirtyRect); |
| } |
| |
| void PluginView::paintMissingPluginIcon(GraphicsContext* context, const IntRect& rect) |
| { |
| static RefPtr<Image> nullPluginImage; |
| if (!nullPluginImage) |
| nullPluginImage = Image::loadPlatformResource("nullPlugin"); |
| |
| IntRect imageRect(frameRect().x(), frameRect().y(), nullPluginImage->width(), nullPluginImage->height()); |
| |
| int xOffset = (frameRect().width() - imageRect.width()) / 2; |
| int yOffset = (frameRect().height() - imageRect.height()) / 2; |
| |
| imageRect.move(xOffset, yOffset); |
| |
| if (!rect.intersects(imageRect)) |
| return; |
| |
| context->save(); |
| context->clip(windowClipRect()); |
| context->drawImage(nullPluginImage.get(), DeviceColorSpace, imageRect.location()); |
| context->restore(); |
| } |
| |
| static const char* MozillaUserAgent = "Mozilla/5.0 (" |
| #if defined(XP_MACOSX) |
| "Macintosh; U; Intel Mac OS X;" |
| #elif defined(XP_WIN) |
| "Windows; U; Windows NT 5.1;" |
| #elif defined(XP_UNIX) |
| // The Gtk port uses X11 plugins in Mac. |
| #if PLATFORM(DARWIN) && PLATFORM(GTK) |
| "X11; U; Intel Mac OS X;" |
| #else |
| "X11; U; Linux i686;" |
| #endif |
| #endif |
| " en-US; rv:1.8.1) Gecko/20061010 Firefox/2.0"; |
| |
| const char* PluginView::userAgent() |
| { |
| #if !PLATFORM(ANDROID) |
| if (m_plugin->quirks().contains(PluginQuirkWantsMozillaUserAgent)) |
| return MozillaUserAgent; |
| #endif |
| |
| if (m_userAgent.isNull()) |
| m_userAgent = m_parentFrame->loader()->userAgent(m_url).utf8(); |
| |
| return m_userAgent.data(); |
| } |
| |
| #if ENABLE(NETSCAPE_PLUGIN_API) |
| const char* PluginView::userAgentStatic() |
| { |
| return MozillaUserAgent; |
| } |
| #endif |
| |
| |
| Node* PluginView::node() const |
| { |
| return m_element; |
| } |
| |
| String PluginView::pluginName() const |
| { |
| return m_plugin->name(); |
| } |
| |
| } // namespace WebCore |