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.
1657 lines
41 KiB
1657 lines
41 KiB
/*++
|
|
|
|
Copyright (c) 1997 Microsoft Corporation
|
|
|
|
Module Name:
|
|
|
|
confaddr.cpp
|
|
|
|
Abstract:
|
|
|
|
This module contains implementation of CIPConfMSP.
|
|
|
|
Author:
|
|
|
|
Mu Han (muhan) 5-September-1997
|
|
|
|
--*/
|
|
#include "stdafx.h"
|
|
|
|
#include <initguid.h>
|
|
DEFINE_GUID(CLSID_IPConfMSP, 0x0F1BE7F7, 0x45CA, 0x11d2,
|
|
0x83, 0x1F, 0x0, 0xA0, 0x24, 0x4D, 0x22, 0x98);
|
|
|
|
#ifdef USEIPADDRTABLE
|
|
|
|
#include <iprtrmib.h>
|
|
|
|
typedef DWORD (WINAPI * PFNGETIPADDRTABLE)(
|
|
OUT PMIB_IPADDRTABLE pIPAddrTable,
|
|
IN OUT PDWORD pdwSize,
|
|
IN BOOL bOrder
|
|
);
|
|
|
|
#define IPHLPAPI_DLL L"IPHLPAPI.DLL"
|
|
|
|
#define GETIPADDRTABLE "GetIpAddrTable"
|
|
|
|
#define IsValidInterface(_dwAddr_) \
|
|
(((_dwAddr_) != 0) && \
|
|
((_dwAddr_) != htonl(INADDR_LOOPBACK)))
|
|
|
|
#endif
|
|
|
|
#define IPCONF_WINSOCKVERSION MAKEWORD(2,0)
|
|
|
|
HRESULT CIPConfMSP::FinalConstruct()
|
|
{
|
|
// initialize winsock stack
|
|
WSADATA wsaData;
|
|
if (WSAStartup(IPCONF_WINSOCKVERSION, &wsaData) != 0)
|
|
{
|
|
LOG((MSP_ERROR, "WSAStartup failed with:%x", WSAGetLastError()));
|
|
return E_FAIL;
|
|
}
|
|
|
|
// allocate control socket
|
|
m_hSocket = WSASocket(
|
|
AF_INET, // af
|
|
SOCK_DGRAM, // type
|
|
IPPROTO_IP, // protocol
|
|
NULL, // lpProtocolInfo
|
|
0, // g
|
|
0 // dwFlags
|
|
);
|
|
|
|
// validate handle
|
|
if (m_hSocket == INVALID_SOCKET) {
|
|
|
|
LOG((
|
|
MSP_ERROR,
|
|
"error %d creating control socket.\n",
|
|
WSAGetLastError()
|
|
));
|
|
|
|
// failure
|
|
WSACleanup();
|
|
|
|
return E_FAIL;
|
|
}
|
|
|
|
HRESULT hr = CMSPAddress::FinalConstruct();
|
|
|
|
if (hr != S_OK)
|
|
{
|
|
// close socket
|
|
closesocket(m_hSocket);
|
|
|
|
// shutdown
|
|
WSACleanup();
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
void CIPConfMSP::FinalRelease()
|
|
{
|
|
CMSPAddress::FinalRelease();
|
|
|
|
if (m_hDxmrtp)
|
|
{
|
|
FreeLibrary(m_hDxmrtp);
|
|
m_hDxmrtp = NULL;
|
|
}
|
|
|
|
if (m_hSocket != INVALID_SOCKET)
|
|
{
|
|
// close socket
|
|
closesocket(m_hSocket);
|
|
}
|
|
|
|
// shutdown
|
|
WSACleanup();
|
|
}
|
|
|
|
DWORD CIPConfMSP::FindLocalInterface(DWORD dwIP)
|
|
{
|
|
|
|
SOCKADDR_IN DestAddr;
|
|
DestAddr.sin_family = AF_INET;
|
|
DestAddr.sin_port = 0;
|
|
DestAddr.sin_addr.s_addr = htonl(dwIP);
|
|
|
|
SOCKADDR_IN LocAddr;
|
|
|
|
// query for default address based on destination
|
|
|
|
DWORD dwStatus;
|
|
DWORD dwLocAddrSize = sizeof(SOCKADDR_IN);
|
|
DWORD dwNumBytesReturned = 0;
|
|
|
|
if ((dwStatus = WSAIoctl(
|
|
m_hSocket, // SOCKET s
|
|
SIO_ROUTING_INTERFACE_QUERY, // DWORD dwIoControlCode
|
|
&DestAddr, // LPVOID lpvInBuffer
|
|
sizeof(SOCKADDR_IN), // DWORD cbInBuffer
|
|
&LocAddr, // LPVOID lpvOUTBuffer
|
|
dwLocAddrSize, // DWORD cbOUTBuffer
|
|
&dwNumBytesReturned, // LPDWORD lpcbBytesReturned
|
|
NULL, // LPWSAOVERLAPPED lpOverlapped
|
|
NULL // LPWSAOVERLAPPED_COMPLETION_ROUTINE lpComplROUTINE
|
|
)) == SOCKET_ERROR)
|
|
{
|
|
|
|
dwStatus = WSAGetLastError();
|
|
|
|
LOG((MSP_ERROR, "WSAIoctl failed: %d (0x%X)", dwStatus, dwStatus));
|
|
|
|
return INADDR_NONE;
|
|
}
|
|
|
|
DWORD dwAddr = ntohl(LocAddr.sin_addr.s_addr);
|
|
|
|
if (dwAddr == 0x7f000001)
|
|
{
|
|
// it is loopback address, just return none.
|
|
return INADDR_NONE;
|
|
}
|
|
|
|
return dwAddr;
|
|
}
|
|
|
|
HRESULT LoadTapiAudioFilterDLL(
|
|
IN const TCHAR * const strDllName,
|
|
IN OUT HMODULE * phModule,
|
|
IN const char * const strAudioGetDeviceInfo,
|
|
IN const char * const strAudioReleaseDeviceInfo,
|
|
OUT PFNAudioGetDeviceInfo * ppfnAudioGetDeviceInfo,
|
|
OUT PFNAudioReleaseDeviceInfo * ppfnAudioReleaseDeviceInfo
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This method enumerate loads the tapi video capture dll.
|
|
|
|
Arguments:
|
|
|
|
str DllName - The name of the dll.
|
|
|
|
phModule - memory to store returned module handle.
|
|
|
|
ppfnAudioGetDeviceInfo - memory to store the address of AudioGetDeviceInfo
|
|
function.
|
|
|
|
ppfnAudioReleaseDeviceInfo - memory to store the address of
|
|
AudioReleaseDeviceInfo function.
|
|
|
|
Return Value:
|
|
|
|
S_OK - success.
|
|
E_FAIL - failure.
|
|
|
|
--*/
|
|
{
|
|
ENTER_FUNCTION("CIPConfMSP::LoadTapiAudioFilterDLL");
|
|
|
|
// dynamically load the video capture filter dll.
|
|
if (*phModule == NULL)
|
|
{
|
|
*phModule = LoadLibrary(strDllName);
|
|
}
|
|
|
|
// validate handle
|
|
if (*phModule == NULL)
|
|
{
|
|
LOG((MSP_ERROR, "%s, could not load %s., error:%d",
|
|
__fxName, strDllName, GetLastError()));
|
|
|
|
return E_FAIL;
|
|
}
|
|
|
|
// retrieve function pointer to retrieve addresses
|
|
PFNAudioGetDeviceInfo pfnAudioGetDeviceInfo
|
|
= (PFNAudioGetDeviceInfo)GetProcAddress(
|
|
*phModule, strAudioGetDeviceInfo
|
|
);
|
|
|
|
// validate function pointer
|
|
if (pfnAudioGetDeviceInfo == NULL)
|
|
{
|
|
LOG((MSP_ERROR, "%s, could not resolve %s, error:%d",
|
|
__fxName, strAudioGetDeviceInfo, GetLastError()));
|
|
|
|
// failure
|
|
return E_FAIL;
|
|
}
|
|
|
|
// retrieve function pointer to retrieve addresses
|
|
PFNAudioReleaseDeviceInfo pfnAudioReleaseDeviceInfo
|
|
= (PFNAudioReleaseDeviceInfo)GetProcAddress(
|
|
*phModule, strAudioReleaseDeviceInfo
|
|
);
|
|
|
|
// validate function pointer
|
|
if (pfnAudioReleaseDeviceInfo == NULL)
|
|
{
|
|
LOG((MSP_ERROR, "%s, could not resolve %s, error:%d",
|
|
__fxName, strAudioReleaseDeviceInfo, GetLastError()));
|
|
|
|
// failure
|
|
return E_FAIL;
|
|
}
|
|
|
|
*ppfnAudioGetDeviceInfo = pfnAudioGetDeviceInfo;
|
|
*ppfnAudioReleaseDeviceInfo = pfnAudioReleaseDeviceInfo;
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
HRESULT CIPConfMSP::CreateAudioCaptureTerminals()
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This method creates audio capture terminals. It uses DShow devenum to
|
|
enumerate all the wavein capture devices first. Then it will enumerate
|
|
all the DSound capture devices and match them up by name.
|
|
|
|
Arguments:
|
|
|
|
nothing
|
|
|
|
Return Value:
|
|
|
|
S_OK
|
|
|
|
--*/
|
|
{
|
|
const TCHAR * const strAudioCaptureDll = TEXT("DXMRTP");
|
|
|
|
ENTER_FUNCTION("CIPConfMSP::CreateAudioCaptureTerminals");
|
|
|
|
// dynamically load the audio capture filter dll.
|
|
PFNAudioGetDeviceInfo pfnAudioGetDeviceInfo;
|
|
PFNAudioReleaseDeviceInfo pfnAudioReleaseDeviceInfo;
|
|
|
|
HRESULT hr = LoadTapiAudioFilterDLL(
|
|
strAudioCaptureDll,
|
|
&m_hDxmrtp,
|
|
"AudioGetCaptureDeviceInfo",
|
|
"AudioReleaseCaptureDeviceInfo",
|
|
&pfnAudioGetDeviceInfo,
|
|
&pfnAudioReleaseDeviceInfo
|
|
);
|
|
|
|
if (FAILED(hr))
|
|
{
|
|
return hr;
|
|
}
|
|
|
|
DWORD dwNumDevices;
|
|
AudioDeviceInfo *pDeviceInfo;
|
|
|
|
hr = (*pfnAudioGetDeviceInfo)(&dwNumDevices, &pDeviceInfo);
|
|
if (FAILED(hr))
|
|
{
|
|
LOG((MSP_ERROR, "%s, AudioGetDeviceInfo failed. hr=%x", __fxName, hr));
|
|
|
|
return hr;
|
|
}
|
|
|
|
// for each devie, create a terminal.
|
|
for (DWORD i = 0; i < dwNumDevices; i ++)
|
|
{
|
|
ITTerminal *pTerminal;
|
|
|
|
hr = CIPConfAudioCaptureTerminal::CreateTerminal(
|
|
&pDeviceInfo[i],
|
|
(MSP_HANDLE) this,
|
|
&pTerminal
|
|
);
|
|
|
|
if (FAILED(hr))
|
|
{
|
|
LOG((MSP_ERROR, "%s, CreateTerminal for device %d failed. hr=%x",
|
|
__fxName, i, hr));
|
|
break;
|
|
}
|
|
|
|
if (!m_Terminals.Add(pTerminal))
|
|
{
|
|
hr = E_OUTOFMEMORY;
|
|
LOG((MSP_ERROR, "%s, out of mem in adding a terminal", __fxName));
|
|
break;
|
|
}
|
|
}
|
|
|
|
// release the device info
|
|
(*pfnAudioReleaseDeviceInfo)(pDeviceInfo);
|
|
|
|
return hr;
|
|
}
|
|
|
|
|
|
HRESULT CIPConfMSP::CreateAudioRenderTerminals()
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This method enumerate all the audio render devices and creates a
|
|
terminal for each of them.
|
|
|
|
Arguments:
|
|
|
|
nothing
|
|
|
|
Return Value:
|
|
|
|
S_OK
|
|
|
|
--*/
|
|
{
|
|
ENTER_FUNCTION("CIPConfMSP::CreateAudioRenderTerminals");
|
|
const TCHAR * const strAudioRenderDll = TEXT("DXMRTP");
|
|
|
|
// dynamically load the audio render filter dll.
|
|
PFNAudioGetDeviceInfo pfnAudioGetDeviceInfo;
|
|
PFNAudioReleaseDeviceInfo pfnAudioReleaseDeviceInfo;
|
|
|
|
HRESULT hr = LoadTapiAudioFilterDLL(
|
|
strAudioRenderDll,
|
|
&m_hDxmrtp,
|
|
"AudioGetRenderDeviceInfo",
|
|
"AudioReleaseRenderDeviceInfo",
|
|
&pfnAudioGetDeviceInfo,
|
|
&pfnAudioReleaseDeviceInfo
|
|
);
|
|
|
|
if (FAILED(hr))
|
|
{
|
|
return hr;
|
|
}
|
|
|
|
DWORD dwNumDevices;
|
|
AudioDeviceInfo *pDeviceInfo;
|
|
|
|
hr = (*pfnAudioGetDeviceInfo)(&dwNumDevices, &pDeviceInfo);
|
|
if (FAILED(hr))
|
|
{
|
|
LOG((MSP_ERROR, "%s, AudioGetDeviceInfo failed. hr=%x", __fxName, hr));
|
|
|
|
return hr;
|
|
}
|
|
|
|
// for each devie, create a terminal.
|
|
for (DWORD i = 0; i < dwNumDevices; i ++)
|
|
{
|
|
ITTerminal *pTerminal;
|
|
|
|
hr = CIPConfAudioRenderTerminal::CreateTerminal(
|
|
&pDeviceInfo[i],
|
|
(MSP_HANDLE) this,
|
|
&pTerminal
|
|
);
|
|
|
|
if (FAILED(hr))
|
|
{
|
|
LOG((MSP_ERROR, "%s, CreateTerminal for device %d failed. hr=%x",
|
|
__fxName, i, hr));
|
|
break;
|
|
}
|
|
|
|
if (!m_Terminals.Add(pTerminal))
|
|
{
|
|
hr = E_OUTOFMEMORY;
|
|
LOG((MSP_ERROR, "%s, out of mem in adding a terminal", __fxName));
|
|
break;
|
|
}
|
|
}
|
|
|
|
// release the device info
|
|
(*pfnAudioReleaseDeviceInfo)(pDeviceInfo);
|
|
|
|
return hr;
|
|
}
|
|
|
|
HRESULT LoadTapiVideoCaptureDLL(
|
|
IN OUT HMODULE * phModule,
|
|
OUT PFNGetNumCapDevices * ppfnGetNumCapDevices,
|
|
OUT PFNGetCapDeviceInfo * ppfnGetCapDeviceInfo
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This method enumerate loads the tapi video capture dll.
|
|
|
|
Arguments:
|
|
|
|
phModule - memory to store returned module handle.
|
|
|
|
ppfnGetNumCapDevices - memory to store the address of GetNumCapDevices
|
|
function.
|
|
|
|
ppfnGetCapDeviceInfo - memory to store the address of GetCapDeviceInfo
|
|
function.
|
|
|
|
Return Value:
|
|
|
|
S_OK - success.
|
|
E_FAIL - failure.
|
|
|
|
--*/
|
|
{
|
|
const TCHAR * const strVideoCaptureDll = TEXT("DXMRTP");
|
|
const char * const strGetNumCapDevices = "GetNumVideoCapDevices";
|
|
const char * const strGetCapDeviceInfo = "GetVideoCapDeviceInfo";
|
|
|
|
ENTER_FUNCTION("CIPConfMSP::LoadTapiVideoCaptureDLL");
|
|
|
|
// dynamically load the video capture filter dll.
|
|
if (*phModule == NULL)
|
|
{
|
|
*phModule = LoadLibrary(strVideoCaptureDll);
|
|
}
|
|
|
|
// validate handle
|
|
if (*phModule == NULL)
|
|
{
|
|
LOG((MSP_ERROR, "%s, could not load %s., error:%d",
|
|
__fxName, strVideoCaptureDll, GetLastError()));
|
|
|
|
return E_FAIL;
|
|
}
|
|
|
|
// retrieve function pointer to retrieve addresses
|
|
PFNGetNumCapDevices pfnGetNumCapDevices
|
|
= (PFNGetNumCapDevices)GetProcAddress(*phModule, strGetNumCapDevices);
|
|
|
|
// validate function pointer
|
|
if (pfnGetNumCapDevices == NULL)
|
|
{
|
|
LOG((MSP_ERROR, "%s, could not resolve %s, error:%d",
|
|
__fxName, strGetNumCapDevices, GetLastError()));
|
|
|
|
// failure
|
|
return E_FAIL;
|
|
}
|
|
|
|
// retrieve function pointer to retrieve addresses
|
|
PFNGetCapDeviceInfo pfnGetCapDeviceInfo
|
|
= (PFNGetCapDeviceInfo)GetProcAddress(*phModule, strGetCapDeviceInfo);
|
|
|
|
// validate function pointer
|
|
if (pfnGetCapDeviceInfo == NULL)
|
|
{
|
|
LOG((MSP_ERROR, "%s, could not resolve %s, error:%d",
|
|
__fxName, strGetCapDeviceInfo, GetLastError()));
|
|
|
|
// failure
|
|
return E_FAIL;
|
|
}
|
|
|
|
*ppfnGetNumCapDevices = pfnGetNumCapDevices;
|
|
*ppfnGetCapDeviceInfo = pfnGetCapDeviceInfo;
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
HRESULT CIPConfMSP::CreateVideoCaptureTerminals()
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This method is called by UpdateTerminalList to create all the video
|
|
capture terminals. It loads the video capture dll and calls its device
|
|
enumeration code to enumerate the devices.
|
|
|
|
Arguments:
|
|
|
|
nothing
|
|
|
|
Return Value:
|
|
|
|
S_OK
|
|
|
|
--*/
|
|
{
|
|
ENTER_FUNCTION("CIPConfMSP::CreateVideoCaptureTerminals");
|
|
|
|
// dynamically load the video capture filter dll.
|
|
PFNGetNumCapDevices pfnGetNumCapDevices;
|
|
PFNGetCapDeviceInfo pfnGetCapDeviceInfo;
|
|
|
|
HRESULT hr = LoadTapiVideoCaptureDLL(
|
|
&m_hDxmrtp,
|
|
&pfnGetNumCapDevices,
|
|
&pfnGetCapDeviceInfo
|
|
);
|
|
|
|
if (FAILED(hr))
|
|
{
|
|
return hr;
|
|
}
|
|
|
|
DWORD dwNumDevices = 0;
|
|
hr = (*pfnGetNumCapDevices)(&dwNumDevices);
|
|
|
|
// we have to check against S_OK because the function returns S_FALSE when
|
|
// there is no device.
|
|
if (hr != S_OK)
|
|
{
|
|
if (FAILED(hr))
|
|
{
|
|
LOG((MSP_ERROR, "%s, GetNumCapDevices failed. hr=%x", __fxName, hr));
|
|
}
|
|
else
|
|
{
|
|
LOG((MSP_WARN, "%s, There is no video device. hr=%x", __fxName, hr));
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
for (DWORD i = 0; i < dwNumDevices; i ++)
|
|
{
|
|
VIDEOCAPTUREDEVICEINFO DeviceInfo;
|
|
hr = (*pfnGetCapDeviceInfo)(i, &DeviceInfo);
|
|
|
|
if (FAILED(hr))
|
|
{
|
|
LOG((MSP_ERROR, "%s, GetNumCapDevices for device %d failed. hr=%x",
|
|
__fxName, i, hr));
|
|
break;
|
|
}
|
|
|
|
ITTerminal * pTerminal;
|
|
|
|
hr = CIPConfVideoCaptureTerminal::CreateTerminal(
|
|
DeviceInfo.szDeviceDescription,
|
|
i,
|
|
(MSP_HANDLE) this,
|
|
&pTerminal
|
|
);
|
|
|
|
if (FAILED(hr))
|
|
{
|
|
LOG((MSP_ERROR, "%s, CreateTerminal for device %d failed. hr=%x",
|
|
__fxName, i, hr));
|
|
break;
|
|
}
|
|
|
|
if (!m_Terminals.Add(pTerminal))
|
|
{
|
|
hr = E_OUTOFMEMORY;
|
|
LOG((MSP_ERROR, "%s, out of mem in adding a terminal", __fxName));
|
|
break;
|
|
}
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
HRESULT CIPConfMSP::UpdateTerminalList(void)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This method is called by the base class when it first tries to enumerate
|
|
all the static terminals. We override this function to create our own
|
|
terminals that use our own filters.
|
|
|
|
Arguments:
|
|
|
|
nothing
|
|
|
|
Return Value:
|
|
|
|
S_OK
|
|
|
|
--*/
|
|
{
|
|
ENTER_FUNCTION("CIPConfMSP::UpdateTerminalList");
|
|
|
|
// the failure of one category of terminals doesn't prevent the enumeration
|
|
// of other categories. So we ignore the return code here. If all
|
|
// categories fail, the app will get an empty list.
|
|
|
|
CreateAudioCaptureTerminals();
|
|
CreateAudioRenderTerminals();
|
|
CreateVideoCaptureTerminals();
|
|
|
|
//
|
|
// Our list is now complete.
|
|
//
|
|
m_fTerminalsUpToDate = TRUE;
|
|
|
|
LOG((MSP_TRACE, "%s, exit S_OK", __fxName));
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
HRESULT CIPConfMSP::UpdateTerminalListForPnp(IN BOOL bDeviceArrival)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This method is called by the base class when it receives pnp event
|
|
and needs to recreate static terminal list. We override this function
|
|
to create our own terminals that use our own filters. terminal list
|
|
locks was acquired when calling this method.
|
|
|
|
Return Value:
|
|
|
|
S_OK
|
|
|
|
--*/
|
|
{
|
|
ENTER_FUNCTION ("CIPConfMSP::UpdateTerminalListForPnp");
|
|
|
|
LOG ((MSP_TRACE, "%s (%d) entered", __fxName, bDeviceArrival));
|
|
|
|
HRESULT hr;
|
|
|
|
// variables for keeping old terminal info
|
|
INT i, iout, iin, count;
|
|
TERMINAL_DIRECTION *ptd = NULL, td;
|
|
LONG *pmedia = NULL, media;
|
|
BSTR *pbstr = NULL, bstr=NULL;
|
|
ITTerminal **ppterminal = NULL;
|
|
|
|
BOOL bmatch;
|
|
MSPEVENTITEM *pevent = NULL;
|
|
|
|
count = m_Terminals.GetSize ();
|
|
|
|
if (count > 0)
|
|
{
|
|
ptd = new TERMINAL_DIRECTION[count];
|
|
pmedia = new LONG[count];
|
|
pbstr = new BSTR[count];
|
|
ppterminal = new ITTerminal* [count];
|
|
|
|
if (ptd == NULL || pmedia == NULL || pbstr == NULL || ppterminal == NULL)
|
|
{
|
|
LOG ((MSP_ERROR, "%s out of memory", __fxName));
|
|
hr = E_OUTOFMEMORY;
|
|
|
|
if (ptd) delete [] ptd;
|
|
if (pmedia) delete [] pmedia;
|
|
if (pbstr) delete [] pbstr;
|
|
if (ppterminal) delete [] ppterminal;
|
|
|
|
return hr;
|
|
}
|
|
|
|
memset (pbstr, 0, count * sizeof(BSTR));
|
|
memset (ppterminal, 0, count * sizeof(ITTerminal*));
|
|
|
|
}
|
|
|
|
// for each old terminal, record
|
|
for (i = 0; i < count; i++)
|
|
{
|
|
if (FAILED (hr = m_Terminals[i]->get_Direction (&ptd[i])))
|
|
{
|
|
LOG ((MSP_ERROR, "%s failed to get terminal direction. %x", __fxName, hr));
|
|
goto Cleanup;
|
|
}
|
|
|
|
if (FAILED (hr = m_Terminals[i]->get_MediaType (&pmedia[i])))
|
|
{
|
|
LOG ((MSP_ERROR, "%s failed to get terminal mediatype. %x", __fxName, hr));
|
|
goto Cleanup;
|
|
}
|
|
|
|
if (FAILED (hr = m_Terminals[i]->get_Name (&pbstr[i])))
|
|
{
|
|
LOG ((MSP_ERROR, "%s failed to get terminal name. %x", __fxName, hr));
|
|
goto Cleanup;
|
|
}
|
|
|
|
m_Terminals[i]->AddRef ();
|
|
ppterminal[i] = m_Terminals[i];
|
|
}
|
|
|
|
// if we release terminal inside previous loop: recording info
|
|
// we would only release some terminals if there is an error
|
|
for (i = 0; i < count; i++)
|
|
{
|
|
m_Terminals[i]->Release ();
|
|
}
|
|
m_Terminals.RemoveAll ();
|
|
|
|
// update terminal list
|
|
/*
|
|
if (FAILED (hr = UpdateTerminalList ()))
|
|
{
|
|
LOG ((MSP_ERROR, "%s failed to update terminal list. %x", __fxName, hr));
|
|
goto Cleanup;
|
|
}
|
|
*/
|
|
|
|
// copy UpdateTerminalList () here
|
|
CreateAudioCaptureTerminals();
|
|
CreateAudioRenderTerminals();
|
|
CreateVideoCaptureTerminals();
|
|
|
|
m_fTerminalsUpToDate = TRUE;
|
|
|
|
// fire event to tapi app
|
|
if (bDeviceArrival)
|
|
{
|
|
// outer loop each new terminal
|
|
for (iout = 0; iout < m_Terminals.GetSize (); iout++)
|
|
{
|
|
if (FAILED (hr = m_Terminals[iout]->get_Direction (&td)))
|
|
{
|
|
LOG ((MSP_ERROR, "%s failed to get terminal direction. %x", __fxName, hr));
|
|
goto Cleanup;
|
|
}
|
|
|
|
if (FAILED (hr = m_Terminals[iout]->get_MediaType (&media)))
|
|
{
|
|
LOG ((MSP_ERROR, "%s failed to get terminal type. %x", __fxName, hr));
|
|
goto Cleanup;
|
|
}
|
|
|
|
if (FAILED (hr = m_Terminals[iout]->get_Name (&bstr)))
|
|
{
|
|
LOG ((MSP_ERROR, "%s failed to get terminal name. %x", __fxName, hr));
|
|
goto Cleanup;
|
|
}
|
|
|
|
// inner loop check if the terminal is new
|
|
for (iin = 0, bmatch = FALSE; iin < count; iin ++)
|
|
{
|
|
if (td == ptd[iin] &&
|
|
media == pmedia[iin] &&
|
|
0 == wcscmp (bstr, pbstr[iin]))
|
|
{
|
|
bmatch = TRUE;
|
|
break;
|
|
}
|
|
}
|
|
|
|
// fire event if not match
|
|
if (!bmatch)
|
|
{
|
|
LOG ((MSP_TRACE, "%s: new device found. name %ws, td %d, media %d",
|
|
__fxName, bstr, td, media));
|
|
|
|
pevent = AllocateEventItem();
|
|
if (pevent == NULL)
|
|
{
|
|
LOG ((MSP_ERROR, "%s failed to new msp event item", __fxName));
|
|
hr = E_OUTOFMEMORY;
|
|
goto Cleanup;
|
|
}
|
|
|
|
m_Terminals[iout]->AddRef ();
|
|
|
|
pevent->MSPEventInfo.dwSize = sizeof(MSP_EVENT_INFO);
|
|
pevent->MSPEventInfo.Event = ME_ADDRESS_EVENT;
|
|
pevent->MSPEventInfo.MSP_ADDRESS_EVENT_INFO.Type = ADDRESS_TERMINAL_AVAILABLE;
|
|
pevent->MSPEventInfo.MSP_ADDRESS_EVENT_INFO.pTerminal = m_Terminals[iout];
|
|
|
|
if (FAILED (hr = PostEvent (pevent)))
|
|
{
|
|
LOG ((MSP_ERROR, "%s failed to post event. %x", __fxName, hr));
|
|
|
|
m_Terminals[iout]->Release ();
|
|
FreeEventItem(pevent);
|
|
pevent = NULL;
|
|
|
|
// we don't return, try next device
|
|
}
|
|
} // outer loop
|
|
|
|
SysFreeString (bstr);
|
|
bstr = NULL;
|
|
}
|
|
}
|
|
else // if (bDeviceArrival)
|
|
{
|
|
// outer loop each old device
|
|
for (iout = 0; iout < count; iout++)
|
|
{
|
|
// inner loop check if the device was removed
|
|
for (iin = 0, bmatch = FALSE; iin < m_Terminals.GetSize (); iin++)
|
|
{
|
|
if (FAILED (hr = m_Terminals[iin]->get_Direction (&td)))
|
|
{
|
|
LOG ((MSP_ERROR, "%s failed to get terminal direction. %x", __fxName, hr));
|
|
goto Cleanup;
|
|
}
|
|
|
|
if (FAILED (hr = m_Terminals[iin]->get_MediaType (&media)))
|
|
{
|
|
LOG ((MSP_ERROR, "%s failed to get terminal type. %x", __fxName, hr));
|
|
goto Cleanup;
|
|
}
|
|
|
|
if (FAILED (hr = m_Terminals[iin]->get_Name (&bstr)))
|
|
{
|
|
LOG ((MSP_ERROR, "%s failed to get terminal name. %x", __fxName, hr));
|
|
goto Cleanup;
|
|
}
|
|
|
|
if (td == ptd[iout] &&
|
|
media == pmedia[iout] &&
|
|
0 == wcscmp (bstr, pbstr[iout]))
|
|
{
|
|
SysFreeString (bstr);
|
|
bstr = NULL;
|
|
|
|
bmatch = TRUE;
|
|
break;
|
|
}
|
|
|
|
SysFreeString (bstr);
|
|
bstr = NULL;
|
|
}
|
|
|
|
// fire event if not match
|
|
if (!bmatch)
|
|
{
|
|
LOG ((MSP_TRACE, "%s: device removed. name %ws, td %d, media %d",
|
|
__fxName, pbstr[iout], ptd[iout], pmedia[iout]));
|
|
|
|
pevent = AllocateEventItem();
|
|
if (pevent == NULL)
|
|
{
|
|
LOG ((MSP_ERROR, "%s failed to new msp event item", __fxName));
|
|
hr = E_OUTOFMEMORY;
|
|
goto Cleanup;
|
|
}
|
|
|
|
ppterminal[iout]->AddRef ();
|
|
|
|
pevent->MSPEventInfo.dwSize = sizeof(MSP_EVENT_INFO);
|
|
pevent->MSPEventInfo.Event = ME_ADDRESS_EVENT;
|
|
pevent->MSPEventInfo.MSP_ADDRESS_EVENT_INFO.Type = ADDRESS_TERMINAL_UNAVAILABLE;
|
|
pevent->MSPEventInfo.MSP_ADDRESS_EVENT_INFO.pTerminal = ppterminal[iout];
|
|
|
|
if (FAILED (hr = PostEvent (pevent)))
|
|
{
|
|
LOG ((MSP_ERROR, "%s failed to post event. %x", __fxName, hr));
|
|
|
|
ppterminal[iout]->Release ();
|
|
FreeEventItem(pevent);
|
|
pevent = NULL;
|
|
|
|
// we don't return, try next device
|
|
}
|
|
}
|
|
} // outer loop
|
|
}
|
|
|
|
Cleanup:
|
|
if (bstr) SysFreeString (bstr);
|
|
|
|
if (count > 0)
|
|
{
|
|
delete [] ptd;
|
|
delete [] pmedia;
|
|
|
|
for (i = 0; i < count; i++)
|
|
{
|
|
if (pbstr[i]) SysFreeString (pbstr[i]);
|
|
if (ppterminal[i]) ppterminal[i]->Release ();
|
|
}
|
|
|
|
delete [] pbstr;
|
|
delete [] ppterminal;
|
|
}
|
|
|
|
LOG ((MSP_TRACE, "%s returns", __fxName));
|
|
|
|
return hr;
|
|
}
|
|
|
|
STDMETHODIMP CIPConfMSP::CreateTerminal(
|
|
IN BSTR pTerminalClass,
|
|
IN long lMediaType,
|
|
IN TERMINAL_DIRECTION Direction,
|
|
OUT ITTerminal ** ppTerminal
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This method is called by TAPI3 to create a dynamic terminal. It asks the
|
|
terminal manager to create a dynamic terminal.
|
|
|
|
Arguments:
|
|
|
|
iidTerminalClass
|
|
IID of the terminal class to be created.
|
|
|
|
dwMediaType
|
|
TAPI media type of the terminal to be created.
|
|
|
|
Direction
|
|
Terminal direction of the terminal to be created.
|
|
|
|
ppTerminal
|
|
Returned created terminal object
|
|
|
|
Return Value:
|
|
|
|
S_OK
|
|
|
|
E_OUTOFMEMORY
|
|
TAPI_E_INVALIDMEDIATYPE
|
|
TAPI_E_INVALIDTERMINALDIRECTION
|
|
TAPI_E_INVALIDTERMINALCLASS
|
|
|
|
--*/
|
|
{
|
|
ENTER_FUNCTION("CIPConfMSP::CreateTerminal");
|
|
LOG((MSP_TRACE, "%s - enter", __fxName));
|
|
|
|
//
|
|
// Check if initialized.
|
|
//
|
|
|
|
// lock the event related data
|
|
m_EventDataLock.Lock();
|
|
|
|
if ( m_htEvent == NULL )
|
|
{
|
|
// unlock the event related data
|
|
m_EventDataLock.Unlock();
|
|
|
|
LOG((MSP_ERROR,
|
|
"%s, not initialized - returning E_UNEXPECTED", __fxName));
|
|
return E_UNEXPECTED;
|
|
}
|
|
|
|
// unlock the event related data
|
|
m_EventDataLock.Unlock();
|
|
|
|
//
|
|
// Get the IID from the BSTR representation.
|
|
//
|
|
|
|
HRESULT hr;
|
|
IID iidTerminalClass;
|
|
|
|
hr = CLSIDFromString(pTerminalClass, &iidTerminalClass);
|
|
|
|
if ( FAILED(hr) )
|
|
{
|
|
LOG((MSP_ERROR, "%s, bad CLSID string", __fxName));
|
|
return E_INVALIDARG;
|
|
}
|
|
|
|
//
|
|
// Make sure we support the requested media type.
|
|
// The terminal manager checks the terminal class, terminal direction,
|
|
// and return pointer.
|
|
//
|
|
|
|
//
|
|
// we don't have any specific req's to terminal's media type.
|
|
// termmgr will check if the media type is valid at all.
|
|
//
|
|
|
|
//
|
|
// Use the terminal manager to create the dynamic terminal.
|
|
//
|
|
|
|
_ASSERTE( m_pITTerminalManager != NULL );
|
|
|
|
hr = m_pITTerminalManager->CreateDynamicTerminal(NULL,
|
|
iidTerminalClass,
|
|
(DWORD) lMediaType,
|
|
Direction,
|
|
(MSP_HANDLE) this,
|
|
ppTerminal);
|
|
|
|
if ( FAILED(hr) )
|
|
{
|
|
LOG((MSP_ERROR,
|
|
"%s create dynamic terminal failed. hr=%x", __fxName, hr));
|
|
|
|
return hr;
|
|
}
|
|
|
|
const DWORD dwAudioCaptureBitPerSample = 16;
|
|
const DWORD dwAudioSampleRate = 8000;
|
|
|
|
if ((iidTerminalClass == CLSID_MediaStreamTerminal)
|
|
&& (lMediaType == TAPIMEDIATYPE_AUDIO))
|
|
{
|
|
// Set the format of the audio to 8KHZ, 16Bit/Sample, MONO.
|
|
hr = ::SetAudioFormat(
|
|
*ppTerminal,
|
|
dwAudioCaptureBitPerSample,
|
|
dwAudioSampleRate
|
|
);
|
|
|
|
if (FAILED(hr))
|
|
{
|
|
LOG((MSP_WARN, "%s, can't set audio format, %x", __fxName, hr));
|
|
}
|
|
}
|
|
|
|
LOG((MSP_TRACE, "%s - exit S_OK", __fxName));
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
STDMETHODIMP CIPConfMSP::CreateMSPCall(
|
|
IN MSP_HANDLE htCall,
|
|
IN DWORD dwReserved,
|
|
IN DWORD dwMediaType,
|
|
IN IUnknown * pOuterUnknown,
|
|
OUT IUnknown ** ppMSPCall
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This method is called by TAPI3 before a call is made or answered. It creates
|
|
a aggregated MSPCall object and returns the IUnknown pointer. It calls the
|
|
helper template function defined in mspaddress.h to handle the real creation.
|
|
|
|
Arguments:
|
|
|
|
htCall
|
|
TAPI 3.0's identifier for this call. Returned in events passed back
|
|
to TAPI.
|
|
|
|
dwReserved
|
|
Reserved parameter. Not currently used.
|
|
|
|
dwMediaType
|
|
Media type of the call being created. These are TAPIMEDIATYPES and more
|
|
than one mediatype can be selected (bitwise).
|
|
|
|
pOuterUnknown
|
|
pointer to the IUnknown interface of the containing object.
|
|
|
|
ppMSPCall
|
|
Returned MSP call that the MSP fills on on success.
|
|
|
|
Return Value:
|
|
|
|
S_OK
|
|
E_OUTOFMEMORY
|
|
E_POINTER
|
|
TAPI_E_INVALIDMEDIATYPE
|
|
|
|
|
|
--*/
|
|
{
|
|
LOG((MSP_TRACE,
|
|
"CreateMSPCall entered. htCall:%x, dwMediaType:%x, ppMSPCall:%x",
|
|
htCall, dwMediaType, ppMSPCall
|
|
));
|
|
|
|
CIPConfMSPCall * pMSPCall = NULL;
|
|
|
|
HRESULT hr = ::CreateMSPCallHelper(
|
|
this,
|
|
htCall,
|
|
dwReserved,
|
|
dwMediaType,
|
|
pOuterUnknown,
|
|
ppMSPCall,
|
|
&pMSPCall
|
|
);
|
|
|
|
if (FAILED(hr))
|
|
{
|
|
LOG((MSP_ERROR, "CreateMSPCallHelper failed:%x", hr));
|
|
return hr;
|
|
}
|
|
|
|
// this function doesn't return anything.
|
|
pMSPCall->SetIPInterface(m_dwIPInterface);
|
|
|
|
return hr;
|
|
}
|
|
|
|
STDMETHODIMP CIPConfMSP::ShutdownMSPCall(
|
|
IN IUnknown * pUnknown
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This method is called by TAPI3 to shutdown a MSPCall. It calls the helper
|
|
function defined in MSPAddress to to the real job.
|
|
|
|
Arguments:
|
|
|
|
pUnknown
|
|
pointer to the IUnknown interface of the contained object. It is a
|
|
CComAggObject that contains our call object.
|
|
|
|
Return Value:
|
|
|
|
S_OK
|
|
E_POINTER
|
|
TAPI_E_INVALIDMEDIATYPE
|
|
|
|
|
|
--*/
|
|
{
|
|
LOG((MSP_TRACE, "ShutDownMSPCall entered. pUnknown:%x", pUnknown));
|
|
|
|
if (IsBadReadPtr(pUnknown, sizeof(VOID *) * 3))
|
|
{
|
|
LOG((MSP_ERROR, "ERROR:pUnknow is a bad pointer"));
|
|
return E_POINTER;
|
|
}
|
|
|
|
|
|
CIPConfMSPCall * pMSPCall = NULL;
|
|
HRESULT hr = ::ShutdownMSPCallHelper(pUnknown, &pMSPCall);
|
|
|
|
if (FAILED(hr))
|
|
{
|
|
LOG((MSP_ERROR, "ShutDownMSPCallhelper failed:: %x", hr));
|
|
return hr;
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
DWORD CIPConfMSP::GetCallMediaTypes(void)
|
|
{
|
|
return IPCONFCALLMEDIATYPES;
|
|
}
|
|
|
|
ULONG CIPConfMSP::MSPAddressAddRef(void)
|
|
{
|
|
return MSPAddRefHelper(this);
|
|
}
|
|
|
|
ULONG CIPConfMSP::MSPAddressRelease(void)
|
|
{
|
|
return MSPReleaseHelper(this);
|
|
}
|
|
|
|
#ifdef USEIPADDRTABLE
|
|
PMIB_IPADDRTABLE GetIPTable()
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This method is used to get the table of local IP interfaces.
|
|
|
|
Arguments:
|
|
|
|
Return Value:
|
|
|
|
NULL - failed.
|
|
Pointer - a memory buffer that contains the IP interface table.
|
|
|
|
|
|
--*/
|
|
{
|
|
// dynamically load iphlpapi.dll
|
|
HMODULE hIPHLPAPI = LoadLibraryW(IPHLPAPI_DLL);
|
|
|
|
// validate handle
|
|
if (hIPHLPAPI == NULL)
|
|
{
|
|
LOG((MSP_ERROR, "could not load %s.\n", IPHLPAPI_DLL));
|
|
// failure
|
|
return NULL;
|
|
}
|
|
|
|
PFNGETIPADDRTABLE pfnGetIpAddrTable = NULL;
|
|
|
|
// retrieve function pointer to retrieve addresses
|
|
pfnGetIpAddrTable = (PFNGETIPADDRTABLE)GetProcAddress(
|
|
hIPHLPAPI,
|
|
GETIPADDRTABLE
|
|
);
|
|
|
|
// validate function pointer
|
|
if (pfnGetIpAddrTable == NULL)
|
|
{
|
|
LOG((MSP_ERROR, "could not resolve GetIpAddrTable.\n"));
|
|
// release
|
|
FreeLibrary(hIPHLPAPI);
|
|
// failure
|
|
return NULL;
|
|
}
|
|
|
|
PMIB_IPADDRTABLE pIPAddrTable = NULL;
|
|
DWORD dwBytesRequired = 0;
|
|
DWORD dwStatus;
|
|
|
|
// determine amount of memory needed for table
|
|
dwStatus = (*pfnGetIpAddrTable)(pIPAddrTable, &dwBytesRequired, FALSE);
|
|
|
|
// validate status is what we expect
|
|
if (dwStatus != ERROR_INSUFFICIENT_BUFFER)
|
|
{
|
|
LOG((MSP_ERROR, "error 0x%08lx calling GetIpAddrTable.\n", dwStatus));
|
|
// release
|
|
FreeLibrary(hIPHLPAPI);
|
|
// failure, but we need to return true to load.
|
|
return NULL;
|
|
}
|
|
|
|
// attempt to allocate memory for table
|
|
pIPAddrTable = (PMIB_IPADDRTABLE)malloc(dwBytesRequired);
|
|
|
|
// validate pointer
|
|
if (pIPAddrTable == NULL)
|
|
{
|
|
LOG((MSP_ERROR, "could not allocate address table.\n"));
|
|
// release
|
|
FreeLibrary(hIPHLPAPI);
|
|
// failure, but we need to return true to load.
|
|
return NULL;
|
|
}
|
|
|
|
// retrieve ip address table from tcp/ip stack via utitity library
|
|
dwStatus = (*pfnGetIpAddrTable)(pIPAddrTable, &dwBytesRequired, FALSE);
|
|
|
|
// validate status
|
|
if (dwStatus != NOERROR)
|
|
{
|
|
LOG((MSP_ERROR, "error 0x%08lx calling GetIpAddrTable.\n", dwStatus));
|
|
// release table
|
|
free(pIPAddrTable);
|
|
// release
|
|
FreeLibrary(hIPHLPAPI);
|
|
// failure, but we need to return true to load.
|
|
return NULL;
|
|
}
|
|
|
|
// release library
|
|
FreeLibrary(hIPHLPAPI);
|
|
|
|
return pIPAddrTable;
|
|
}
|
|
|
|
BSTR IPToBstr(
|
|
DWORD dwIP
|
|
)
|
|
{
|
|
struct in_addr Addr;
|
|
Addr.s_addr = dwIP;
|
|
|
|
// convert the interface to a string.
|
|
CHAR *pChar = inet_ntoa(Addr);
|
|
if (pChar == NULL)
|
|
{
|
|
LOG((MSP_ERROR, "bad IP address:%x", dwIP));
|
|
return NULL;
|
|
}
|
|
|
|
// convert the ascii string to WCHAR.
|
|
WCHAR szAddressName[MAXIPADDRLEN + 1];
|
|
wsprintfW(szAddressName, L"%hs", pChar);
|
|
|
|
// create a BSTR.
|
|
BSTR bAddress = SysAllocString(szAddressName);
|
|
if (bAddress == NULL)
|
|
{
|
|
LOG((MSP_ERROR, "out of mem in allocation address name"));
|
|
return NULL;
|
|
}
|
|
|
|
return bAddress;
|
|
}
|
|
|
|
STDMETHODIMP CIPConfMSP::get_DefaultIPInterface(
|
|
OUT BSTR * ppIPAddress
|
|
)
|
|
{
|
|
LOG((MSP_TRACE, "get_DefaultIPInterface, ppIPAddress:%p", ppIPAddress));
|
|
|
|
if (IsBadWritePtr(ppIPAddress, sizeof(BSTR)))
|
|
{
|
|
LOG((MSP_ERROR,
|
|
"get_DefaultIPInterface, ppIPAddress is bad:%p", ppIPAddress));
|
|
return E_POINTER;
|
|
}
|
|
|
|
// get the current local interface.
|
|
m_Lock.Lock();
|
|
DWORD dwIP= m_dwIPInterface;
|
|
m_Lock.Unlock();
|
|
|
|
BSTR bAddress = IPToBstr(dwIP);
|
|
|
|
if (bAddress == NULL)
|
|
{
|
|
return E_OUTOFMEMORY;
|
|
}
|
|
|
|
*ppIPAddress = bAddress;
|
|
|
|
LOG((MSP_TRACE, "get_DefaultIPInterface, returning %ws", bAddress));
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
STDMETHODIMP CIPConfMSP::put_DefaultIPInterface(
|
|
IN BSTR pIPAddress
|
|
)
|
|
{
|
|
LOG((MSP_TRACE, "put_DefaultIPInterface, pIPAddress:%p", pIPAddress));
|
|
|
|
if (IsBadStringPtrW(pIPAddress, MAXIPADDRLEN))
|
|
{
|
|
LOG((MSP_ERROR,
|
|
"put_DefaultIPInterface, invalid pointer:%p", pIPAddress));
|
|
return E_POINTER;
|
|
}
|
|
|
|
char buffer[MAXIPADDRLEN + 1];
|
|
|
|
if (WideCharToMultiByte(
|
|
GetACP(),
|
|
0,
|
|
pIPAddress,
|
|
-1,
|
|
buffer,
|
|
MAXIPADDRLEN,
|
|
NULL,
|
|
NULL
|
|
) == 0)
|
|
{
|
|
LOG((MSP_ERROR, "put_DefaultIPInterface, can't covert:%ws", pIPAddress));
|
|
return E_INVALIDARG;
|
|
}
|
|
|
|
DWORD dwAddr;
|
|
if ((dwAddr = inet_addr(buffer)) == INADDR_NONE)
|
|
{
|
|
LOG((MSP_ERROR, "put_DefaultIPInterface, bad address:%s", buffer));
|
|
return E_INVALIDARG;
|
|
}
|
|
|
|
// set the current local interface.
|
|
m_Lock.Lock();
|
|
m_dwIPInterface = dwAddr;
|
|
m_Lock.Unlock();
|
|
|
|
|
|
LOG((MSP_TRACE, "put_DefaultIPInterface, set to %s", buffer));
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
HRESULT CreateBstrCollection(
|
|
IN BSTR * pBstr,
|
|
IN DWORD dwCount,
|
|
OUT VARIANT * pVariant
|
|
)
|
|
{
|
|
//
|
|
// create the collection object - see mspcoll.h
|
|
//
|
|
|
|
CComObject<CTapiBstrCollection> * pCollection;
|
|
|
|
HRESULT hr;
|
|
|
|
hr = ::CreateCComObjectInstance(&pCollection);
|
|
|
|
if ( FAILED(hr) )
|
|
{
|
|
LOG((MSP_ERROR, "get_IPInterfaces - "
|
|
"can't create collection - exit 0x%08x", hr));
|
|
|
|
return hr;
|
|
}
|
|
|
|
//
|
|
// get the Collection's IDispatch interface
|
|
//
|
|
|
|
IDispatch * pDispatch;
|
|
|
|
hr = pCollection->_InternalQueryInterface(__uuidof(IDispatch),
|
|
(void **) &pDispatch );
|
|
|
|
if ( FAILED(hr) )
|
|
{
|
|
LOG((MSP_ERROR, "get_IPInterfaces - "
|
|
"QI for IDispatch on collection failed - exit 0x%08x", hr));
|
|
|
|
delete pCollection;
|
|
|
|
return hr;
|
|
}
|
|
|
|
//
|
|
// Init the collection using an iterator -- pointers to the beginning and
|
|
// the ending element plus one.
|
|
//
|
|
|
|
hr = pCollection->Initialize( dwCount,
|
|
pBstr,
|
|
pBstr + dwCount);
|
|
|
|
if ( FAILED(hr) )
|
|
{
|
|
LOG((MSP_ERROR, "get_IPInterfaces - "
|
|
"Initialize on collection failed - exit 0x%08x", hr));
|
|
|
|
pDispatch->Release();
|
|
|
|
return hr;
|
|
}
|
|
|
|
//
|
|
// put the IDispatch interface pointer into the variant
|
|
//
|
|
|
|
LOG((MSP_ERROR, "get_IPInterfaces - "
|
|
"placing IDispatch value %08x in variant", pDispatch));
|
|
|
|
VariantInit(pVariant);
|
|
pVariant->vt = VT_DISPATCH;
|
|
pVariant->pdispVal = pDispatch;
|
|
|
|
LOG((MSP_TRACE, "get_IPInterfaces - exit S_OK"));
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
|
|
STDMETHODIMP CIPConfMSP::get_IPInterfaces(
|
|
OUT VARIANT * pVariant
|
|
)
|
|
{
|
|
PMIB_IPADDRTABLE pIPAddrTable = GetIPTable();
|
|
|
|
if (pIPAddrTable == NULL)
|
|
{
|
|
return E_FAIL;
|
|
}
|
|
|
|
BSTR *Addresses =
|
|
(BSTR *)malloc(sizeof(BSTR *) * pIPAddrTable->dwNumEntries);
|
|
|
|
if (Addresses == NULL)
|
|
{
|
|
return E_OUTOFMEMORY;
|
|
}
|
|
|
|
HRESULT hr = S_OK;
|
|
DWORD dwCount = 0;
|
|
|
|
// loop through the interfaces and find the valid ones.
|
|
for (DWORD i = 0; i < pIPAddrTable->dwNumEntries; i++)
|
|
{
|
|
if (IsValidInterface(pIPAddrTable->table[i].dwAddr))
|
|
{
|
|
DWORD dwIPAddr = ntohl(pIPAddrTable->table[i].dwAddr);
|
|
Addresses[i] = IPToBstr(dwIPAddr);
|
|
if (Addresses[i] == NULL)
|
|
{
|
|
hr = E_OUTOFMEMORY;
|
|
break;
|
|
}
|
|
|
|
dwCount ++;
|
|
}
|
|
}
|
|
|
|
// release table memory
|
|
free(pIPAddrTable);
|
|
|
|
if (FAILED(hr))
|
|
{
|
|
// release all the BSTRs and the array.
|
|
for (i = 0; i < dwCount; i ++)
|
|
{
|
|
SysFreeString(Addresses[i]);
|
|
}
|
|
free(Addresses);
|
|
return hr;
|
|
}
|
|
|
|
hr = CreateBstrCollection(Addresses, dwCount, pVariant);
|
|
|
|
// if the collection is not created, release all the BSTRs.
|
|
if (FAILED(hr))
|
|
{
|
|
for (i = 0; i < dwCount; i ++)
|
|
{
|
|
SysFreeString(Addresses[i]);
|
|
}
|
|
}
|
|
|
|
// delete the pointer array.
|
|
free(Addresses);
|
|
|
|
return hr;
|
|
}
|
|
|
|
HRESULT CreateBstrEnumerator(
|
|
IN BSTR * begin,
|
|
IN BSTR * end,
|
|
OUT IEnumBstr ** ppIEnum
|
|
)
|
|
{
|
|
typedef CSafeComEnum<IEnumBstr, &__uuidof(IEnumBstr), BSTR, _CopyBSTR>> CEnumerator;
|
|
|
|
HRESULT hr;
|
|
|
|
CComObject<CEnumerator> *pEnum = NULL;
|
|
|
|
hr = ::CreateCComObjectInstance(&pEnum);
|
|
|
|
if (pEnum == NULL)
|
|
{
|
|
LOG((MSP_ERROR, "Could not create enumerator object, %x", hr));
|
|
return hr;
|
|
}
|
|
|
|
IEnumBstr * pIEnum;
|
|
|
|
// query for the __uuidof(IEnumDirectory) i/f
|
|
hr = pEnum->_InternalQueryInterface(
|
|
__uuidof(IEnumBstr),
|
|
(void**)&pIEnum
|
|
);
|
|
|
|
if (FAILED(hr))
|
|
{
|
|
LOG((MSP_ERROR, "query enum interface failed, %x", hr));
|
|
delete pEnum;
|
|
return hr;
|
|
}
|
|
|
|
hr = pEnum->Init(begin, end, NULL, AtlFlagTakeOwnership);
|
|
|
|
if (FAILED(hr))
|
|
{
|
|
LOG((MSP_ERROR, "init enumerator object failed, %x", hr));
|
|
pIEnum->Release();
|
|
return hr;
|
|
}
|
|
|
|
*ppIEnum = pIEnum;
|
|
|
|
return hr;
|
|
}
|
|
|
|
STDMETHODIMP CIPConfMSP::EnumerateIPInterfaces(
|
|
OUT IEnumBstr ** ppIEnumBstr
|
|
)
|
|
{
|
|
PMIB_IPADDRTABLE pIPAddrTable = GetIPTable();
|
|
|
|
if (pIPAddrTable == NULL)
|
|
{
|
|
return E_FAIL;
|
|
}
|
|
|
|
BSTR *Addresses =
|
|
(BSTR *)malloc(sizeof(BSTR *) * pIPAddrTable->dwNumEntries);
|
|
|
|
if (Addresses == NULL)
|
|
{
|
|
return E_OUTOFMEMORY;
|
|
}
|
|
|
|
HRESULT hr = S_OK;
|
|
DWORD dwCount = 0;
|
|
|
|
// loop through the interfaces and find the valid ones.
|
|
for (DWORD i = 0; i < pIPAddrTable->dwNumEntries; i++)
|
|
{
|
|
if (IsValidInterface(pIPAddrTable->table[i].dwAddr))
|
|
{
|
|
DWORD dwIPAddr = ntohl(pIPAddrTable->table[i].dwAddr);
|
|
Addresses[i] = IPToBstr(dwIPAddr);
|
|
if (Addresses[i] == NULL)
|
|
{
|
|
hr = E_OUTOFMEMORY;
|
|
break;
|
|
}
|
|
|
|
dwCount ++;
|
|
}
|
|
}
|
|
|
|
// release table memory
|
|
free(pIPAddrTable);
|
|
|
|
if (FAILED(hr))
|
|
{
|
|
// release all the BSTRs and the array.
|
|
for (i = 0; i < dwCount; i ++)
|
|
{
|
|
SysFreeString(Addresses[i]);
|
|
}
|
|
free(Addresses);
|
|
return hr;
|
|
}
|
|
|
|
hr = CreateBstrEnumerator(Addresses, Addresses + dwCount, ppIEnumBstr);
|
|
|
|
// if the collection is not created, release all the BSTRs.
|
|
if (FAILED(hr))
|
|
{
|
|
for (i = 0; i < dwCount; i ++)
|
|
{
|
|
SysFreeString(Addresses[i]);
|
|
}
|
|
free(Addresses);
|
|
return hr;
|
|
}
|
|
|
|
// the enumerator will destroy the bstr array eventually,
|
|
// so no need to free anything here. Even if we tell it to hand
|
|
// out zero objects, it will delete the array on destruction.
|
|
|
|
return hr;
|
|
}
|
|
#endif
|