|
|
/*****************************************************************************
* * ftp.cpp - FTP folder bookkeeping * *****************************************************************************/
#include "priv.h"
#include "ftpinet.h"
#include "ftpsite.h"
#include "ftplist.h"
#include "msieftp.h"
#include "cookie.h"
extern CFtpList * g_FtpSiteCache; extern DWORD g_dwOpenConnections;
/*****************************************************************************
* * Dynamic Globals. There should be as few of these as possible. * * All access to dynamic globals must be thread-safe. * *****************************************************************************/
ULONG g_cRef = 0; /* Global reference count */ CRITICAL_SECTION g_csDll; /* The shared critical section */
extern HANDLE g_hthWorker; // Background worker thread
#ifdef DEBUG
DWORD g_TlsMem = 0xffffffff; extern DWORD g_TLSliStopWatchStartHi; extern DWORD g_TLSliStopWatchStartLo;
LEAKSTRUCT g_LeakList[] = { {0, "CFtpFolder"}, {0, "CFtpDir"}, {0, "CFtpSite"}, {0, "CFtpObj"}, {0, "CFtpEidl"}, {0, "CFtpDrop"}, {0, "CFtpList"}, {0, "CFtpStm"}, {0, "CAccount"}, {0, "CFtpFactory"}, {0, "CFtpContextMenu"}, {0, "CFtpEfe"}, {0, "CFtpGlob"}, {0, "CFtpIcon"}, {0, "CMallocItem"}, {0, "CFtpPidlList"}, {0, "CFtpProp"}, {0, "CStatusBar"}, {0, "CFtpView"}, {0, "CFtpWebView"}, {0, "CCookieList"}, {0, "CDropOperation"} }; #endif // DEBUG
ULONG g_cRef_CFtpView = 0; // Needed to determine when to purge cache.
/*****************************************************************************
* * DllAddRef / DllRelease * * Maintain the DLL reference count. * *****************************************************************************/
void DllAddRef(void) { CREATE_CALLERS_ADDRESS; // For debug spew.
InterlockedIncrement((LPLONG)&g_cRef); TraceMsg(TF_FTPREF, "DllAddRef() g_cRef=%d, called from=%#08lx.", g_cRef, GET_CALLERS_ADDRESS); }
void DllRelease(void) { CREATE_CALLERS_ADDRESS; // For debug spew.
TraceMsg(TF_FTPREF, "DllRelease() g_cRef=%d, called from=%#08lx.", g_cRef-1, GET_CALLERS_ADDRESS); InterlockedDecrement((LPLONG)&g_cRef); }
/*****************************************************************************
* * DllGetClassObject * * OLE entry point. Produces an IClassFactory for the indicated GUID. * * The artificial refcount inside DllGetClassObject helps to * avoid the race condition described in DllCanUnloadNow. It's * not perfect, but it makes the race window much smaller. * *****************************************************************************/
STDAPI DllGetClassObject(REFCLSID rclsid, REFIID riid, LPVOID * ppvObj) { HRESULT hres;
DllAddRef(); if (IsEqualIID(rclsid, CLSID_FtpFolder) || IsEqualIID(rclsid, CLSID_FtpWebView) || IsEqualIID(rclsid, CLSID_FtpDataObject) || IsEqualIID(rclsid, CLSID_FtpInstaller)) { hres = CFtpFactory_Create(rclsid, riid, ppvObj); } else { *ppvObj = NULL; hres = CLASS_E_CLASSNOTAVAILABLE; }
DllRelease(); return hres; }
/*****************************************************************************
* * DllCanUnloadNow * * OLE entry point. Fail iff there are outstanding refs. * * There is an unavoidable race condition between DllCanUnloadNow * and the creation of a new IClassFactory: Between the time we * return from DllCanUnloadNow() and the caller inspects the value, * another thread in the same process may decide to call * DllGetClassObject, thus suddenly creating an object in this DLL * when there previously was none. * * It is the caller's responsibility to prepare for this possibility; * there is nothing we can do about it. * *****************************************************************************/
STDMETHODIMP DllCanUnloadNow(void) { HRESULT hres;
ENTERCRITICALNOASSERT;
// Purge Cache if there aren't any FtpViews open.
if ((0 == g_cRef_CFtpView)) { // Since no views are open, we want to try to purge
// the Delayed Actions so we can closed down the background
// thread. Is it running?
if (AreOutstandingDelayedActions()) { LEAVECRITICALNOASSERT; PurgeDelayedActions(); // Try to close it down.
ENTERCRITICALNOASSERT; }
if (!AreOutstandingDelayedActions()) // Did it close down?
{ // We need to purge the session key because we lost the password
// redirects in the CFtpSites. So we would login but later fail
// when we try to fish out the password when falling back to
// URLMON/shdocfl for file downloads. (NT #362108)
PurgeSessionKey(); CFtpPunkList_Purge(&g_FtpSiteCache); // Yes so purge the cache...
} }
hres = g_cRef ? S_FALSE : S_OK; TraceMsg(TF_FTP_DLLLOADING, "DllCanUnloadNow() DllRefs=%d, returning hres=%#08lx. (S_OK means yes)", g_cRef, hres);
LEAVECRITICALNOASSERT;
return hres; }
void CheckForLeaks(BOOL fForce) { #ifdef DEBUG
DWORD dwLeakCount = 0;
if (fForce) { // Let's free our stuff so we can make sure not to leak it.
// This is done more to force our selves to be w/o leaks
// than anything else.
DllCanUnloadNow(); }
for (int nIndex = 0; nIndex < ARRAYSIZE(g_LeakList); nIndex++) dwLeakCount += g_LeakList[nIndex].dwRef;
if ((!g_FtpSiteCache || fForce) && (dwLeakCount || g_dwOpenConnections || g_cRef)) { TraceMsg(TF_ALWAYS, "***********************************************"); TraceMsg(TF_ALWAYS, "* LEAK - LEAK - LEAK - LEAK - LEAK *"); TraceMsg(TF_ALWAYS, "* *"); TraceMsg(TF_ALWAYS, "* WARNING: The FTP Shell Extension Leaked *"); TraceMsg(TF_ALWAYS, "* one or more objects *"); TraceMsg(TF_ALWAYS, "***********************************************"); TraceMsg(TF_ALWAYS, "* *"); for (int nIndex = 0; nIndex < ARRAYSIZE(g_LeakList); nIndex++) { if (g_LeakList[nIndex].dwRef) TraceMsg(TF_ALWAYS, "* %hs, Leaked=%d *", g_LeakList[nIndex].szObject, g_LeakList[nIndex].dwRef); } TraceMsg(TF_ALWAYS, "* *"); TraceMsg(TF_ALWAYS, "* Open Wininet Connections=%d *", g_dwOpenConnections); TraceMsg(TF_ALWAYS, "* DLL Refs=%d *", g_cRef); TraceMsg(TF_ALWAYS, "* *"); TraceMsg(TF_ALWAYS, "***********************************************"); ASSERT(0); }
#endif // DEBUG
}
// Globals to free.
extern CCookieList * g_pCookieList;
/*****************************************************************************\
DESCRIPTION: DLL entry point. \*****************************************************************************/ STDAPI_(BOOL) DllEntry(HINSTANCE hinst, DWORD dwReason, LPVOID lpReserved) { // This is called in two situations, FreeLibrary() is called and lpReserved is
// NULL, or the process is shutting down and lpReserved is not NULL.
BOOL fIsProcessShuttingDown = (lpReserved ? TRUE : FALSE);
switch (dwReason) { case DLL_PROCESS_ATTACH: SHFusionInitializeFromModule(hinst); InitializeCriticalSection(&g_csDll); #ifdef DEBUG
g_TlsMem = TlsAlloc(); g_TLSliStopWatchStartHi = TlsAlloc(); g_TLSliStopWatchStartLo = TlsAlloc(); #endif
// Don't put it under #ifdef DEBUG
CcshellGetDebugFlags(); DisableThreadLibraryCalls(hinst);
g_hthWorker = NULL;
g_hinst = hinst; g_formatEtcOffsets.cfFormat = (CLIPFORMAT)RegisterClipboardFormat(CFSTR_SHELLIDLISTOFFSET); g_formatPasteSucceeded.cfFormat = (CLIPFORMAT)RegisterClipboardFormat(CFSTR_PASTESUCCEEDED); g_cfTargetCLSID = (CLIPFORMAT)RegisterClipboardFormat(CFSTR_TARGETCLSID);
g_dropTypes[DROP_FCont].cfFormat = (CLIPFORMAT)RegisterClipboardFormat(CFSTR_FILECONTENTS); g_dropTypes[DROP_FGDW].cfFormat = (CLIPFORMAT)RegisterClipboardFormat(CFSTR_FILEDESCRIPTORW); g_dropTypes[DROP_FGDA].cfFormat = (CLIPFORMAT)RegisterClipboardFormat(CFSTR_FILEDESCRIPTORA); g_dropTypes[DROP_IDList].cfFormat = (CLIPFORMAT)RegisterClipboardFormat(CFSTR_SHELLIDLIST); g_dropTypes[DROP_FNMA].cfFormat = (CLIPFORMAT)RegisterClipboardFormat(CFSTR_FILENAMEMAPA); g_dropTypes[DROP_FNMW].cfFormat = (CLIPFORMAT)RegisterClipboardFormat(CFSTR_FILENAMEMAPW); g_dropTypes[DROP_PrefDe].cfFormat = (CLIPFORMAT)RegisterClipboardFormat(CFSTR_PREFERREDDROPEFFECT); g_dropTypes[DROP_PerfDe].cfFormat = (CLIPFORMAT)RegisterClipboardFormat(CFSTR_PERFORMEDDROPEFFECT); g_dropTypes[DROP_FTP_PRIVATE].cfFormat = (CLIPFORMAT)RegisterClipboardFormat(TEXT("FtpPrivateData")); g_dropTypes[DROP_URL].cfFormat = (CLIPFORMAT)RegisterClipboardFormat(TEXT("UniformResourceLocator")); g_dropTypes[DROP_OLEPERSIST].cfFormat = (CLIPFORMAT)RegisterClipboardFormat(TEXT("OleClipboardPersistOnFlush"));
GetModuleFileNameA(GetModuleHandle(TEXT("SHELL32")), g_szShell32, ARRAYSIZE(g_szShell32));
if (FAILED(CFtpSite_Init())) return 0;
break;
case DLL_PROCESS_DETACH: { CCookieList * pCookieList = (CCookieList *) InterlockedExchangePointer((void **) &g_pCookieList, NULL); if (pCookieList) delete pCookieList;
// Yes, so we need to make sure all of the CFtpView's have closed down
// or it's really bad to purge the FTP cache of FTP Servers (CFtpSite) and
// their directories (CFtpDir).
ASSERT(0 == g_cRef_CFtpView);
// Now force the Delayed Actions to happen now instead of waiting.
PurgeDelayedActions();
// OndrejS turned this on. It's firing but I think they are false positives. Since FTP
// Folders does so much caching, this is non-trivial to track down. I will turn this off
// until Ondrej has time to verify.
// CheckForLeaks(fIsProcessShuttingDown);
UnloadWininet(); DeleteCriticalSection(&g_csDll); #ifdef DEBUG
if (g_TLSliStopWatchStartHi) { TlsFree(g_TLSliStopWatchStartHi); g_TLSliStopWatchStartHi = NULL; } if (g_TLSliStopWatchStartLo) { TlsFree(g_TLSliStopWatchStartLo); g_TLSliStopWatchStartLo = NULL; } #endif
SHFusionUninitialize(); } break; } return 1; }
|