Source code of Windows XP (NT5)
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.
 
 
 
 
 
 

874 lines
20 KiB

///////////////////////////////////////////////////////////////////////////////
//
// Copyright (c) 1998, Microsoft Corp. All rights reserved.
//
// FILE
//
// iastlimp.cpp
//
// SYNOPSIS
//
// Provides function definitions for the Internet Authentication Service
// Template Library (IASTL).
//
// MODIFICATION HISTORY
//
// 08/09/1998 Original version.
// 10/13/1998 Drop null terminator when converting strings to octets.
// 10/27/1998 Added IASDictionaryView.
// 06/01/1999 Fix bug computing length in setOctetString.
// 01/25/2000 Removed IASDictionaryView.
// 04/14/2000 Added IASDictionary.
// 06/26/2000 Relax the type checking in IASPeekAttribute.
//
///////////////////////////////////////////////////////////////////////////////
//////////
// Must be NT5.0 or higher.
//////////
#if (_WIN32_WINNT < 0x0500)
#error iastlimp.cpp requires NT5.0 or later.
#endif
//////////
// IASTL must be used in conjuction with ATL.
//////////
#ifndef __ATLCOM_H__
#error iastlimp.cpp requires atlcom.h to be included first
#endif
#include <windows.h>
//////////
// IASTL declarations.
//////////
#include <iastl.h>
//////////
// The entire library is contained within the IASTL namespace.
//////////
namespace IASTL {
//////////
// FSM governing IAS components.
//////////
const IASComponent::State IASComponent::fsm[NUM_EVENTS][NUM_STATES] =
{
{ STATE_UNINITIALIZED, STATE_UNEXPECTED, STATE_UNEXPECTED, STATE_UNEXPECTED },
{ STATE_UNEXPECTED, STATE_INITIALIZED, STATE_INITIALIZED, STATE_UNEXPECTED },
{ STATE_UNEXPECTED, STATE_UNEXPECTED, STATE_SUSPENDED, STATE_SUSPENDED },
{ STATE_UNEXPECTED, STATE_UNEXPECTED, STATE_INITIALIZED, STATE_INITIALIZED },
{ STATE_SHUTDOWN, STATE_SHUTDOWN, STATE_UNEXPECTED, STATE_SHUTDOWN }
};
///////////////////////////////////////////////////////////////////////////////
//
// IASComponent
//
///////////////////////////////////////////////////////////////////////////////
HRESULT IASComponent::fireEvent(Event event) throw ()
{
// Check the input parameters.
if (event >= NUM_EVENTS) { return E_UNEXPECTED; }
// Compute the next state.
State next = fsm[event][state];
Lock();
HRESULT hr;
if (next == state)
{
// If we're already in that state, there's nothing to do.
hr = S_OK;
}
else if (next == STATE_UNEXPECTED)
{
// We received an unexpected event.
hr = E_UNEXPECTED;
}
else
{
// Attempt the transition.
hr = attemptTransition(event);
// Only change state if the transition was successful.
if (SUCCEEDED(hr))
{
state = next;
}
}
Unlock();
return hr;
}
///////////////////////////////////////////////////////////////////////////////
//
// IASRequestHandler
//
///////////////////////////////////////////////////////////////////////////////
STDMETHODIMP IASRequestHandler::OnRequest(IRequest* pRequest)
{
if (getState() != IASComponent::STATE_INITIALIZED)
{
pRequest->SetResponse(IAS_RESPONSE_DISCARD_PACKET, IAS_INTERNAL_ERROR);
pRequest->ReturnToSource(IAS_REQUEST_STATUS_ABORT);
}
else
{
onAsyncRequest(pRequest);
}
return S_OK;
}
///////////////////////////////////////////////////////////////////////////////
//
// IASRequestHandlerSync
//
///////////////////////////////////////////////////////////////////////////////
void IASRequestHandlerSync::onAsyncRequest(IRequest* pRequest) throw ()
{
pRequest->ReturnToSource(onSyncRequest(pRequest));
}
//////////
// End of the IASTL namespace.
//////////
}
//////////
// Only pull in the utility classes if necessary.
//////////
#ifdef _IASTLUTL_H_
//////////
// The utility classes are also contained within the IASTL namespace.
//////////
namespace IASTL {
///////////////////////////////////////////////////////////////////////////////
//
// IASAttribute
//
///////////////////////////////////////////////////////////////////////////////
IASAttribute& IASAttribute::operator=(PIASATTRIBUTE attr) throw ()
{
// Check for self-assignment.
if (p != attr)
{
_release();
p = attr;
_addref();
}
return *this;
}
void IASAttribute::attach(PIASATTRIBUTE attr, bool addRef) throw ()
{
_release();
p = attr;
if (addRef) { _addref(); }
}
bool IASAttribute::load(IAttributesRaw* request, DWORD dwId)
{
// Release any existing attribute.
release();
DWORD posCount = 1;
ATTRIBUTEPOSITION pos;
HRESULT hr = request->GetAttributes(&posCount, &pos, 1, &dwId);
if (FAILED(hr))
{
if (hr == HRESULT_FROM_WIN32(ERROR_MORE_DATA))
{
// There was more than one, so release the partial result.
IASAttributeRelease(pos.pAttribute);
}
issue_error(hr);
}
p = (posCount ? pos.pAttribute : NULL);
return p != NULL;
}
bool IASAttribute::load(IAttributesRaw* request, DWORD dwId, IASTYPE itType)
{
// Get an attribute with the given ID.
load(request, dwId);
// Attributes have a fixed type, so if the type doesn't match it must
// be an error.
if (p && p->Value.itType != itType)
{
release();
issue_error(DISP_E_TYPEMISMATCH);
}
return p != NULL;
}
void IASAttribute::store(IAttributesRaw* request) const
{
if (p)
{
ATTRIBUTEPOSITION pos;
pos.pAttribute = p;
HRESULT hr = request->AddAttributes(1, &pos);
if (FAILED(hr))
{
issue_error(hr);
}
}
}
void IASAttribute::setOctetString(DWORD dwLength, const BYTE* lpValue)
{
PBYTE newVal = NULL;
if (dwLength)
{
// Allocate a buffer for the octet string ...
newVal = (PBYTE)CoTaskMemAlloc(dwLength);
if (!newVal) { issue_error(E_OUTOFMEMORY); }
// ... and copy it in.
memcpy(newVal, lpValue, dwLength);
}
// Clear the old value.
clearValue();
// Store the new.
p->Value.OctetString.lpValue = newVal;
p->Value.OctetString.dwLength = dwLength;
p->Value.itType = IASTYPE_OCTET_STRING;
}
void IASAttribute::setOctetString(PCSTR szAnsi)
{
setOctetString((szAnsi ? strlen(szAnsi) : 0), (const BYTE*)szAnsi);
}
void IASAttribute::setOctetString(PCWSTR szWide)
{
// Allocate a buffer for the conversion.
int len = WideCharToMultiByte(CP_ACP, 0, szWide, -1, NULL, 0, NULL, NULL);
PSTR ansi = (PSTR)_alloca(len);
// Convert from wide to ansi.
len = WideCharToMultiByte(CP_ACP, 0, szWide, -1, ansi, len, NULL, NULL);
// Don't include the null-terminator.
if (len) { --len; }
// Set the octet string.
setOctetString(len, (const BYTE*)ansi);
}
void IASAttribute::setString(DWORD dwLength, const BYTE* lpValue)
{
// Reserve space for a null terminator.
LPSTR newVal = (LPSTR)CoTaskMemAlloc(dwLength + 1);
if (!newVal) { issue_error(E_OUTOFMEMORY); }
// Copy in the string ...
memcpy(newVal, lpValue, dwLength);
// ... and add a null terminator.
newVal[dwLength] = 0;
// Clear the old value.
clearValue();
// Store the new value.
p->Value.String.pszAnsi = newVal;
p->Value.String.pszWide = NULL;
p->Value.itType = IASTYPE_STRING;
}
void IASAttribute::setString(PCSTR szAnsi)
{
LPSTR newVal = NULL;
if (szAnsi)
{
// Allocate a buffer for the string ...
size_t nbyte = strlen(szAnsi) + 1;
newVal = (LPSTR)CoTaskMemAlloc(nbyte);
if (!newVal) { issue_error(E_OUTOFMEMORY); }
// ... and copy it in.
memcpy(newVal, szAnsi, nbyte);
}
// Clear the old value.
clearValue();
// Store the new value.
p->Value.String.pszAnsi = newVal;
p->Value.String.pszWide = NULL;
p->Value.itType = IASTYPE_STRING;
}
void IASAttribute::setString(PCWSTR szWide)
{
LPWSTR newVal = NULL;
if (szWide)
{
// Allocate a buffer for the string ...
size_t nbyte = sizeof(WCHAR) * (wcslen(szWide) + 1);
newVal = (LPWSTR)CoTaskMemAlloc(nbyte);
if (!newVal) { issue_error(E_OUTOFMEMORY); }
// ... and copy it in.
memcpy(newVal, szWide, nbyte);
}
// Clear the old value.
clearValue();
// Store the new value.
p->Value.String.pszWide = newVal;
p->Value.String.pszAnsi = NULL;
p->Value.itType = IASTYPE_STRING;
}
void IASAttribute::clearValue() throw ()
{
switch (p->Value.itType)
{
case IASTYPE_STRING:
CoTaskMemFree(p->Value.String.pszAnsi);
CoTaskMemFree(p->Value.String.pszWide);
break;
case IASTYPE_OCTET_STRING:
CoTaskMemFree(p->Value.OctetString.lpValue);
break;
}
p->Value.itType = IASTYPE_INVALID;
}
///////////////////////////////////////////////////////////////////////////////
//
// IASAttributeVector
//
///////////////////////////////////////////////////////////////////////////////
IASAttributeVector::IASAttributeVector() throw ()
: begin_(NULL), end_(NULL), capacity_(0), owner(false)
{ }
IASAttributeVector::IASAttributeVector(size_type N)
: begin_(NULL), end_(NULL), capacity_(0), owner(false)
{
reserve(N);
}
IASAttributeVector::IASAttributeVector(
PATTRIBUTEPOSITION init,
size_type initCap
) throw ()
: begin_(init), end_(begin_), capacity_(initCap), owner(false)
{ }
IASAttributeVector::IASAttributeVector(const IASAttributeVector& v)
: begin_(NULL), end_(NULL), capacity_(0), owner(false)
{
reserve(v.size());
for (const_iterator i = v.begin(); i != v.end(); ++i, ++end_)
{
*end_ = *i;
IASAttributeAddRef(end_->pAttribute);
}
}
IASAttributeVector& IASAttributeVector::operator=(const IASAttributeVector& v)
{
if (this != &v)
{
clear();
reserve(v.size());
for (const_iterator i = v.begin(); i != v.end(); ++i, ++end_)
{
*end_ = *i;
IASAttributeAddRef(end_->pAttribute);
}
}
return *this;
}
IASAttributeVector::~IASAttributeVector() throw ()
{
clear();
if (owner && begin_)
{
CoTaskMemFree(begin_);
}
}
IASAttributeVector::iterator IASAttributeVector::discard(iterator p) throw ()
{
// We now have one less attribute.
--end_;
// Shift over one.
memmove(p, p + 1, (size_t)((PBYTE)end_ - (PBYTE)p));
// The iterator now points to the next element, so no need to update.
return p;
}
IASAttributeVector::iterator IASAttributeVector::fast_discard(iterator p) throw ()
{
// We now have one less attribute.
--end_;
// Use the attribute from the end to fill the empty slot.
*p = *end_;
return p;
}
DWORD IASAttributeVector::load(
IAttributesRaw* request,
DWORD attrIDCount,
LPDWORD attrIDs
)
{
clear();
// Get the desired attributes.
DWORD posCount = capacity_;
HRESULT hr = request->GetAttributes(&posCount,
begin_,
attrIDCount,
attrIDs);
end_ = begin_ + posCount;
if (FAILED(hr))
{
// Maybe the array just wasn't big enough.
if (hr == HRESULT_FROM_WIN32(ERROR_MORE_DATA))
{
// Clear the partial result.
clear();
// Find out how much space we really need.
DWORD needed = 0;
hr = request->GetAttributes(&needed, NULL, attrIDCount, attrIDs);
if (FAILED(hr)) { issue_error(hr); }
// Reserve the necessary space ...
reserve(needed);
// ... and try again.
return load(request, attrIDCount, attrIDs);
}
end_ = begin_;
issue_error(hr);
}
return posCount;
}
DWORD IASAttributeVector::load(IAttributesRaw* request)
{
// Find out how much space we need.
DWORD needed;
HRESULT hr = request->GetAttributeCount(&needed);
if (FAILED(hr)) { issue_error(hr); }
// Make sure we have enough space.
reserve(needed);
return load(request, 0, NULL);
}
void IASAttributeVector::push_back(ATTRIBUTEPOSITION& p, bool addRef) throw ()
{
// Make sure we have enough room for one more attribute.
if (size() == capacity())
{
reserve(empty() ? 1 : 2 * size());
}
if (addRef) { IASAttributeAddRef(p.pAttribute); }
// Store the attribute at the end.
*end_ = p;
// Move the end.
++end_;
}
void IASAttributeVector::remove(IAttributesRaw* request)
{
if (begin_ != end_)
{
HRESULT hr = request->RemoveAttributes(size(), begin_);
if (FAILED(hr)) { issue_error(hr); }
}
}
void IASAttributeVector::store(IAttributesRaw* request) const
{
if (begin_ != end_)
{
HRESULT hr = request->AddAttributes(size(), begin_);
if (FAILED(hr)) { issue_error(hr); }
}
}
void IASAttributeVector::clear() throw ()
{
while (end_ != begin_)
{
--end_;
IASAttributeRelease(end_->pAttribute);
}
}
void IASAttributeVector::reserve(size_type N)
{
// We only worry about growing; shrinking is a nop.
if (N > capacity_)
{
// Allocate memory for the new vector.
PATTRIBUTEPOSITION tmp =
(PATTRIBUTEPOSITION)CoTaskMemAlloc(N * sizeof(ATTRIBUTEPOSITION));
if (tmp == NULL) { issue_error(E_OUTOFMEMORY); }
// Copy the existing attributes into the new array.
size_type size_ = size();
memcpy(tmp, begin_, size_ * sizeof(ATTRIBUTEPOSITION));
// Free the old array if necessary.
if (owner) { CoTaskMemFree(begin_); }
// Update our state to point at the new array.
begin_ = tmp;
end_ = begin_ + size_;
capacity_ = N;
owner = true;
}
}
///////////////////////////////////////////////////////////////////////////////
//
// IASRequest
//
///////////////////////////////////////////////////////////////////////////////
IASRequest::IASRequest(IRequest* request)
: req(request)
{
if (!req)
{
// We don't allow NULL request objects.
issue_error(E_POINTER);
}
// Get the 'raw' counterpart.
checkError(req->QueryInterface(__uuidof(IAttributesRaw), (PVOID*)&raw));
}
IASRequest& IASRequest::operator=(const IASRequest& request) throw ()
{
// Check for self-assignment.
if (this != &request)
{
_release();
req = request.req;
raw = request.raw;
_addref();
}
return *this;
}
DWORD IASRequest::GetAttributes(DWORD dwPosCount,
PATTRIBUTEPOSITION pPositions,
DWORD dwAttrIDCount,
LPDWORD lpdwAttrIDs)
{
DWORD count = dwPosCount;
HRESULT hr = raw->GetAttributes(&count,
pPositions,
dwAttrIDCount,
lpdwAttrIDs);
if (FAILED(hr))
{
if (hr == HRESULT_FROM_WIN32(ERROR_MORE_DATA))
{
// Free the partial results.
while (count--)
{
IASAttributeRelease(pPositions->pAttribute);
pPositions->pAttribute = NULL;
++pPositions;
}
}
issue_error(hr);
}
return count;
}
IASDictionary::IASDictionary(
const WCHAR* const* selectNames,
PCWSTR path
)
: mapSize(0),
nextRowNumber(0),
currentRow(NULL)
{
if (!path)
{
// No path, so get the cached local dictionary.
table = IASGetLocalDictionary();
if (!table) { issue_error(GetLastError()); }
}
else
{
// Otherwise, get an arbitrary dictionary.
HRESULT hr = IASGetDictionary(path, &data, &storage);
if (FAILED(hr)) { issue_error(hr); }
table = &data;
}
// How many columns are selected ?
for (const WCHAR* const* p = selectNames; *p; ++p)
{
++mapSize;
}
// Allocate memory for the map.
selectMap = (PULONG)CoTaskMemAlloc(mapSize * sizeof(ULONG));
if (!selectMap) { issue_error(E_OUTOFMEMORY); }
// Lookup the column names.
for (ULONG i = 0; i < mapSize; ++i)
{
for (ULONG j = 0; j < table->numColumns; ++j)
{
if (!_wcsicmp(selectNames[i], table->columnNames[j]))
{
selectMap[i] = j;
break;
}
}
if (j == table->numColumns)
{
// We didn't find the column.
CoTaskMemFree(selectMap);
issue_error(E_INVALIDARG);
}
}
}
IASDictionary::~IASDictionary() throw ()
{
CoTaskMemFree(selectMap);
}
bool IASDictionary::next() throw ()
{
// Are there any rows left ?
if (nextRowNumber >= table->numRows) { return false; }
// Set currentRow to the next row.
currentRow = table->table + nextRowNumber * table->numColumns;
// Advance nextRowNumber.
++nextRowNumber;
return true;
}
void IASDictionary::reset() throw ()
{
nextRowNumber = 0;
currentRow = NULL;
}
bool IASDictionary::isEmpty(ULONG ordinal) const
{
return V_VT(getVariant(ordinal)) == VT_EMPTY;
}
VARIANT_BOOL IASDictionary::getBool(ULONG ordinal) const
{
const VARIANT* v = getVariant(ordinal);
if (V_VT(v) == VT_BOOL)
{
return V_BOOL(v);
}
else if (V_VT(v) != VT_EMPTY)
{
issue_error(DISP_E_TYPEMISMATCH);
}
return VARIANT_FALSE;
}
BSTR IASDictionary::getBSTR(ULONG ordinal) const
{
const VARIANT* v = getVariant(ordinal);
if (V_VT(v) == VT_BSTR)
{
return V_BSTR(v);
}
else if (V_VT(v) != VT_EMPTY)
{
issue_error(DISP_E_TYPEMISMATCH);
}
return NULL;
}
LONG IASDictionary::getLong(ULONG ordinal) const
{
const VARIANT* v = getVariant(ordinal);
if (V_VT(v) == VT_I4)
{
return V_I4(v);
}
else if (V_VT(v) != VT_EMPTY)
{
issue_error(DISP_E_TYPEMISMATCH);
}
return 0L;
}
const VARIANT* IASDictionary::getVariant(ULONG ordinal) const
{
// Are we positioned on a row ?
if (!currentRow) { issue_error(E_UNEXPECTED); }
// Is the ordinal valid ?
if (ordinal >= mapSize) { issue_error(E_INVALIDARG); }
return currentRow + selectMap[ordinal];
}
//////////
// End of the IASTL namespace.
//////////
}
///////////////////////////////////////////////////////////////////////////////
//
// OctetString conversion macros and functions.
//
///////////////////////////////////////////////////////////////////////////////
PSTR IASOctetStringToAnsi(const IAS_OCTET_STRING& src, PSTR dst) throw ()
{
dst[src.dwLength] = '\0';
return (PSTR)memcpy(dst, src.lpValue, src.dwLength);
}
PWSTR IASOctetStringToWide(const IAS_OCTET_STRING& src, PWSTR dst) throw ()
{
DWORD nChar = MultiByteToWideChar(CP_ACP,
0,
(PSTR)src.lpValue,
src.dwLength,
dst,
src.dwLength);
dst[nChar] = L'\0';
return dst;
}
///////////////////////////////////////////////////////////////////////////////
//
// Miscellaneous functions.
//
///////////////////////////////////////////////////////////////////////////////
// Returns 'true' if the IASTYPE maps to a RADIUS integer attribute.
bool isRadiusInteger(IASTYPE type) throw ()
{
bool retval;
switch (type)
{
case IASTYPE_BOOLEAN:
case IASTYPE_INTEGER:
case IASTYPE_ENUM:
retval = true;
break;
default:
retval = false;
}
return retval;
}
PIASATTRIBUTE IASPeekAttribute(
IAttributesRaw* request,
DWORD dwId,
IASTYPE itType
) throw ()
{
if (request)
{
DWORD posCount = 1;
ATTRIBUTEPOSITION pos;
HRESULT hr = request->GetAttributes(&posCount, &pos, 1, &dwId);
if (posCount == 1)
{
IASAttributeRelease(pos.pAttribute);
if (SUCCEEDED(hr))
{
// There is some confusion between RAS and IAS regarding which
// IASTYPEs to use for various RADIUS attributes, so we'll assume
// any of the RADIUS integer types are interchangeable.
if (itType == pos.pAttribute->Value.itType ||
(isRadiusInteger(itType) &&
isRadiusInteger(pos.pAttribute->Value.itType)))
{
return pos.pAttribute;
}
}
}
}
return NULL;
}
//////////
// End of utility classe.
//////////
#endif // _IASTLUTL_H_