| /* |
| * Copyright (C) 2006, 2008 Apple Inc. All rights reserved. |
| * Copyright (C) 2008 Collabora Ltd. All rights reserved. |
| * Copyright (C) 2009 Holger Hans Peter Freyther |
| * |
| * 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 "PluginPackage.h" |
| |
| #include "MIMETypeRegistry.h" |
| #include "PluginDatabase.h" |
| #include "PluginDebug.h" |
| #include "Timer.h" |
| #include "npruntime_impl.h" |
| #include <string.h> |
| #include <wtf/OwnArrayPtr.h> |
| #include <wtf/text/CString.h> |
| |
| namespace WebCore { |
| |
| PluginPackage::~PluginPackage() |
| { |
| // This destructor gets called during refresh() if PluginDatabase's |
| // PluginSet hash is already populated, as it removes items from |
| // the hash table. Calling the destructor on a loaded plug-in of |
| // course would cause a crash, so we check to call unload before we |
| // ASSERT. |
| // FIXME: There is probably a better way to fix this. |
| if (!m_loadCount) |
| unloadWithoutShutdown(); |
| else |
| unload(); |
| |
| ASSERT(!m_isLoaded); |
| } |
| |
| void PluginPackage::freeLibrarySoon() |
| { |
| ASSERT(!m_freeLibraryTimer.isActive()); |
| ASSERT(m_module); |
| ASSERT(!m_loadCount); |
| |
| m_freeLibraryTimer.startOneShot(0); |
| } |
| |
| void PluginPackage::freeLibraryTimerFired(Timer<PluginPackage>*) |
| { |
| ASSERT(m_module); |
| ASSERT(!m_loadCount); |
| |
| unloadModule(m_module); |
| m_module = 0; |
| } |
| |
| |
| int PluginPackage::compare(const PluginPackage& compareTo) const |
| { |
| // Sort plug-ins that allow multiple instances first. |
| bool AallowsMultipleInstances = !quirks().contains(PluginQuirkDontAllowMultipleInstances); |
| bool BallowsMultipleInstances = !compareTo.quirks().contains(PluginQuirkDontAllowMultipleInstances); |
| if (AallowsMultipleInstances != BallowsMultipleInstances) |
| return AallowsMultipleInstances ? -1 : 1; |
| |
| // Sort plug-ins in a preferred path first. |
| bool AisInPreferredDirectory = PluginDatabase::isPreferredPluginDirectory(parentDirectory()); |
| bool BisInPreferredDirectory = PluginDatabase::isPreferredPluginDirectory(compareTo.parentDirectory()); |
| if (AisInPreferredDirectory != BisInPreferredDirectory) |
| return AisInPreferredDirectory ? -1 : 1; |
| |
| int diff = strcmp(name().utf8().data(), compareTo.name().utf8().data()); |
| if (diff) |
| return diff; |
| |
| diff = compareFileVersion(compareTo.version()); |
| if (diff) |
| return diff; |
| |
| return strcmp(parentDirectory().utf8().data(), compareTo.parentDirectory().utf8().data()); |
| } |
| |
| PluginPackage::PluginPackage(const String& path, const time_t& lastModified) |
| : m_isEnabled(true) |
| , m_isLoaded(false) |
| , m_loadCount(0) |
| , m_path(path) |
| , m_moduleVersion(0) |
| , m_module(0) |
| , m_lastModified(lastModified) |
| , m_freeLibraryTimer(this, &PluginPackage::freeLibraryTimerFired) |
| #if ENABLE(NETSCAPE_PLUGIN_METADATA_CACHE) |
| , m_infoIsFromCache(true) |
| #endif |
| { |
| m_fileName = pathGetFileName(m_path); |
| m_parentDirectory = m_path.left(m_path.length() - m_fileName.length() - 1); |
| } |
| |
| #if !OS(SYMBIAN) |
| void PluginPackage::unload() |
| { |
| if (!m_isLoaded) |
| return; |
| |
| if (--m_loadCount > 0) |
| return; |
| |
| m_NPP_Shutdown(); |
| |
| unloadWithoutShutdown(); |
| } |
| #endif // !OS(SYMBIAN) |
| |
| void PluginPackage::unloadWithoutShutdown() |
| { |
| if (!m_isLoaded) |
| return; |
| |
| ASSERT(!m_loadCount); |
| ASSERT(m_module); |
| |
| // <rdar://5530519>: Crash when closing tab with pdf file (Reader 7 only) |
| // If the plugin has subclassed its parent window, as with Reader 7, we may have |
| // gotten here by way of the plugin's internal window proc forwarding a message to our |
| // original window proc. If we free the plugin library from here, we will jump back |
| // to code we just freed when we return, so delay calling FreeLibrary at least until |
| // the next message loop |
| freeLibrarySoon(); |
| |
| m_isLoaded = false; |
| } |
| |
| void PluginPackage::setEnabled(bool enabled) |
| { |
| m_isEnabled = enabled; |
| } |
| |
| PassRefPtr<PluginPackage> PluginPackage::createPackage(const String& path, const time_t& lastModified) |
| { |
| RefPtr<PluginPackage> package = adoptRef(new PluginPackage(path, lastModified)); |
| |
| if (!package->fetchInfo()) |
| return 0; |
| |
| return package.release(); |
| } |
| |
| #if ENABLE(NETSCAPE_PLUGIN_METADATA_CACHE) |
| PassRefPtr<PluginPackage> PluginPackage::createPackageFromCache(const String& path, const time_t& lastModified, const String& name, const String& description, const String& mimeDescription) |
| { |
| RefPtr<PluginPackage> package = adoptRef(new PluginPackage(path, lastModified)); |
| package->m_name = name; |
| package->m_description = description; |
| package->determineModuleVersionFromDescription(); |
| package->setMIMEDescription(mimeDescription); |
| package->m_infoIsFromCache = true; |
| return package.release(); |
| } |
| #endif |
| |
| #if defined(XP_UNIX) |
| void PluginPackage::determineQuirks(const String& mimeType) |
| { |
| if (MIMETypeRegistry::isJavaAppletMIMEType(mimeType)) { |
| // Because a single process cannot create multiple VMs, and we cannot reliably unload a |
| // Java VM, we cannot unload the Java plugin, or we'll lose reference to our only VM |
| m_quirks.add(PluginQuirkDontUnloadPlugin); |
| |
| // Setting the window region to an empty region causes bad scrolling repaint problems |
| // with the Java plug-in. |
| m_quirks.add(PluginQuirkDontClipToZeroRectWhenScrolling); |
| return; |
| } |
| |
| if (mimeType == "application/x-shockwave-flash") { |
| static const PlatformModuleVersion flashTenVersion(0x0a000000); |
| |
| if (compareFileVersion(flashTenVersion) >= 0) { |
| // Flash 10.0 b218 doesn't like having a NULL window handle |
| m_quirks.add(PluginQuirkDontSetNullWindowHandleOnDestroy); |
| #if PLATFORM(QT) |
| m_quirks.add(PluginQuirkRequiresGtkToolKit); |
| #endif |
| } else { |
| // Flash 9 and older requests windowless plugins if we return a mozilla user agent |
| m_quirks.add(PluginQuirkWantsMozillaUserAgent); |
| } |
| |
| #if PLATFORM(QT) |
| // Flash will crash on repeated calls to SetWindow in windowed mode |
| m_quirks.add(PluginQuirkDontCallSetWindowMoreThanOnce); |
| |
| #if CPU(X86_64) |
| // 64-bit Flash freezes if right-click is sent in windowless mode |
| m_quirks.add(PluginQuirkIgnoreRightClickInWindowlessMode); |
| #endif |
| #endif |
| |
| m_quirks.add(PluginQuirkRequiresDefaultScreenDepth); |
| m_quirks.add(PluginQuirkThrottleInvalidate); |
| m_quirks.add(PluginQuirkThrottleWMUserPlusOneMessages); |
| m_quirks.add(PluginQuirkFlashURLNotifyBug); |
| } |
| |
| #if defined(Q_WS_MAEMO_5) || defined(Q_WS_MAEMO_6) |
| // Passing a 32-bit depth pixmap to NPAPI plugins is too inefficient. Instead, pass a X Pixmap |
| // that has same depth as the screen depth since graphics operations are optimized |
| // for this depth. |
| m_quirks.add(PluginQuirkRequiresDefaultScreenDepth); |
| #endif |
| } |
| #endif |
| |
| #if !OS(WINDOWS) |
| void PluginPackage::determineModuleVersionFromDescription() |
| { |
| // It's a bit lame to detect the plugin version by parsing it |
| // from the plugin description string, but it doesn't seem that |
| // version information is available in any standardized way at |
| // the module level, like in Windows |
| |
| if (m_description.isEmpty()) |
| return; |
| |
| if (m_description.startsWith("Shockwave Flash") && m_description.length() >= 19) { |
| // The flash version as a PlatformModuleVersion differs on Unix from Windows |
| // since the revision can be larger than a 8 bits, so we allow it 16 here and |
| // push the major/minor up 8 bits. Thus on Unix, Flash's version may be |
| // 0x0a000000 instead of 0x000a0000. |
| |
| Vector<String> versionParts; |
| m_description.substring(16).split(' ', /*allowEmptyEntries =*/ false, versionParts); |
| if (versionParts.isEmpty()) |
| return; |
| |
| if (versionParts.size() >= 1) { |
| Vector<String> majorMinorParts; |
| versionParts[0].split('.', majorMinorParts); |
| if (majorMinorParts.size() >= 1) { |
| bool converted = false; |
| unsigned major = majorMinorParts[0].toUInt(&converted); |
| if (converted) |
| m_moduleVersion = (major & 0xff) << 24; |
| } |
| if (majorMinorParts.size() == 2) { |
| bool converted = false; |
| unsigned minor = majorMinorParts[1].toUInt(&converted); |
| if (converted) |
| m_moduleVersion |= (minor & 0xff) << 16; |
| } |
| } |
| |
| if (versionParts.size() >= 2) { |
| String revision = versionParts[1]; |
| if (revision.length() > 1 && (revision[0] == 'r' || revision[0] == 'b')) { |
| revision.remove(0, 1); |
| m_moduleVersion |= revision.toInt() & 0xffff; |
| } |
| } |
| } |
| } |
| #endif |
| |
| #if ENABLE(NETSCAPE_PLUGIN_API) |
| void PluginPackage::initializeBrowserFuncs() |
| { |
| memset(&m_browserFuncs, 0, sizeof(m_browserFuncs)); |
| m_browserFuncs.size = sizeof(m_browserFuncs); |
| m_browserFuncs.version = NPVersion(); |
| |
| m_browserFuncs.geturl = NPN_GetURL; |
| m_browserFuncs.posturl = NPN_PostURL; |
| m_browserFuncs.requestread = NPN_RequestRead; |
| m_browserFuncs.newstream = NPN_NewStream; |
| m_browserFuncs.write = NPN_Write; |
| m_browserFuncs.destroystream = NPN_DestroyStream; |
| m_browserFuncs.status = NPN_Status; |
| m_browserFuncs.uagent = NPN_UserAgent; |
| m_browserFuncs.memalloc = NPN_MemAlloc; |
| m_browserFuncs.memfree = NPN_MemFree; |
| m_browserFuncs.memflush = NPN_MemFlush; |
| m_browserFuncs.reloadplugins = NPN_ReloadPlugins; |
| m_browserFuncs.geturlnotify = NPN_GetURLNotify; |
| m_browserFuncs.posturlnotify = NPN_PostURLNotify; |
| m_browserFuncs.getvalue = NPN_GetValue; |
| m_browserFuncs.setvalue = NPN_SetValue; |
| m_browserFuncs.invalidaterect = NPN_InvalidateRect; |
| m_browserFuncs.invalidateregion = NPN_InvalidateRegion; |
| m_browserFuncs.forceredraw = NPN_ForceRedraw; |
| m_browserFuncs.getJavaEnv = NPN_GetJavaEnv; |
| m_browserFuncs.getJavaPeer = NPN_GetJavaPeer; |
| m_browserFuncs.pushpopupsenabledstate = NPN_PushPopupsEnabledState; |
| m_browserFuncs.poppopupsenabledstate = NPN_PopPopupsEnabledState; |
| m_browserFuncs.pluginthreadasynccall = NPN_PluginThreadAsyncCall; |
| |
| m_browserFuncs.releasevariantvalue = _NPN_ReleaseVariantValue; |
| m_browserFuncs.getstringidentifier = _NPN_GetStringIdentifier; |
| m_browserFuncs.getstringidentifiers = _NPN_GetStringIdentifiers; |
| m_browserFuncs.getintidentifier = _NPN_GetIntIdentifier; |
| m_browserFuncs.identifierisstring = _NPN_IdentifierIsString; |
| m_browserFuncs.utf8fromidentifier = _NPN_UTF8FromIdentifier; |
| m_browserFuncs.intfromidentifier = _NPN_IntFromIdentifier; |
| m_browserFuncs.createobject = _NPN_CreateObject; |
| m_browserFuncs.retainobject = _NPN_RetainObject; |
| m_browserFuncs.releaseobject = _NPN_ReleaseObject; |
| m_browserFuncs.invoke = _NPN_Invoke; |
| m_browserFuncs.invokeDefault = _NPN_InvokeDefault; |
| m_browserFuncs.evaluate = _NPN_Evaluate; |
| m_browserFuncs.getproperty = _NPN_GetProperty; |
| m_browserFuncs.setproperty = _NPN_SetProperty; |
| m_browserFuncs.removeproperty = _NPN_RemoveProperty; |
| m_browserFuncs.hasproperty = _NPN_HasProperty; |
| m_browserFuncs.hasmethod = _NPN_HasMethod; |
| m_browserFuncs.setexception = _NPN_SetException; |
| m_browserFuncs.enumerate = _NPN_Enumerate; |
| m_browserFuncs.construct = _NPN_Construct; |
| m_browserFuncs.getvalueforurl = NPN_GetValueForURL; |
| m_browserFuncs.setvalueforurl = NPN_SetValueForURL; |
| m_browserFuncs.getauthenticationinfo = NPN_GetAuthenticationInfo; |
| } |
| #endif |
| |
| #if ENABLE(PLUGIN_PACKAGE_SIMPLE_HASH) |
| unsigned PluginPackage::hash() const |
| { |
| unsigned hashCodes[] = { |
| m_path.impl()->hash(), |
| m_lastModified |
| }; |
| |
| return StringHasher::hashMemory<sizeof(hashCodes)>(hashCodes); |
| } |
| |
| bool PluginPackage::equal(const PluginPackage& a, const PluginPackage& b) |
| { |
| return a.m_description == b.m_description; |
| } |
| #endif |
| |
| int PluginPackage::compareFileVersion(const PlatformModuleVersion& compareVersion) const |
| { |
| // return -1, 0, or 1 if plug-in version is less than, equal to, or greater than |
| // the passed version |
| |
| #if OS(WINDOWS) |
| if (m_moduleVersion.mostSig != compareVersion.mostSig) |
| return m_moduleVersion.mostSig > compareVersion.mostSig ? 1 : -1; |
| if (m_moduleVersion.leastSig != compareVersion.leastSig) |
| return m_moduleVersion.leastSig > compareVersion.leastSig ? 1 : -1; |
| #else |
| if (m_moduleVersion != compareVersion) |
| return m_moduleVersion > compareVersion ? 1 : -1; |
| #endif |
| |
| return 0; |
| } |
| |
| #if ENABLE(NETSCAPE_PLUGIN_METADATA_CACHE) |
| bool PluginPackage::ensurePluginLoaded() |
| { |
| if (!m_infoIsFromCache) |
| return m_isLoaded; |
| |
| m_quirks = PluginQuirkSet(); |
| m_name = String(); |
| m_description = String(); |
| m_fullMIMEDescription = String(); |
| m_moduleVersion = 0; |
| |
| return fetchInfo(); |
| } |
| #endif |
| |
| } |