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.
1706 lines
39 KiB
1706 lines
39 KiB
/*++
|
|
|
|
Copyright (c) 1999 Microsoft Corporation
|
|
|
|
|
|
Module Name:
|
|
|
|
interface.c
|
|
|
|
Abstract:
|
|
|
|
This module contains all of the code to drive
|
|
the interface list management of IPSecSPD Service.
|
|
|
|
Author:
|
|
|
|
abhisheV 30-September-1999
|
|
|
|
Environment
|
|
|
|
User Level: Win32
|
|
|
|
Revision History:
|
|
|
|
|
|
--*/
|
|
|
|
|
|
#include "precomp.h"
|
|
#ifdef TRACE_ON
|
|
#include "interface.tmh"
|
|
#endif
|
|
|
|
DWORD
|
|
CreateInterfaceList(
|
|
OUT PIPSEC_INTERFACE * ppIfListToCreate
|
|
)
|
|
{
|
|
DWORD dwError = 0;
|
|
PIPSEC_INTERFACE pIfList = NULL;
|
|
|
|
|
|
dwError = GetInterfaceListFromStack(
|
|
&pIfList
|
|
);
|
|
|
|
ENTER_SPD_SECTION();
|
|
|
|
*ppIfListToCreate = pIfList;
|
|
|
|
LEAVE_SPD_SECTION();
|
|
|
|
return (dwError);
|
|
}
|
|
|
|
|
|
VOID
|
|
DestroyInterfaceList(
|
|
IN PIPSEC_INTERFACE pIfListToDelete
|
|
)
|
|
{
|
|
PIPSEC_INTERFACE pIf = NULL;
|
|
PIPSEC_INTERFACE pTempIf = NULL;
|
|
|
|
pIf = pIfListToDelete;
|
|
|
|
while (pIf) {
|
|
pTempIf = pIf;
|
|
pIf = pIf->pNext;
|
|
FreeIpsecInterface(pTempIf);
|
|
}
|
|
}
|
|
|
|
|
|
DWORD
|
|
OnInterfaceChangeEvent(
|
|
)
|
|
{
|
|
DWORD dwError = 0;
|
|
PIPSEC_INTERFACE pIfList = NULL;
|
|
PIPSEC_INTERFACE pObseleteIfList = NULL;
|
|
PIPSEC_INTERFACE pNewIfList = NULL;
|
|
PIPSEC_INTERFACE pExistingIfList = NULL;
|
|
PSPECIAL_ADDR pLatestSpecialAddrsList;
|
|
DWORD dwMMError = 0;
|
|
DWORD dwTxError = 0;
|
|
DWORD dwTnError = 0;
|
|
|
|
|
|
dwError = ResetInterfaceChangeEvent();
|
|
|
|
(VOID) GetInterfaceListFromStack(
|
|
&pIfList
|
|
);
|
|
|
|
(VOID) GetSpecialAddrsList(
|
|
&pLatestSpecialAddrsList
|
|
);
|
|
|
|
ENTER_SPD_SECTION();
|
|
|
|
(VOID) FreeSpecialAddrList(
|
|
&gpSpecialAddrsList
|
|
);
|
|
|
|
gpSpecialAddrsList = pLatestSpecialAddrsList;
|
|
|
|
pExistingIfList = gpInterfaceList;
|
|
|
|
// Interface List from Stack can be NULL.
|
|
|
|
FormObseleteAndNewIfLists(
|
|
pIfList,
|
|
&pExistingIfList,
|
|
&pObseleteIfList,
|
|
&pNewIfList
|
|
);
|
|
|
|
if (pNewIfList) {
|
|
AddToInterfaceList(
|
|
pNewIfList,
|
|
&pExistingIfList
|
|
);
|
|
}
|
|
|
|
if (pObseleteIfList) {
|
|
DestroyInterfaceList(
|
|
pObseleteIfList
|
|
);
|
|
}
|
|
|
|
gpInterfaceList = pExistingIfList;
|
|
|
|
(VOID) ApplyIfChangeToIniMMFilters(
|
|
&dwMMError,
|
|
pExistingIfList,
|
|
gpSpecialAddrsList
|
|
);
|
|
|
|
(VOID) ApplyIfChangeToIniTxFilters(
|
|
&dwTxError,
|
|
pExistingIfList,
|
|
gpSpecialAddrsList
|
|
);
|
|
|
|
(VOID) ApplyIfChangeToIniTnFilters(
|
|
&dwTnError,
|
|
pExistingIfList,
|
|
gpSpecialAddrsList
|
|
);
|
|
|
|
LEAVE_SPD_SECTION();
|
|
|
|
if (dwMMError || dwTxError || dwTnError) {
|
|
AuditEvent(
|
|
SE_CATEGID_POLICY_CHANGE,
|
|
SE_AUDITID_IPSEC_POLICY_CHANGED,
|
|
IPSECSVC_FAILED_PNP_FILTER_PROCESSING,
|
|
NULL,
|
|
FALSE,
|
|
TRUE
|
|
);
|
|
}
|
|
|
|
return (dwError);
|
|
}
|
|
|
|
|
|
DWORD
|
|
OnSpecialAddrsChange(
|
|
)
|
|
{
|
|
DWORD dwError = 0;
|
|
DWORD dwMMError = 0;
|
|
DWORD dwTxError = 0;
|
|
DWORD dwTnError = 0;
|
|
PSPECIAL_ADDR pOldSpecialAddrsList = NULL;
|
|
PSPECIAL_ADDR pLatestSpecialAddrsList = NULL;
|
|
BOOL bListSame = FALSE;
|
|
|
|
dwError = GetSpecialAddrsList(
|
|
&pLatestSpecialAddrsList
|
|
);
|
|
BAIL_ON_WIN32_ERROR(dwError);
|
|
|
|
ENTER_SPD_SECTION();
|
|
|
|
bListSame = IsSpecialListSame(
|
|
pLatestSpecialAddrsList,
|
|
gpSpecialAddrsList
|
|
);
|
|
if (bListSame) {
|
|
(VOID) FreeSpecialAddrList(
|
|
&pLatestSpecialAddrsList
|
|
);
|
|
|
|
LEAVE_SPD_SECTION();
|
|
return ERROR_SUCCESS;
|
|
}
|
|
|
|
pOldSpecialAddrsList = gpSpecialAddrsList;
|
|
gpSpecialAddrsList = pLatestSpecialAddrsList;
|
|
|
|
(VOID) FreeSpecialAddrList(
|
|
&pOldSpecialAddrsList
|
|
);
|
|
|
|
(VOID) ApplyIfChangeToIniMMFilters(
|
|
&dwMMError,
|
|
gpInterfaceList,
|
|
gpSpecialAddrsList
|
|
);
|
|
|
|
(VOID) ApplyIfChangeToIniTxFilters(
|
|
&dwTxError,
|
|
gpInterfaceList,
|
|
gpSpecialAddrsList
|
|
);
|
|
|
|
(VOID) ApplyIfChangeToIniTnFilters(
|
|
&dwTnError,
|
|
gpInterfaceList,
|
|
gpSpecialAddrsList
|
|
);
|
|
|
|
LEAVE_SPD_SECTION();
|
|
|
|
if (dwMMError || dwTxError || dwTnError) {
|
|
AuditEvent(
|
|
SE_CATEGID_POLICY_CHANGE,
|
|
SE_AUDITID_IPSEC_POLICY_CHANGED,
|
|
IPSECSVC_FAILED_PNP_FILTER_PROCESSING,
|
|
NULL,
|
|
FALSE,
|
|
TRUE
|
|
);
|
|
}
|
|
|
|
error:
|
|
return (dwError);
|
|
}
|
|
|
|
VOID
|
|
FormObseleteAndNewIfLists(
|
|
IN PIPSEC_INTERFACE pIfList,
|
|
IN OUT PIPSEC_INTERFACE * ppExistingIfList,
|
|
OUT PIPSEC_INTERFACE * ppObseleteIfList,
|
|
OUT PIPSEC_INTERFACE * ppNewIfList
|
|
)
|
|
{
|
|
PIPSEC_INTERFACE pObseleteIfList = NULL;
|
|
PIPSEC_INTERFACE pNewIfList = NULL;
|
|
PIPSEC_INTERFACE pIf = NULL;
|
|
PIPSEC_INTERFACE pNewIf = NULL;
|
|
PIPSEC_INTERFACE pTempIf = NULL;
|
|
BOOL bInterfaceExists = FALSE;
|
|
PIPSEC_INTERFACE pExistingIf = NULL;
|
|
PIPSEC_INTERFACE pExistingIfList = NULL;
|
|
|
|
pExistingIfList = *ppExistingIfList;
|
|
|
|
MarkInterfaceListSuspect(
|
|
pExistingIfList
|
|
);
|
|
|
|
pIf = pIfList;
|
|
|
|
while (pIf) {
|
|
|
|
bInterfaceExists = InterfaceExistsInList(
|
|
pIf,
|
|
pExistingIfList,
|
|
&pExistingIf
|
|
);
|
|
|
|
if (bInterfaceExists) {
|
|
|
|
// Interface already exists in the list.
|
|
// Delete the interface.
|
|
|
|
pTempIf = pIf;
|
|
pIf = pIf->pNext;
|
|
FreeIpsecInterface(pTempIf);
|
|
|
|
// The corresponding entry in the original interface list
|
|
// is not a suspect any more.
|
|
|
|
pExistingIf->bIsASuspect = FALSE;
|
|
|
|
}
|
|
else {
|
|
|
|
// This is a new interface.
|
|
// Add it to the list of new interfaces.
|
|
|
|
pNewIf = pIf;
|
|
pIf = pIf->pNext;
|
|
|
|
pTempIf = pNewIfList;
|
|
pNewIfList = pNewIf;
|
|
pNewIfList->pNext = pTempIf;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
DeleteObseleteInterfaces(
|
|
&pExistingIfList,
|
|
&pObseleteIfList
|
|
);
|
|
|
|
*ppExistingIfList = pExistingIfList;
|
|
*ppObseleteIfList = pObseleteIfList;
|
|
*ppNewIfList = pNewIfList;
|
|
}
|
|
|
|
|
|
VOID
|
|
AddToInterfaceList(
|
|
IN PIPSEC_INTERFACE pIfListToAppend,
|
|
OUT PIPSEC_INTERFACE * ppOriginalIfList
|
|
)
|
|
{
|
|
PIPSEC_INTERFACE pIf = NULL;
|
|
PIPSEC_INTERFACE pIfToAppend = NULL;
|
|
PIPSEC_INTERFACE pTempIf = NULL;
|
|
|
|
pIf = pIfListToAppend;
|
|
|
|
while (pIf) {
|
|
|
|
pIfToAppend = pIf;
|
|
pIf = pIf->pNext;
|
|
|
|
pTempIf = *ppOriginalIfList;
|
|
*ppOriginalIfList = pIfToAppend;
|
|
(*ppOriginalIfList)->pNext = pTempIf;
|
|
|
|
}
|
|
}
|
|
|
|
|
|
VOID
|
|
MarkInterfaceListSuspect(
|
|
IN PIPSEC_INTERFACE pExistingIfList
|
|
)
|
|
{
|
|
PIPSEC_INTERFACE pIf = NULL;
|
|
|
|
pIf = pExistingIfList;
|
|
|
|
while (pIf) {
|
|
pIf->bIsASuspect = TRUE;
|
|
pIf = pIf->pNext;
|
|
}
|
|
}
|
|
|
|
|
|
VOID
|
|
DeleteObseleteInterfaces(
|
|
IN OUT PIPSEC_INTERFACE * ppExistingIfList,
|
|
OUT PIPSEC_INTERFACE * ppObseleteIfList
|
|
)
|
|
{
|
|
PIPSEC_INTERFACE pCurIf = NULL;
|
|
PIPSEC_INTERFACE pPreIf = NULL;
|
|
PIPSEC_INTERFACE pStartIf = NULL;
|
|
PIPSEC_INTERFACE pObseleteIfList = NULL;
|
|
PIPSEC_INTERFACE pObseleteIf = NULL;
|
|
PIPSEC_INTERFACE pTempIf = NULL;
|
|
|
|
pCurIf = *ppExistingIfList;
|
|
pStartIf = pCurIf;
|
|
|
|
while (pCurIf) {
|
|
|
|
if (pCurIf->bIsASuspect) {
|
|
|
|
pObseleteIf = pCurIf;
|
|
pCurIf = pCurIf->pNext;
|
|
|
|
if (pPreIf) {
|
|
pPreIf->pNext = pCurIf;
|
|
}
|
|
else {
|
|
pStartIf = pCurIf;
|
|
}
|
|
|
|
pTempIf = pObseleteIfList;
|
|
pObseleteIfList = pObseleteIf;
|
|
pObseleteIfList->pNext = pTempIf;
|
|
|
|
}
|
|
else {
|
|
|
|
pPreIf = pCurIf;
|
|
pCurIf = pCurIf->pNext;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
*ppObseleteIfList = pObseleteIfList;
|
|
*ppExistingIfList = pStartIf;
|
|
}
|
|
|
|
|
|
BOOL
|
|
InterfaceExistsInList(
|
|
IN PIPSEC_INTERFACE pTestIf,
|
|
IN PIPSEC_INTERFACE pExistingIfList,
|
|
OUT PIPSEC_INTERFACE * ppExistingIf
|
|
)
|
|
{
|
|
PIPSEC_INTERFACE pIf = NULL;
|
|
PIPSEC_INTERFACE pExistingIf = NULL;
|
|
BOOL bIfExists = FALSE;
|
|
|
|
pIf = pExistingIfList;
|
|
|
|
while (pIf) {
|
|
|
|
if ((pIf->dwIndex == pTestIf->dwIndex) &&
|
|
(pIf->IpAddress == pTestIf->IpAddress)) {
|
|
|
|
bIfExists = TRUE;
|
|
pExistingIf = pIf;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
pIf = pIf->pNext;
|
|
|
|
}
|
|
|
|
*ppExistingIf = pExistingIf;
|
|
return (bIfExists);
|
|
}
|
|
|
|
|
|
DWORD
|
|
GetInterfaceListFromStack(
|
|
OUT PIPSEC_INTERFACE *ppIfList
|
|
)
|
|
{
|
|
DWORD dwError = 0;
|
|
PMIB_IPADDRTABLE pMibIpAddrTable = NULL;
|
|
PMIB_IFTABLE pMibIfTable = NULL;
|
|
PIPSEC_INTERFACE pIfList = NULL;
|
|
|
|
dwError = PaPNPGetIpAddrTable(
|
|
&pMibIpAddrTable
|
|
);
|
|
BAIL_ON_WIN32_ERROR(dwError);
|
|
|
|
dwError = PaPNPGetIfTable(
|
|
&pMibIfTable
|
|
);
|
|
BAIL_ON_WIN32_ERROR(dwError);
|
|
|
|
dwError = GenerateInterfaces(
|
|
pMibIpAddrTable,
|
|
pMibIfTable,
|
|
&pIfList
|
|
);
|
|
BAIL_ON_WIN32_ERROR(dwError);
|
|
|
|
*ppIfList = pIfList;
|
|
|
|
cleanup:
|
|
|
|
if (pMibIfTable) {
|
|
LocalFree(pMibIfTable);
|
|
}
|
|
|
|
if (pMibIpAddrTable) {
|
|
LocalFree(pMibIpAddrTable);
|
|
}
|
|
|
|
return (dwError);
|
|
|
|
error:
|
|
|
|
AuditEvent(
|
|
SE_CATEGID_POLICY_CHANGE,
|
|
SE_AUDITID_IPSEC_POLICY_CHANGED,
|
|
IPSECSVC_INTERFACE_LIST_INCOMPLETE,
|
|
NULL,
|
|
FALSE,
|
|
TRUE
|
|
);
|
|
*ppIfList = NULL;
|
|
|
|
TRACE(TRC_ERROR, (L"Failed to get interface list from stack: %!winerr!", dwError));
|
|
goto cleanup;
|
|
}
|
|
|
|
|
|
DWORD
|
|
GenerateInterfaces(
|
|
IN PMIB_IPADDRTABLE pMibIpAddrTable,
|
|
IN PMIB_IFTABLE pMibIfTable,
|
|
OUT PIPSEC_INTERFACE * ppIfList
|
|
)
|
|
{
|
|
DWORD dwError = 0;
|
|
DWORD dwInterfaceType = 0;
|
|
ULONG IpAddress = 0;
|
|
DWORD dwIndex = 0;
|
|
DWORD dwNumEntries = 0;
|
|
DWORD dwCnt = 0;
|
|
PMIB_IFROW pMibIfRow = NULL;
|
|
PIPSEC_INTERFACE pNewIf = NULL;
|
|
PIPSEC_INTERFACE pTempIf = NULL;
|
|
PIPSEC_INTERFACE pIfList = NULL;
|
|
DWORD dwNewIfsCnt = 0;
|
|
|
|
dwNumEntries = pMibIpAddrTable->dwNumEntries;
|
|
|
|
for (dwCnt = 0; dwCnt < dwNumEntries; dwCnt++) {
|
|
|
|
dwIndex = pMibIpAddrTable->table[dwCnt].dwIndex;
|
|
|
|
pMibIfRow = GetMibIfRow(
|
|
pMibIfTable,
|
|
dwIndex
|
|
);
|
|
if (!pMibIfRow) {
|
|
continue;
|
|
}
|
|
|
|
IpAddress = pMibIpAddrTable->table[dwCnt].dwAddr;
|
|
dwInterfaceType = pMibIfRow->dwType;
|
|
|
|
dwError = CreateNewInterface(
|
|
dwInterfaceType,
|
|
IpAddress,
|
|
dwIndex,
|
|
pMibIfRow,
|
|
&pNewIf
|
|
);
|
|
if (dwError) {
|
|
continue;
|
|
}
|
|
|
|
pTempIf = pIfList;
|
|
pIfList = pNewIf;
|
|
pIfList->pNext = pTempIf;
|
|
dwNewIfsCnt++;
|
|
|
|
}
|
|
|
|
if (dwNewIfsCnt) {
|
|
*ppIfList = pIfList;
|
|
dwError = ERROR_SUCCESS;
|
|
}
|
|
else {
|
|
*ppIfList = NULL;
|
|
dwError = ERROR_INVALID_DATA;
|
|
}
|
|
|
|
return (dwError);
|
|
}
|
|
|
|
|
|
BOOL
|
|
IsInSpecialAddrList(
|
|
PSPECIAL_ADDR pSpecialAddrList,
|
|
PSPECIAL_ADDR pInSpecialAddr
|
|
)
|
|
{
|
|
PSPECIAL_ADDR pSpecialAddr;
|
|
|
|
for (pSpecialAddr = pSpecialAddrList;
|
|
pSpecialAddr;
|
|
pSpecialAddr = pSpecialAddr->pNext) {
|
|
if (pSpecialAddr->AddrType == pInSpecialAddr->AddrType
|
|
&& pSpecialAddr->uIpAddr == pInSpecialAddr->uIpAddr
|
|
&& pSpecialAddr->InterfaceType
|
|
== pInSpecialAddr->InterfaceType) {
|
|
|
|
return TRUE;
|
|
}
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
BOOL
|
|
IsSpecialListSame(
|
|
PSPECIAL_ADDR pSpecialAddrList1,
|
|
PSPECIAL_ADDR pSpecialAddrList2
|
|
)
|
|
{
|
|
PSPECIAL_ADDR pSpecialAddr;
|
|
|
|
for (pSpecialAddr = pSpecialAddrList1;
|
|
pSpecialAddr;
|
|
pSpecialAddr = pSpecialAddr->pNext) {
|
|
if (!IsInSpecialAddrList(
|
|
pSpecialAddrList2,
|
|
pSpecialAddr
|
|
)) {
|
|
|
|
return FALSE;
|
|
}
|
|
}
|
|
|
|
for (pSpecialAddr = pSpecialAddrList2;
|
|
pSpecialAddr;
|
|
pSpecialAddr = pSpecialAddr->pNext) {
|
|
if (!IsInSpecialAddrList(
|
|
pSpecialAddrList1,
|
|
pSpecialAddr
|
|
)) {
|
|
|
|
return FALSE;
|
|
}
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
DWORD
|
|
NoDupAddSpecialAddr(
|
|
PSPECIAL_ADDR *ppSpecialAddrList,
|
|
ADDR_TYPE AddrType,
|
|
IP_ADDRESS_STRING IpAddr,
|
|
DWORD dwInterfaceType
|
|
)
|
|
{
|
|
PSPECIAL_ADDR pSpecialAddr;
|
|
BOOL Found = FALSE;
|
|
BOOL bDupInterface = FALSE;
|
|
PSPECIAL_ADDR pSpecialAddrList;
|
|
DWORD dwError = ERROR_SUCCESS;
|
|
IF_TYPE IfType = 0;
|
|
ULONG uIpAddr;
|
|
|
|
uIpAddr = inet_addr(
|
|
IpAddr.String
|
|
);
|
|
|
|
if (uIpAddr == INADDR_NONE || !uIpAddr) {
|
|
dwError = ERROR_INVALID_PARAMETER;
|
|
BAIL_ON_WIN32_ERROR(dwError);
|
|
}
|
|
|
|
if (IsDialUp(dwInterfaceType)) {
|
|
IfType = INTERFACE_TYPE_DIALUP;
|
|
}
|
|
else if (IsLAN(dwInterfaceType)) {
|
|
IfType = INTERFACE_TYPE_LAN;
|
|
}
|
|
|
|
//
|
|
// Search if an exact entry already exists or if a similar entry
|
|
// with different InterfaceType exists
|
|
//
|
|
for (pSpecialAddr = *ppSpecialAddrList;
|
|
pSpecialAddr;
|
|
pSpecialAddr = pSpecialAddr->pNext) {
|
|
if (pSpecialAddr->AddrType == AddrType
|
|
&& pSpecialAddr->uIpAddr == uIpAddr) {
|
|
if (pSpecialAddr->InterfaceType == IfType) {
|
|
Found = TRUE;
|
|
|
|
break;
|
|
}
|
|
else {
|
|
bDupInterface = TRUE;
|
|
}
|
|
}
|
|
}
|
|
|
|
//
|
|
// Add new address to list head if it doesn't exist
|
|
//
|
|
if (!Found) {
|
|
dwError = AllocateSPDMemory(
|
|
sizeof(SPECIAL_ADDR),
|
|
&pSpecialAddr);
|
|
BAIL_ON_WIN32_ERROR(dwError);
|
|
pSpecialAddr->AddrType = AddrType;
|
|
pSpecialAddr->uIpAddr = uIpAddr;
|
|
pSpecialAddr->InterfaceType = IfType;
|
|
pSpecialAddr->bDupInterface = bDupInterface;
|
|
pSpecialAddr->pNext = *ppSpecialAddrList;
|
|
*ppSpecialAddrList = pSpecialAddr;
|
|
}
|
|
|
|
error:
|
|
return dwError;
|
|
}
|
|
|
|
DWORD
|
|
FreeSpecialAddrList(
|
|
PSPECIAL_ADDR *ppSpecialAddrList
|
|
)
|
|
{
|
|
PSPECIAL_ADDR pSpecialAddr;
|
|
PSPECIAL_ADDR pTemp;
|
|
|
|
for (pSpecialAddr = *ppSpecialAddrList;
|
|
pSpecialAddr;
|
|
pSpecialAddr = pTemp) {
|
|
pTemp = pSpecialAddr->pNext;
|
|
FreeSPDMemory(pSpecialAddr);
|
|
}
|
|
|
|
*ppSpecialAddrList = NULL;
|
|
|
|
return ERROR_SUCCESS;
|
|
}
|
|
|
|
|
|
DWORD
|
|
AllocAndGetAdaptersInfo(
|
|
PIP_ADAPTER_INFO *ppAdapterInfo,
|
|
ULONG *pulOutBufLen
|
|
)
|
|
{
|
|
DWORD dwError = ERROR_SUCCESS;
|
|
PIP_ADAPTER_INFO pAdapterInfo = NULL;
|
|
ULONG ulOutBufLen = 0;
|
|
|
|
dwError = GetAdaptersInfo(
|
|
pAdapterInfo,
|
|
&ulOutBufLen
|
|
);
|
|
|
|
if (dwError == ERROR_BUFFER_OVERFLOW) {
|
|
dwError = AllocateSPDMemory(
|
|
ulOutBufLen,
|
|
&pAdapterInfo
|
|
);
|
|
BAIL_ON_WIN32_ERROR(dwError);
|
|
|
|
dwError = GetAdaptersInfo(
|
|
pAdapterInfo,
|
|
&ulOutBufLen
|
|
);
|
|
BAIL_ON_WIN32_ERROR(dwError);
|
|
}
|
|
else {
|
|
BAIL_ON_WIN32_ERROR(dwError);
|
|
}
|
|
|
|
*ppAdapterInfo = pAdapterInfo;
|
|
*pulOutBufLen = ulOutBufLen;
|
|
|
|
return dwError;
|
|
error:
|
|
if (pAdapterInfo) {
|
|
FreeSPDMemory(pAdapterInfo);
|
|
}
|
|
*ppAdapterInfo = NULL;
|
|
*pulOutBufLen =0;
|
|
return dwError;
|
|
}
|
|
|
|
DWORD
|
|
AllocAndGetPerAdapterInfo(
|
|
ULONG IfIndex,
|
|
PIP_PER_ADAPTER_INFO *ppPerAdapterInfo,
|
|
ULONG *pulOutBufLen
|
|
)
|
|
{
|
|
DWORD dwError = ERROR_SUCCESS;
|
|
PIP_PER_ADAPTER_INFO pPerAdapterInfo = NULL;
|
|
ULONG ulOutBufLen = 0;
|
|
|
|
dwError = GetPerAdapterInfo(
|
|
IfIndex,
|
|
pPerAdapterInfo,
|
|
&ulOutBufLen
|
|
);
|
|
if (dwError == ERROR_BUFFER_OVERFLOW) {
|
|
dwError = AllocateSPDMemory(
|
|
ulOutBufLen,
|
|
&pPerAdapterInfo
|
|
);
|
|
BAIL_ON_WIN32_ERROR(dwError);
|
|
|
|
dwError = GetPerAdapterInfo(
|
|
IfIndex,
|
|
pPerAdapterInfo,
|
|
&ulOutBufLen
|
|
);
|
|
BAIL_ON_WIN32_ERROR(dwError);
|
|
}
|
|
else {
|
|
BAIL_ON_WIN32_ERROR(dwError);
|
|
}
|
|
|
|
*ppPerAdapterInfo = pPerAdapterInfo;
|
|
*pulOutBufLen = ulOutBufLen;
|
|
|
|
return dwError;
|
|
error:
|
|
if (pPerAdapterInfo) {
|
|
FreeSPDMemory(pPerAdapterInfo);
|
|
}
|
|
*ppPerAdapterInfo = NULL;
|
|
*pulOutBufLen =0;
|
|
return dwError;
|
|
|
|
}
|
|
|
|
|
|
DWORD
|
|
GetSpecialAddrsList(
|
|
OUT PSPECIAL_ADDR *ppSpecialAddrsList
|
|
)
|
|
{
|
|
PSPECIAL_ADDR pAddrs = NULL;
|
|
DWORD dwAddrCnt = 0;
|
|
PIP_ADAPTER_INFO pAdapterInfo = NULL;
|
|
PIP_ADAPTER_INFO pAdapterInfoEnum = NULL;
|
|
PIP_PER_ADAPTER_INFO pPerAdapterInfo = NULL;
|
|
ULONG ulOutBufLen = 0;
|
|
PIP_ADDR_STRING pIPAddrStr = NULL;
|
|
PSPECIAL_ADDR pSpecialAddrsList = NULL;
|
|
|
|
DWORD dwError = ERROR_SUCCESS;
|
|
DWORD i;
|
|
|
|
dwError = AllocAndGetAdaptersInfo(
|
|
&pAdapterInfo,
|
|
&ulOutBufLen
|
|
);
|
|
BAIL_ON_WIN32_ERROR(dwError);
|
|
|
|
//
|
|
// Fill in special addresses
|
|
//
|
|
|
|
pAdapterInfoEnum = pAdapterInfo;
|
|
while (pAdapterInfoEnum) {
|
|
if (pAdapterInfoEnum->DhcpEnabled) {
|
|
(VOID) NoDupAddSpecialAddr(
|
|
&pSpecialAddrsList,
|
|
IP_ADDR_DHCP_SERVER,
|
|
pAdapterInfoEnum->DhcpServer.IpAddress,
|
|
pAdapterInfoEnum->Type
|
|
);
|
|
}
|
|
|
|
(VOID) NoDupAddSpecialAddr(
|
|
&pSpecialAddrsList,
|
|
IP_ADDR_DEFAULT_GATEWAY,
|
|
pAdapterInfoEnum->GatewayList.IpAddress,
|
|
pAdapterInfoEnum->Type
|
|
);
|
|
|
|
if (pAdapterInfoEnum->HaveWins) {
|
|
(VOID) NoDupAddSpecialAddr(
|
|
&pSpecialAddrsList,
|
|
IP_ADDR_WINS_SERVER,
|
|
pAdapterInfoEnum->PrimaryWinsServer.IpAddress,
|
|
pAdapterInfoEnum->Type
|
|
);
|
|
// Get Secondary WINS
|
|
pIPAddrStr = &pAdapterInfoEnum->SecondaryWinsServer;
|
|
while (pIPAddrStr) {
|
|
(VOID) NoDupAddSpecialAddr(
|
|
&pSpecialAddrsList,
|
|
IP_ADDR_WINS_SERVER,
|
|
pIPAddrStr->IpAddress,
|
|
pAdapterInfoEnum->Type
|
|
);
|
|
|
|
pIPAddrStr = pIPAddrStr->Next;
|
|
}
|
|
|
|
}
|
|
|
|
//
|
|
// Get DNS servers
|
|
//
|
|
dwError = AllocAndGetPerAdapterInfo(
|
|
pAdapterInfoEnum->Index,
|
|
&pPerAdapterInfo,
|
|
&ulOutBufLen
|
|
);
|
|
|
|
if (dwError == ERROR_SUCCESS) {
|
|
pIPAddrStr = &pPerAdapterInfo->DnsServerList;
|
|
while (pIPAddrStr) {
|
|
(VOID) NoDupAddSpecialAddr(
|
|
&pSpecialAddrsList,
|
|
IP_ADDR_DNS_SERVER,
|
|
pIPAddrStr->IpAddress,
|
|
pAdapterInfoEnum->Type
|
|
);
|
|
|
|
pIPAddrStr = pIPAddrStr->Next;
|
|
}
|
|
|
|
FreeSPDMemory(
|
|
pPerAdapterInfo
|
|
);
|
|
pPerAdapterInfo = NULL;
|
|
}
|
|
|
|
pAdapterInfoEnum = pAdapterInfoEnum->Next;
|
|
}
|
|
|
|
FreeSPDMemory(
|
|
pAdapterInfo
|
|
);
|
|
*ppSpecialAddrsList = pSpecialAddrsList;
|
|
|
|
return dwError;
|
|
|
|
error:
|
|
FreeSPDMemory(
|
|
pAdapterInfo
|
|
);
|
|
*ppSpecialAddrsList = NULL;
|
|
|
|
TRACE(TRC_WARNING, (L"Failed to create list of special servers: %!winerr!", dwError));
|
|
return dwError;
|
|
}
|
|
|
|
PMIB_IFROW
|
|
GetMibIfRow(
|
|
IN PMIB_IFTABLE pMibIfTable,
|
|
IN DWORD dwIndex
|
|
)
|
|
{
|
|
DWORD i = 0;
|
|
PMIB_IFROW pMibIfRow = NULL;
|
|
|
|
for (i = 0; i < pMibIfTable->dwNumEntries; i++) {
|
|
|
|
if (pMibIfTable->table[i].dwIndex == dwIndex) {
|
|
pMibIfRow = &(pMibIfTable->table[i]);
|
|
break;
|
|
}
|
|
|
|
}
|
|
|
|
return (pMibIfRow);
|
|
}
|
|
|
|
|
|
DWORD
|
|
CreateNewInterface(
|
|
IN DWORD dwInterfaceType,
|
|
IN ULONG IpAddress,
|
|
IN DWORD dwIndex,
|
|
IN PMIB_IFROW pMibIfRow,
|
|
OUT PIPSEC_INTERFACE * ppNewInterface
|
|
)
|
|
{
|
|
DWORD dwError = 0;
|
|
PIPSEC_INTERFACE pNewInterface = NULL;
|
|
LPWSTR pszString = NULL;
|
|
LPWSTR pszTemp = NULL;
|
|
WCHAR szDeviceName[MAXLEN_IFDESCR*sizeof(WCHAR)];
|
|
GUID gInterfaceID;
|
|
|
|
|
|
szDeviceName[0] = L'\0';
|
|
|
|
if (IpAddress == INADDR_ANY) {
|
|
dwError = ERROR_INVALID_DATA;
|
|
BAIL_ON_WIN32_ERROR(dwError);
|
|
}
|
|
else {
|
|
if (dwInterfaceType == MIB_IF_TYPE_LOOPBACK) {
|
|
dwError = ERROR_INVALID_DATA;
|
|
BAIL_ON_WIN32_ERROR(dwError);
|
|
}
|
|
}
|
|
|
|
pszString = AllocSPDStr(pMibIfRow->wszName);
|
|
if (!pszString) {
|
|
dwError = ERROR_OUTOFMEMORY;
|
|
BAIL_ON_WIN32_ERROR(dwError);
|
|
}
|
|
|
|
if (wcslen(pszString) <= wcslen(L"\\DEVICE\\TCPIP_")) {
|
|
dwError = ERROR_INVALID_DATA;
|
|
BAIL_ON_WIN32_ERROR(dwError);
|
|
}
|
|
|
|
pszTemp = pszString + wcslen(L"\\DEVICE\\TCPIP_");
|
|
|
|
wGUIDFromString(pszTemp, &gInterfaceID);
|
|
|
|
pNewInterface = (PIPSEC_INTERFACE) AllocSPDMem(
|
|
sizeof(IPSEC_INTERFACE)
|
|
);
|
|
if (!pNewInterface) {
|
|
dwError = ERROR_OUTOFMEMORY;
|
|
BAIL_ON_WIN32_ERROR(dwError);
|
|
}
|
|
|
|
pNewInterface->dwInterfaceType = dwInterfaceType;
|
|
pNewInterface->IpAddress = IpAddress;
|
|
pNewInterface->dwIndex = dwIndex;
|
|
pNewInterface->bIsASuspect = FALSE;
|
|
|
|
memcpy(
|
|
&pNewInterface->gInterfaceID,
|
|
&gInterfaceID,
|
|
sizeof(GUID)
|
|
);
|
|
|
|
pNewInterface->pszInterfaceName = NULL;
|
|
|
|
mbstowcs(
|
|
szDeviceName,
|
|
pMibIfRow->bDescr,
|
|
MAXLEN_IFDESCR
|
|
);
|
|
|
|
pNewInterface->pszDeviceName = AllocSPDStr(
|
|
szDeviceName
|
|
);
|
|
if (!pNewInterface->pszDeviceName) {
|
|
dwError = ERROR_OUTOFMEMORY;
|
|
BAIL_ON_WIN32_ERROR(dwError);
|
|
}
|
|
|
|
pNewInterface->pNext = NULL;
|
|
|
|
*ppNewInterface = pNewInterface;
|
|
|
|
cleanup:
|
|
|
|
if (pszString) {
|
|
FreeSPDStr(pszString);
|
|
}
|
|
|
|
return(dwError);
|
|
|
|
error:
|
|
|
|
*ppNewInterface = NULL;
|
|
|
|
if (pNewInterface) {
|
|
FreeIpsecInterface(pNewInterface);
|
|
}
|
|
|
|
goto cleanup;
|
|
}
|
|
|
|
|
|
BOOL
|
|
MatchInterfaceType(
|
|
IN DWORD dwIfListEntryIfType,
|
|
IN IF_TYPE FilterIfType
|
|
)
|
|
{
|
|
BOOL bMatchesType = FALSE;
|
|
|
|
if (FilterIfType == INTERFACE_TYPE_ALL) {
|
|
bMatchesType = TRUE;
|
|
}
|
|
else if (FilterIfType == INTERFACE_TYPE_LAN) {
|
|
bMatchesType = IsLAN(dwIfListEntryIfType);
|
|
}
|
|
else if (FilterIfType == INTERFACE_TYPE_DIALUP) {
|
|
bMatchesType = IsDialUp(dwIfListEntryIfType);
|
|
}
|
|
|
|
return (bMatchesType);
|
|
}
|
|
|
|
BOOL
|
|
IsLAN(
|
|
IN DWORD dwInterfaceType
|
|
)
|
|
{
|
|
BOOL bIsLAN = FALSE;
|
|
|
|
if ((dwInterfaceType == MIB_IF_TYPE_ETHERNET) ||
|
|
(dwInterfaceType == MIB_IF_TYPE_FDDI) ||
|
|
(dwInterfaceType == MIB_IF_TYPE_TOKENRING)) {
|
|
bIsLAN = TRUE;
|
|
}
|
|
|
|
return (bIsLAN);
|
|
}
|
|
|
|
|
|
BOOL
|
|
IsDialUp(
|
|
IN DWORD dwInterfaceType
|
|
)
|
|
{
|
|
BOOL bIsDialUp = FALSE;
|
|
|
|
if ((dwInterfaceType == MIB_IF_TYPE_PPP) ||
|
|
(dwInterfaceType == MIB_IF_TYPE_SLIP)) {
|
|
bIsDialUp = TRUE;
|
|
}
|
|
|
|
return (bIsDialUp);
|
|
}
|
|
|
|
|
|
DWORD
|
|
InitializeInterfaceChangeEvent(
|
|
)
|
|
{
|
|
DWORD dwError = 0;
|
|
WORD wsaVersion = MAKEWORD(2,0);
|
|
|
|
memset(&gwsaOverlapped, 0, sizeof(WSAOVERLAPPED));
|
|
|
|
// Start up WinSock.
|
|
|
|
dwError = WSAStartup(
|
|
wsaVersion,
|
|
&gwsaData
|
|
);
|
|
BAIL_ON_WIN32_ERROR(dwError);
|
|
|
|
gbwsaStarted = TRUE;
|
|
|
|
if ((LOBYTE(gwsaData.wVersion) != LOBYTE(wsaVersion)) ||
|
|
(HIBYTE(gwsaData.wVersion) != HIBYTE(wsaVersion))) {
|
|
dwError = WSAVERNOTSUPPORTED;
|
|
BAIL_ON_WIN32_ERROR(dwError);
|
|
}
|
|
|
|
// Set up the Socket.
|
|
|
|
gIfChangeEventSocket = WSASocket(
|
|
AF_INET,
|
|
SOCK_DGRAM,
|
|
0,
|
|
NULL,
|
|
0,
|
|
WSA_FLAG_OVERLAPPED
|
|
);
|
|
if (gIfChangeEventSocket == INVALID_SOCKET) {
|
|
dwError = WSAGetLastError();
|
|
BAIL_ON_WIN32_ERROR(dwError);
|
|
}
|
|
|
|
ghIfChangeEvent = WSACreateEvent();
|
|
if (ghIfChangeEvent == WSA_INVALID_EVENT) {
|
|
dwError = WSAGetLastError();
|
|
BAIL_ON_WIN32_ERROR(dwError);
|
|
}
|
|
|
|
ghOverlapEvent = WSACreateEvent();
|
|
if (ghOverlapEvent == WSA_INVALID_EVENT) {
|
|
dwError = WSAGetLastError();
|
|
BAIL_ON_WIN32_ERROR(dwError);
|
|
}
|
|
gwsaOverlapped.hEvent = ghOverlapEvent;
|
|
return (dwError);
|
|
error:
|
|
TRACE(TRC_ERROR, (L"Failed to initialize interface change event: %!winerr!", dwError));
|
|
return (dwError);
|
|
}
|
|
|
|
|
|
DWORD
|
|
ResetInterfaceChangeEvent(
|
|
)
|
|
{
|
|
DWORD dwError = 0;
|
|
LONG lNetworkEvents = FD_ADDRESS_LIST_CHANGE;
|
|
DWORD dwOutSize = 0;
|
|
|
|
ResetEvent(ghIfChangeEvent);
|
|
gbIsIoctlPended = FALSE;
|
|
|
|
dwError = WSAIoctl(
|
|
gIfChangeEventSocket,
|
|
SIO_ADDRESS_LIST_CHANGE,
|
|
NULL,
|
|
0,
|
|
NULL,
|
|
0,
|
|
&dwOutSize,
|
|
&gwsaOverlapped,
|
|
NULL
|
|
);
|
|
|
|
if (dwError == SOCKET_ERROR) {
|
|
dwError = WSAGetLastError();
|
|
if (dwError != ERROR_IO_PENDING) {
|
|
TRACE(TRC_ERROR, (L"Failed to register for interface change event: %!winerr!", dwError));
|
|
return (dwError);
|
|
}
|
|
else {
|
|
gbIsIoctlPended = TRUE;
|
|
}
|
|
}
|
|
|
|
dwError = WSAEventSelect(
|
|
gIfChangeEventSocket,
|
|
ghIfChangeEvent,
|
|
lNetworkEvents
|
|
);
|
|
#ifdef TRACE_ON
|
|
if (dwError) {
|
|
TRACE(TRC_ERROR, (L"Failed to associate socket with interface change event: %!winerr!", dwError));
|
|
}
|
|
#endif
|
|
return (dwError);
|
|
}
|
|
|
|
|
|
VOID
|
|
DestroyInterfaceChangeEvent(
|
|
)
|
|
{
|
|
DWORD dwStatus = 0;
|
|
BOOL bDoneWaiting = FALSE;
|
|
|
|
|
|
if (gIfChangeEventSocket) {
|
|
closesocket(gIfChangeEventSocket);
|
|
if (gbIsIoctlPended) {
|
|
while (!bDoneWaiting) {
|
|
dwStatus = WaitForSingleObject(
|
|
ghOverlapEvent,
|
|
1000
|
|
);
|
|
switch (dwStatus) {
|
|
case WAIT_OBJECT_0:
|
|
bDoneWaiting = TRUE;
|
|
break;
|
|
case WAIT_TIMEOUT:
|
|
ASSERT(FALSE);
|
|
break;
|
|
default:
|
|
bDoneWaiting = TRUE;
|
|
ASSERT(FALSE);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if (ghIfChangeEvent) {
|
|
CloseHandle(ghIfChangeEvent);
|
|
}
|
|
|
|
if (ghOverlapEvent) {
|
|
CloseHandle(ghOverlapEvent);
|
|
}
|
|
|
|
if (gbwsaStarted) {
|
|
WSACleanup();
|
|
}
|
|
}
|
|
|
|
|
|
HANDLE
|
|
GetInterfaceChangeEvent(
|
|
)
|
|
{
|
|
return ghOverlapEvent;
|
|
}
|
|
|
|
|
|
BOOL
|
|
IsMyAddress(
|
|
IN ULONG IpAddrToCheck,
|
|
IN ULONG IpAddrMask,
|
|
IN PIPSEC_INTERFACE pExistingIfList
|
|
)
|
|
{
|
|
BOOL bIsMyAddress = FALSE;
|
|
PIPSEC_INTERFACE pIf = NULL;
|
|
|
|
pIf = pExistingIfList;
|
|
|
|
while (pIf) {
|
|
|
|
if ((pIf->IpAddress & IpAddrMask) ==
|
|
(IpAddrToCheck & IpAddrMask)) {
|
|
|
|
bIsMyAddress = TRUE;
|
|
break;
|
|
|
|
}
|
|
|
|
pIf = pIf->pNext;
|
|
|
|
}
|
|
|
|
return (bIsMyAddress);
|
|
}
|
|
|
|
|
|
VOID
|
|
PrintInterfaceList(
|
|
IN PIPSEC_INTERFACE pInterfaceList
|
|
)
|
|
{
|
|
WCHAR PrintData[256];
|
|
PIPSEC_INTERFACE pInterface = NULL;
|
|
DWORD i = 0;
|
|
|
|
pInterface = pInterfaceList;
|
|
|
|
while (pInterface) {
|
|
|
|
wsprintf(PrintData, L"Interface Entry no. %d\n", i+1);
|
|
OutputDebugString((LPCTSTR) PrintData);
|
|
|
|
wsprintf(PrintData, L"\tInterface Type: %d\n", pInterface->dwInterfaceType);
|
|
OutputDebugString((LPCTSTR) PrintData);
|
|
|
|
wsprintf(PrintData, L"\tIP Address: %d\n", pInterface->IpAddress);
|
|
OutputDebugString((LPCTSTR) PrintData);
|
|
|
|
wsprintf(PrintData, L"\tIndex: %d\n", pInterface->dwIndex);
|
|
OutputDebugString((LPCTSTR) PrintData);
|
|
|
|
wsprintf(PrintData, L"\tIs a suspect: %d\n", pInterface->bIsASuspect);
|
|
OutputDebugString((LPCTSTR) PrintData);
|
|
|
|
i++;
|
|
pInterface = pInterface->pNext;
|
|
|
|
}
|
|
}
|
|
|
|
|
|
DWORD
|
|
GetMatchingInterfaces(
|
|
IF_TYPE FilterInterfaceType,
|
|
PIPSEC_INTERFACE pExistingIfList,
|
|
MATCHING_ADDR ** ppMatchingAddresses,
|
|
DWORD * pdwAddrCnt
|
|
)
|
|
{
|
|
DWORD dwError = 0;
|
|
MATCHING_ADDR * pMatchingAddresses = NULL;
|
|
PIPSEC_INTERFACE pTempIf = NULL;
|
|
BOOL bMatches = FALSE;
|
|
DWORD dwCnt = 0;
|
|
DWORD i = 0;
|
|
|
|
pTempIf = pExistingIfList;
|
|
while (pTempIf) {
|
|
|
|
bMatches = MatchInterfaceType(
|
|
pTempIf->dwInterfaceType,
|
|
FilterInterfaceType
|
|
);
|
|
if (bMatches) {
|
|
dwCnt++;
|
|
}
|
|
pTempIf = pTempIf->pNext;
|
|
|
|
}
|
|
|
|
if (!dwCnt) {
|
|
dwError = ERROR_SUCCESS;
|
|
BAIL_ON_WIN32_SUCCESS(dwError);
|
|
}
|
|
|
|
dwError = AllocateSPDMemory(
|
|
sizeof(MATCHING_ADDR) * dwCnt,
|
|
(LPVOID *) &pMatchingAddresses
|
|
);
|
|
BAIL_ON_WIN32_ERROR(dwError);
|
|
|
|
pTempIf = pExistingIfList;
|
|
while (pTempIf) {
|
|
|
|
bMatches = MatchInterfaceType(
|
|
pTempIf->dwInterfaceType,
|
|
FilterInterfaceType
|
|
);
|
|
if (bMatches) {
|
|
pMatchingAddresses[i].uIpAddr = pTempIf->IpAddress;
|
|
memcpy(
|
|
&pMatchingAddresses[i].gInterfaceID,
|
|
&pTempIf->gInterfaceID,
|
|
sizeof(GUID)
|
|
);
|
|
i++;
|
|
}
|
|
pTempIf = pTempIf->pNext;
|
|
|
|
}
|
|
|
|
*ppMatchingAddresses = pMatchingAddresses;
|
|
*pdwAddrCnt = i;
|
|
return (dwError);
|
|
|
|
success:
|
|
error:
|
|
|
|
*ppMatchingAddresses = NULL;
|
|
*pdwAddrCnt = 0;
|
|
return (dwError);
|
|
}
|
|
|
|
|
|
BOOL
|
|
InterfaceAddrIsLocal(
|
|
ULONG uIpAddr,
|
|
ULONG uIpAddrMask,
|
|
MATCHING_ADDR * pLocalAddresses,
|
|
DWORD dwAddrCnt
|
|
)
|
|
{
|
|
BOOL bIsLocal = FALSE;
|
|
DWORD i = 0;
|
|
|
|
|
|
for (i = 0; i < dwAddrCnt; i++) {
|
|
|
|
if ((pLocalAddresses[i].uIpAddr & uIpAddrMask) ==
|
|
(uIpAddr & uIpAddrMask)) {
|
|
|
|
bIsLocal = TRUE;
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return (bIsLocal);
|
|
}
|
|
|
|
|
|
VOID
|
|
FreeIpsecInterface(
|
|
PIPSEC_INTERFACE pIpsecInterface
|
|
)
|
|
{
|
|
if (pIpsecInterface) {
|
|
|
|
if (pIpsecInterface->pszInterfaceName) {
|
|
FreeSPDStr(pIpsecInterface->pszInterfaceName);
|
|
}
|
|
|
|
if (pIpsecInterface->pszDeviceName) {
|
|
FreeSPDStr(pIpsecInterface->pszDeviceName);
|
|
}
|
|
|
|
FreeSPDMem(pIpsecInterface);
|
|
|
|
}
|
|
}
|
|
|
|
|
|
DWORD
|
|
EnumIPSecInterfaces(
|
|
LPWSTR pServerName,
|
|
DWORD dwVersion,
|
|
PIPSEC_INTERFACE_INFO pIpsecIfTemplate,
|
|
DWORD dwFlags,
|
|
DWORD dwPreferredNumEntries,
|
|
PIPSEC_INTERFACE_INFO * ppIpsecInterfaces,
|
|
LPDWORD pdwNumInterfaces,
|
|
LPDWORD pdwNumTotalInterfaces,
|
|
LPDWORD pdwResumeHandle,
|
|
LPVOID pvReserved
|
|
)
|
|
{
|
|
DWORD dwError = 0;
|
|
DWORD dwResumeHandle = 0;
|
|
DWORD dwNumToEnum = 0;
|
|
PIPSEC_INTERFACE pIpsecIf = NULL;
|
|
DWORD dwNumTotalInterfaces = 0;
|
|
DWORD i = 0;
|
|
PIPSEC_INTERFACE pTempIf = NULL;
|
|
DWORD dwNumInterfaces = 0;
|
|
PIPSEC_INTERFACE_INFO pIpsecInterfaces = NULL;
|
|
PIPSEC_INTERFACE_INFO pTempInterface = NULL;
|
|
|
|
|
|
dwResumeHandle = *pdwResumeHandle;
|
|
|
|
if (!dwPreferredNumEntries || (dwPreferredNumEntries > MAX_INTERFACE_ENUM_COUNT)) {
|
|
dwNumToEnum = MAX_INTERFACE_ENUM_COUNT;
|
|
}
|
|
else {
|
|
dwNumToEnum = dwPreferredNumEntries;
|
|
}
|
|
|
|
ENTER_SPD_SECTION();
|
|
|
|
dwError = ValidateSecurity(
|
|
SPD_OBJECT_SERVER,
|
|
SERVER_ACCESS_ADMINISTER,
|
|
NULL,
|
|
NULL
|
|
);
|
|
BAIL_ON_LOCK_ERROR(dwError);
|
|
|
|
pIpsecIf = gpInterfaceList;
|
|
|
|
for (i = 0; (i < dwResumeHandle) && (pIpsecIf != NULL); i++) {
|
|
dwNumTotalInterfaces++;
|
|
pIpsecIf = pIpsecIf->pNext;
|
|
}
|
|
|
|
if (!pIpsecIf) {
|
|
dwError = ERROR_NO_DATA;
|
|
BAIL_ON_LOCK_ERROR(dwError);
|
|
}
|
|
|
|
pTempIf = pIpsecIf;
|
|
|
|
while (pTempIf && (dwNumInterfaces < dwNumToEnum)) {
|
|
dwNumTotalInterfaces++;
|
|
dwNumInterfaces++;
|
|
pTempIf = pTempIf->pNext;
|
|
}
|
|
|
|
while (pTempIf) {
|
|
dwNumTotalInterfaces++;
|
|
pTempIf = pTempIf->pNext;
|
|
}
|
|
|
|
dwError = SPDApiBufferAllocate(
|
|
sizeof(IPSEC_INTERFACE_INFO)*dwNumInterfaces,
|
|
&pIpsecInterfaces
|
|
);
|
|
BAIL_ON_LOCK_ERROR(dwError);
|
|
|
|
pTempIf = pIpsecIf;
|
|
pTempInterface = pIpsecInterfaces;
|
|
|
|
for (i = 0; i < dwNumInterfaces; i++) {
|
|
|
|
dwError = CopyIpsecInterface(
|
|
pTempIf,
|
|
pTempInterface
|
|
);
|
|
BAIL_ON_LOCK_ERROR(dwError);
|
|
|
|
pTempIf = pTempIf->pNext;
|
|
pTempInterface++;
|
|
|
|
}
|
|
|
|
*ppIpsecInterfaces = pIpsecInterfaces;
|
|
*pdwNumInterfaces = dwNumInterfaces;
|
|
*pdwNumTotalInterfaces = dwNumTotalInterfaces;
|
|
*pdwResumeHandle = dwResumeHandle + dwNumInterfaces;
|
|
|
|
LEAVE_SPD_SECTION();
|
|
|
|
return (dwError);
|
|
|
|
lock:
|
|
|
|
LEAVE_SPD_SECTION();
|
|
|
|
if (pIpsecInterfaces) {
|
|
FreeIpsecInterfaceInfos(
|
|
i,
|
|
pIpsecInterfaces
|
|
);
|
|
}
|
|
|
|
*ppIpsecInterfaces = NULL;
|
|
*pdwNumInterfaces = 0;
|
|
*pdwNumTotalInterfaces = 0;
|
|
*pdwResumeHandle = dwResumeHandle;
|
|
|
|
return (dwError);
|
|
}
|
|
|
|
|
|
DWORD
|
|
CopyIpsecInterface(
|
|
PIPSEC_INTERFACE pIpsecIf,
|
|
PIPSEC_INTERFACE_INFO pIpsecInterface
|
|
)
|
|
{
|
|
DWORD dwError = 0;
|
|
|
|
|
|
memcpy(
|
|
&(pIpsecInterface->gInterfaceID),
|
|
&(pIpsecIf->gInterfaceID),
|
|
sizeof(GUID)
|
|
);
|
|
|
|
pIpsecInterface->dwIndex = pIpsecIf->dwIndex;
|
|
|
|
if (!(pIpsecIf->pszInterfaceName)) {
|
|
(VOID) GetInterfaceName(
|
|
pIpsecIf->gInterfaceID,
|
|
&pIpsecIf->pszInterfaceName
|
|
);
|
|
}
|
|
|
|
if (pIpsecIf->pszInterfaceName) {
|
|
|
|
dwError = SPDApiBufferAllocate(
|
|
wcslen(pIpsecIf->pszInterfaceName)*sizeof(WCHAR)
|
|
+ sizeof(WCHAR),
|
|
&(pIpsecInterface->pszInterfaceName)
|
|
);
|
|
BAIL_ON_WIN32_ERROR(dwError);
|
|
|
|
wcscpy(pIpsecInterface->pszInterfaceName, pIpsecIf->pszInterfaceName);
|
|
|
|
}
|
|
|
|
dwError = SPDApiBufferAllocate(
|
|
wcslen(pIpsecIf->pszDeviceName)*sizeof(WCHAR)
|
|
+ sizeof(WCHAR),
|
|
&(pIpsecInterface->pszDeviceName)
|
|
);
|
|
BAIL_ON_WIN32_ERROR(dwError);
|
|
|
|
wcscpy(pIpsecInterface->pszDeviceName, pIpsecIf->pszDeviceName);
|
|
|
|
pIpsecInterface->dwInterfaceType = pIpsecIf->dwInterfaceType;
|
|
|
|
pIpsecInterface->IpVersion = IPSEC_PROTOCOL_V4;
|
|
pIpsecInterface->uIpAddr = pIpsecIf->IpAddress;
|
|
|
|
return (dwError);
|
|
|
|
error:
|
|
|
|
if (pIpsecInterface->pszInterfaceName) {
|
|
SPDApiBufferFree(pIpsecInterface->pszInterfaceName);
|
|
}
|
|
|
|
return (dwError);
|
|
}
|
|
|
|
|
|
VOID
|
|
FreeIpsecInterfaceInfos(
|
|
DWORD dwNumInterfaces,
|
|
PIPSEC_INTERFACE_INFO pIpsecInterfaces
|
|
)
|
|
{
|
|
PIPSEC_INTERFACE_INFO pTempInterface = NULL;
|
|
DWORD i = 0;
|
|
|
|
|
|
if (!pIpsecInterfaces) {
|
|
return;
|
|
}
|
|
|
|
pTempInterface = pIpsecInterfaces;
|
|
|
|
for (i = 0; i < dwNumInterfaces; i++) {
|
|
|
|
if (pTempInterface->pszInterfaceName) {
|
|
SPDApiBufferFree(pTempInterface->pszInterfaceName);
|
|
}
|
|
|
|
if (pTempInterface->pszDeviceName) {
|
|
SPDApiBufferFree(pTempInterface->pszDeviceName);
|
|
}
|
|
|
|
pTempInterface++;
|
|
|
|
}
|
|
|
|
SPDApiBufferFree(pIpsecInterfaces);
|
|
}
|
|
|
|
|
|
DWORD
|
|
GetInterfaceName(
|
|
GUID gInterfaceID,
|
|
LPWSTR * ppszInterfaceName
|
|
)
|
|
{
|
|
DWORD dwError = 0;
|
|
DWORD dwSize = 0;
|
|
WCHAR szInterfaceName[512];
|
|
|
|
|
|
*ppszInterfaceName = NULL;
|
|
szInterfaceName[0] = L'\0';
|
|
|
|
dwSize = sizeof(szInterfaceName)/sizeof(WCHAR);
|
|
|
|
dwError = NhGetInterfaceNameFromGuid(
|
|
&gInterfaceID,
|
|
szInterfaceName,
|
|
&dwSize,
|
|
FALSE,
|
|
FALSE
|
|
);
|
|
BAIL_ON_WIN32_ERROR(dwError);
|
|
|
|
*ppszInterfaceName = AllocSPDStr(
|
|
szInterfaceName
|
|
);
|
|
|
|
error:
|
|
|
|
return (dwError);
|
|
}
|
|
|