1187 lines
38 KiB
C
1187 lines
38 KiB
C
/* Copyright (C) 2004 Juan Lang
|
|
*
|
|
* This file implements loading of SSP DLLs.
|
|
*
|
|
* This library is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
* License as published by the Free Software Foundation; either
|
|
* version 2.1 of the License, or (at your option) any later version.
|
|
*
|
|
* This library is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
* Lesser General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
* License along with this library; if not, write to the Free Software
|
|
* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
|
|
*/
|
|
#include <assert.h>
|
|
#include <stdarg.h>
|
|
|
|
#include "ntstatus.h"
|
|
#define WIN32_NO_STATUS
|
|
#include "windef.h"
|
|
#include "winbase.h"
|
|
#include "winnls.h"
|
|
#include "winreg.h"
|
|
#include "winternl.h"
|
|
#include "shlwapi.h"
|
|
#include "sspi.h"
|
|
#include "secur32_priv.h"
|
|
#include "secext.h"
|
|
#include "ntsecapi.h"
|
|
#include "thunks.h"
|
|
#include "lmcons.h"
|
|
|
|
#include "wine/list.h"
|
|
#include "wine/debug.h"
|
|
|
|
WINE_DEFAULT_DEBUG_CHANNEL(secur32);
|
|
|
|
/**
|
|
* Type definitions
|
|
*/
|
|
|
|
typedef struct _SecurePackageTable
|
|
{
|
|
DWORD numPackages;
|
|
DWORD numAllocated;
|
|
struct list table;
|
|
} SecurePackageTable;
|
|
|
|
typedef struct _SecureProviderTable
|
|
{
|
|
DWORD numProviders;
|
|
DWORD numAllocated;
|
|
struct list table;
|
|
} SecureProviderTable;
|
|
|
|
/**
|
|
* Prototypes
|
|
*/
|
|
|
|
/* Tries to load moduleName as a provider. If successful, enumerates what
|
|
* packages it can and adds them to the package and provider tables. Resizes
|
|
* tables as necessary.
|
|
*/
|
|
static void _tryLoadProvider(PWSTR moduleName);
|
|
|
|
/* Initialization: read securityproviders value and attempt to open each dll
|
|
* there. For each DLL, call _tryLoadProvider to see if it's really an SSP.
|
|
* Two undocumented functions, AddSecurityPackage(A/W) and
|
|
* DeleteSecurityPackage(A/W), seem suspiciously like they'd register or
|
|
* unregister a dll, but I'm not sure.
|
|
*/
|
|
static void SECUR32_initializeProviders(void);
|
|
|
|
/* Frees all loaded packages and providers */
|
|
static void SECUR32_freeProviders(void);
|
|
|
|
/**
|
|
* Globals
|
|
*/
|
|
|
|
static CRITICAL_SECTION cs;
|
|
static SecurePackageTable *packageTable = NULL;
|
|
static SecureProviderTable *providerTable = NULL;
|
|
|
|
static SecurityFunctionTableA securityFunctionTableA = {
|
|
SECURITY_SUPPORT_PROVIDER_INTERFACE_VERSION,
|
|
EnumerateSecurityPackagesA,
|
|
QueryCredentialsAttributesA,
|
|
AcquireCredentialsHandleA,
|
|
FreeCredentialsHandle,
|
|
NULL, /* Reserved2 */
|
|
InitializeSecurityContextA,
|
|
AcceptSecurityContext,
|
|
CompleteAuthToken,
|
|
DeleteSecurityContext,
|
|
ApplyControlToken,
|
|
QueryContextAttributesA,
|
|
ImpersonateSecurityContext,
|
|
RevertSecurityContext,
|
|
MakeSignature,
|
|
VerifySignature,
|
|
FreeContextBuffer,
|
|
QuerySecurityPackageInfoA,
|
|
EncryptMessage, /* Reserved3 */
|
|
DecryptMessage, /* Reserved4 */
|
|
ExportSecurityContext,
|
|
ImportSecurityContextA,
|
|
AddCredentialsA,
|
|
NULL, /* Reserved8 */
|
|
QuerySecurityContextToken,
|
|
EncryptMessage,
|
|
DecryptMessage,
|
|
SetContextAttributesA
|
|
};
|
|
|
|
static SecurityFunctionTableW securityFunctionTableW = {
|
|
SECURITY_SUPPORT_PROVIDER_INTERFACE_VERSION,
|
|
EnumerateSecurityPackagesW,
|
|
QueryCredentialsAttributesW,
|
|
AcquireCredentialsHandleW,
|
|
FreeCredentialsHandle,
|
|
NULL, /* Reserved2 */
|
|
InitializeSecurityContextW,
|
|
AcceptSecurityContext,
|
|
CompleteAuthToken,
|
|
DeleteSecurityContext,
|
|
ApplyControlToken,
|
|
QueryContextAttributesW,
|
|
ImpersonateSecurityContext,
|
|
RevertSecurityContext,
|
|
MakeSignature,
|
|
VerifySignature,
|
|
FreeContextBuffer,
|
|
QuerySecurityPackageInfoW,
|
|
EncryptMessage, /* Reserved3 */
|
|
DecryptMessage, /* Reserved4 */
|
|
ExportSecurityContext,
|
|
ImportSecurityContextW,
|
|
AddCredentialsW,
|
|
NULL, /* Reserved8 */
|
|
QuerySecurityContextToken,
|
|
EncryptMessage,
|
|
DecryptMessage,
|
|
SetContextAttributesW
|
|
};
|
|
|
|
/***********************************************************************
|
|
* InitSecurityInterfaceA (SECUR32.@)
|
|
*/
|
|
PSecurityFunctionTableA WINAPI InitSecurityInterfaceA(void)
|
|
{
|
|
return &securityFunctionTableA;
|
|
}
|
|
|
|
/***********************************************************************
|
|
* InitSecurityInterfaceW (SECUR32.@)
|
|
*/
|
|
PSecurityFunctionTableW WINAPI InitSecurityInterfaceW(void)
|
|
{
|
|
return &securityFunctionTableW;
|
|
}
|
|
|
|
static PWSTR SECUR32_strdupW(PCWSTR str)
|
|
{
|
|
PWSTR ret;
|
|
|
|
if (str)
|
|
{
|
|
ret = HeapAlloc(GetProcessHeap(), 0, (lstrlenW(str) + 1) * sizeof(WCHAR));
|
|
if (ret)
|
|
lstrcpyW(ret, str);
|
|
}
|
|
else
|
|
ret = NULL;
|
|
return ret;
|
|
}
|
|
|
|
PWSTR SECUR32_AllocWideFromMultiByte(PCSTR str)
|
|
{
|
|
PWSTR ret;
|
|
|
|
if (str)
|
|
{
|
|
int charsNeeded = MultiByteToWideChar(CP_ACP, 0, str, -1, NULL, 0);
|
|
|
|
if (charsNeeded)
|
|
{
|
|
ret = HeapAlloc(GetProcessHeap(), 0, charsNeeded * sizeof(WCHAR));
|
|
if (ret)
|
|
MultiByteToWideChar(CP_ACP, 0, str, -1, ret, charsNeeded);
|
|
}
|
|
else
|
|
ret = NULL;
|
|
}
|
|
else
|
|
ret = NULL;
|
|
return ret;
|
|
}
|
|
|
|
PSTR SECUR32_AllocMultiByteFromWide(PCWSTR str)
|
|
{
|
|
PSTR ret;
|
|
|
|
if (str)
|
|
{
|
|
int charsNeeded = WideCharToMultiByte(CP_ACP, 0, str, -1, NULL, 0,
|
|
NULL, NULL);
|
|
|
|
if (charsNeeded)
|
|
{
|
|
ret = HeapAlloc(GetProcessHeap(), 0, charsNeeded);
|
|
if (ret)
|
|
WideCharToMultiByte(CP_ACP, 0, str, -1, ret, charsNeeded,
|
|
NULL, NULL);
|
|
}
|
|
else
|
|
ret = NULL;
|
|
}
|
|
else
|
|
ret = NULL;
|
|
return ret;
|
|
}
|
|
|
|
static void _makeFnTableA(PSecurityFunctionTableA fnTableA,
|
|
const SecurityFunctionTableA *inFnTableA,
|
|
const SecurityFunctionTableW *inFnTableW)
|
|
{
|
|
if (fnTableA)
|
|
{
|
|
if (inFnTableA)
|
|
{
|
|
/* The size of the version 1 table is based on platform sdk's
|
|
* sspi.h, though the sample ssp also provided with platform sdk
|
|
* implies only functions through QuerySecurityPackageInfoA are
|
|
* implemented (yikes)
|
|
*/
|
|
size_t tableSize = inFnTableA->dwVersion == 1 ?
|
|
(const BYTE *)&inFnTableA->SetContextAttributesA -
|
|
(const BYTE *)inFnTableA : sizeof(SecurityFunctionTableA);
|
|
|
|
memcpy(fnTableA, inFnTableA, tableSize);
|
|
/* override this, since we can do it internally anyway */
|
|
fnTableA->QuerySecurityPackageInfoA =
|
|
QuerySecurityPackageInfoA;
|
|
}
|
|
else if (inFnTableW)
|
|
{
|
|
/* functions with thunks */
|
|
if (inFnTableW->AcquireCredentialsHandleW)
|
|
fnTableA->AcquireCredentialsHandleA =
|
|
thunk_AcquireCredentialsHandleA;
|
|
if (inFnTableW->InitializeSecurityContextW)
|
|
fnTableA->InitializeSecurityContextA =
|
|
thunk_InitializeSecurityContextA;
|
|
if (inFnTableW->ImportSecurityContextW)
|
|
fnTableA->ImportSecurityContextA =
|
|
thunk_ImportSecurityContextA;
|
|
if (inFnTableW->AddCredentialsW)
|
|
fnTableA->AddCredentialsA =
|
|
thunk_AddCredentialsA;
|
|
if (inFnTableW->QueryCredentialsAttributesW)
|
|
fnTableA->QueryCredentialsAttributesA =
|
|
thunk_QueryCredentialsAttributesA;
|
|
if (inFnTableW->QueryContextAttributesW)
|
|
fnTableA->QueryContextAttributesA =
|
|
thunk_QueryContextAttributesA;
|
|
if (inFnTableW->SetContextAttributesW)
|
|
fnTableA->SetContextAttributesA =
|
|
thunk_SetContextAttributesA;
|
|
/* this can't be thunked, there's no extra param to know which
|
|
* package to forward to */
|
|
fnTableA->EnumerateSecurityPackagesA = NULL;
|
|
/* functions with no thunks needed */
|
|
fnTableA->AcceptSecurityContext = inFnTableW->AcceptSecurityContext;
|
|
fnTableA->CompleteAuthToken = inFnTableW->CompleteAuthToken;
|
|
fnTableA->DeleteSecurityContext = inFnTableW->DeleteSecurityContext;
|
|
fnTableA->ImpersonateSecurityContext =
|
|
inFnTableW->ImpersonateSecurityContext;
|
|
fnTableA->RevertSecurityContext = inFnTableW->RevertSecurityContext;
|
|
fnTableA->MakeSignature = inFnTableW->MakeSignature;
|
|
fnTableA->VerifySignature = inFnTableW->VerifySignature;
|
|
fnTableA->FreeContextBuffer = inFnTableW->FreeContextBuffer;
|
|
fnTableA->QuerySecurityPackageInfoA =
|
|
QuerySecurityPackageInfoA;
|
|
fnTableA->ExportSecurityContext =
|
|
inFnTableW->ExportSecurityContext;
|
|
fnTableA->QuerySecurityContextToken =
|
|
inFnTableW->QuerySecurityContextToken;
|
|
fnTableA->EncryptMessage = inFnTableW->EncryptMessage;
|
|
fnTableA->DecryptMessage = inFnTableW->DecryptMessage;
|
|
}
|
|
}
|
|
}
|
|
|
|
static void _makeFnTableW(PSecurityFunctionTableW fnTableW,
|
|
const SecurityFunctionTableA *inFnTableA,
|
|
const SecurityFunctionTableW *inFnTableW)
|
|
{
|
|
if (fnTableW)
|
|
{
|
|
if (inFnTableW)
|
|
{
|
|
/* The size of the version 1 table is based on platform sdk's
|
|
* sspi.h, though the sample ssp also provided with platform sdk
|
|
* implies only functions through QuerySecurityPackageInfoA are
|
|
* implemented (yikes)
|
|
*/
|
|
size_t tableSize = inFnTableW->dwVersion == 1 ?
|
|
(const BYTE *)&inFnTableW->SetContextAttributesW -
|
|
(const BYTE *)inFnTableW : sizeof(SecurityFunctionTableW);
|
|
|
|
memcpy(fnTableW, inFnTableW, tableSize);
|
|
/* override this, since we can do it internally anyway */
|
|
fnTableW->QuerySecurityPackageInfoW =
|
|
QuerySecurityPackageInfoW;
|
|
}
|
|
else if (inFnTableA)
|
|
{
|
|
/* functions with thunks */
|
|
if (inFnTableA->AcquireCredentialsHandleA)
|
|
fnTableW->AcquireCredentialsHandleW =
|
|
thunk_AcquireCredentialsHandleW;
|
|
if (inFnTableA->InitializeSecurityContextA)
|
|
fnTableW->InitializeSecurityContextW =
|
|
thunk_InitializeSecurityContextW;
|
|
if (inFnTableA->ImportSecurityContextA)
|
|
fnTableW->ImportSecurityContextW =
|
|
thunk_ImportSecurityContextW;
|
|
if (inFnTableA->AddCredentialsA)
|
|
fnTableW->AddCredentialsW =
|
|
thunk_AddCredentialsW;
|
|
if (inFnTableA->QueryCredentialsAttributesA)
|
|
fnTableW->QueryCredentialsAttributesW =
|
|
thunk_QueryCredentialsAttributesW;
|
|
if (inFnTableA->QueryContextAttributesA)
|
|
fnTableW->QueryContextAttributesW =
|
|
thunk_QueryContextAttributesW;
|
|
if (inFnTableA->SetContextAttributesA)
|
|
fnTableW->SetContextAttributesW =
|
|
thunk_SetContextAttributesW;
|
|
/* this can't be thunked, there's no extra param to know which
|
|
* package to forward to */
|
|
fnTableW->EnumerateSecurityPackagesW = NULL;
|
|
/* functions with no thunks needed */
|
|
fnTableW->AcceptSecurityContext = inFnTableA->AcceptSecurityContext;
|
|
fnTableW->CompleteAuthToken = inFnTableA->CompleteAuthToken;
|
|
fnTableW->DeleteSecurityContext = inFnTableA->DeleteSecurityContext;
|
|
fnTableW->ImpersonateSecurityContext =
|
|
inFnTableA->ImpersonateSecurityContext;
|
|
fnTableW->RevertSecurityContext = inFnTableA->RevertSecurityContext;
|
|
fnTableW->MakeSignature = inFnTableA->MakeSignature;
|
|
fnTableW->VerifySignature = inFnTableA->VerifySignature;
|
|
fnTableW->FreeContextBuffer = inFnTableA->FreeContextBuffer;
|
|
fnTableW->QuerySecurityPackageInfoW =
|
|
QuerySecurityPackageInfoW;
|
|
fnTableW->ExportSecurityContext =
|
|
inFnTableA->ExportSecurityContext;
|
|
fnTableW->QuerySecurityContextToken =
|
|
inFnTableA->QuerySecurityContextToken;
|
|
fnTableW->EncryptMessage = inFnTableA->EncryptMessage;
|
|
fnTableW->DecryptMessage = inFnTableA->DecryptMessage;
|
|
}
|
|
}
|
|
}
|
|
|
|
static void _copyPackageInfo(PSecPkgInfoW info, const SecPkgInfoA *inInfoA,
|
|
const SecPkgInfoW *inInfoW)
|
|
{
|
|
if (info && (inInfoA || inInfoW))
|
|
{
|
|
/* odd, I know, but up until Name and Comment the structures are
|
|
* identical
|
|
*/
|
|
memcpy(info, inInfoW ? inInfoW : (const SecPkgInfoW *)inInfoA, sizeof(*info));
|
|
if (inInfoW)
|
|
{
|
|
info->Name = SECUR32_strdupW(inInfoW->Name);
|
|
info->Comment = SECUR32_strdupW(inInfoW->Comment);
|
|
}
|
|
else
|
|
{
|
|
info->Name = SECUR32_AllocWideFromMultiByte(inInfoA->Name);
|
|
info->Comment = SECUR32_AllocWideFromMultiByte(inInfoA->Comment);
|
|
}
|
|
}
|
|
}
|
|
|
|
SecureProvider *SECUR32_addProvider(const SecurityFunctionTableA *fnTableA,
|
|
const SecurityFunctionTableW *fnTableW, PCWSTR moduleName)
|
|
{
|
|
SecureProvider *ret;
|
|
|
|
EnterCriticalSection(&cs);
|
|
|
|
if (!providerTable)
|
|
{
|
|
providerTable = HeapAlloc(GetProcessHeap(), 0, sizeof(SecureProviderTable));
|
|
if (!providerTable)
|
|
{
|
|
LeaveCriticalSection(&cs);
|
|
return NULL;
|
|
}
|
|
|
|
list_init(&providerTable->table);
|
|
}
|
|
|
|
ret = HeapAlloc(GetProcessHeap(), 0, sizeof(SecureProvider));
|
|
if (!ret)
|
|
{
|
|
LeaveCriticalSection(&cs);
|
|
return NULL;
|
|
}
|
|
|
|
list_add_tail(&providerTable->table, &ret->entry);
|
|
ret->lib = NULL;
|
|
|
|
if (fnTableA || fnTableW)
|
|
{
|
|
ret->moduleName = moduleName ? SECUR32_strdupW(moduleName) : NULL;
|
|
_makeFnTableA(&ret->fnTableA, fnTableA, fnTableW);
|
|
_makeFnTableW(&ret->fnTableW, fnTableA, fnTableW);
|
|
ret->loaded = moduleName ? FALSE : TRUE;
|
|
}
|
|
else
|
|
{
|
|
ret->moduleName = SECUR32_strdupW(moduleName);
|
|
ret->loaded = FALSE;
|
|
}
|
|
|
|
LeaveCriticalSection(&cs);
|
|
return ret;
|
|
}
|
|
|
|
void SECUR32_addPackages(SecureProvider *provider, ULONG toAdd,
|
|
const SecPkgInfoA *infoA, const SecPkgInfoW *infoW)
|
|
{
|
|
ULONG i;
|
|
|
|
assert(provider);
|
|
assert(infoA || infoW);
|
|
|
|
EnterCriticalSection(&cs);
|
|
|
|
if (!packageTable)
|
|
{
|
|
packageTable = HeapAlloc(GetProcessHeap(), 0, sizeof(SecurePackageTable));
|
|
if (!packageTable)
|
|
{
|
|
LeaveCriticalSection(&cs);
|
|
return;
|
|
}
|
|
|
|
packageTable->numPackages = 0;
|
|
list_init(&packageTable->table);
|
|
}
|
|
|
|
for (i = 0; i < toAdd; i++)
|
|
{
|
|
SecurePackage *package = HeapAlloc(GetProcessHeap(), 0, sizeof(SecurePackage));
|
|
if (!package)
|
|
continue;
|
|
|
|
list_add_tail(&packageTable->table, &package->entry);
|
|
|
|
package->provider = provider;
|
|
_copyPackageInfo(&package->infoW,
|
|
infoA ? &infoA[i] : NULL,
|
|
infoW ? &infoW[i] : NULL);
|
|
}
|
|
packageTable->numPackages += toAdd;
|
|
|
|
LeaveCriticalSection(&cs);
|
|
}
|
|
|
|
static void _tryLoadProvider(PWSTR moduleName)
|
|
{
|
|
HMODULE lib = LoadLibraryW(moduleName);
|
|
|
|
if (lib)
|
|
{
|
|
INIT_SECURITY_INTERFACE_W pInitSecurityInterfaceW =
|
|
(INIT_SECURITY_INTERFACE_W)GetProcAddress(lib,
|
|
SECURITY_ENTRYPOINT_ANSIW);
|
|
INIT_SECURITY_INTERFACE_A pInitSecurityInterfaceA =
|
|
(INIT_SECURITY_INTERFACE_A)GetProcAddress(lib,
|
|
SECURITY_ENTRYPOINT_ANSIA);
|
|
|
|
TRACE("loaded %s, InitSecurityInterfaceA is %p, InitSecurityInterfaceW is %p\n",
|
|
debugstr_w(moduleName), pInitSecurityInterfaceA,
|
|
pInitSecurityInterfaceW);
|
|
if (pInitSecurityInterfaceW || pInitSecurityInterfaceA)
|
|
{
|
|
PSecurityFunctionTableA fnTableA = NULL;
|
|
PSecurityFunctionTableW fnTableW = NULL;
|
|
ULONG toAdd = 0;
|
|
PSecPkgInfoA infoA = NULL;
|
|
PSecPkgInfoW infoW = NULL;
|
|
SECURITY_STATUS ret = SEC_E_OK;
|
|
|
|
if (pInitSecurityInterfaceA)
|
|
fnTableA = pInitSecurityInterfaceA();
|
|
if (pInitSecurityInterfaceW)
|
|
fnTableW = pInitSecurityInterfaceW();
|
|
if (fnTableW && fnTableW->EnumerateSecurityPackagesW)
|
|
{
|
|
if (fnTableW != &securityFunctionTableW)
|
|
ret = fnTableW->EnumerateSecurityPackagesW(&toAdd, &infoW);
|
|
else
|
|
TRACE("%s has built-in providers, skip adding\n", debugstr_w(moduleName));
|
|
}
|
|
else if (fnTableA && fnTableA->EnumerateSecurityPackagesA)
|
|
{
|
|
if (fnTableA != &securityFunctionTableA)
|
|
ret = fnTableA->EnumerateSecurityPackagesA(&toAdd, &infoA);
|
|
else
|
|
TRACE("%s has built-in providers, skip adding\n", debugstr_w(moduleName));
|
|
}
|
|
if (ret == SEC_E_OK && toAdd > 0 && (infoW || infoA))
|
|
{
|
|
SecureProvider *provider = SECUR32_addProvider(NULL, NULL,
|
|
moduleName);
|
|
|
|
if (provider)
|
|
SECUR32_addPackages(provider, toAdd, infoA, infoW);
|
|
if (infoW)
|
|
fnTableW->FreeContextBuffer(infoW);
|
|
else
|
|
fnTableA->FreeContextBuffer(infoA);
|
|
}
|
|
}
|
|
FreeLibrary(lib);
|
|
}
|
|
else
|
|
WARN("failed to load %s\n", debugstr_w(moduleName));
|
|
}
|
|
|
|
static const WCHAR securityProvidersKeyW[] = {
|
|
'S','Y','S','T','E','M','\\','C','u','r','r','e','n','t','C','o','n','t','r',
|
|
'o','l','S','e','t','\\','C','o','n','t','r','o','l','\\','S','e','c','u','r',
|
|
'i','t','y','P','r','o','v','i','d','e','r','s','\0'
|
|
};
|
|
static const WCHAR securityProvidersW[] = {
|
|
'S','e','c','u','r','i','t','y','P','r','o','v','i','d','e','r','s',0
|
|
};
|
|
|
|
static void SECUR32_initializeProviders(void)
|
|
{
|
|
HKEY key;
|
|
long apiRet;
|
|
|
|
TRACE("\n");
|
|
InitializeCriticalSection(&cs);
|
|
cs.DebugInfo->Spare[0] = (DWORD_PTR)(__FILE__ ": cs");
|
|
/* First load built-in providers */
|
|
SECUR32_initSchannelSP();
|
|
SECUR32_initNTLMSP();
|
|
/* Load the Negotiate provider last so apps stumble over the working NTLM
|
|
* provider first. Attempting to fix bug #16905 while keeping the
|
|
* application reported on wine-users on 2006-09-12 working. */
|
|
SECUR32_initNegotiateSP();
|
|
/* Now load providers from registry */
|
|
apiRet = RegOpenKeyExW(HKEY_LOCAL_MACHINE, securityProvidersKeyW, 0,
|
|
KEY_READ, &key);
|
|
if (apiRet == ERROR_SUCCESS)
|
|
{
|
|
WCHAR securityPkgNames[MAX_PATH]; /* arbitrary len */
|
|
DWORD size = sizeof(securityPkgNames) / sizeof(WCHAR), type;
|
|
|
|
apiRet = RegQueryValueExW(key, securityProvidersW, NULL, &type,
|
|
(PBYTE)securityPkgNames, &size);
|
|
if (apiRet == ERROR_SUCCESS && type == REG_SZ)
|
|
{
|
|
WCHAR *ptr;
|
|
|
|
size = size / sizeof(WCHAR);
|
|
for (ptr = securityPkgNames;
|
|
ptr < securityPkgNames + size; )
|
|
{
|
|
WCHAR *comma;
|
|
|
|
for (comma = ptr; *comma && *comma != ','; comma++)
|
|
;
|
|
if (*comma == ',')
|
|
*comma = '\0';
|
|
for (; *ptr && isspace(*ptr) && ptr < securityPkgNames + size;
|
|
ptr++)
|
|
;
|
|
if (*ptr)
|
|
_tryLoadProvider(ptr);
|
|
ptr += lstrlenW(ptr) + 1;
|
|
}
|
|
}
|
|
RegCloseKey(key);
|
|
}
|
|
}
|
|
|
|
SecurePackage *SECUR32_findPackageW(PCWSTR packageName)
|
|
{
|
|
SecurePackage *ret = NULL;
|
|
BOOL matched = FALSE;
|
|
|
|
if (packageTable && packageName)
|
|
{
|
|
LIST_FOR_EACH_ENTRY(ret, &packageTable->table, SecurePackage, entry)
|
|
{
|
|
matched = !lstrcmpiW(ret->infoW.Name, packageName);
|
|
if (matched)
|
|
break;
|
|
}
|
|
|
|
if (!matched)
|
|
return NULL;
|
|
|
|
if (ret->provider && !ret->provider->loaded)
|
|
{
|
|
ret->provider->lib = LoadLibraryW(ret->provider->moduleName);
|
|
if (ret->provider->lib)
|
|
{
|
|
INIT_SECURITY_INTERFACE_W pInitSecurityInterfaceW =
|
|
(INIT_SECURITY_INTERFACE_W)GetProcAddress(ret->provider->lib,
|
|
SECURITY_ENTRYPOINT_ANSIW);
|
|
INIT_SECURITY_INTERFACE_A pInitSecurityInterfaceA =
|
|
(INIT_SECURITY_INTERFACE_A)GetProcAddress(ret->provider->lib,
|
|
SECURITY_ENTRYPOINT_ANSIA);
|
|
PSecurityFunctionTableA fnTableA = NULL;
|
|
PSecurityFunctionTableW fnTableW = NULL;
|
|
|
|
if (pInitSecurityInterfaceA)
|
|
fnTableA = pInitSecurityInterfaceA();
|
|
if (pInitSecurityInterfaceW)
|
|
fnTableW = pInitSecurityInterfaceW();
|
|
/* don't update built-in SecurityFunctionTable */
|
|
if (fnTableA != &securityFunctionTableA)
|
|
_makeFnTableA(&ret->provider->fnTableA, fnTableA, fnTableW);
|
|
if (fnTableW != &securityFunctionTableW)
|
|
_makeFnTableW(&ret->provider->fnTableW, fnTableA, fnTableW);
|
|
ret->provider->loaded = TRUE;
|
|
}
|
|
else
|
|
ret = NULL;
|
|
}
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
SecurePackage *SECUR32_findPackageA(PCSTR packageName)
|
|
{
|
|
SecurePackage *ret;
|
|
|
|
if (packageTable && packageName)
|
|
{
|
|
UNICODE_STRING package;
|
|
|
|
RtlCreateUnicodeStringFromAsciiz(&package, packageName);
|
|
ret = SECUR32_findPackageW(package.Buffer);
|
|
RtlFreeUnicodeString(&package);
|
|
}
|
|
else
|
|
ret = NULL;
|
|
return ret;
|
|
}
|
|
|
|
static void SECUR32_freeProviders(void)
|
|
{
|
|
SecurePackage *package;
|
|
SecureProvider *provider;
|
|
|
|
TRACE("\n");
|
|
EnterCriticalSection(&cs);
|
|
|
|
SECUR32_deinitSchannelSP();
|
|
|
|
if (packageTable)
|
|
{
|
|
LIST_FOR_EACH_ENTRY(package, &packageTable->table, SecurePackage, entry)
|
|
{
|
|
HeapFree(GetProcessHeap(), 0, package->infoW.Name);
|
|
HeapFree(GetProcessHeap(), 0, package->infoW.Comment);
|
|
}
|
|
|
|
HeapFree(GetProcessHeap(), 0, packageTable);
|
|
packageTable = NULL;
|
|
}
|
|
|
|
if (providerTable)
|
|
{
|
|
LIST_FOR_EACH_ENTRY(provider, &providerTable->table, SecureProvider, entry)
|
|
{
|
|
HeapFree(GetProcessHeap(), 0, provider->moduleName);
|
|
if (provider->lib)
|
|
FreeLibrary(provider->lib);
|
|
}
|
|
|
|
HeapFree(GetProcessHeap(), 0, providerTable);
|
|
providerTable = NULL;
|
|
}
|
|
|
|
LeaveCriticalSection(&cs);
|
|
cs.DebugInfo->Spare[0] = 0;
|
|
DeleteCriticalSection(&cs);
|
|
}
|
|
|
|
/***********************************************************************
|
|
* FreeContextBuffer (SECUR32.@)
|
|
*
|
|
* Doh--if pv was allocated by a crypto package, this may not be correct.
|
|
* The sample ssp seems to use LocalAlloc/LocalFee, but there doesn't seem to
|
|
* be any guarantee, nor is there an alloc function in secur32.
|
|
*/
|
|
SECURITY_STATUS WINAPI FreeContextBuffer(PVOID pv)
|
|
{
|
|
HeapFree(GetProcessHeap(), 0, pv);
|
|
|
|
return SEC_E_OK;
|
|
}
|
|
|
|
/***********************************************************************
|
|
* EnumerateSecurityPackagesW (SECUR32.@)
|
|
*/
|
|
SECURITY_STATUS WINAPI EnumerateSecurityPackagesW(PULONG pcPackages,
|
|
PSecPkgInfoW *ppPackageInfo)
|
|
{
|
|
SECURITY_STATUS ret = SEC_E_OK;
|
|
|
|
TRACE("(%p, %p)\n", pcPackages, ppPackageInfo);
|
|
|
|
/* windows just crashes if pcPackages or ppPackageInfo is NULL, so will I */
|
|
*pcPackages = 0;
|
|
EnterCriticalSection(&cs);
|
|
if (packageTable)
|
|
{
|
|
SecurePackage *package;
|
|
size_t bytesNeeded;
|
|
|
|
bytesNeeded = packageTable->numPackages * sizeof(SecPkgInfoW);
|
|
LIST_FOR_EACH_ENTRY(package, &packageTable->table, SecurePackage, entry)
|
|
{
|
|
if (package->infoW.Name)
|
|
bytesNeeded += (lstrlenW(package->infoW.Name) + 1) * sizeof(WCHAR);
|
|
if (package->infoW.Comment)
|
|
bytesNeeded += (lstrlenW(package->infoW.Comment) + 1) * sizeof(WCHAR);
|
|
}
|
|
if (bytesNeeded)
|
|
{
|
|
*ppPackageInfo = HeapAlloc(GetProcessHeap(), 0, bytesNeeded);
|
|
if (*ppPackageInfo)
|
|
{
|
|
ULONG i = 0;
|
|
PWSTR nextString;
|
|
|
|
*pcPackages = packageTable->numPackages;
|
|
nextString = (PWSTR)((PBYTE)*ppPackageInfo +
|
|
packageTable->numPackages * sizeof(SecPkgInfoW));
|
|
LIST_FOR_EACH_ENTRY(package, &packageTable->table, SecurePackage, entry)
|
|
{
|
|
PSecPkgInfoW pkgInfo = *ppPackageInfo + i++;
|
|
|
|
*pkgInfo = package->infoW;
|
|
if (package->infoW.Name)
|
|
{
|
|
TRACE("Name[%d] = %s\n", i - 1, debugstr_w(package->infoW.Name));
|
|
pkgInfo->Name = nextString;
|
|
lstrcpyW(nextString, package->infoW.Name);
|
|
nextString += lstrlenW(nextString) + 1;
|
|
}
|
|
else
|
|
pkgInfo->Name = NULL;
|
|
if (package->infoW.Comment)
|
|
{
|
|
TRACE("Comment[%d] = %s\n", i - 1, debugstr_w(package->infoW.Comment));
|
|
pkgInfo->Comment = nextString;
|
|
lstrcpyW(nextString, package->infoW.Comment);
|
|
nextString += lstrlenW(nextString) + 1;
|
|
}
|
|
else
|
|
pkgInfo->Comment = NULL;
|
|
}
|
|
}
|
|
else
|
|
ret = SEC_E_INSUFFICIENT_MEMORY;
|
|
}
|
|
}
|
|
LeaveCriticalSection(&cs);
|
|
TRACE("<-- 0x%08x\n", ret);
|
|
return ret;
|
|
}
|
|
|
|
/* Converts info (which is assumed to be an array of cPackages SecPkgInfoW
|
|
* structures) into an array of SecPkgInfoA structures, which it returns.
|
|
*/
|
|
static PSecPkgInfoA thunk_PSecPkgInfoWToA(ULONG cPackages,
|
|
const SecPkgInfoW *info)
|
|
{
|
|
PSecPkgInfoA ret;
|
|
|
|
if (info)
|
|
{
|
|
size_t bytesNeeded = cPackages * sizeof(SecPkgInfoA);
|
|
ULONG i;
|
|
|
|
for (i = 0; i < cPackages; i++)
|
|
{
|
|
if (info[i].Name)
|
|
bytesNeeded += WideCharToMultiByte(CP_ACP, 0, info[i].Name,
|
|
-1, NULL, 0, NULL, NULL);
|
|
if (info[i].Comment)
|
|
bytesNeeded += WideCharToMultiByte(CP_ACP, 0, info[i].Comment,
|
|
-1, NULL, 0, NULL, NULL);
|
|
}
|
|
ret = HeapAlloc(GetProcessHeap(), 0, bytesNeeded);
|
|
if (ret)
|
|
{
|
|
PSTR nextString;
|
|
|
|
nextString = (PSTR)((PBYTE)ret + cPackages * sizeof(SecPkgInfoA));
|
|
for (i = 0; i < cPackages; i++)
|
|
{
|
|
PSecPkgInfoA pkgInfo = ret + i;
|
|
int bytes;
|
|
|
|
memcpy(pkgInfo, &info[i], sizeof(SecPkgInfoA));
|
|
if (info[i].Name)
|
|
{
|
|
pkgInfo->Name = nextString;
|
|
/* just repeat back to WideCharToMultiByte how many bytes
|
|
* it requires, since we asked it earlier
|
|
*/
|
|
bytes = WideCharToMultiByte(CP_ACP, 0, info[i].Name, -1,
|
|
NULL, 0, NULL, NULL);
|
|
WideCharToMultiByte(CP_ACP, 0, info[i].Name, -1,
|
|
pkgInfo->Name, bytes, NULL, NULL);
|
|
nextString += lstrlenA(nextString) + 1;
|
|
}
|
|
else
|
|
pkgInfo->Name = NULL;
|
|
if (info[i].Comment)
|
|
{
|
|
pkgInfo->Comment = nextString;
|
|
/* just repeat back to WideCharToMultiByte how many bytes
|
|
* it requires, since we asked it earlier
|
|
*/
|
|
bytes = WideCharToMultiByte(CP_ACP, 0, info[i].Comment, -1,
|
|
NULL, 0, NULL, NULL);
|
|
WideCharToMultiByte(CP_ACP, 0, info[i].Comment, -1,
|
|
pkgInfo->Comment, bytes, NULL, NULL);
|
|
nextString += lstrlenA(nextString) + 1;
|
|
}
|
|
else
|
|
pkgInfo->Comment = NULL;
|
|
}
|
|
}
|
|
}
|
|
else
|
|
ret = NULL;
|
|
return ret;
|
|
}
|
|
|
|
/***********************************************************************
|
|
* EnumerateSecurityPackagesA (SECUR32.@)
|
|
*/
|
|
SECURITY_STATUS WINAPI EnumerateSecurityPackagesA(PULONG pcPackages,
|
|
PSecPkgInfoA *ppPackageInfo)
|
|
{
|
|
SECURITY_STATUS ret;
|
|
PSecPkgInfoW info;
|
|
|
|
ret = EnumerateSecurityPackagesW(pcPackages, &info);
|
|
if (ret == SEC_E_OK && *pcPackages && info)
|
|
{
|
|
*ppPackageInfo = thunk_PSecPkgInfoWToA(*pcPackages, info);
|
|
if (*pcPackages && !*ppPackageInfo)
|
|
{
|
|
*pcPackages = 0;
|
|
ret = SEC_E_INSUFFICIENT_MEMORY;
|
|
}
|
|
FreeContextBuffer(info);
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
/***********************************************************************
|
|
* GetComputerObjectNameA (SECUR32.@)
|
|
*
|
|
* Get the local computer's name using the format specified.
|
|
*
|
|
* PARAMS
|
|
* NameFormat [I] The format for the name.
|
|
* lpNameBuffer [O] Pointer to buffer to receive the name.
|
|
* nSize [I/O] Size in characters of buffer.
|
|
*
|
|
* RETURNS
|
|
* TRUE If the name was written to lpNameBuffer.
|
|
* FALSE If the name couldn't be written.
|
|
*
|
|
* NOTES
|
|
* If lpNameBuffer is NULL, then the size of the buffer needed to hold the
|
|
* name will be returned in *nSize.
|
|
*
|
|
* nSize returns the number of characters written when lpNameBuffer is not
|
|
* NULL or the size of the buffer needed to hold the name when the buffer
|
|
* is too short or lpNameBuffer is NULL.
|
|
*
|
|
* It the buffer is too short, ERROR_INSUFFICIENT_BUFFER error will be set.
|
|
*/
|
|
BOOLEAN WINAPI GetComputerObjectNameA(
|
|
EXTENDED_NAME_FORMAT NameFormat, LPSTR lpNameBuffer, PULONG nSize)
|
|
{
|
|
BOOLEAN rc;
|
|
LPWSTR bufferW = NULL;
|
|
ULONG sizeW = *nSize;
|
|
TRACE("(%d %p %p)\n", NameFormat, lpNameBuffer, nSize);
|
|
if (lpNameBuffer) {
|
|
bufferW = HeapAlloc(GetProcessHeap(), 0, sizeW * sizeof(WCHAR));
|
|
if (bufferW == NULL) {
|
|
SetLastError(ERROR_NOT_ENOUGH_MEMORY);
|
|
return FALSE;
|
|
}
|
|
}
|
|
rc = GetComputerObjectNameW(NameFormat, bufferW, &sizeW);
|
|
if (rc && bufferW) {
|
|
ULONG len = WideCharToMultiByte(CP_ACP, 0, bufferW, -1, NULL, 0, NULL, NULL);
|
|
WideCharToMultiByte(CP_ACP, 0, bufferW, -1, lpNameBuffer, *nSize, NULL, NULL);
|
|
*nSize = len;
|
|
}
|
|
else
|
|
*nSize = sizeW;
|
|
HeapFree(GetProcessHeap(), 0, bufferW);
|
|
return rc;
|
|
}
|
|
|
|
/***********************************************************************
|
|
* GetComputerObjectNameW (SECUR32.@)
|
|
*/
|
|
BOOLEAN WINAPI GetComputerObjectNameW(
|
|
EXTENDED_NAME_FORMAT NameFormat, LPWSTR lpNameBuffer, PULONG nSize)
|
|
{
|
|
LSA_HANDLE policyHandle;
|
|
LSA_OBJECT_ATTRIBUTES objectAttributes;
|
|
PPOLICY_DNS_DOMAIN_INFO domainInfo;
|
|
NTSTATUS ntStatus;
|
|
BOOLEAN status;
|
|
TRACE("(%d %p %p)\n", NameFormat, lpNameBuffer, nSize);
|
|
|
|
if (NameFormat == NameUnknown)
|
|
{
|
|
SetLastError(ERROR_INVALID_PARAMETER);
|
|
return FALSE;
|
|
}
|
|
|
|
ZeroMemory(&objectAttributes, sizeof(objectAttributes));
|
|
objectAttributes.Length = sizeof(objectAttributes);
|
|
|
|
ntStatus = LsaOpenPolicy(NULL, &objectAttributes,
|
|
POLICY_VIEW_LOCAL_INFORMATION,
|
|
&policyHandle);
|
|
if (ntStatus != STATUS_SUCCESS)
|
|
{
|
|
SetLastError(LsaNtStatusToWinError(ntStatus));
|
|
WARN("LsaOpenPolicy failed with NT status %u\n", GetLastError());
|
|
return FALSE;
|
|
}
|
|
|
|
ntStatus = LsaQueryInformationPolicy(policyHandle,
|
|
PolicyDnsDomainInformation,
|
|
(PVOID *)&domainInfo);
|
|
if (ntStatus != STATUS_SUCCESS)
|
|
{
|
|
SetLastError(LsaNtStatusToWinError(ntStatus));
|
|
WARN("LsaQueryInformationPolicy failed with NT status %u\n",
|
|
GetLastError());
|
|
LsaClose(policyHandle);
|
|
return FALSE;
|
|
}
|
|
|
|
if (domainInfo->Sid)
|
|
{
|
|
switch (NameFormat)
|
|
{
|
|
case NameSamCompatible:
|
|
{
|
|
WCHAR name[MAX_COMPUTERNAME_LENGTH + 1];
|
|
DWORD size = sizeof(name)/sizeof(name[0]);
|
|
if (GetComputerNameW(name, &size))
|
|
{
|
|
DWORD len = domainInfo->Name.Length + size + 3;
|
|
if (lpNameBuffer)
|
|
{
|
|
if (*nSize < len)
|
|
{
|
|
*nSize = len;
|
|
SetLastError(ERROR_INSUFFICIENT_BUFFER);
|
|
status = FALSE;
|
|
}
|
|
else
|
|
{
|
|
WCHAR bs[] = { '\\', 0 };
|
|
WCHAR ds[] = { '$', 0 };
|
|
lstrcpyW(lpNameBuffer, domainInfo->Name.Buffer);
|
|
lstrcatW(lpNameBuffer, bs);
|
|
lstrcatW(lpNameBuffer, name);
|
|
lstrcatW(lpNameBuffer, ds);
|
|
status = TRUE;
|
|
}
|
|
}
|
|
else /* just requesting length required */
|
|
{
|
|
*nSize = len;
|
|
status = TRUE;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
SetLastError(ERROR_INTERNAL_ERROR);
|
|
status = FALSE;
|
|
}
|
|
}
|
|
break;
|
|
case NameFullyQualifiedDN:
|
|
case NameDisplay:
|
|
case NameUniqueId:
|
|
case NameCanonical:
|
|
case NameUserPrincipal:
|
|
case NameCanonicalEx:
|
|
case NameServicePrincipal:
|
|
case NameDnsDomain:
|
|
FIXME("NameFormat %d not implemented\n", NameFormat);
|
|
SetLastError(ERROR_CANT_ACCESS_DOMAIN_INFO);
|
|
status = FALSE;
|
|
break;
|
|
default:
|
|
SetLastError(ERROR_INVALID_PARAMETER);
|
|
status = FALSE;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
SetLastError(ERROR_CANT_ACCESS_DOMAIN_INFO);
|
|
status = FALSE;
|
|
}
|
|
|
|
LsaFreeMemory(domainInfo);
|
|
LsaClose(policyHandle);
|
|
|
|
return status;
|
|
}
|
|
|
|
/***********************************************************************
|
|
* GetUserNameExA (SECUR32.@)
|
|
*/
|
|
BOOLEAN WINAPI GetUserNameExA(
|
|
EXTENDED_NAME_FORMAT NameFormat, LPSTR lpNameBuffer, PULONG nSize)
|
|
{
|
|
BOOLEAN rc;
|
|
LPWSTR bufferW = NULL;
|
|
ULONG sizeW = *nSize;
|
|
TRACE("(%d %p %p)\n", NameFormat, lpNameBuffer, nSize);
|
|
if (lpNameBuffer) {
|
|
bufferW = HeapAlloc(GetProcessHeap(), 0, sizeW * sizeof(WCHAR));
|
|
if (bufferW == NULL) {
|
|
SetLastError(ERROR_NOT_ENOUGH_MEMORY);
|
|
return FALSE;
|
|
}
|
|
}
|
|
rc = GetUserNameExW(NameFormat, bufferW, &sizeW);
|
|
if (rc) {
|
|
ULONG len = WideCharToMultiByte(CP_ACP, 0, bufferW, -1, NULL, 0, NULL, NULL);
|
|
if (len <= *nSize)
|
|
{
|
|
WideCharToMultiByte(CP_ACP, 0, bufferW, -1, lpNameBuffer, *nSize, NULL, NULL);
|
|
*nSize = len - 1;
|
|
}
|
|
else
|
|
{
|
|
*nSize = len;
|
|
rc = FALSE;
|
|
SetLastError(ERROR_MORE_DATA);
|
|
}
|
|
}
|
|
else
|
|
*nSize = sizeW;
|
|
HeapFree(GetProcessHeap(), 0, bufferW);
|
|
return rc;
|
|
}
|
|
|
|
BOOLEAN WINAPI GetUserNameExW(
|
|
EXTENDED_NAME_FORMAT NameFormat, LPWSTR lpNameBuffer, PULONG nSize)
|
|
{
|
|
BOOLEAN status;
|
|
WCHAR samname[UNLEN + 1 + MAX_COMPUTERNAME_LENGTH + 1];
|
|
LPWSTR out;
|
|
DWORD len;
|
|
TRACE("(%d %p %p)\n", NameFormat, lpNameBuffer, nSize);
|
|
|
|
switch (NameFormat)
|
|
{
|
|
case NameSamCompatible:
|
|
{
|
|
/* This assumes the current user is always a local account */
|
|
len = MAX_COMPUTERNAME_LENGTH + 1;
|
|
if (GetComputerNameW(samname, &len))
|
|
{
|
|
out = samname + lstrlenW(samname);
|
|
*out++ = '\\';
|
|
len = UNLEN + 1;
|
|
if (GetUserNameW(out, &len))
|
|
{
|
|
status = (lstrlenW(samname) < *nSize);
|
|
if (status)
|
|
{
|
|
lstrcpyW(lpNameBuffer, samname);
|
|
*nSize = lstrlenW(samname);
|
|
}
|
|
else
|
|
{
|
|
SetLastError(ERROR_MORE_DATA);
|
|
*nSize = lstrlenW(samname) + 1;
|
|
}
|
|
}
|
|
else
|
|
status = FALSE;
|
|
}
|
|
else
|
|
status = FALSE;
|
|
}
|
|
break;
|
|
case NameUnknown:
|
|
case NameFullyQualifiedDN:
|
|
case NameDisplay:
|
|
case NameUniqueId:
|
|
case NameCanonical:
|
|
case NameUserPrincipal:
|
|
case NameCanonicalEx:
|
|
case NameServicePrincipal:
|
|
case NameDnsDomain:
|
|
SetLastError(ERROR_NONE_MAPPED);
|
|
status = FALSE;
|
|
break;
|
|
default:
|
|
SetLastError(ERROR_INVALID_PARAMETER);
|
|
status = FALSE;
|
|
}
|
|
|
|
return status;
|
|
}
|
|
|
|
BOOLEAN WINAPI TranslateNameA(
|
|
LPCSTR lpAccountName, EXTENDED_NAME_FORMAT AccountNameFormat,
|
|
EXTENDED_NAME_FORMAT DesiredNameFormat, LPSTR lpTranslatedName,
|
|
PULONG nSize)
|
|
{
|
|
FIXME("%p %d %d %p %p\n", lpAccountName, AccountNameFormat,
|
|
DesiredNameFormat, lpTranslatedName, nSize);
|
|
return FALSE;
|
|
}
|
|
|
|
BOOLEAN WINAPI TranslateNameW(
|
|
LPCWSTR lpAccountName, EXTENDED_NAME_FORMAT AccountNameFormat,
|
|
EXTENDED_NAME_FORMAT DesiredNameFormat, LPWSTR lpTranslatedName,
|
|
PULONG nSize)
|
|
{
|
|
FIXME("%p %d %d %p %p\n", lpAccountName, AccountNameFormat,
|
|
DesiredNameFormat, lpTranslatedName, nSize);
|
|
return FALSE;
|
|
}
|
|
|
|
/***********************************************************************
|
|
* DllMain (SECUR32.0)
|
|
*/
|
|
BOOL WINAPI DllMain(HINSTANCE hinstDLL, DWORD fdwReason, LPVOID lpvReserved)
|
|
{
|
|
if (fdwReason == DLL_PROCESS_ATTACH)
|
|
{
|
|
DisableThreadLibraryCalls(hinstDLL);
|
|
SECUR32_initializeProviders();
|
|
}
|
|
else if (fdwReason == DLL_PROCESS_DETACH)
|
|
{
|
|
SECUR32_freeProviders();
|
|
}
|
|
|
|
return TRUE;
|
|
}
|