Windows2000/private/shell/ext/url/init.cpp
2020-09-30 17:12:32 +02:00

396 lines
9.6 KiB
C++

/*
* init.c - DLL startup routines module.
*/
/* Headers
*/
#include "project.h"
#pragma hdrstop
#include "shlwapi.h"
#include "shlobj.h"
#include "init.h"
#include "clsfact.h"
#include "inetcpl.h"
#include "refcount.hpp"
#include "shlstock.h"
#include "olestock.h"
#include "ftps.hpp" /* for CLSID_MIMEFileTypesPropSheetHook */
#include "inetps.hpp" /* for CLSID_Internet */
#include "shguidp.h" // for CLSID_URLExecHook
#include "cfmacros.h" // static class factory macros
#define MLUI_INIT
#include <mluisupp.h>
/****** Public Functions *****/
/* Declare _main() so we can link with the CRT lib, but not have to
** use the DllMainCRTStartup entry point.
*/
void
_cdecl
main(void)
{
}
#pragma warning(disable:4100) /* "unreferenced formal parameter" warning */
#ifdef MAINWIN
BOOL WINAPI DllMain(HINSTANCE hModule, DWORD dwReason, PVOID pvReserved);
extern "C" BOOL url_DllMain(
IN HINSTANCE DllHandle,
IN DWORD Reason,
IN LPVOID Reserved
)
{
return DllMain(DllHandle,Reason,Reserved);
}
#endif
/*
** DllMain()
**
**
**
** Arguments:
**
** Returns:
**
** Side Effects: none
*/
PUBLIC_CODE BOOL WINAPI DllMain(HINSTANCE hModule, DWORD dwReason,
PVOID pvReserved)
{
BOOL bResult;
DebugEntry(DllMain);
/* Validate dwReason below. */
/* pvReserved may be any value. */
ASSERT(IS_VALID_HANDLE(hModule, MODULE));
switch (dwReason)
{
case DLL_PROCESS_ATTACH:
MLLoadResources(hModule, TEXT("urllc.dll"));
bResult = AttachProcess(hModule);
break;
case DLL_PROCESS_DETACH:
MLFreeResources(hModule);
bResult = DetachProcess(hModule);
break;
case DLL_THREAD_ATTACH:
bResult = AttachThread(hModule);
break;
case DLL_THREAD_DETACH:
bResult = DetachThread(hModule);
break;
default:
ERROR_OUT(("LibMain() called with unrecognized dwReason %lu.",
dwReason));
bResult = FALSE;
break;
}
DebugExitBOOL(DllMain, bResult);
return(bResult);
}
UINT
WhichPlatform(void)
{
HINSTANCE hinst;
// in retail we cache this info
// in debug we always re-fetch it (so people can switch)
#ifdef DEBUG
UINT uInstall = PLATFORM_UNKNOWN;
#else
static UINT uInstall = PLATFORM_UNKNOWN;
if (uInstall != PLATFORM_UNKNOWN)
return uInstall;
#endif
hinst = GetModuleHandle(TEXT("SHDOCVW.DLL"));
if (hinst)
{
// NOTE: GetProcAddress always takes ANSI strings!
DLLGETVERSIONPROC pfnGetVersion =
(DLLGETVERSIONPROC)GetProcAddress(hinst, "DllGetVersion");
if (pfnGetVersion)
{
DLLVERSIONINFO info;
info.cbSize = sizeof(info);
if (SUCCEEDED(pfnGetVersion(&info)))
{
if (4 <= info.dwMajorVersion &&
71 <= info.dwMinorVersion &&
429 <= info.dwBuildNumber)
{
uInstall = PLATFORM_INTEGRATED;
}
else
uInstall = PLATFORM_IE3;
}
}
#ifdef DEBUG
// To allow easier debugging, we can override the platform by
// setting InstallPlatform value in ccshell.ini.
{
UINT uInstallT = GetPrivateProfileInt(TEXT("urldebugoptions"),
TEXT("InstallPlatform"),
PLATFORM_UNKNOWN,
TEXT("ohare.ini"));
if (PLATFORM_UNKNOWN != uInstallT)
{
TRACE_OUT((" ***Overriding real platform installation***\r\n"));
uInstall = uInstallT;
}
}
switch (uInstall)
{
case PLATFORM_IE3:
TRACE_OUT((" ***Assuming IE3***\r\n"));
break;
case PLATFORM_INTEGRATED:
TRACE_OUT((" ***Assuming Nashville***\r\n"));
break;
}
#endif
}
return uInstall;
}
#pragma data_seg(DATA_SEG_PER_INSTANCE)
// DLL reference count == number of class factories +
// number of URLs +
// LockServer() count
PRIVATE_DATA ULONG s_ulcDLLRef = 0;
#pragma data_seg()
PUBLIC_CODE ULONG DLLAddRef(void)
{
ULONG ulcRef;
ASSERT(s_ulcDLLRef < ULONG_MAX);
ulcRef = ++s_ulcDLLRef;
TRACE_OUT(("DLLAddRef(): DLL reference count is now %lu.",
ulcRef));
return(ulcRef);
}
PUBLIC_CODE ULONG DLLRelease(void)
{
ULONG ulcRef;
if (EVAL(s_ulcDLLRef > 0))
s_ulcDLLRef--;
ulcRef = s_ulcDLLRef;
TRACE_OUT(("DLLRelease(): DLL reference count is now %lu.",
ulcRef));
return(ulcRef);
}
PUBLIC_CODE PULONG GetDLLRefCountPtr(void)
{
// BUGBUG: this is dangerous. Shouldn't be used like this.
return(&s_ulcDLLRef);
}
typedef HRESULT (CALLBACK* DLLGETCLASSOBJECTPROC)(REFCLSID, REFIID, void**);
STDMETHODIMP_(BOOL)
PatchForNashville(
REFCLSID rclsid,
REFIID riid,
void **ppv,
HRESULT * phres)
{
BOOL bRet = FALSE;
*phres = CLASS_E_CLASSNOTAVAILABLE; // assume error
if (IsEqualIID(rclsid, CLSID_InternetShortcut) &&
PLATFORM_INTEGRATED == WhichPlatform())
{
HINSTANCE hinst;
// Normally we can just patch the registry. But there is a valid
// case where url.dll is the InprocServer, and that is when the
// user has chosen to uninstall IE 4.0 and we haven't restarted
// the machine yet. In this case, we don't want to patch the
// registry. Use the "MayChangeDefaultMenu" as an indication
// of whether we should really patch it or not.
// Are we uninstalling IE 4.0?
if (NO_ERROR == SHGetValue(HKEY_CLASSES_ROOT,
TEXT("CLSID\\{FBF23B40-E3F0-101B-8488-00AA003E56F8}\\shellex\\MayChangeDefaultMenu"),
TEXT(""),
NULL, NULL, NULL))
{
// No; patch the registry so shdocvw is the handler again
SetRegKeyValue(HKEY_CLASSES_ROOT,
"CLSID\\{FBF23B40-E3F0-101B-8488-00AA003E56F8}\\InProcServer32",
NULL,
REG_SZ,
(PCBYTE)"shdocvw.dll", sizeof("shdocvw.dll"));
// Now call shdocvw's DllGetClassObject
hinst = GetModuleHandle(TEXT("SHDOCVW.DLL"));
if (hinst)
{
DLLGETCLASSOBJECTPROC pfn = (DLLGETCLASSOBJECTPROC)GetProcAddress(hinst, "DllGetClassObject");
if (pfn)
{
*phres = pfn(rclsid, riid, ppv);
bRet = TRUE;
}
}
}
}
return bRet;
}
STDAPI CreateInstance_Intshcut(IUnknown *punkOuter, REFIID riid, void **ppvOut);
STDAPI CreateInstance_MIMEHook(IUnknown *punkOuter, REFIID riid, void **ppvOut);
STDAPI CreateInstance_Internet(IUnknown *punkOuter, REFIID riid, void **ppvOut);
STDAPI CreateInstance_URLExec(IUnknown *punkOuter, REFIID riid, void **ppvOut);
// ClassFactory methods.
STDMETHODIMP CClassFactory::QueryInterface(REFIID riid, void **ppvObj)
{
if (IsEqualIID(riid, IID_IClassFactory) || IsEqualIID(riid, IID_IUnknown))
{
*ppvObj = (void *)GET_ICLASSFACTORY(this);
DLLAddRef();
return NOERROR;
}
*ppvObj = NULL;
return E_NOINTERFACE;
}
STDMETHODIMP_(ULONG) CClassFactory::AddRef()
{
return DLLAddRef();
}
STDMETHODIMP_(ULONG) CClassFactory::Release()
{
return DLLRelease();
}
STDMETHODIMP CClassFactory::CreateInstance(IUnknown *punkOuter, REFIID riid, void **ppvObject)
{
CObjectInfo *localthis = (CObjectInfo*)(this);
return localthis->pfnCreate(punkOuter, riid, ppvObject);
}
STDMETHODIMP CClassFactory::LockServer(BOOL fLock)
{
if (fLock)
DLLAddRef();
else
DLLRelease();
return S_OK;
}
// we always do a linear search here so put your most often used things first
CF_TABLE_BEGIN(c_clsmap)
CF_TABLE_ENTRY(&CLSID_URLExecHook, CreateInstance_URLExec)
CF_TABLE_ENTRY(&CLSID_InternetShortcut, CreateInstance_Intshcut)
CF_TABLE_ENTRY(&CLSID_MIMEFileTypesPropSheetHook, CreateInstance_MIMEHook)
CF_TABLE_ENTRY(&CLSID_Internet, CreateInstance_Internet)
CF_TABLE_END(c_clsmap)
STDAPI DllGetClassObject(REFCLSID rclsid, REFIID riid, void **ppv)
{
HRESULT hres = CLASS_E_CLASSNOTAVAILABLE;
*ppv = NULL; // assume error
if (IsEqualIID(riid, IID_IClassFactory))
{
// Under Nashville, the internet shortcuts are handled by shdocvw.
// It is possible that the user installed Netscape after installing
// Nashville, which would cause url.dll to be the handler again
// for internet shortcuts. This patches the registry and calls
// shdocvw's DllGetClassObject if we're in that scenario.
// Did we patch for nashville?
if ( !PatchForNashville(rclsid, riid, ppv, &hres) )
{
// No; carry on...
const CObjectInfo *pcls;
for (pcls = c_clsmap; pcls->pclsid; pcls++)
{
if (IsEqualIID(rclsid, *(pcls->pclsid)))
{
*ppv = (void *)GET_ICLASSFACTORY(pcls);
DLLAddRef(); // creation of the CF, CF holds DLL Ref count
return NOERROR;
}
}
}
}
return hres;
}
STDAPI DllCanUnloadNow(void)
{
if (s_ulcDLLRef > 0)
return S_FALSE;
return InternetCPLCanUnloadNow();
}
#pragma warning(default:4100) /* "unreferenced formal parameter" warning */