mirror of https://github.com/tongzx/nt5src
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.
1464 lines
44 KiB
1464 lines
44 KiB
//#--------------------------------------------------------------
|
|
//
|
|
// File: recvfrompipe.cpp
|
|
//
|
|
// Synopsis: Implementation of CRecvFromPip class methods
|
|
//
|
|
//
|
|
// History: 10/22/97 MKarki Created
|
|
//
|
|
// Copyright (C) 1997-98 Microsoft Corporation
|
|
// All rights reserved.
|
|
//
|
|
//----------------------------------------------------------------
|
|
#include "radcommon.h"
|
|
#include "preprocessor.h"
|
|
#include "recvfrompipe.h"
|
|
#include "logresult.h"
|
|
|
|
#include <new>
|
|
#include <iastlutl.h>
|
|
|
|
const CHAR NUL = '\0';
|
|
|
|
extern LONG g_lPacketCount;
|
|
//++--------------------------------------------------------------
|
|
//
|
|
// Function: CRecvFromPipe
|
|
//
|
|
// Synopsis: This is the constructor of the CRecvFromPipe
|
|
// class
|
|
//
|
|
// Arguments: none
|
|
//
|
|
// Returns: none
|
|
//
|
|
//
|
|
// History: MKarki Created 10/22/97
|
|
//
|
|
//----------------------------------------------------------------
|
|
CRecvFromPipe::CRecvFromPipe(
|
|
CPreProcessor *pCPreProcessor,
|
|
CHashMD5 *pCHashMD5,
|
|
CHashHmacMD5 *pCHashHmacMD5,
|
|
CClients *pCClients,
|
|
VSAFilter *pCVSAFilter,
|
|
CTunnelPassword *pCTunnelPassword,
|
|
CReportEvent *pCReportEvent
|
|
)
|
|
:m_pCPreProcessor (pCPreProcessor),
|
|
m_pCHashMD5 (pCHashMD5),
|
|
m_pCHashHmacMD5 (pCHashHmacMD5),
|
|
m_pCClients (pCClients),
|
|
m_pCVSAFilter (pCVSAFilter),
|
|
m_pCTunnelPassword (pCTunnelPassword),
|
|
m_pCReportEvent (pCReportEvent)
|
|
{
|
|
_ASSERT (
|
|
(NULL != pCPreProcessor) &&
|
|
(NULL != pCHashMD5) &&
|
|
(NULL != pCHashHmacMD5) &&
|
|
(NULL != pCClients) &&
|
|
(NULL != pCVSAFilter) &&
|
|
(NULL != pCTunnelPassword) &&
|
|
(NULL != pCReportEvent)
|
|
);
|
|
|
|
} // end of CRecvFromPipe constructor
|
|
|
|
//++--------------------------------------------------------------
|
|
//
|
|
// Function: ~CRecvFromPipe
|
|
//
|
|
// Synopsis: This is the destructor of the CRecvFromPipe
|
|
// class
|
|
//
|
|
// Arguments: none
|
|
//
|
|
// Returns: none
|
|
//
|
|
//
|
|
// History: MKarki Created 10/22/97
|
|
//
|
|
//----------------------------------------------------------------
|
|
CRecvFromPipe::~CRecvFromPipe()
|
|
{
|
|
} // end of CRecvFromPipe destructor
|
|
|
|
//++--------------------------------------------------------------
|
|
//
|
|
// Function: Process
|
|
//
|
|
// Synopsis: This is the CRecvFromPipe class public method
|
|
// to start processing the packet on its way out
|
|
//
|
|
// Arguments:
|
|
// [in] CPacketRadius*
|
|
//
|
|
// Returns: HRESULT
|
|
//
|
|
//
|
|
// History: MKarki Created 10/22/97
|
|
//
|
|
// Called By:
|
|
// 1) CController::CRequestSource::OnRequest Method
|
|
//
|
|
//----------------------------------------------------------------
|
|
HRESULT
|
|
CRecvFromPipe::Process (
|
|
IRequest *pIRequest
|
|
)
|
|
{
|
|
BOOL bStatus = FALSE;
|
|
HRESULT hr = S_OK;
|
|
DWORD dwCode = 0;
|
|
DWORD dwCount = 0;
|
|
DWORD dwAttribCount = 0;
|
|
IAttributesRaw *pIAttributesRaw = NULL;
|
|
IRequestState *pIRequestState = NULL;
|
|
CPacketRadius *pCPacketRadius = NULL;
|
|
LONG iasResponse, iasReason;
|
|
PACKETTYPE ePacketType;
|
|
PATTRIBUTEPOSITION pAttribPosition = NULL;
|
|
unsigned hyper uhyPacketAddress = 0;
|
|
RADIUSLOGTYPE RadiusError = RADIUS_DROPPED_PACKET;
|
|
|
|
_ASSERT (pIRequest);
|
|
|
|
__try
|
|
{
|
|
//
|
|
// get the IAttributesRaw interface now
|
|
//
|
|
hr = pIRequest->QueryInterface (
|
|
__uuidof(IAttributesRaw),
|
|
reinterpret_cast <PVOID*> (&pIAttributesRaw)
|
|
);
|
|
if (FAILED (hr))
|
|
{
|
|
IASTracePrintf (
|
|
"Unable to obtain Attributes interface in request "
|
|
"received from backend"
|
|
);
|
|
__leave;
|
|
}
|
|
|
|
//
|
|
// split the attributes which can not fit in a radius packet
|
|
//
|
|
hr = SplitAttributes (pIAttributesRaw);
|
|
if (FAILED (hr))
|
|
{
|
|
IASTracePrintf (
|
|
"Unable to split IAS attribute received from backend"
|
|
);
|
|
__leave;
|
|
}
|
|
|
|
//
|
|
// convert the IAS VSA attributes to RADIUS format
|
|
//
|
|
hr = m_pCVSAFilter->radiusFromIAS (pIAttributesRaw);
|
|
if (FAILED (hr))
|
|
{
|
|
IASTracePrintf (
|
|
"Unable to convert IAS attribute to Radius VSAs in request "
|
|
"received from backend"
|
|
);
|
|
__leave;
|
|
}
|
|
|
|
//
|
|
// get the IRequestState interface now
|
|
//
|
|
hr = pIRequest->QueryInterface (
|
|
__uuidof(IRequestState),
|
|
reinterpret_cast <PVOID*> (&pIRequestState)
|
|
);
|
|
if (FAILED (hr))
|
|
{
|
|
IASTracePrintf (
|
|
"Unable to obtain request state in request received from "
|
|
"backend"
|
|
);
|
|
__leave;
|
|
}
|
|
|
|
//
|
|
// get the CPacketRadius class object
|
|
//
|
|
hr = pIRequestState->Pop (
|
|
reinterpret_cast <unsigned hyper*> (&uhyPacketAddress)
|
|
);
|
|
if (FAILED (hr))
|
|
{
|
|
IASTracePrintf (
|
|
"Unable to obtain information from request state received "
|
|
"from backend"
|
|
);
|
|
__leave;
|
|
}
|
|
|
|
pCPacketRadius = reinterpret_cast <CPacketRadius*> (uhyPacketAddress);
|
|
|
|
//
|
|
// if this Request object has been generated by the backend then we
|
|
// don't have a CPacketRadius class object
|
|
//
|
|
if (NULL == pCPacketRadius)
|
|
{
|
|
//
|
|
// we most probably are sending out an EAP-Challenge
|
|
//
|
|
hr= GeneratePacketRadius (
|
|
&pCPacketRadius,
|
|
pIAttributesRaw
|
|
);
|
|
if (FAILED (hr)) { __leave; }
|
|
}
|
|
|
|
//
|
|
// get the outbound packet code
|
|
//
|
|
hr = pIRequest->get_Response (&iasResponse);
|
|
if (FAILED (hr))
|
|
{
|
|
IASTracePrintf (
|
|
"Unable to obtain response code in request recieved from "
|
|
"backend"
|
|
);
|
|
__leave;
|
|
}
|
|
|
|
//
|
|
// get the outbound reason code
|
|
//
|
|
hr = pIRequest->get_Reason (&iasReason);
|
|
if (FAILED (hr))
|
|
{
|
|
IASTracePrintf (
|
|
"Unable to obtain reason code in request recieved from "
|
|
"backend"
|
|
);
|
|
__leave;
|
|
}
|
|
|
|
// Log the result of the request.
|
|
IASRadiusLogResult(pIRequest, pIAttributesRaw);
|
|
|
|
// If it failed convert the reason code.
|
|
if (iasReason != S_OK)
|
|
{
|
|
ConvertReasonToRadiusError (iasReason, &RadiusError);
|
|
}
|
|
|
|
//
|
|
// convert the IASRESPONSE type to RADIUS type
|
|
//
|
|
hr = ConvertResponseToRadiusCode (
|
|
iasResponse,
|
|
&ePacketType,
|
|
pCPacketRadius
|
|
);
|
|
if (FAILED (hr)) { __leave; }
|
|
|
|
//
|
|
// check if we have Tunnel-Password attribute, if yes
|
|
// then encrypt it
|
|
//
|
|
hr = m_pCTunnelPassword->Process (
|
|
ePacketType,
|
|
pIAttributesRaw,
|
|
pCPacketRadius
|
|
);
|
|
if (FAILED (hr)) { __leave; }
|
|
|
|
//
|
|
// inject Signature Attribute if needed
|
|
//
|
|
hr = InjectSignatureIfNeeded (
|
|
ePacketType,
|
|
pIAttributesRaw,
|
|
pCPacketRadius
|
|
);
|
|
if (FAILED (hr)) { __leave; }
|
|
|
|
//
|
|
// get the count of number of request
|
|
//
|
|
hr = pIAttributesRaw->GetAttributeCount (&dwAttribCount);
|
|
if (FAILED (hr))
|
|
{
|
|
IASTracePrintf (
|
|
"Unable to obtain attribute count in request received from "
|
|
"backend"
|
|
);
|
|
__leave;
|
|
}
|
|
|
|
//
|
|
// allocate the attribposition structures
|
|
//
|
|
pAttribPosition = reinterpret_cast <PATTRIBUTEPOSITION> (
|
|
::CoTaskMemAlloc (
|
|
sizeof (ATTRIBUTEPOSITION)*dwAttribCount
|
|
));
|
|
if (NULL == pAttribPosition)
|
|
{
|
|
IASTracePrintf (
|
|
"Unable to allocate memory for attribute postion array "
|
|
"while processing request recieved from backend"
|
|
);
|
|
hr = E_OUTOFMEMORY;
|
|
__leave;
|
|
}
|
|
|
|
//
|
|
// get the attributes from the collection
|
|
//
|
|
hr = pIAttributesRaw->GetAttributes (
|
|
&dwAttribCount,
|
|
pAttribPosition,
|
|
0,
|
|
NULL
|
|
);
|
|
if (FAILED (hr))
|
|
{
|
|
IASTracePrintf (
|
|
"Unable to get attribute in request received from backend"
|
|
);
|
|
__leave;
|
|
}
|
|
|
|
//
|
|
// remove the attributes from the collection now
|
|
//
|
|
hr = pIAttributesRaw->RemoveAttributes (
|
|
dwAttribCount,
|
|
pAttribPosition
|
|
);
|
|
if (FAILED (hr))
|
|
{
|
|
IASTracePrintf (
|
|
"Unable to remove attribute in request received from backend"
|
|
);
|
|
__leave;
|
|
}
|
|
|
|
//
|
|
// carry out the generic packet building here
|
|
//
|
|
hr = pCPacketRadius->BuildOutPacket (
|
|
ePacketType,
|
|
pAttribPosition,
|
|
dwAttribCount
|
|
);
|
|
if (FAILED (hr)) { __leave; }
|
|
|
|
//
|
|
// sending out packet now
|
|
//
|
|
hr = m_pCPreProcessor->StartOutProcessing (pCPacketRadius);
|
|
if (FAILED (hr)) { __leave; }
|
|
|
|
}
|
|
__finally
|
|
{
|
|
|
|
//
|
|
// log event in case of both success and failure
|
|
//
|
|
if (SUCCEEDED (hr))
|
|
{
|
|
//
|
|
// log an event that inbound packet has been processed
|
|
// successfully
|
|
//
|
|
m_pCReportEvent->Process (
|
|
RADIUS_LOG_PACKET,
|
|
pCPacketRadius->GetInCode (),
|
|
pCPacketRadius->GetInLength (),
|
|
pCPacketRadius->GetInAddress (),
|
|
NULL,
|
|
static_cast <LPVOID> (pCPacketRadius->GetInPacket())
|
|
);
|
|
|
|
//
|
|
// log an event for the outbound packet successfully send
|
|
// out
|
|
//
|
|
m_pCReportEvent->Process (
|
|
RADIUS_LOG_PACKET,
|
|
pCPacketRadius->GetOutCode (),
|
|
pCPacketRadius->GetOutLength (),
|
|
pCPacketRadius->GetOutAddress (),
|
|
NULL,
|
|
static_cast <LPVOID> (pCPacketRadius->GetOutPacket())
|
|
);
|
|
}
|
|
else
|
|
{
|
|
if (hr != RADIUS_E_ERRORS_OCCURRED)
|
|
{
|
|
IASReportEvent(
|
|
RADIUS_E_INTERNAL_ERROR,
|
|
0,
|
|
sizeof(hr),
|
|
NULL,
|
|
&hr
|
|
);
|
|
}
|
|
|
|
//
|
|
// generate event that inbound packet has been dropped
|
|
//
|
|
m_pCReportEvent->Process (
|
|
RadiusError,
|
|
pCPacketRadius->GetInCode (),
|
|
pCPacketRadius->GetInLength (),
|
|
pCPacketRadius->GetInAddress (),
|
|
NULL,
|
|
static_cast <LPVOID> (pCPacketRadius->GetInPacket())
|
|
);
|
|
}
|
|
|
|
|
|
//
|
|
// now delete the dynamically allocated memory
|
|
//
|
|
if (NULL != pAttribPosition)
|
|
{
|
|
//
|
|
// release the attributes first
|
|
//
|
|
for (dwCount = 0; dwCount < dwAttribCount; dwCount++)
|
|
{
|
|
::IASAttributeRelease (pAttribPosition[dwCount].pAttribute);
|
|
}
|
|
|
|
::CoTaskMemFree (pAttribPosition);
|
|
}
|
|
|
|
if (pIRequestState) { pIRequestState->Release (); }
|
|
|
|
if (pIAttributesRaw) { pIAttributesRaw->Release (); }
|
|
|
|
//
|
|
// delete the packet
|
|
//
|
|
if (pCPacketRadius) { delete pCPacketRadius; }
|
|
|
|
//
|
|
// now decrement the global packet reference count
|
|
//
|
|
InterlockedDecrement (&g_lPacketCount);
|
|
}
|
|
|
|
return (hr);
|
|
|
|
} // end of CRecvFromPipe::Process method
|
|
|
|
//++--------------------------------------------------------------
|
|
//
|
|
// Function: ConvertResponseToRadiusCode
|
|
//
|
|
// Synopsis: This is the CRecvFromPipe class private method
|
|
// that converts the IASRESPONSE code to RADIUS
|
|
// packet type
|
|
//
|
|
// Arguments:
|
|
// [in] IASRESPONSE
|
|
// [out] PPACKETTYPE
|
|
//
|
|
// Returns: HRESULT - status
|
|
//
|
|
// History: MKarki Created 12/12/97
|
|
//
|
|
// Called By: CRecvFromPipe::Process method
|
|
//
|
|
//----------------------------------------------------------------
|
|
HRESULT
|
|
CRecvFromPipe::ConvertResponseToRadiusCode (
|
|
LONG iasResponse,
|
|
PPACKETTYPE pPacketType,
|
|
CPacketRadius *pCPacketRadius
|
|
)
|
|
{
|
|
HRESULT hr = S_OK;
|
|
|
|
_ASSERT (pPacketType && pCPacketRadius);
|
|
|
|
switch (iasResponse)
|
|
{
|
|
case IAS_RESPONSE_ACCESS_ACCEPT:
|
|
*pPacketType = ACCESS_ACCEPT;
|
|
break;
|
|
case IAS_RESPONSE_ACCESS_REJECT:
|
|
*pPacketType = ACCESS_REJECT;
|
|
break;
|
|
case IAS_RESPONSE_ACCESS_CHALLENGE:
|
|
*pPacketType = ACCESS_CHALLENGE;
|
|
break;
|
|
case IAS_RESPONSE_ACCOUNTING:
|
|
*pPacketType = ACCOUNTING_RESPONSE;
|
|
break;
|
|
case IAS_RESPONSE_FORWARD_PACKET:
|
|
//
|
|
// if we are forwarding this packet
|
|
// the the packet type remains the same
|
|
//
|
|
*pPacketType = pCPacketRadius->GetInCode ();
|
|
break;
|
|
case IAS_RESPONSE_DISCARD_PACKET:
|
|
hr = RADIUS_E_ERRORS_OCCURRED;
|
|
break;
|
|
|
|
case IAS_RESPONSE_INVALID:
|
|
default:
|
|
hr = E_FAIL;
|
|
break;
|
|
}
|
|
|
|
return (hr);
|
|
|
|
} // end of CRecvFromPipe::ConvertResponseToRadiusCode method
|
|
|
|
//++--------------------------------------------------------------
|
|
//
|
|
// Function: GetOutPacketInfo
|
|
//
|
|
// Synopsis: This is the CRecvFromPipe class private method
|
|
// that is used to used to get out bound packet
|
|
// information from the IAS attribute collection
|
|
//
|
|
// Arguments:
|
|
// [out] PDWORD - IP address
|
|
// [out] PWORD - UDP port
|
|
// [out] Client** - reference to CClient object
|
|
// [out] PBYTE - packet header
|
|
// [in] IAttributesRaw*
|
|
//
|
|
// Returns: BOOL status
|
|
//
|
|
// History: MKarki Created 1/9/97
|
|
//
|
|
// Called By: CRecvFromPipe::Process method
|
|
//
|
|
//----------------------------------------------------------------
|
|
HRESULT
|
|
CRecvFromPipe::GetOutPacketInfo (
|
|
PDWORD pdwIPAddress,
|
|
PWORD pwPort,
|
|
IIasClient **ppIIasClient,
|
|
PBYTE pPacketHeader,
|
|
IAttributesRaw *pIAttributesRaw
|
|
)
|
|
{
|
|
BOOL bStatus = TRUE;
|
|
HRESULT hr = S_OK;
|
|
DWORD dwCount = 0;
|
|
PIASATTRIBUTE pIasAttribute = NULL;
|
|
DWORD dwAttribPosCount = COMPONENT_SPECIFIC_ATTRIBUTE_COUNT;
|
|
DWORD dwAttribIDCount = COMPONENT_SPECIFIC_ATTRIBUTE_COUNT;
|
|
ATTRIBUTEPOSITION AttribPos[COMPONENT_SPECIFIC_ATTRIBUTE_COUNT];
|
|
static DWORD AttribIDs [] =
|
|
{
|
|
IAS_ATTRIBUTE_CLIENT_IP_ADDRESS,
|
|
IAS_ATTRIBUTE_CLIENT_UDP_PORT,
|
|
IAS_ATTRIBUTE_CLIENT_PACKET_HEADER
|
|
};
|
|
|
|
_ASSERT (
|
|
(NULL != pdwIPAddress) &&
|
|
(NULL != pwPort) &&
|
|
(NULL != pPacketHeader) &&
|
|
(NULL != ppIIasClient) &&
|
|
(NULL != pIAttributesRaw)
|
|
);
|
|
|
|
__try
|
|
{
|
|
|
|
//
|
|
// get client info
|
|
//
|
|
// get the attributes from the collection
|
|
//
|
|
hr = pIAttributesRaw->GetAttributes (
|
|
&dwAttribPosCount,
|
|
AttribPos,
|
|
dwAttribIDCount,
|
|
reinterpret_cast <LPDWORD> (AttribIDs)
|
|
);
|
|
if (FAILED (hr))
|
|
{
|
|
IASTracePrintf (
|
|
"Unable to get attributes while obtaining out-bound packet "
|
|
"information"
|
|
);
|
|
__leave;
|
|
}
|
|
else if (COMPONENT_SPECIFIC_ATTRIBUTE_COUNT != dwAttribPosCount)
|
|
{
|
|
IASTracePrintf (
|
|
"Request received from backend does not have all the "
|
|
"all the required attributes"
|
|
);
|
|
hr = E_FAIL;
|
|
__leave;
|
|
}
|
|
|
|
|
|
//
|
|
// go through the attributes and get values out
|
|
//
|
|
for (dwCount = 0; dwCount < dwAttribPosCount; dwCount++)
|
|
{
|
|
pIasAttribute = AttribPos[dwCount].pAttribute;
|
|
switch (pIasAttribute->dwId)
|
|
{
|
|
case IAS_ATTRIBUTE_CLIENT_IP_ADDRESS:
|
|
_ASSERT (IASTYPE_INET_ADDR == pIasAttribute->Value.itType),
|
|
*pdwIPAddress = pIasAttribute->Value.InetAddr;
|
|
break;
|
|
|
|
case IAS_ATTRIBUTE_CLIENT_UDP_PORT:
|
|
_ASSERT (IASTYPE_INTEGER == pIasAttribute->Value.itType);
|
|
*pwPort = pIasAttribute->Value.Integer;
|
|
break;
|
|
|
|
case IAS_ATTRIBUTE_CLIENT_PACKET_HEADER:
|
|
_ASSERT (
|
|
(IASTYPE_OCTET_STRING == pIasAttribute->Value.itType) &&
|
|
(PACKET_HEADER_SIZE == pIasAttribute->Value.OctetString.dwLength)
|
|
);
|
|
//
|
|
// copy the value into the buffer provided
|
|
//
|
|
CopyMemory (
|
|
pPacketHeader,
|
|
pIasAttribute->Value.OctetString.lpValue,
|
|
PACKET_HEADER_SIZE
|
|
);
|
|
break;
|
|
default:
|
|
_ASSERT (0);
|
|
IASTracePrintf (
|
|
"Attribute:%d, not requested, is present "
|
|
"in request received from backend",
|
|
pIasAttribute->dwId
|
|
);
|
|
hr = E_FAIL;
|
|
__leave;
|
|
break;
|
|
}
|
|
|
|
} // end of for loop
|
|
|
|
//
|
|
// get client information for this RADIUS packet
|
|
//
|
|
bStatus = m_pCClients->FindObject (
|
|
*pdwIPAddress,
|
|
ppIIasClient
|
|
);
|
|
if (FALSE == bStatus)
|
|
{
|
|
in_addr sin;
|
|
sin.s_addr = *pdwIPAddress;
|
|
IASTracePrintf (
|
|
"Unable to get information for client:%s "
|
|
"while processing request received from backend",
|
|
inet_ntoa (sin)
|
|
);
|
|
hr = E_FAIL;
|
|
__leave;
|
|
}
|
|
|
|
}
|
|
__finally
|
|
{
|
|
if (SUCCEEDED (hr))
|
|
{
|
|
for (dwCount = 0; dwCount < dwAttribPosCount; dwCount++)
|
|
{
|
|
//
|
|
// now release the reference to the attributes
|
|
//
|
|
::IASAttributeRelease (AttribPos[dwCount].pAttribute);
|
|
}
|
|
}
|
|
}
|
|
|
|
return (hr);
|
|
|
|
} // end of CPacketRadius::GetOutPacketInfo method
|
|
|
|
//++--------------------------------------------------------------
|
|
//
|
|
// Function: GeneratePacketRadius
|
|
//
|
|
// Synopsis: This is the CRecvFromPipe class private method
|
|
// that is used generate a new CPacketRadius
|
|
// class object and initialize it
|
|
//
|
|
// Arguments:
|
|
// [out] CPacketRadius**
|
|
// [in] IAttributesRaw*
|
|
//
|
|
// Returns: HRESULT - status
|
|
//
|
|
// History: MKarki Created 2/6/98
|
|
//
|
|
// Called By: CRecvFromPipe::Process method
|
|
//
|
|
//----------------------------------------------------------------
|
|
HRESULT
|
|
CRecvFromPipe:: GeneratePacketRadius (
|
|
CPacketRadius **ppCPacketRadius,
|
|
IAttributesRaw *pIAttributesRaw
|
|
)
|
|
{
|
|
PBYTE pPacketHeader = NULL;
|
|
DWORD dwAddress = 0;
|
|
WORD wPort = 0;
|
|
IIasClient *pIIasClient = NULL;
|
|
HRESULT hr = S_OK;
|
|
PATTRIBUTEPOSITION pAttribPosition = NULL;
|
|
|
|
|
|
_ASSERT (ppCPacketRadius && pIAttributesRaw);
|
|
|
|
//
|
|
// allocate memory for packet header
|
|
//
|
|
pPacketHeader =
|
|
reinterpret_cast <PBYTE> (::CoTaskMemAlloc (PACKET_HEADER_SIZE));
|
|
if (NULL == pPacketHeader)
|
|
{
|
|
IASTracePrintf (
|
|
"Unable to allocate memory for packet header information "
|
|
"while generating out-bound packet"
|
|
);
|
|
hr = E_OUTOFMEMORY;
|
|
goto Cleanup;
|
|
}
|
|
|
|
//
|
|
// we need to gather information from the attribute collection
|
|
// needed to get create and initialize a CPacketRadius class
|
|
// object
|
|
//
|
|
hr = GetOutPacketInfo (
|
|
&dwAddress,
|
|
&wPort,
|
|
&pIIasClient,
|
|
pPacketHeader,
|
|
pIAttributesRaw
|
|
);
|
|
if (FAILED (hr)) { goto Cleanup; }
|
|
|
|
|
|
//
|
|
// create a new CPacketRadius class object
|
|
//
|
|
*ppCPacketRadius = new (std::nothrow) CPacketRadius (
|
|
m_pCHashMD5,
|
|
m_pCHashHmacMD5,
|
|
pIIasClient,
|
|
m_pCReportEvent,
|
|
pPacketHeader,
|
|
PACKET_HEADER_SIZE,
|
|
dwAddress,
|
|
wPort,
|
|
INVALID_SOCKET,
|
|
AUTH_PORTTYPE
|
|
);
|
|
if (NULL == *ppCPacketRadius)
|
|
{
|
|
IASTracePrintf (
|
|
"Unable to create a Packet-Radius object "
|
|
"while generating an out-bound packet"
|
|
);
|
|
pIIasClient->Release ();
|
|
hr = E_OUTOFMEMORY;
|
|
goto Cleanup;
|
|
}
|
|
|
|
Cleanup:
|
|
|
|
if ((FAILED (hr)) && (pPacketHeader))
|
|
{
|
|
::CoTaskMemFree (pPacketHeader);
|
|
}
|
|
|
|
return (hr);
|
|
|
|
} // end of CRecvFromPipe::GeneratePacketRadius method
|
|
|
|
//++--------------------------------------------------------------
|
|
//
|
|
// Function: InjectSignatureIfNeeded
|
|
//
|
|
// Synopsis: This method is used to add a blank Signature attribute
|
|
// into the response if we see an EAP-Message attribute
|
|
// present
|
|
//
|
|
// Arguments:
|
|
// [in] PACKETTYPE
|
|
// [in] IAttributesRaw*
|
|
// [in] CPacketRadius*
|
|
//
|
|
// Returns: HRESULT - status
|
|
//
|
|
// History: MKarki Created 11/17/98
|
|
//
|
|
// Called By: CRecvFromPipe::Process method
|
|
//
|
|
//----------------------------------------------------------------
|
|
HRESULT CRecvFromPipe::InjectSignatureIfNeeded (
|
|
/*[in]*/ PACKETTYPE ePacketType,
|
|
/*[in]*/ IAttributesRaw *pIAttributesRaw,
|
|
/*[in]*/ CPacketRadius *pCPacketRadius
|
|
)
|
|
{
|
|
HRESULT hr = S_OK;
|
|
PATTRIBUTEPOSITION pAttribPos = NULL;
|
|
|
|
_ASSERT (pIAttributesRaw && pCPacketRadius);
|
|
|
|
__try
|
|
{
|
|
if (
|
|
(ACCESS_ACCEPT != ePacketType) &&
|
|
(ACCESS_REJECT != ePacketType) &&
|
|
(ACCESS_CHALLENGE != ePacketType)
|
|
)
|
|
{__leave;}
|
|
|
|
//
|
|
// get the count of the total attributes in the collection
|
|
//
|
|
DWORD dwAttributeCount = 0;
|
|
hr = pIAttributesRaw->GetAttributeCount (&dwAttributeCount);
|
|
if (FAILED (hr))
|
|
{
|
|
IASTracePrintf (
|
|
"Unable to obtain attribute count in request while "
|
|
"processing signature attribute in out-bound packet "
|
|
);
|
|
__leave;
|
|
}
|
|
else if (0 == dwAttributeCount)
|
|
{
|
|
__leave;
|
|
}
|
|
|
|
//
|
|
// allocate memory for the ATTRIBUTEPOSITION array
|
|
//
|
|
pAttribPos = reinterpret_cast <PATTRIBUTEPOSITION> (
|
|
::CoTaskMemAlloc (
|
|
sizeof (ATTRIBUTEPOSITION)*dwAttributeCount));
|
|
if (NULL == pAttribPos)
|
|
{
|
|
IASTracePrintf (
|
|
"Unable to allocate memory for attribute position array "
|
|
"while processing signature attribute in out-bound packet"
|
|
);
|
|
hr = E_OUTOFMEMORY;
|
|
__leave;
|
|
}
|
|
|
|
//
|
|
// get the EAP-Message attribute from the interface
|
|
//
|
|
DWORD dwAttrId = RADIUS_ATTRIBUTE_EAP_MESSAGE;
|
|
hr = pIAttributesRaw->GetAttributes (
|
|
&dwAttributeCount,
|
|
pAttribPos,
|
|
1,
|
|
&dwAttrId
|
|
);
|
|
if (FAILED (hr))
|
|
{
|
|
IASTracePrintf (
|
|
"Unable to obtain information about EAP-Message attribute "
|
|
"while processing out-bound RADIUS packet"
|
|
);
|
|
}
|
|
else if (0 == dwAttributeCount)
|
|
{
|
|
__leave;
|
|
}
|
|
|
|
bool bFound = false;
|
|
for (DWORD dwCount = 0; dwCount < dwAttributeCount; dwCount++)
|
|
{
|
|
if (
|
|
(!bFound) &&
|
|
(pCPacketRadius->IsOutBoundAttribute (
|
|
ePacketType,
|
|
pAttribPos[dwCount].pAttribute
|
|
))
|
|
)
|
|
{
|
|
bFound = true;
|
|
}
|
|
|
|
::IASAttributeRelease (pAttribPos[dwCount].pAttribute);
|
|
}
|
|
|
|
if (bFound)
|
|
{
|
|
//
|
|
// if we have an out-bound EAP-Message attribute then
|
|
// we need to insert a Signature attribute too
|
|
//
|
|
|
|
//
|
|
// create a new blank attribute
|
|
//
|
|
PIASATTRIBUTE pIasAttrib = NULL;
|
|
DWORD dwRetVal = ::IASAttributeAlloc ( 1, &pIasAttrib);
|
|
if (0 != dwRetVal)
|
|
{
|
|
IASTracePrintf (
|
|
"Unable to allocate IAS attribute for Signature "
|
|
"while processing out-bound RADIUS packet"
|
|
);
|
|
hr = HRESULT_FROM_WIN32 (dwRetVal);
|
|
__leave;
|
|
}
|
|
|
|
//
|
|
// allocate dynamic memory for the Signature
|
|
//
|
|
pIasAttrib->Value.OctetString.lpValue =
|
|
reinterpret_cast <PBYTE>
|
|
(::CoTaskMemAlloc (SIGNATURE_SIZE));
|
|
if (NULL == pIasAttrib->Value.OctetString.lpValue)
|
|
{
|
|
IASTracePrintf (
|
|
"Unable to allocate dynamic memory for Signature "
|
|
"attribute value while processing out-bound RADIUS packet"
|
|
);
|
|
hr = E_OUTOFMEMORY;
|
|
::IASAttributeRelease (pIasAttrib);
|
|
__leave;
|
|
}
|
|
else
|
|
{
|
|
//
|
|
// put the signature attribute with no value
|
|
// but correct size
|
|
//
|
|
pIasAttrib->dwId = RADIUS_ATTRIBUTE_SIGNATURE;
|
|
pIasAttrib->Value.itType = IASTYPE_OCTET_STRING;
|
|
pIasAttrib->Value.OctetString.dwLength = SIGNATURE_SIZE;
|
|
pIasAttrib->dwFlags = IAS_INCLUDE_IN_RESPONSE;
|
|
|
|
//
|
|
// add the attribute to the collection now
|
|
//
|
|
ATTRIBUTEPOSITION attrPos;
|
|
attrPos.pAttribute = pIasAttrib;
|
|
hr = pIAttributesRaw->AddAttributes (1, &attrPos);
|
|
if (FAILED (hr))
|
|
{
|
|
IASTracePrintf (
|
|
"Unable to add signature attribute to request while "
|
|
"processing out-bound RADIUS packet"
|
|
);
|
|
::IASAttributeRelease (pIasAttrib);
|
|
__leave;
|
|
}
|
|
|
|
IASTracePrintf (
|
|
"Signature Attribute added to out-bound RADIUS packet"
|
|
);
|
|
}
|
|
}
|
|
}
|
|
__finally
|
|
{
|
|
if (pAttribPos) { ::CoTaskMemFree (pAttribPos); }
|
|
}
|
|
|
|
return (hr);
|
|
|
|
} // end of CRecvFromPipe::InjectSignatureIfNeeded method
|
|
|
|
//++--------------------------------------------------------------
|
|
//
|
|
// Function: SplitAttributes
|
|
//
|
|
// Synopsis: This method is used to split up the following
|
|
// out-bound attributes:
|
|
// 1) Reply-Message attribute
|
|
// 1) MS-Filter-VSA Attribute
|
|
//
|
|
// Arguments:
|
|
// [in] IAttributesRaw*
|
|
//
|
|
// Returns: HRESULT - status
|
|
//
|
|
// History: MKarki Created 1/19/99
|
|
//
|
|
// Called By: CRecvFromPipe::Process method
|
|
//
|
|
//----------------------------------------------------------------
|
|
HRESULT CRecvFromPipe::SplitAttributes (
|
|
/*[in]*/ IAttributesRaw *pIAttributesRaw
|
|
)
|
|
{
|
|
const DWORD SPLIT_ATTRIBUTE_COUNT = 2;
|
|
static DWORD AttribIds [] = {
|
|
RADIUS_ATTRIBUTE_REPLY_MESSAGE,
|
|
MS_ATTRIBUTE_FILTER
|
|
};
|
|
|
|
HRESULT hr = S_OK;
|
|
DWORD dwAttributesFound = 0;
|
|
PATTRIBUTEPOSITION pAttribPos = NULL;
|
|
|
|
_ASSERT (pIAttributesRaw);
|
|
|
|
__try
|
|
{
|
|
//
|
|
// get the count of the total attributes in the collection
|
|
//
|
|
DWORD dwAttributeCount = 0;
|
|
hr = pIAttributesRaw->GetAttributeCount (&dwAttributeCount);
|
|
if (FAILED (hr))
|
|
{
|
|
IASTracePrintf (
|
|
"Unable to obtain attribute count in request while "
|
|
"splitting attributes in out-bound packet "
|
|
);
|
|
__leave;
|
|
}
|
|
else if (0 == dwAttributeCount)
|
|
{
|
|
__leave;
|
|
}
|
|
|
|
//
|
|
// allocate memory for the ATTRIBUTEPOSITION array
|
|
//
|
|
pAttribPos = reinterpret_cast <PATTRIBUTEPOSITION> (
|
|
::CoTaskMemAlloc (
|
|
sizeof (ATTRIBUTEPOSITION)*dwAttributeCount)
|
|
);
|
|
if (NULL == pAttribPos)
|
|
{
|
|
IASTracePrintf (
|
|
"Unable to allocate memory for attribute position array "
|
|
"while splitting attributes in out-bound packet"
|
|
);
|
|
hr = E_OUTOFMEMORY;
|
|
__leave;
|
|
}
|
|
|
|
//
|
|
// get the attributes we are interested in from the interface
|
|
//
|
|
hr = pIAttributesRaw->GetAttributes (
|
|
&dwAttributeCount,
|
|
pAttribPos,
|
|
SPLIT_ATTRIBUTE_COUNT,
|
|
static_cast <PDWORD> (AttribIds)
|
|
);
|
|
if (FAILED (hr))
|
|
{
|
|
IASTracePrintf (
|
|
"Unable to obtain information about attributes"
|
|
"while splitting attributes in out-bound RADIUS packet"
|
|
);
|
|
__leave;
|
|
}
|
|
else if (0 == dwAttributeCount)
|
|
{
|
|
__leave;
|
|
}
|
|
|
|
//
|
|
// save the count of attributes returned
|
|
//
|
|
dwAttributesFound = dwAttributeCount;
|
|
|
|
DWORD dwAttribLength = 0;
|
|
DWORD dwMaxPossibleLength = 0;
|
|
IASTYPE iasType = IASTYPE_INVALID;
|
|
//
|
|
// evaluate each attribute now
|
|
//
|
|
for (DWORD dwCount = 0; dwCount < dwAttributeCount; dwCount++)
|
|
{
|
|
if ((pAttribPos[dwCount].pAttribute)->dwFlags &
|
|
IAS_INCLUDE_IN_RESPONSE)
|
|
{
|
|
//
|
|
// get attribute type and length
|
|
//
|
|
if (
|
|
(iasType = (pAttribPos[dwCount].pAttribute)->Value.itType) ==
|
|
IASTYPE_STRING
|
|
)
|
|
{
|
|
::IASAttributeAnsiAlloc (pAttribPos[dwCount].pAttribute);
|
|
dwAttribLength =
|
|
strlen (
|
|
(pAttribPos[dwCount].pAttribute)->Value.String.pszAnsi);
|
|
|
|
}
|
|
else if (
|
|
(iasType = (pAttribPos[dwCount].pAttribute)->Value.itType) ==
|
|
IASTYPE_OCTET_STRING
|
|
)
|
|
{
|
|
dwAttribLength =
|
|
(pAttribPos[dwCount].pAttribute)->Value.OctetString.dwLength;
|
|
}
|
|
else
|
|
{
|
|
//
|
|
// only string values need to be split
|
|
//
|
|
continue;
|
|
}
|
|
|
|
//
|
|
// get max possible attribute length
|
|
//
|
|
if ((pAttribPos[dwCount].pAttribute)->dwId > MAX_ATTRIBUTE_TYPE)
|
|
{
|
|
dwMaxPossibleLength = MAX_VSA_ATTRIBUTE_LENGTH;
|
|
}
|
|
else
|
|
{
|
|
dwMaxPossibleLength = MAX_ATTRIBUTE_LENGTH;
|
|
}
|
|
|
|
//
|
|
// check if we need to split this attribute
|
|
//
|
|
if (dwAttribLength <= dwMaxPossibleLength) {continue;}
|
|
|
|
|
|
//
|
|
// split the attribute now
|
|
//
|
|
hr = SplitAndAdd (
|
|
pIAttributesRaw,
|
|
pAttribPos[dwCount].pAttribute,
|
|
iasType,
|
|
dwAttribLength,
|
|
dwMaxPossibleLength
|
|
);
|
|
if (SUCCEEDED (hr))
|
|
{
|
|
//
|
|
// remove this attribute from the collection now
|
|
//
|
|
hr = pIAttributesRaw->RemoveAttributes (
|
|
1,
|
|
&(pAttribPos[dwCount])
|
|
);
|
|
if (FAILED (hr))
|
|
{
|
|
IASTracePrintf (
|
|
"Unable to remove attribute from collection"
|
|
"while splitting out-bound attributes"
|
|
);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
__finally
|
|
{
|
|
if (pAttribPos)
|
|
{
|
|
for (DWORD dwCount = 0; dwCount < dwAttributesFound; dwCount++)
|
|
{
|
|
::IASAttributeRelease (pAttribPos[dwCount].pAttribute);
|
|
}
|
|
|
|
::CoTaskMemFree (pAttribPos);
|
|
}
|
|
}
|
|
|
|
return (hr);
|
|
|
|
} // end of CRecvFromPipe::SplitAttributes method
|
|
|
|
//++--------------------------------------------------------------
|
|
//
|
|
// Function: SplitAndAdd
|
|
//
|
|
// Synopsis: This method is used to remove the original attribute
|
|
// and add new ones
|
|
// Arguments:
|
|
// [in] IAttributesRaw*
|
|
// [in] PIASATTRIBUTE
|
|
// [in] IASTYPE
|
|
// [in] DWORD - attribute length
|
|
// [in] DWORD - max attribute length
|
|
//
|
|
// Returns: HRESULT - status
|
|
//
|
|
// History: MKarki Created 1/19/99
|
|
//
|
|
// Called By: CRecvFromPipe::SplitAttributes method
|
|
//
|
|
//----------------------------------------------------------------
|
|
HRESULT CRecvFromPipe::SplitAndAdd (
|
|
/*[in]*/ IAttributesRaw *pIAttributesRaw,
|
|
/*[in]*/ PIASATTRIBUTE pIasAttribute,
|
|
/*[in]*/ IASTYPE iasType,
|
|
/*[in]*/ DWORD dwAttributeLength,
|
|
/*[in]*/ DWORD dwMaxLength
|
|
)
|
|
{
|
|
HRESULT hr = S_OK;
|
|
DWORD dwPacketsNeeded = 0;
|
|
DWORD dwFailed = 0;
|
|
PIASATTRIBUTE *ppAttribArray = NULL;
|
|
PATTRIBUTEPOSITION pAttribPos = NULL;
|
|
|
|
_ASSERT (pIAttributesRaw && pIasAttribute);
|
|
|
|
__try
|
|
{
|
|
dwPacketsNeeded = dwAttributeLength / dwMaxLength;
|
|
if (dwAttributeLength % dwMaxLength) {++dwPacketsNeeded;}
|
|
|
|
//
|
|
// allocate memory for the ATTRIBUTEPOSITION array
|
|
//
|
|
pAttribPos = reinterpret_cast <PATTRIBUTEPOSITION> (
|
|
::CoTaskMemAlloc (
|
|
sizeof (ATTRIBUTEPOSITION)*dwPacketsNeeded));
|
|
if (NULL == pAttribPos)
|
|
{
|
|
IASTracePrintf (
|
|
"Unable to allocate memory for attribute position array "
|
|
"while split and add of attributese in out-bound packet"
|
|
);
|
|
hr = E_OUTOFMEMORY;
|
|
__leave;
|
|
}
|
|
|
|
//
|
|
// allocate array to store the attributes in
|
|
//
|
|
ppAttribArray =
|
|
reinterpret_cast <PIASATTRIBUTE*> (
|
|
::CoTaskMemAlloc (sizeof (PIASATTRIBUTE)*dwPacketsNeeded));
|
|
if (NULL == ppAttribArray)
|
|
{
|
|
IASTracePrintf (
|
|
"Unable to allocate memory"
|
|
"while split and add of out-bound attribues"
|
|
);
|
|
hr = E_OUTOFMEMORY;
|
|
__leave;
|
|
}
|
|
|
|
DWORD dwFailed =
|
|
::IASAttributeAlloc (dwPacketsNeeded, ppAttribArray);
|
|
if (0 != dwFailed)
|
|
{
|
|
IASTracePrintf (
|
|
"Unable to allocate attributes while splitting out-bound"
|
|
"attributes"
|
|
);
|
|
hr = HRESULT_FROM_WIN32 (dwFailed);
|
|
__leave;
|
|
}
|
|
|
|
if (IASTYPE_STRING == iasType)
|
|
{
|
|
PCHAR pStart = (pIasAttribute->Value).String.pszAnsi;
|
|
DWORD dwCopySize = dwMaxLength;
|
|
|
|
//
|
|
// set value in each of the new attributes
|
|
//
|
|
for (DWORD dwCount1 = 0; dwCount1 < dwPacketsNeeded; dwCount1++)
|
|
{
|
|
(ppAttribArray[dwCount1])->Value.String.pszWide = NULL;
|
|
(ppAttribArray[dwCount1])->Value.String.pszAnsi =
|
|
reinterpret_cast <PCHAR>
|
|
(::CoTaskMemAlloc ((dwCopySize + 1)*sizeof (CHAR)));
|
|
if (NULL == (ppAttribArray[dwCount1])->Value.String.pszAnsi)
|
|
{
|
|
IASTracePrintf (
|
|
"Unable to allocate memory for new attribute values"
|
|
"while split and add of out-bound attribues"
|
|
);
|
|
hr = E_OUTOFMEMORY;
|
|
__leave;
|
|
}
|
|
|
|
//
|
|
// set the value now
|
|
//
|
|
::CopyMemory (
|
|
(ppAttribArray[dwCount1])->Value.String.pszAnsi,
|
|
pStart,
|
|
dwCopySize
|
|
);
|
|
//
|
|
// nul terminate the values
|
|
//
|
|
((ppAttribArray[dwCount1])->Value.String.pszAnsi)[dwCopySize]=NUL;
|
|
(ppAttribArray[dwCount1])->Value.itType = iasType;
|
|
(ppAttribArray[dwCount1])->dwId = pIasAttribute->dwId;
|
|
(ppAttribArray[dwCount1])->dwFlags = pIasAttribute->dwFlags;
|
|
|
|
//
|
|
// calculate for next attribute
|
|
//
|
|
pStart = pStart + dwCopySize;
|
|
dwAttributeLength -= dwCopySize;
|
|
dwCopySize = (dwAttributeLength > dwMaxLength) ?
|
|
dwMaxLength : dwAttributeLength;
|
|
|
|
//
|
|
// add attribute to position array
|
|
//
|
|
pAttribPos[dwCount1].pAttribute = ppAttribArray[dwCount1];
|
|
}
|
|
}
|
|
else
|
|
{
|
|
PBYTE pStart = (pIasAttribute->Value).OctetString.lpValue;
|
|
DWORD dwCopySize = dwMaxLength;
|
|
|
|
//
|
|
// fill the new attributes now
|
|
//
|
|
for (DWORD dwCount1 = 0; dwCount1 < dwPacketsNeeded; dwCount1++)
|
|
{
|
|
(ppAttribArray[dwCount1])->Value.OctetString.lpValue =
|
|
reinterpret_cast <PBYTE> (::CoTaskMemAlloc (dwCopySize));
|
|
if (NULL ==(ppAttribArray[dwCount1])->Value.OctetString.lpValue)
|
|
{
|
|
IASTracePrintf (
|
|
"Unable to allocate memory for new attribute values"
|
|
"while split and add of out-bound attribues"
|
|
);
|
|
hr = E_OUTOFMEMORY;
|
|
__leave;
|
|
}
|
|
|
|
//
|
|
// set the value now
|
|
//
|
|
::CopyMemory (
|
|
(ppAttribArray[dwCount1])->Value.OctetString.lpValue,
|
|
pStart,
|
|
dwCopySize
|
|
);
|
|
|
|
(ppAttribArray[dwCount1])->Value.OctetString.dwLength = dwCopySize;
|
|
(ppAttribArray[dwCount1])->Value.itType = iasType;
|
|
(ppAttribArray[dwCount1])->dwId = pIasAttribute->dwId;
|
|
(ppAttribArray[dwCount1])->dwFlags = pIasAttribute->dwFlags;
|
|
|
|
//
|
|
// calculate for next attribute
|
|
//
|
|
pStart = pStart + dwCopySize;
|
|
dwAttributeLength -= dwCopySize;
|
|
dwCopySize = (dwAttributeLength > dwMaxLength) ?
|
|
dwMaxLength :
|
|
dwAttributeLength;
|
|
|
|
//
|
|
// add attribute to position array
|
|
//
|
|
pAttribPos[dwCount1].pAttribute = ppAttribArray[dwCount1];
|
|
}
|
|
}
|
|
|
|
//
|
|
// add the attribute to the collection
|
|
//
|
|
hr = pIAttributesRaw->AddAttributes (dwPacketsNeeded, pAttribPos);
|
|
if (FAILED (hr))
|
|
{
|
|
IASTracePrintf (
|
|
"Failed to add attributes to the collection"
|
|
"on split and add out-bound attributes"
|
|
);
|
|
__leave;
|
|
}
|
|
}
|
|
__finally
|
|
{
|
|
if (ppAttribArray && !dwFailed)
|
|
{
|
|
for (DWORD dwCount = 0; dwCount < dwPacketsNeeded; dwCount++)
|
|
{
|
|
::IASAttributeRelease (ppAttribArray[dwCount]);
|
|
}
|
|
}
|
|
|
|
if (ppAttribArray) {::CoTaskMemFree (ppAttribArray);}
|
|
|
|
if (pAttribPos) {::CoTaskMemFree (pAttribPos);}
|
|
}
|
|
|
|
return (hr);
|
|
|
|
} // end of CRecvFromPipe::SplitAndAdd method
|
|
|
|
//++--------------------------------------------------------------
|
|
//
|
|
// Function: ConvertReasonToRadiusError
|
|
//
|
|
// Synopsis:
|
|
//
|
|
// Arguments:
|
|
// [in] iasReason
|
|
// [out] Radius Error
|
|
//
|
|
// Returns: HRESULT - status
|
|
//
|
|
// History: MKarki Created 12/31/98
|
|
//
|
|
// Called By: CRecvFromPipe::Process method
|
|
//
|
|
//----------------------------------------------------------------
|
|
HRESULT
|
|
CRecvFromPipe::ConvertReasonToRadiusError (
|
|
/*[in]*/ LONG iasReason,
|
|
/*[out]*/ PRADIUSLOGTYPE pRadError
|
|
)
|
|
{
|
|
HRESULT hr = S_OK;
|
|
|
|
_ASSERT (pRadError);
|
|
|
|
switch (iasReason)
|
|
{
|
|
case IAS_NO_RECORD:
|
|
*pRadError = RADIUS_NO_RECORD;
|
|
break;
|
|
|
|
case IAS_MALFORMED_REQUEST:
|
|
*pRadError = RADIUS_MALFORMED_PACKET;
|
|
break;
|
|
|
|
default:
|
|
hr = E_FAIL;
|
|
break;
|
|
}
|
|
|
|
return (hr);
|
|
|
|
} // end of CRecvFromPipe::ConvertReasonToRadiusError method
|