You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
303 lines
7.4 KiB
303 lines
7.4 KiB
//+--------------------------------------------------------------------------
|
|
//
|
|
// Microsoft Windows
|
|
// Copyright (C) Microsoft Corporation, 1996 - 1999
|
|
//
|
|
// File: mapi.cpp
|
|
//
|
|
// Contents: Cert Server wrapper routines
|
|
//
|
|
//---------------------------------------------------------------------------
|
|
|
|
#include <pch.cpp>
|
|
|
|
#pragma hdrstop
|
|
|
|
#include <ntlsa.h>
|
|
|
|
#define __dwFILE__ __dwFILE_CERTLIB_MAPI_CPP__
|
|
|
|
|
|
#define wszCERTMAPIINFO L"CertServerMapiInfo"
|
|
|
|
|
|
void
|
|
InitLsaString(
|
|
OUT LSA_UNICODE_STRING *plus,
|
|
IN WCHAR const *pwsz,
|
|
IN DWORD cb)
|
|
{
|
|
if (MAXDWORD == cb)
|
|
{
|
|
cb = lstrlenW(pwsz) * sizeof(WCHAR);
|
|
}
|
|
|
|
plus->Buffer = const_cast<WCHAR *>(pwsz);
|
|
plus->Length = (USHORT) cb;
|
|
plus->MaximumLength = plus->Length + sizeof(WCHAR);
|
|
}
|
|
|
|
|
|
HRESULT
|
|
OpenPolicy(
|
|
IN WCHAR const *pwszServerName,
|
|
DWORD DesiredAccess,
|
|
LSA_HANDLE *phPolicy)
|
|
{
|
|
HRESULT hr;
|
|
LSA_OBJECT_ATTRIBUTES oa;
|
|
LSA_UNICODE_STRING ServerString;
|
|
LSA_UNICODE_STRING *plusServer;
|
|
|
|
ZeroMemory(&oa, sizeof(oa));
|
|
|
|
plusServer = NULL;
|
|
if (NULL != pwszServerName)
|
|
{
|
|
InitLsaString(&ServerString, pwszServerName, MAXDWORD);
|
|
plusServer = &ServerString;
|
|
}
|
|
hr = LsaOpenPolicy(plusServer, &oa, DesiredAccess, phPolicy);
|
|
if (!NT_SUCCESS(hr))
|
|
{
|
|
_JumpError(hr, error, "LsaOpenPolicy");
|
|
}
|
|
|
|
error:
|
|
return(hr);
|
|
}
|
|
|
|
|
|
HRESULT
|
|
SaveString(
|
|
IN WCHAR const *pwszIn,
|
|
IN DWORD cwcIn,
|
|
OUT WCHAR **ppwszOut)
|
|
{
|
|
HRESULT hr;
|
|
WCHAR *pwsz;
|
|
|
|
*ppwszOut = NULL;
|
|
pwsz = (WCHAR *) LocalAlloc(LMEM_FIXED, (cwcIn + 1) * sizeof(WCHAR));
|
|
if (NULL == pwsz)
|
|
{
|
|
hr = E_OUTOFMEMORY;
|
|
_JumpError(hr, error, "LocalAlloc");
|
|
}
|
|
CopyMemory(pwsz, pwszIn, cwcIn * sizeof(WCHAR));
|
|
pwsz[cwcIn] = L'\0';
|
|
*ppwszOut = pwsz;
|
|
hr = S_OK;
|
|
|
|
error:
|
|
return(hr);
|
|
}
|
|
|
|
|
|
//+--------------------------------------------------------------------------
|
|
// myGetMapiInfo -- Retrieve a Name/Password from a global LSA secret
|
|
//+--------------------------------------------------------------------------
|
|
|
|
HRESULT
|
|
myGetMapiInfo(
|
|
OPTIONAL IN WCHAR const *pwszServerName,
|
|
OUT WCHAR **ppwszProfileName,
|
|
OUT WCHAR **ppwszLogonName,
|
|
OUT WCHAR **ppwszPassword)
|
|
{
|
|
HRESULT hr;
|
|
LSA_HANDLE hPolicy;
|
|
UNICODE_STRING lusSecretKeyName;
|
|
UNICODE_STRING *plusSecretData = NULL;
|
|
DWORD cwc;
|
|
DWORD cwcProfileName;
|
|
DWORD cwcLogonName;
|
|
DWORD cwcPassword;
|
|
WCHAR const *pwsz;
|
|
WCHAR *pwszProfileName = NULL;
|
|
WCHAR *pwszLogonName = NULL;
|
|
WCHAR *pwszPassword = NULL;
|
|
|
|
cwcProfileName = 0;
|
|
cwcLogonName = 0;
|
|
cwcPassword = 0;
|
|
if (NULL == ppwszProfileName ||
|
|
NULL == ppwszLogonName ||
|
|
NULL == ppwszPassword)
|
|
{
|
|
hr = E_POINTER;
|
|
_JumpError(hr, error, "no data");
|
|
}
|
|
|
|
InitLsaString(&lusSecretKeyName, wszCERTMAPIINFO, MAXDWORD);
|
|
|
|
hr = OpenPolicy(pwszServerName, POLICY_GET_PRIVATE_INFORMATION, &hPolicy);
|
|
if (!NT_SUCCESS(hr))
|
|
{
|
|
_JumpError(hr, error, "OpenPolicy");
|
|
}
|
|
|
|
hr = LsaRetrievePrivateData(hPolicy, &lusSecretKeyName, &plusSecretData);
|
|
LsaClose(hPolicy);
|
|
if (!NT_SUCCESS(hr))
|
|
{
|
|
_PrintError2(hr, "LsaRetrievePrivateData", STATUS_OBJECT_NAME_NOT_FOUND);
|
|
if ((HRESULT) STATUS_OBJECT_NAME_NOT_FOUND == hr)
|
|
{
|
|
hr = HRESULT_FROM_WIN32(ERROR_FILE_NOT_FOUND);
|
|
}
|
|
_JumpError2(hr, error, "LsaRetrievePrivateData", hr);
|
|
}
|
|
|
|
if (NULL == plusSecretData || NULL == plusSecretData->Buffer)
|
|
{
|
|
hr = HRESULT_FROM_WIN32(ERROR_FILE_NOT_FOUND);
|
|
_JumpError(hr, error, "no data");
|
|
}
|
|
|
|
pwsz = (WCHAR const *) plusSecretData->Buffer;
|
|
cwc = plusSecretData->Length / sizeof(WCHAR);
|
|
|
|
for (cwcProfileName = 0; cwcProfileName < cwc; cwcProfileName++)
|
|
{
|
|
if (L'\0' == pwsz[cwcProfileName])
|
|
{
|
|
break;
|
|
}
|
|
}
|
|
if (cwcProfileName == cwc)
|
|
{
|
|
hr = HRESULT_FROM_WIN32(ERROR_INVALID_DATA);
|
|
_JumpError(hr, error, "bad data");
|
|
}
|
|
|
|
for (cwcLogonName = cwcProfileName + 1; cwcLogonName < cwc; cwcLogonName++)
|
|
{
|
|
if (L'\0' == pwsz[cwcLogonName])
|
|
{
|
|
break;
|
|
}
|
|
}
|
|
if (cwcLogonName == cwc)
|
|
{
|
|
hr = HRESULT_FROM_WIN32(ERROR_INVALID_DATA);
|
|
_JumpError(hr, error, "bad data");
|
|
}
|
|
cwcLogonName -= cwcProfileName + 1;
|
|
|
|
cwcPassword = cwc - (cwcProfileName + 1 + cwcLogonName + 1);
|
|
|
|
hr = SaveString(pwsz, cwcProfileName, &pwszProfileName);
|
|
_JumpIfError(hr, error, "SaveString");
|
|
|
|
hr = SaveString(&pwsz[cwcProfileName + 1], cwcLogonName, &pwszLogonName);
|
|
_JumpIfError(hr, error, "SaveString");
|
|
|
|
hr = SaveString(
|
|
&pwsz[cwcProfileName + 1 + cwcLogonName + 1],
|
|
cwcPassword,
|
|
&pwszPassword);
|
|
_JumpIfError(hr, error, "SaveString");
|
|
|
|
*ppwszProfileName = pwszProfileName;
|
|
pwszProfileName = NULL;
|
|
|
|
*ppwszLogonName = pwszLogonName;
|
|
pwszLogonName = NULL;
|
|
|
|
*ppwszPassword = pwszPassword;
|
|
pwszPassword = NULL;
|
|
|
|
error:
|
|
if (NULL != pwszProfileName)
|
|
{
|
|
SecureZeroMemory(pwszProfileName, cwcProfileName * sizeof(WCHAR));
|
|
LocalFree(pwszProfileName);
|
|
}
|
|
if (NULL != pwszLogonName)
|
|
{
|
|
SecureZeroMemory(pwszLogonName, cwcLogonName * sizeof(WCHAR));
|
|
LocalFree(pwszLogonName);
|
|
}
|
|
if (NULL != pwszPassword)
|
|
{
|
|
SecureZeroMemory(pwszPassword, cwcPassword * sizeof(WCHAR));
|
|
LocalFree(pwszPassword);
|
|
}
|
|
if (NULL != plusSecretData)
|
|
{
|
|
if (NULL != plusSecretData->Buffer)
|
|
{
|
|
SecureZeroMemory(plusSecretData->Buffer, plusSecretData->Length);
|
|
}
|
|
LsaFreeMemory(plusSecretData);
|
|
}
|
|
return(hr);
|
|
}
|
|
|
|
|
|
//+--------------------------------------------------------------------------
|
|
// mySaveMapiInfo -- Persist the specified Name/Password to a global LSA secret
|
|
//+--------------------------------------------------------------------------
|
|
|
|
HRESULT
|
|
mySaveMapiInfo(
|
|
OPTIONAL IN WCHAR const *pwszServerName,
|
|
OUT WCHAR const *pwszProfileName,
|
|
OUT WCHAR const *pwszLogonName,
|
|
OUT WCHAR const *pwszPassword)
|
|
{
|
|
HRESULT hr;
|
|
LSA_HANDLE hPolicy;
|
|
UNICODE_STRING lusSecretKeyName;
|
|
UNICODE_STRING lusSecretData;
|
|
WCHAR wszSecret[MAX_PATH];
|
|
DWORD cwc;
|
|
WCHAR *pwsz;
|
|
|
|
if (NULL == pwszProfileName ||
|
|
NULL == pwszLogonName ||
|
|
NULL == pwszPassword)
|
|
{
|
|
hr = E_POINTER;
|
|
_JumpError(hr, error, "NULL parm");
|
|
}
|
|
cwc = lstrlen(pwszProfileName) + 1 +
|
|
lstrlen(pwszLogonName) + 1 +
|
|
lstrlen(pwszPassword);
|
|
|
|
if (ARRAYSIZE(wszSecret) <= cwc)
|
|
{
|
|
hr = HRESULT_FROM_WIN32(ERROR_BUFFER_OVERFLOW);
|
|
_JumpError(hr, error, "overflow");
|
|
}
|
|
pwsz = wszSecret;
|
|
wcscpy(pwsz, pwszProfileName);
|
|
|
|
pwsz += lstrlen(pwsz) + 1;
|
|
wcscpy(pwsz, pwszLogonName);
|
|
|
|
pwsz += lstrlen(pwsz) + 1;
|
|
wcscpy(pwsz, pwszPassword);
|
|
|
|
InitLsaString(&lusSecretData, wszSecret, cwc * sizeof(WCHAR));
|
|
InitLsaString(&lusSecretKeyName, wszCERTMAPIINFO, MAXDWORD);
|
|
|
|
hr = OpenPolicy(pwszServerName, POLICY_CREATE_SECRET, &hPolicy);
|
|
if (!NT_SUCCESS(hr))
|
|
{
|
|
_JumpError(hr, error, "OpenPolicy");
|
|
}
|
|
|
|
hr = LsaStorePrivateData(hPolicy, &lusSecretKeyName, &lusSecretData);
|
|
LsaClose(hPolicy);
|
|
if (!NT_SUCCESS(hr))
|
|
{
|
|
_JumpError(hr, error, "LsaStorePrivateData");
|
|
}
|
|
hr = S_OK;
|
|
|
|
error:
|
|
SecureZeroMemory(wszSecret, sizeof(wszSecret));
|
|
return(hr);
|
|
}
|