Home | History | Annotate | Download | only in plugins
      1 /*
      2  * Copyright (C) 2006, 2008 Apple Inc. All rights reserved.
      3  * Copyright (C) 2008 Collabora Ltd.  All rights reserved.
      4  * Copyright (C) 2009 Holger Hans Peter Freyther
      5  *
      6  * Redistribution and use in source and binary forms, with or without
      7  * modification, are permitted provided that the following conditions
      8  * are met:
      9  * 1. Redistributions of source code must retain the above copyright
     10  *    notice, this list of conditions and the following disclaimer.
     11  * 2. Redistributions in binary form must reproduce the above copyright
     12  *    notice, this list of conditions and the following disclaimer in the
     13  *    documentation and/or other materials provided with the distribution.
     14  *
     15  * THIS SOFTWARE IS PROVIDED BY APPLE COMPUTER, INC. ``AS IS'' AND ANY
     16  * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
     17  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
     18  * PURPOSE ARE DISCLAIMED.  IN NO EVENT SHALL APPLE COMPUTER, INC. OR
     19  * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
     20  * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
     21  * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
     22  * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY
     23  * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
     24  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
     25  * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
     26  */
     27 
     28 #include "config.h"
     29 #include "PluginPackage.h"
     30 
     31 #include "MIMETypeRegistry.h"
     32 #include "PluginDatabase.h"
     33 #include "PluginDebug.h"
     34 #include "Timer.h"
     35 #include "npruntime_impl.h"
     36 #include <string.h>
     37 #include <wtf/OwnArrayPtr.h>
     38 #include <wtf/text/CString.h>
     39 
     40 namespace WebCore {
     41 
     42 PluginPackage::~PluginPackage()
     43 {
     44     // This destructor gets called during refresh() if PluginDatabase's
     45     // PluginSet hash is already populated, as it removes items from
     46     // the hash table. Calling the destructor on a loaded plug-in of
     47     // course would cause a crash, so we check to call unload before we
     48     // ASSERT.
     49     // FIXME: There is probably a better way to fix this.
     50     if (!m_loadCount)
     51         unloadWithoutShutdown();
     52     else
     53         unload();
     54 
     55     ASSERT(!m_isLoaded);
     56 }
     57 
     58 void PluginPackage::freeLibrarySoon()
     59 {
     60     ASSERT(!m_freeLibraryTimer.isActive());
     61     ASSERT(m_module);
     62     ASSERT(!m_loadCount);
     63 
     64     m_freeLibraryTimer.startOneShot(0);
     65 }
     66 
     67 void PluginPackage::freeLibraryTimerFired(Timer<PluginPackage>*)
     68 {
     69     ASSERT(m_module);
     70     ASSERT(!m_loadCount);
     71 
     72     unloadModule(m_module);
     73     m_module = 0;
     74 }
     75 
     76 
     77 int PluginPackage::compare(const PluginPackage& compareTo) const
     78 {
     79     // Sort plug-ins that allow multiple instances first.
     80     bool AallowsMultipleInstances = !quirks().contains(PluginQuirkDontAllowMultipleInstances);
     81     bool BallowsMultipleInstances = !compareTo.quirks().contains(PluginQuirkDontAllowMultipleInstances);
     82     if (AallowsMultipleInstances != BallowsMultipleInstances)
     83         return AallowsMultipleInstances ? -1 : 1;
     84 
     85     // Sort plug-ins in a preferred path first.
     86     bool AisInPreferredDirectory = PluginDatabase::isPreferredPluginDirectory(parentDirectory());
     87     bool BisInPreferredDirectory = PluginDatabase::isPreferredPluginDirectory(compareTo.parentDirectory());
     88     if (AisInPreferredDirectory != BisInPreferredDirectory)
     89         return AisInPreferredDirectory ? -1 : 1;
     90 
     91     int diff = strcmp(name().utf8().data(), compareTo.name().utf8().data());
     92     if (diff)
     93         return diff;
     94 
     95     diff = compareFileVersion(compareTo.version());
     96     if (diff)
     97         return diff;
     98 
     99     return strcmp(parentDirectory().utf8().data(), compareTo.parentDirectory().utf8().data());
    100 }
    101 
    102 PluginPackage::PluginPackage(const String& path, const time_t& lastModified)
    103     : m_isEnabled(true)
    104     , m_isLoaded(false)
    105     , m_loadCount(0)
    106     , m_path(path)
    107     , m_moduleVersion(0)
    108     , m_module(0)
    109     , m_lastModified(lastModified)
    110     , m_freeLibraryTimer(this, &PluginPackage::freeLibraryTimerFired)
    111 #if ENABLE(NETSCAPE_PLUGIN_METADATA_CACHE)
    112     , m_infoIsFromCache(true)
    113 #endif
    114 {
    115     m_fileName = pathGetFileName(m_path);
    116     m_parentDirectory = m_path.left(m_path.length() - m_fileName.length() - 1);
    117 }
    118 
    119 #if !OS(SYMBIAN)
    120 void PluginPackage::unload()
    121 {
    122     if (!m_isLoaded)
    123         return;
    124 
    125     if (--m_loadCount > 0)
    126         return;
    127 
    128     m_NPP_Shutdown();
    129 
    130     unloadWithoutShutdown();
    131 }
    132 #endif // !OS(SYMBIAN)
    133 
    134 void PluginPackage::unloadWithoutShutdown()
    135 {
    136     if (!m_isLoaded)
    137         return;
    138 
    139     ASSERT(!m_loadCount);
    140     ASSERT(m_module);
    141 
    142     // <rdar://5530519>: Crash when closing tab with pdf file (Reader 7 only)
    143     // If the plugin has subclassed its parent window, as with Reader 7, we may have
    144     // gotten here by way of the plugin's internal window proc forwarding a message to our
    145     // original window proc. If we free the plugin library from here, we will jump back
    146     // to code we just freed when we return, so delay calling FreeLibrary at least until
    147     // the next message loop
    148     freeLibrarySoon();
    149 
    150     m_isLoaded = false;
    151 }
    152 
    153 void PluginPackage::setEnabled(bool enabled)
    154 {
    155     m_isEnabled = enabled;
    156 }
    157 
    158 PassRefPtr<PluginPackage> PluginPackage::createPackage(const String& path, const time_t& lastModified)
    159 {
    160     RefPtr<PluginPackage> package = adoptRef(new PluginPackage(path, lastModified));
    161 
    162     if (!package->fetchInfo())
    163         return 0;
    164 
    165     return package.release();
    166 }
    167 
    168 #if ENABLE(NETSCAPE_PLUGIN_METADATA_CACHE)
    169 PassRefPtr<PluginPackage> PluginPackage::createPackageFromCache(const String& path, const time_t& lastModified, const String& name, const String& description, const String& mimeDescription)
    170 {
    171     RefPtr<PluginPackage> package = adoptRef(new PluginPackage(path, lastModified));
    172     package->m_name = name;
    173     package->m_description = description;
    174     package->determineModuleVersionFromDescription();
    175     package->setMIMEDescription(mimeDescription);
    176     package->m_infoIsFromCache = true;
    177     return package.release();
    178 }
    179 #endif
    180 
    181 #if defined(XP_UNIX)
    182 void PluginPackage::determineQuirks(const String& mimeType)
    183 {
    184     if (MIMETypeRegistry::isJavaAppletMIMEType(mimeType)) {
    185         // Because a single process cannot create multiple VMs, and we cannot reliably unload a
    186         // Java VM, we cannot unload the Java plugin, or we'll lose reference to our only VM
    187         m_quirks.add(PluginQuirkDontUnloadPlugin);
    188 
    189         // Setting the window region to an empty region causes bad scrolling repaint problems
    190         // with the Java plug-in.
    191         m_quirks.add(PluginQuirkDontClipToZeroRectWhenScrolling);
    192         return;
    193     }
    194 
    195     if (mimeType == "application/x-shockwave-flash") {
    196         static const PlatformModuleVersion flashTenVersion(0x0a000000);
    197 
    198         if (compareFileVersion(flashTenVersion) >= 0) {
    199             // Flash 10.0 b218 doesn't like having a NULL window handle
    200             m_quirks.add(PluginQuirkDontSetNullWindowHandleOnDestroy);
    201 #if PLATFORM(QT)
    202             m_quirks.add(PluginQuirkRequiresGtkToolKit);
    203 #endif
    204         } else {
    205             // Flash 9 and older requests windowless plugins if we return a mozilla user agent
    206             m_quirks.add(PluginQuirkWantsMozillaUserAgent);
    207         }
    208 
    209 #if PLATFORM(QT)
    210         // Flash will crash on repeated calls to SetWindow in windowed mode
    211         m_quirks.add(PluginQuirkDontCallSetWindowMoreThanOnce);
    212 
    213 #if CPU(X86_64)
    214         // 64-bit Flash freezes if right-click is sent in windowless mode
    215         m_quirks.add(PluginQuirkIgnoreRightClickInWindowlessMode);
    216 #endif
    217 #endif
    218 
    219         m_quirks.add(PluginQuirkRequiresDefaultScreenDepth);
    220         m_quirks.add(PluginQuirkThrottleInvalidate);
    221         m_quirks.add(PluginQuirkThrottleWMUserPlusOneMessages);
    222         m_quirks.add(PluginQuirkFlashURLNotifyBug);
    223     }
    224 
    225 #if defined(Q_WS_MAEMO_5) || defined(Q_WS_MAEMO_6)
    226     // Passing a 32-bit depth pixmap to NPAPI plugins is too inefficient. Instead, pass a X Pixmap
    227     // that has same depth as the screen depth since graphics operations are optimized
    228     // for this depth.
    229     m_quirks.add(PluginQuirkRequiresDefaultScreenDepth);
    230 #endif
    231 }
    232 #endif
    233 
    234 #if !OS(WINDOWS)
    235 void PluginPackage::determineModuleVersionFromDescription()
    236 {
    237     // It's a bit lame to detect the plugin version by parsing it
    238     // from the plugin description string, but it doesn't seem that
    239     // version information is available in any standardized way at
    240     // the module level, like in Windows
    241 
    242     if (m_description.isEmpty())
    243         return;
    244 
    245     if (m_description.startsWith("Shockwave Flash") && m_description.length() >= 19) {
    246         // The flash version as a PlatformModuleVersion differs on Unix from Windows
    247         // since the revision can be larger than a 8 bits, so we allow it 16 here and
    248         // push the major/minor up 8 bits. Thus on Unix, Flash's version may be
    249         // 0x0a000000 instead of 0x000a0000.
    250 
    251         Vector<String> versionParts;
    252         m_description.substring(16).split(' ', /*allowEmptyEntries =*/ false, versionParts);
    253         if (versionParts.isEmpty())
    254             return;
    255 
    256         if (versionParts.size() >= 1) {
    257             Vector<String> majorMinorParts;
    258             versionParts[0].split('.', majorMinorParts);
    259             if (majorMinorParts.size() >= 1) {
    260                 bool converted = false;
    261                 unsigned major = majorMinorParts[0].toUInt(&converted);
    262                 if (converted)
    263                     m_moduleVersion = (major & 0xff) << 24;
    264             }
    265             if (majorMinorParts.size() == 2) {
    266                 bool converted = false;
    267                 unsigned minor = majorMinorParts[1].toUInt(&converted);
    268                 if (converted)
    269                     m_moduleVersion |= (minor & 0xff) << 16;
    270             }
    271         }
    272 
    273         if (versionParts.size() >= 2) {
    274             String revision = versionParts[1];
    275             if (revision.length() > 1 && (revision[0] == 'r' || revision[0] == 'b')) {
    276                 revision.remove(0, 1);
    277                 m_moduleVersion |= revision.toInt() & 0xffff;
    278             }
    279         }
    280     }
    281 }
    282 #endif
    283 
    284 #if ENABLE(NETSCAPE_PLUGIN_API)
    285 void PluginPackage::initializeBrowserFuncs()
    286 {
    287     memset(&m_browserFuncs, 0, sizeof(m_browserFuncs));
    288     m_browserFuncs.size = sizeof(m_browserFuncs);
    289     m_browserFuncs.version = NPVersion();
    290 
    291     m_browserFuncs.geturl = NPN_GetURL;
    292     m_browserFuncs.posturl = NPN_PostURL;
    293     m_browserFuncs.requestread = NPN_RequestRead;
    294     m_browserFuncs.newstream = NPN_NewStream;
    295     m_browserFuncs.write = NPN_Write;
    296     m_browserFuncs.destroystream = NPN_DestroyStream;
    297     m_browserFuncs.status = NPN_Status;
    298     m_browserFuncs.uagent = NPN_UserAgent;
    299     m_browserFuncs.memalloc = NPN_MemAlloc;
    300     m_browserFuncs.memfree = NPN_MemFree;
    301     m_browserFuncs.memflush = NPN_MemFlush;
    302     m_browserFuncs.reloadplugins = NPN_ReloadPlugins;
    303     m_browserFuncs.geturlnotify = NPN_GetURLNotify;
    304     m_browserFuncs.posturlnotify = NPN_PostURLNotify;
    305     m_browserFuncs.getvalue = NPN_GetValue;
    306     m_browserFuncs.setvalue = NPN_SetValue;
    307     m_browserFuncs.invalidaterect = NPN_InvalidateRect;
    308     m_browserFuncs.invalidateregion = NPN_InvalidateRegion;
    309     m_browserFuncs.forceredraw = NPN_ForceRedraw;
    310     m_browserFuncs.getJavaEnv = NPN_GetJavaEnv;
    311     m_browserFuncs.getJavaPeer = NPN_GetJavaPeer;
    312     m_browserFuncs.pushpopupsenabledstate = NPN_PushPopupsEnabledState;
    313     m_browserFuncs.poppopupsenabledstate = NPN_PopPopupsEnabledState;
    314     m_browserFuncs.pluginthreadasynccall = NPN_PluginThreadAsyncCall;
    315 
    316     m_browserFuncs.releasevariantvalue = _NPN_ReleaseVariantValue;
    317     m_browserFuncs.getstringidentifier = _NPN_GetStringIdentifier;
    318     m_browserFuncs.getstringidentifiers = _NPN_GetStringIdentifiers;
    319     m_browserFuncs.getintidentifier = _NPN_GetIntIdentifier;
    320     m_browserFuncs.identifierisstring = _NPN_IdentifierIsString;
    321     m_browserFuncs.utf8fromidentifier = _NPN_UTF8FromIdentifier;
    322     m_browserFuncs.intfromidentifier = _NPN_IntFromIdentifier;
    323     m_browserFuncs.createobject = _NPN_CreateObject;
    324     m_browserFuncs.retainobject = _NPN_RetainObject;
    325     m_browserFuncs.releaseobject = _NPN_ReleaseObject;
    326     m_browserFuncs.invoke = _NPN_Invoke;
    327     m_browserFuncs.invokeDefault = _NPN_InvokeDefault;
    328     m_browserFuncs.evaluate = _NPN_Evaluate;
    329     m_browserFuncs.getproperty = _NPN_GetProperty;
    330     m_browserFuncs.setproperty = _NPN_SetProperty;
    331     m_browserFuncs.removeproperty = _NPN_RemoveProperty;
    332     m_browserFuncs.hasproperty = _NPN_HasProperty;
    333     m_browserFuncs.hasmethod = _NPN_HasMethod;
    334     m_browserFuncs.setexception = _NPN_SetException;
    335     m_browserFuncs.enumerate = _NPN_Enumerate;
    336     m_browserFuncs.construct = _NPN_Construct;
    337     m_browserFuncs.getvalueforurl = NPN_GetValueForURL;
    338     m_browserFuncs.setvalueforurl = NPN_SetValueForURL;
    339     m_browserFuncs.getauthenticationinfo = NPN_GetAuthenticationInfo;
    340 }
    341 #endif
    342 
    343 #if ENABLE(PLUGIN_PACKAGE_SIMPLE_HASH)
    344 unsigned PluginPackage::hash() const
    345 {
    346     unsigned hashCodes[] = {
    347         m_path.impl()->hash(),
    348         m_lastModified
    349     };
    350 
    351     return StringHasher::hashMemory<sizeof(hashCodes)>(hashCodes);
    352 }
    353 
    354 bool PluginPackage::equal(const PluginPackage& a, const PluginPackage& b)
    355 {
    356     return a.m_description == b.m_description;
    357 }
    358 #endif
    359 
    360 int PluginPackage::compareFileVersion(const PlatformModuleVersion& compareVersion) const
    361 {
    362     // return -1, 0, or 1 if plug-in version is less than, equal to, or greater than
    363     // the passed version
    364 
    365 #if OS(WINDOWS)
    366     if (m_moduleVersion.mostSig != compareVersion.mostSig)
    367         return m_moduleVersion.mostSig > compareVersion.mostSig ? 1 : -1;
    368     if (m_moduleVersion.leastSig != compareVersion.leastSig)
    369         return m_moduleVersion.leastSig > compareVersion.leastSig ? 1 : -1;
    370 #else
    371     if (m_moduleVersion != compareVersion)
    372         return m_moduleVersion > compareVersion ? 1 : -1;
    373 #endif
    374 
    375     return 0;
    376 }
    377 
    378 #if ENABLE(NETSCAPE_PLUGIN_METADATA_CACHE)
    379 bool PluginPackage::ensurePluginLoaded()
    380 {
    381     if (!m_infoIsFromCache)
    382         return m_isLoaded;
    383 
    384     m_quirks = PluginQuirkSet();
    385     m_name = String();
    386     m_description = String();
    387     m_fullMIMEDescription = String();
    388     m_moduleVersion = 0;
    389 
    390     return fetchInfo();
    391 }
    392 #endif
    393 
    394 }
    395