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.
1151 lines
29 KiB
1151 lines
29 KiB
/*++ BUILD Version: 0000 // Increment this if a change has global effects
|
|
|
|
Copyright (c) 2000-2002 Microsoft Corporation
|
|
|
|
Module Name:
|
|
|
|
mmcmgmt.cpp
|
|
|
|
Abstract:
|
|
|
|
Source file module for MMC manipulation
|
|
|
|
Author:
|
|
|
|
Xiaohai Zhang (xzhang) 22-March-2000
|
|
|
|
Revision History:
|
|
|
|
--*/
|
|
#include <stdio.h>
|
|
#include "windows.h"
|
|
#include "objbase.h"
|
|
#include "tapi.h"
|
|
|
|
#include "mmcmgmt.h"
|
|
#include "error.h"
|
|
#include "tchar.h"
|
|
|
|
///////////////////////////////////////////////////////////
|
|
//
|
|
// CMMCManagement implementation
|
|
//
|
|
///////////////////////////////////////////////////////////
|
|
|
|
HRESULT CMMCManagement::GetMMCData ()
|
|
{
|
|
HRESULT hr = S_OK;
|
|
DWORD dwAPIVersion = TAPI_CURRENT_VERSION;
|
|
DWORD dw, dw1, dw2;
|
|
LPDEVICEINFO pDeviceInfo;
|
|
TAPISERVERCONFIG cfg;
|
|
LINEPROVIDERLIST tapiProviderList = {0};
|
|
LPLINEPROVIDERENTRY pProvider;
|
|
AVAILABLEPROVIDERLIST tapiAvailProvList = {0};
|
|
LPAVAILABLEPROVIDERLIST pAvailProvList = NULL;
|
|
AVAILABLEPROVIDERENTRY *pAvailProv;
|
|
|
|
hr = MMCInitialize (
|
|
NULL, // Local computer
|
|
&m_hMmc, // HMMCAPP to return
|
|
&dwAPIVersion, // API version
|
|
NULL
|
|
);
|
|
if (FAILED(hr) || m_hMmc == NULL)
|
|
{
|
|
goto ExitHere;
|
|
}
|
|
|
|
// Mark MMC to be busy
|
|
cfg.dwTotalSize = sizeof(TAPISERVERCONFIG);
|
|
hr = MMCGetServerConfig (m_hMmc, &cfg);
|
|
if (FAILED(hr))
|
|
{
|
|
goto ExitHere;
|
|
}
|
|
cfg.dwFlags |= TAPISERVERCONFIGFLAGS_LOCKMMCWRITE;
|
|
hr = MMCSetServerConfig (m_hMmc, &cfg);
|
|
if (FAILED(hr))
|
|
{
|
|
goto ExitHere;
|
|
}
|
|
m_bMarkedBusy = TRUE;
|
|
|
|
// Get the DEVICEINFOLIST structure
|
|
m_pDeviceInfoList = (LPDEVICEINFOLIST) new BYTE[sizeof(DEVICEINFOLIST)];
|
|
if (m_pDeviceInfoList == NULL)
|
|
{
|
|
hr = TSECERR_NOMEM;
|
|
goto ExitHere;
|
|
}
|
|
m_pDeviceInfoList->dwTotalSize = sizeof(DEVICEINFOLIST);
|
|
hr = MMCGetLineInfo (
|
|
m_hMmc,
|
|
m_pDeviceInfoList
|
|
);
|
|
if (FAILED(hr))
|
|
{
|
|
goto ExitHere;
|
|
}
|
|
else if (m_pDeviceInfoList->dwNeededSize > m_pDeviceInfoList->dwTotalSize)
|
|
{
|
|
dw = m_pDeviceInfoList->dwNeededSize;
|
|
delete [] m_pDeviceInfoList;
|
|
m_pDeviceInfoList = (LPDEVICEINFOLIST) new BYTE[dw];
|
|
if (m_pDeviceInfoList == NULL)
|
|
{
|
|
hr = TSECERR_NOMEM;
|
|
goto ExitHere;
|
|
}
|
|
m_pDeviceInfoList->dwTotalSize = dw;
|
|
hr = MMCGetLineInfo (
|
|
m_hMmc,
|
|
m_pDeviceInfoList
|
|
);
|
|
if (FAILED(hr))
|
|
{
|
|
goto ExitHere;
|
|
}
|
|
}
|
|
if (m_pDeviceInfoList->dwNumDeviceInfoEntries == 0)
|
|
{
|
|
delete [] m_pDeviceInfoList;
|
|
if (m_hMmc)
|
|
{
|
|
cfg.dwFlags &= (~TAPISERVERCONFIGFLAGS_LOCKMMCWRITE);
|
|
cfg.dwFlags |= TAPISERVERCONFIGFLAGS_UNLOCKMMCWRITE;
|
|
MMCSetServerConfig (m_hMmc, &cfg);
|
|
m_bMarkedBusy = FALSE;
|
|
MMCShutdown (m_hMmc);
|
|
m_hMmc = NULL;
|
|
}
|
|
goto ExitHere;
|
|
}
|
|
|
|
// Build the user name tuple
|
|
m_pUserTuple =
|
|
new USERNAME_TUPLE[m_pDeviceInfoList->dwNumDeviceInfoEntries];
|
|
if (m_pUserTuple == NULL)
|
|
{
|
|
hr = TSECERR_NOMEM;
|
|
goto ExitHere;
|
|
}
|
|
pDeviceInfo = (LPDEVICEINFO)(((LPBYTE)m_pDeviceInfoList) +
|
|
m_pDeviceInfoList->dwDeviceInfoOffset);
|
|
for (dw = 0;
|
|
dw < m_pDeviceInfoList->dwNumDeviceInfoEntries;
|
|
++dw, ++pDeviceInfo)
|
|
{
|
|
if (pDeviceInfo->dwDomainUserNamesSize == 0)
|
|
{
|
|
m_pUserTuple[dw].pDomainUserNames = NULL;
|
|
m_pUserTuple[dw].pFriendlyUserNames = NULL;
|
|
}
|
|
else
|
|
{
|
|
m_pUserTuple[dw].pDomainUserNames =
|
|
(LPTSTR) new BYTE[pDeviceInfo->dwDomainUserNamesSize];
|
|
m_pUserTuple[dw].pFriendlyUserNames =
|
|
(LPTSTR) new BYTE[pDeviceInfo->dwFriendlyUserNamesSize];
|
|
if (m_pUserTuple[dw].pDomainUserNames == NULL ||
|
|
m_pUserTuple[dw].pFriendlyUserNames == NULL)
|
|
{
|
|
hr = TSECERR_NOMEM;
|
|
goto ExitHere;
|
|
}
|
|
memcpy (
|
|
m_pUserTuple[dw].pDomainUserNames,
|
|
(((LPBYTE)m_pDeviceInfoList) +
|
|
pDeviceInfo->dwDomainUserNamesOffset),
|
|
pDeviceInfo->dwDomainUserNamesSize
|
|
);
|
|
memcpy (
|
|
m_pUserTuple[dw].pFriendlyUserNames,
|
|
(((LPBYTE)m_pDeviceInfoList) +
|
|
pDeviceInfo->dwFriendlyUserNamesOffset),
|
|
pDeviceInfo->dwFriendlyUserNamesSize
|
|
);
|
|
}
|
|
}
|
|
|
|
// Get the provider list
|
|
tapiProviderList.dwTotalSize = sizeof(LINEPROVIDERLIST);
|
|
hr = MMCGetProviderList( m_hMmc, &tapiProviderList);
|
|
if (FAILED(hr))
|
|
{
|
|
goto ExitHere;
|
|
}
|
|
|
|
m_pProviderList = (LPLINEPROVIDERLIST) new BYTE[tapiProviderList.dwNeededSize];
|
|
if (NULL == m_pProviderList)
|
|
{
|
|
hr = TSECERR_NOMEM;
|
|
goto ExitHere;
|
|
}
|
|
|
|
memset(m_pProviderList, 0, tapiProviderList.dwNeededSize);
|
|
m_pProviderList->dwTotalSize = tapiProviderList.dwNeededSize;
|
|
|
|
hr = MMCGetProviderList( m_hMmc, m_pProviderList);
|
|
if (FAILED(hr) || !m_pProviderList->dwNumProviders)
|
|
{
|
|
goto ExitHere;
|
|
}
|
|
|
|
// Get the available providers
|
|
|
|
tapiAvailProvList.dwTotalSize = sizeof(LINEPROVIDERLIST);
|
|
hr = MMCGetAvailableProviders (m_hMmc, &tapiAvailProvList);
|
|
if (FAILED(hr))
|
|
{
|
|
goto ExitHere;
|
|
}
|
|
|
|
pAvailProvList = (LPAVAILABLEPROVIDERLIST) new BYTE[tapiAvailProvList.dwNeededSize];
|
|
if (NULL == pAvailProvList)
|
|
{
|
|
hr = TSECERR_NOMEM;
|
|
goto ExitHere;
|
|
}
|
|
|
|
memset(pAvailProvList, 0, tapiAvailProvList.dwNeededSize);
|
|
pAvailProvList->dwTotalSize = tapiAvailProvList.dwNeededSize;
|
|
|
|
hr = MMCGetAvailableProviders (m_hMmc, pAvailProvList);
|
|
if (FAILED(hr) || !pAvailProvList->dwNumProviderListEntries)
|
|
{
|
|
delete [] pAvailProvList;
|
|
goto ExitHere;
|
|
}
|
|
|
|
m_pProviderName = new LPTSTR[ m_pProviderList->dwNumProviders ];
|
|
if (NULL == m_pProviderName)
|
|
{
|
|
hr = TSECERR_NOMEM;
|
|
delete [] pAvailProvList;
|
|
goto ExitHere;
|
|
}
|
|
memset(m_pProviderName, 0, m_pProviderList->dwNumProviders * sizeof (LPTSTR) );
|
|
|
|
// find providers friendly name
|
|
LPTSTR szAvailProvFilename;
|
|
LPTSTR szProviderFilename;
|
|
LPTSTR szAvailProvFriendlyName;
|
|
|
|
pProvider = (LPLINEPROVIDERENTRY)
|
|
((LPBYTE) m_pProviderList + m_pProviderList->dwProviderListOffset);
|
|
|
|
for( dw1=0; dw1 < m_pProviderList->dwNumProviders; dw1++, pProvider++ )
|
|
{
|
|
szProviderFilename =
|
|
(LPTSTR) ((LPBYTE) m_pProviderList + pProvider->dwProviderFilenameOffset);
|
|
|
|
for ( dw2=0; dw2 < pAvailProvList->dwNumProviderListEntries; dw2++ )
|
|
{
|
|
pAvailProv = (LPAVAILABLEPROVIDERENTRY)
|
|
(( LPBYTE) pAvailProvList + pAvailProvList->dwProviderListOffset) + dw2;
|
|
szAvailProvFilename =
|
|
(LPTSTR) ((LPBYTE) pAvailProvList + pAvailProv->dwFileNameOffset);
|
|
if (_tcsicmp(szAvailProvFilename, szProviderFilename) == 0)
|
|
{
|
|
szAvailProvFriendlyName =
|
|
(LPTSTR) ((LPBYTE) pAvailProvList + pAvailProv->dwFriendlyNameOffset);
|
|
m_pProviderName[ dw1 ] = new TCHAR[ _tcslen( szAvailProvFriendlyName ) + 1 ];
|
|
if (m_pProviderName[ dw1 ])
|
|
{
|
|
_tcscpy( m_pProviderName[ dw1 ], szAvailProvFriendlyName );
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
delete [] pAvailProvList;
|
|
|
|
ExitHere:
|
|
if (FAILED(hr))
|
|
{
|
|
FreeMMCData ();
|
|
}
|
|
return hr;
|
|
}
|
|
|
|
HRESULT CMMCManagement::FreeMMCData ()
|
|
{
|
|
DWORD dw;
|
|
|
|
if (m_pUserTuple && m_pDeviceInfoList)
|
|
{
|
|
for (dw = 0;
|
|
dw < m_pDeviceInfoList->dwNumDeviceInfoEntries;
|
|
++dw)
|
|
{
|
|
if (m_pUserTuple[dw].pDomainUserNames)
|
|
{
|
|
delete [] m_pUserTuple[dw].pDomainUserNames;
|
|
}
|
|
if (m_pUserTuple[dw].pFriendlyUserNames)
|
|
{
|
|
delete [] m_pUserTuple[dw].pFriendlyUserNames;
|
|
}
|
|
}
|
|
delete [] m_pUserTuple;
|
|
m_pUserTuple = NULL;
|
|
}
|
|
if (m_pDeviceInfoList)
|
|
{
|
|
delete [] m_pDeviceInfoList;
|
|
m_pDeviceInfoList = NULL;
|
|
}
|
|
if (m_pProviderList)
|
|
{
|
|
delete [] m_pProviderList;
|
|
m_pProviderList = NULL;
|
|
}
|
|
if (m_pProviderName)
|
|
{
|
|
for ( DWORD dw=0; dw < sizeof( m_pProviderName ) / sizeof(LPTSTR); dw++ )
|
|
{
|
|
if (m_pProviderName[ dw ])
|
|
{
|
|
delete [] m_pProviderName[ dw ];
|
|
}
|
|
}
|
|
delete [] m_pProviderName;
|
|
}
|
|
|
|
if (m_bMarkedBusy && m_hMmc)
|
|
{
|
|
TAPISERVERCONFIG cfg;
|
|
|
|
cfg.dwTotalSize = sizeof(TAPISERVERCONFIG);
|
|
if (S_OK == MMCGetServerConfig (m_hMmc, &cfg))
|
|
{
|
|
cfg.dwFlags |= TAPISERVERCONFIGFLAGS_UNLOCKMMCWRITE;
|
|
MMCSetServerConfig (m_hMmc, &cfg);
|
|
}
|
|
m_bMarkedBusy = FALSE;
|
|
}
|
|
if (m_hMmc)
|
|
{
|
|
MMCShutdown (m_hMmc);
|
|
m_hMmc = NULL;
|
|
}
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
HRESULT CMMCManagement::RemoveLinesForUser (LPTSTR szDomainUser)
|
|
{
|
|
HRESULT hr = S_OK;
|
|
LPDWORD adwIndex = NULL;
|
|
DWORD dwNumEntries;
|
|
DWORD dw;
|
|
|
|
// Ensure we are properly initialized
|
|
if (m_hMmc == NULL)
|
|
{
|
|
goto ExitHere;
|
|
}
|
|
hr = FindEntriesForUser (szDomainUser, &adwIndex, &dwNumEntries);
|
|
if (hr)
|
|
{
|
|
goto ExitHere;
|
|
}
|
|
for (dw = 0; dw < dwNumEntries; ++dw)
|
|
{
|
|
hr = RemoveEntryForUser (
|
|
adwIndex[dw],
|
|
szDomainUser
|
|
);
|
|
if(FAILED(hr))
|
|
{
|
|
goto ExitHere;
|
|
}
|
|
}
|
|
|
|
ExitHere:
|
|
if (adwIndex)
|
|
{
|
|
delete [] adwIndex;
|
|
}
|
|
return hr;
|
|
}
|
|
|
|
HRESULT CMMCManagement::IsValidPID (
|
|
DWORD dwPermanentID
|
|
)
|
|
{
|
|
DWORD dwEntry;
|
|
|
|
return FindEntryFromPID (dwPermanentID, &dwEntry);
|
|
}
|
|
|
|
HRESULT CMMCManagement::IsValidAddress (
|
|
LPTSTR szAddr
|
|
)
|
|
{
|
|
DWORD dwEntry;
|
|
|
|
return FindEntryFromAddr (szAddr, &dwEntry);
|
|
}
|
|
|
|
HRESULT CMMCManagement::AddLinePIDForUser (
|
|
DWORD dwPermanentID,
|
|
LPTSTR szDomainUser,
|
|
LPTSTR szFriendlyName
|
|
)
|
|
{
|
|
HRESULT hr = S_OK;
|
|
DWORD dwEntry;
|
|
|
|
// Ensure we are properly initialized
|
|
if (m_hMmc == NULL)
|
|
{
|
|
goto ExitHere;
|
|
}
|
|
hr = FindEntryFromPID (dwPermanentID, &dwEntry);
|
|
if (hr)
|
|
{
|
|
goto ExitHere;
|
|
}
|
|
hr = AddEntryForUser (dwEntry, szDomainUser, szFriendlyName);
|
|
|
|
ExitHere:
|
|
return hr;
|
|
}
|
|
|
|
HRESULT CMMCManagement::AddLineAddrForUser (
|
|
LPTSTR szAddr,
|
|
LPTSTR szDomainUser,
|
|
LPTSTR szFriendlyName
|
|
)
|
|
{
|
|
HRESULT hr = S_OK;
|
|
DWORD dwEntry;
|
|
|
|
// Ensure we are properly initialized
|
|
if (m_hMmc == NULL)
|
|
{
|
|
goto ExitHere;
|
|
}
|
|
hr = FindEntryFromAddr (szAddr, &dwEntry);
|
|
if (hr)
|
|
{
|
|
goto ExitHere;
|
|
}
|
|
hr = AddEntryForUser (dwEntry, szDomainUser, szFriendlyName);
|
|
|
|
ExitHere:
|
|
return hr;
|
|
}
|
|
|
|
HRESULT CMMCManagement::RemoveLinePIDForUser (
|
|
DWORD dwPermanentID,
|
|
LPTSTR szDomainUser
|
|
)
|
|
{
|
|
HRESULT hr = S_OK;
|
|
DWORD dwEntry;
|
|
|
|
// Ensure we are properly initialized
|
|
if (m_hMmc == NULL)
|
|
{
|
|
goto ExitHere;
|
|
}
|
|
hr = FindEntryFromPID (dwPermanentID, &dwEntry);
|
|
if (hr)
|
|
{
|
|
goto ExitHere;
|
|
}
|
|
hr = RemoveEntryForUser (dwEntry, szDomainUser);
|
|
|
|
ExitHere:
|
|
return hr;
|
|
}
|
|
|
|
HRESULT CMMCManagement::RemoveLineAddrForUser (
|
|
LPTSTR szAddr,
|
|
LPTSTR szDomainUser
|
|
)
|
|
{
|
|
HRESULT hr = S_OK;
|
|
DWORD dwEntry;
|
|
|
|
// Ensure we are properly initialized
|
|
if (m_hMmc == NULL)
|
|
{
|
|
goto ExitHere;
|
|
}
|
|
hr = FindEntryFromAddr (szAddr, &dwEntry);
|
|
if (hr)
|
|
{
|
|
goto ExitHere;
|
|
}
|
|
hr = RemoveEntryForUser (dwEntry, szDomainUser);
|
|
|
|
ExitHere:
|
|
return hr;
|
|
}
|
|
|
|
BOOL CMMCManagement::IsDeviceLocalOnly (DWORD dwProviderID, DWORD dwDeviceID)
|
|
{
|
|
HRESULT hr;
|
|
DWORD dwFlags, dwDev;
|
|
|
|
if (m_pFuncGetDeviceFlags == NULL)
|
|
{
|
|
return TRUE;
|
|
}
|
|
|
|
hr = (*m_pFuncGetDeviceFlags)(
|
|
m_hMmc,
|
|
TRUE,
|
|
dwProviderID,
|
|
dwDeviceID,
|
|
&dwFlags,
|
|
&dwDev
|
|
);
|
|
if (hr || dwFlags & LINEDEVCAPFLAGS_LOCAL)
|
|
{
|
|
return TRUE;
|
|
}
|
|
else
|
|
{
|
|
return FALSE;
|
|
}
|
|
}
|
|
|
|
|
|
HRESULT CMMCManagement::FindEntryFromAddr (LPTSTR szAddr, DWORD * pdwIndex)
|
|
{
|
|
HRESULT hr = S_FALSE;
|
|
DWORD dw;
|
|
LPDEVICEINFO pDevInfo;
|
|
LPTSTR szAddr2;
|
|
|
|
pDevInfo = (LPDEVICEINFO) (((LPBYTE)m_pDeviceInfoList) +
|
|
m_pDeviceInfoList->dwDeviceInfoOffset);
|
|
for (dw = 0;
|
|
dw < m_pDeviceInfoList->dwNumDeviceInfoEntries;
|
|
++dw, ++pDevInfo)
|
|
{
|
|
szAddr2 = (LPTSTR)(((LPBYTE)m_pDeviceInfoList) +
|
|
pDevInfo->dwAddressesOffset);
|
|
while (*szAddr2)
|
|
{
|
|
if (_tcsicmp (szAddr, szAddr2) == 0)
|
|
{
|
|
if (IsDeviceLocalOnly (
|
|
pDevInfo->dwProviderID,
|
|
pDevInfo->dwPermanentDeviceID
|
|
))
|
|
{
|
|
hr = TSECERR_DEVLOCALONLY;
|
|
}
|
|
else
|
|
{
|
|
hr = S_OK;
|
|
*pdwIndex = dw;
|
|
}
|
|
goto ExitHere;
|
|
}
|
|
szAddr2 += _tcslen (szAddr2) + 1;
|
|
}
|
|
}
|
|
|
|
ExitHere:
|
|
return hr;
|
|
}
|
|
|
|
HRESULT CMMCManagement::FindEntryFromPID (DWORD dwPID, DWORD * pdwIndex)
|
|
{
|
|
HRESULT hr = S_FALSE;
|
|
DWORD dw;
|
|
LPDEVICEINFO pDevInfo;
|
|
|
|
pDevInfo = (LPDEVICEINFO) (((LPBYTE)m_pDeviceInfoList) +
|
|
m_pDeviceInfoList->dwDeviceInfoOffset);
|
|
for (dw = 0;
|
|
dw < m_pDeviceInfoList->dwNumDeviceInfoEntries;
|
|
++dw, ++pDevInfo)
|
|
{
|
|
if (dwPID == pDevInfo->dwPermanentDeviceID)
|
|
{
|
|
if (IsDeviceLocalOnly (
|
|
pDevInfo->dwProviderID,
|
|
pDevInfo->dwPermanentDeviceID
|
|
))
|
|
{
|
|
hr = TSECERR_DEVLOCALONLY;
|
|
}
|
|
else
|
|
{
|
|
*pdwIndex = dw;
|
|
hr = S_OK;
|
|
}
|
|
goto ExitHere;
|
|
}
|
|
}
|
|
|
|
ExitHere:
|
|
return hr;
|
|
}
|
|
|
|
HRESULT CMMCManagement::FindEntriesForUser (
|
|
LPTSTR szDomainUser,
|
|
LPDWORD * padwIndex,
|
|
DWORD * pdwNumEntries
|
|
)
|
|
{
|
|
HRESULT hr = S_OK;
|
|
DWORD dw;
|
|
LPTSTR szUsers;
|
|
LPDWORD adw;
|
|
|
|
*padwIndex = NULL;
|
|
*pdwNumEntries = 0;
|
|
for (dw = 0;
|
|
dw < m_pDeviceInfoList->dwNumDeviceInfoEntries;
|
|
++dw)
|
|
{
|
|
szUsers = m_pUserTuple[dw].pDomainUserNames;
|
|
while (szUsers && *szUsers)
|
|
{
|
|
if (_tcsicmp (szDomainUser, szUsers) == 0)
|
|
{
|
|
hr = S_OK;
|
|
adw = new DWORD[*pdwNumEntries + 1];
|
|
if (adw == NULL)
|
|
{
|
|
hr = TSECERR_NOMEM;
|
|
goto ExitHere;
|
|
}
|
|
if (*pdwNumEntries > 0)
|
|
{
|
|
memcpy (adw, *padwIndex, sizeof(DWORD) * (*pdwNumEntries));
|
|
}
|
|
if (*padwIndex)
|
|
{
|
|
delete [] (*padwIndex);
|
|
}
|
|
*padwIndex = adw;
|
|
adw[*pdwNumEntries] = dw;
|
|
*pdwNumEntries += 1;
|
|
}
|
|
szUsers += _tcslen (szUsers) + 1;
|
|
}
|
|
}
|
|
|
|
ExitHere:
|
|
if (FAILED(hr))
|
|
{
|
|
*pdwNumEntries = 0;
|
|
if (*padwIndex)
|
|
{
|
|
delete [] (*padwIndex);
|
|
}
|
|
}
|
|
else if (*pdwNumEntries == 0)
|
|
{
|
|
hr = S_FALSE;
|
|
}
|
|
return hr;
|
|
}
|
|
|
|
HRESULT CMMCManagement::AddEntryForUser (
|
|
DWORD dwIndex,
|
|
LPTSTR szDomainUser,
|
|
LPTSTR szFriendlyName
|
|
)
|
|
{
|
|
HRESULT hr = S_OK;
|
|
DWORD dwSize, dw;
|
|
LPTSTR szUsers, szNewUsers = NULL, szNewFriendlyNames = NULL;
|
|
|
|
if (dwIndex >= m_pDeviceInfoList->dwNumDeviceInfoEntries ||
|
|
szDomainUser[0] == 0 ||
|
|
szFriendlyName[0] == 0)
|
|
{
|
|
hr = S_FALSE;
|
|
goto ExitHere;
|
|
}
|
|
|
|
//
|
|
// Add szDomainUser into the user tuple
|
|
//
|
|
|
|
// Computer the existing domain user size and make sure
|
|
// this user is not there already
|
|
dwSize = 0;
|
|
szUsers = m_pUserTuple[dwIndex].pDomainUserNames;
|
|
while (szUsers && *szUsers)
|
|
{
|
|
if (_tcsicmp (szDomainUser, szUsers) == 0)
|
|
{
|
|
goto ExitHere;
|
|
}
|
|
dw = _tcslen (szUsers) + 1;
|
|
dwSize += dw;
|
|
szUsers += dw;
|
|
}
|
|
|
|
// Extra space for double zero terminating
|
|
dw = _tcslen (szDomainUser);
|
|
szNewUsers = new TCHAR[dwSize + dw + 2];
|
|
if (szNewUsers == NULL)
|
|
{
|
|
hr = TSECERR_NOMEM;
|
|
goto ExitHere;
|
|
}
|
|
|
|
// copy over the old domain users
|
|
if (dwSize > 0)
|
|
{
|
|
memcpy (
|
|
szNewUsers,
|
|
m_pUserTuple[dwIndex].pDomainUserNames,
|
|
dwSize * sizeof(TCHAR)
|
|
);
|
|
}
|
|
|
|
// Append the new domain user
|
|
memcpy (
|
|
szNewUsers + dwSize,
|
|
szDomainUser,
|
|
(dw + 1) * sizeof(TCHAR)
|
|
);
|
|
|
|
// double zero terminate and assign the data
|
|
szNewUsers[dwSize + dw + 1] = 0;
|
|
|
|
//
|
|
// Add the szFriendlyName into the user tuple
|
|
//
|
|
|
|
// Compute the existing friendly names size
|
|
dwSize = 0;
|
|
szUsers = m_pUserTuple[dwIndex].pFriendlyUserNames;
|
|
while (szUsers && *szUsers)
|
|
{
|
|
dw = _tcslen (szUsers) + 1;
|
|
dwSize += dw;
|
|
szUsers += dw;
|
|
}
|
|
|
|
// Extra space for double zero terminating
|
|
dw = _tcslen (szFriendlyName);
|
|
szNewFriendlyNames = new TCHAR[dwSize + dw + 2];
|
|
if (szNewFriendlyNames == NULL)
|
|
{
|
|
hr = TSECERR_NOMEM;
|
|
goto ExitHere;
|
|
}
|
|
|
|
// Copy over the old friendly names
|
|
if (dwSize > 0)
|
|
{
|
|
memcpy (
|
|
szNewFriendlyNames,
|
|
m_pUserTuple[dwIndex].pFriendlyUserNames,
|
|
dwSize * sizeof(TCHAR)
|
|
);
|
|
}
|
|
|
|
// Append the new friendly name
|
|
memcpy (
|
|
szNewFriendlyNames + dwSize,
|
|
szFriendlyName,
|
|
(dw + 1) * sizeof(TCHAR)
|
|
);
|
|
|
|
// Double zero terminate the friend names
|
|
szNewFriendlyNames[dwSize + dw + 1] = 0;
|
|
|
|
//
|
|
// Everything is fine, set the new data in
|
|
//
|
|
if (m_pUserTuple[dwIndex].pDomainUserNames)
|
|
{
|
|
delete [] m_pUserTuple[dwIndex].pDomainUserNames;
|
|
}
|
|
m_pUserTuple[dwIndex].pDomainUserNames = szNewUsers;
|
|
if (m_pUserTuple[dwIndex].pFriendlyUserNames)
|
|
{
|
|
delete [] m_pUserTuple[dwIndex].pFriendlyUserNames;
|
|
}
|
|
m_pUserTuple[dwIndex].pFriendlyUserNames = szNewFriendlyNames;
|
|
|
|
// Call WriteMMCEntry
|
|
hr = WriteMMCEntry (dwIndex);
|
|
|
|
ExitHere:
|
|
return hr;
|
|
}
|
|
|
|
HRESULT CMMCManagement::RemoveEntryForUser (
|
|
DWORD dwIndex,
|
|
LPTSTR szDomainUser
|
|
)
|
|
{
|
|
HRESULT hr = S_OK;
|
|
LPTSTR szLoc, szUsers;
|
|
DWORD dwLoc, dw, dwSize;
|
|
BOOL bFound;
|
|
|
|
if (dwIndex >= m_pDeviceInfoList->dwNumDeviceInfoEntries ||
|
|
szDomainUser[0] == 0)
|
|
{
|
|
hr = S_FALSE;
|
|
goto ExitHere;
|
|
}
|
|
|
|
// Locate the domain user and its index in the array
|
|
// of domain users
|
|
szUsers = m_pUserTuple[dwIndex].pDomainUserNames;
|
|
dwLoc = 0;
|
|
dwSize = 0;
|
|
bFound = FALSE;
|
|
while (szUsers && *szUsers)
|
|
{
|
|
dw = _tcslen (szUsers) + 1;
|
|
if (bFound)
|
|
{
|
|
dwSize += dw;
|
|
}
|
|
else
|
|
{
|
|
if (_tcsicmp (szDomainUser, szUsers) == 0)
|
|
{
|
|
bFound = TRUE;
|
|
szLoc = szUsers;
|
|
}
|
|
else
|
|
{
|
|
++dwLoc;
|
|
}
|
|
}
|
|
szUsers += dw;
|
|
}
|
|
if (!bFound)
|
|
{
|
|
goto ExitHere;
|
|
}
|
|
|
|
// Move down the pszDomainUserNames
|
|
if (dwSize > 0)
|
|
{
|
|
dw = _tcslen (szDomainUser);
|
|
// Memory copy includes the double zero terminator
|
|
memmove (szLoc, szLoc + dw + 1, (dwSize + 1) * sizeof(TCHAR));
|
|
}
|
|
else
|
|
{
|
|
// The is the last item, simple double zero terminate
|
|
*szLoc = 0;
|
|
}
|
|
|
|
// Now find corresponding friendly name based on dwLoc
|
|
szUsers = m_pUserTuple[dwIndex].pFriendlyUserNames;
|
|
while (szUsers && *szUsers && dwLoc > 0)
|
|
{
|
|
--dwLoc;
|
|
szUsers += _tcslen (szUsers) + 1;
|
|
}
|
|
// bail if not exist, otherwise, remember the location
|
|
if (szUsers == NULL || *szUsers == 0)
|
|
{
|
|
goto ExitHere;
|
|
}
|
|
szLoc = szUsers;
|
|
// Go to the next item
|
|
szUsers += _tcslen (szUsers) + 1;
|
|
// This is the last item
|
|
if (*szUsers == 0)
|
|
{
|
|
*szLoc = 0;
|
|
}
|
|
// Otherwise compute the remaining size to move
|
|
else
|
|
{
|
|
dwSize = 0;
|
|
while (*szUsers)
|
|
{
|
|
dw = _tcslen (szUsers) + 1;
|
|
dwSize += dw;
|
|
szUsers += dw;
|
|
}
|
|
// Compensate for the double zero terminating
|
|
dwSize++;
|
|
// Do the memory move
|
|
memmove (
|
|
szLoc,
|
|
szLoc + _tcslen (szLoc) + 1,
|
|
dwSize * sizeof(TCHAR)
|
|
);
|
|
}
|
|
|
|
// Call WriteMMCEntry
|
|
hr = WriteMMCEntry (dwIndex);
|
|
|
|
ExitHere:
|
|
return hr;
|
|
}
|
|
|
|
HRESULT CMMCManagement::WriteMMCEntry (DWORD dwIndex)
|
|
{
|
|
HRESULT hr = S_OK;
|
|
DWORD dwSize, dwSizeDU, dwSizeFU;
|
|
LPUSERNAME_TUPLE pUserTuple;
|
|
LPTSTR szUsers;
|
|
DWORD dw;
|
|
LPDEVICEINFOLIST pDevList = NULL;
|
|
LPDEVICEINFO pDevInfo, pDevInfoOld;
|
|
LPBYTE pDest;
|
|
|
|
if (dwIndex >= m_pDeviceInfoList->dwNumDeviceInfoEntries)
|
|
{
|
|
hr = S_FALSE;
|
|
goto ExitHere;
|
|
}
|
|
pUserTuple = m_pUserTuple + dwIndex;
|
|
|
|
// Computer domain user name size
|
|
dwSizeDU = 0;
|
|
if (pUserTuple->pDomainUserNames != NULL &&
|
|
*pUserTuple->pDomainUserNames != 0)
|
|
{
|
|
szUsers = pUserTuple->pDomainUserNames;
|
|
while (*szUsers)
|
|
{
|
|
dw = _tcslen (szUsers) + 1;
|
|
szUsers += dw;
|
|
dwSizeDU += dw;
|
|
}
|
|
dwSizeDU++; // double zero terminator
|
|
}
|
|
|
|
// Computer the friendly user name size
|
|
dwSizeFU = 0;
|
|
if (pUserTuple->pFriendlyUserNames != NULL &&
|
|
*pUserTuple->pFriendlyUserNames != 0)
|
|
{
|
|
szUsers = pUserTuple->pFriendlyUserNames;
|
|
while (*szUsers)
|
|
{
|
|
dw = _tcslen (szUsers) + 1;
|
|
szUsers += dw;
|
|
dwSizeFU += dw;
|
|
}
|
|
dwSizeFU++; // double zero terminator
|
|
}
|
|
|
|
// Computer the total size
|
|
dwSize = sizeof(DEVICEINFOLIST) + sizeof(DEVICEINFO) +
|
|
(dwSizeDU + dwSizeFU) * sizeof(TCHAR);
|
|
|
|
// Allocate the structure
|
|
pDevList = (LPDEVICEINFOLIST) new BYTE[dwSize];
|
|
if (pDevList == NULL)
|
|
{
|
|
hr = TSECERR_NOMEM;
|
|
goto ExitHere;
|
|
}
|
|
|
|
// Set the data member of DEVICEINFOLIST
|
|
pDevList->dwTotalSize = dwSize;
|
|
pDevList->dwNeededSize = dwSize;
|
|
pDevList->dwUsedSize = dwSize;
|
|
pDevList->dwNumDeviceInfoEntries = 1;
|
|
pDevList->dwDeviceInfoSize = sizeof(DEVICEINFO);
|
|
pDevList->dwDeviceInfoOffset = sizeof(DEVICEINFOLIST);
|
|
|
|
// Set the member of DEVICEINFO
|
|
pDevInfo = (LPDEVICEINFO)(((LPBYTE)pDevList) +
|
|
pDevList->dwDeviceInfoOffset);
|
|
pDevInfoOld = (LPDEVICEINFO)(((LPBYTE)m_pDeviceInfoList) +
|
|
m_pDeviceInfoList->dwDeviceInfoOffset) + dwIndex;
|
|
memset (pDevInfo, 0, sizeof(DEVICEINFO));
|
|
|
|
pDevInfo->dwPermanentDeviceID = pDevInfoOld->dwPermanentDeviceID;
|
|
pDevInfo->dwProviderID = pDevInfoOld->dwProviderID;
|
|
|
|
if (dwSizeDU > 0)
|
|
{
|
|
pDevInfo->dwDomainUserNamesSize = dwSizeDU * sizeof(TCHAR);
|
|
pDevInfo->dwDomainUserNamesOffset =
|
|
sizeof(DEVICEINFOLIST) + sizeof(DEVICEINFO);
|
|
pDest = ((LPBYTE)pDevList) + pDevInfo->dwDomainUserNamesOffset;
|
|
memcpy (
|
|
pDest,
|
|
pUserTuple->pDomainUserNames,
|
|
dwSizeDU * sizeof(TCHAR)
|
|
);
|
|
}
|
|
|
|
if (dwSizeFU)
|
|
{
|
|
pDevInfo->dwFriendlyUserNamesSize = dwSizeFU * sizeof(TCHAR);
|
|
pDevInfo->dwFriendlyUserNamesOffset =
|
|
pDevInfo->dwDomainUserNamesOffset + dwSizeDU * sizeof(TCHAR);
|
|
pDest = ((LPBYTE)pDevList) + pDevInfo->dwFriendlyUserNamesOffset;
|
|
memcpy (
|
|
pDest,
|
|
pUserTuple->pFriendlyUserNames,
|
|
dwSizeFU * sizeof(TCHAR)
|
|
);
|
|
}
|
|
|
|
hr = MMCSetLineInfo (
|
|
m_hMmc,
|
|
pDevList
|
|
);
|
|
|
|
ExitHere:
|
|
if (pDevList)
|
|
{
|
|
delete [] pDevList;
|
|
}
|
|
return hr;
|
|
}
|
|
|
|
HRESULT CMMCManagement::DisplayMMCData ()
|
|
{
|
|
HRESULT hr = S_OK;
|
|
LPDEVICEINFO pDeviceInfo;
|
|
DWORD * pdwIndex = NULL;
|
|
DWORD dwProviderId;
|
|
DWORD dwAddressCount,dw1,dw2;
|
|
LPLINEPROVIDERENTRY pProvider;
|
|
LPTSTR pUserName;
|
|
LPTSTR pAddress,
|
|
pAddressFirst,
|
|
pAddressLast;
|
|
|
|
if ( !m_pDeviceInfoList || !m_pDeviceInfoList->dwNumDeviceInfoEntries )
|
|
{
|
|
CIds IdsNoDevices (IDS_NODEVICES);
|
|
_tprintf ( IdsNoDevices.GetString () );
|
|
return hr;
|
|
}
|
|
|
|
//
|
|
// Build an index by provider ID
|
|
//
|
|
pdwIndex = new DWORD [ m_pDeviceInfoList->dwNumDeviceInfoEntries ];
|
|
if ( !pdwIndex )
|
|
{
|
|
hr = TSECERR_NOMEM;
|
|
return hr;
|
|
}
|
|
|
|
for ( dw1=0; dw1 < m_pDeviceInfoList->dwNumDeviceInfoEntries; dw1++ )
|
|
{
|
|
pdwIndex[ dw1 ] = dw1;
|
|
}
|
|
|
|
dw1 = 0;
|
|
while ( dw1 < m_pDeviceInfoList->dwNumDeviceInfoEntries )
|
|
{
|
|
|
|
pDeviceInfo = (LPDEVICEINFO)((LPBYTE)m_pDeviceInfoList +
|
|
m_pDeviceInfoList->dwDeviceInfoOffset) + pdwIndex[ dw1 ];
|
|
dwProviderId = pDeviceInfo->dwProviderID;
|
|
dw1++;
|
|
|
|
for ( dw2=dw1; dw2 < m_pDeviceInfoList->dwNumDeviceInfoEntries; dw2++ )
|
|
{
|
|
pDeviceInfo = (LPDEVICEINFO)((LPBYTE)m_pDeviceInfoList +
|
|
m_pDeviceInfoList->dwDeviceInfoOffset) + pdwIndex[ dw2 ];
|
|
|
|
if ( dwProviderId == pDeviceInfo->dwProviderID )
|
|
{
|
|
DWORD dwTemp = pdwIndex[ dw2 ];
|
|
pdwIndex[ dw2 ] = pdwIndex[ dw1 ];
|
|
pdwIndex[ dw1 ] = dwTemp;
|
|
dw1++;
|
|
}
|
|
}
|
|
}
|
|
|
|
//
|
|
// Display the device list
|
|
//
|
|
dw1 = 0;
|
|
while ( dw1 < m_pDeviceInfoList->dwNumDeviceInfoEntries )
|
|
{
|
|
pDeviceInfo = (LPDEVICEINFO)((LPBYTE)m_pDeviceInfoList +
|
|
m_pDeviceInfoList->dwDeviceInfoOffset) + pdwIndex[ dw1 ];
|
|
dwProviderId = pDeviceInfo->dwProviderID;
|
|
|
|
// find the provider entry
|
|
pProvider = (LPLINEPROVIDERENTRY) ((LPBYTE) m_pProviderList + m_pProviderList->dwProviderListOffset);
|
|
for( dw2=0; dw2 < m_pProviderList->dwNumProviders; dw2++, pProvider++ )
|
|
{
|
|
if ( dwProviderId == pProvider->dwPermanentProviderID )
|
|
{
|
|
break;
|
|
}
|
|
}
|
|
|
|
// display the provider name
|
|
if ( dw2 < m_pProviderList->dwNumProviders )
|
|
{
|
|
// provider entry found
|
|
_tprintf(
|
|
_T("\n%s\n"),
|
|
m_pProviderName[ dw2 ] ? m_pProviderName[ dw2 ] :
|
|
(LPTSTR)((LPBYTE) m_pProviderList + pProvider->dwProviderFilenameOffset)
|
|
);
|
|
}
|
|
else
|
|
{
|
|
CIds IdsProvider (IDS_PROVIDER);
|
|
_tprintf( IdsProvider.GetString(), dwProviderId );
|
|
}
|
|
|
|
// list devices / users for this provider
|
|
do
|
|
{
|
|
CIds IdsLine (IDS_LINE);
|
|
CIds IdsPermanentID (IDS_PID);
|
|
CIds IdsAddresses (IDS_ADDRESSES);
|
|
|
|
_tprintf( IdsLine.GetString(),
|
|
(LPTSTR)((LPBYTE)m_pDeviceInfoList + pDeviceInfo->dwDeviceNameOffset));
|
|
_tprintf( IdsPermanentID.GetString(), pDeviceInfo->dwPermanentDeviceID);
|
|
if ( pDeviceInfo->dwFriendlyUserNamesSize )
|
|
{
|
|
CIds IdsUsers (IDS_USERS);
|
|
_tprintf ( IdsUsers.GetString () );
|
|
|
|
pUserName = (LPTSTR) (((LPBYTE) m_pDeviceInfoList) +
|
|
pDeviceInfo->dwFriendlyUserNamesOffset);
|
|
while ( *pUserName != _T('\0') )
|
|
{
|
|
_tprintf( _T("\t\t\t%s\n"), pUserName );
|
|
pUserName += _tcslen (pUserName) + 1;
|
|
}
|
|
}
|
|
if ( pDeviceInfo->dwAddressesSize )
|
|
{
|
|
_tprintf( IdsAddresses.GetString() );
|
|
pAddress = (LPTSTR) (((LPBYTE) m_pDeviceInfoList) +
|
|
pDeviceInfo->dwAddressesOffset);
|
|
while ( *pAddress != _T('\0') )
|
|
{
|
|
_tprintf( _T("\t\t\t%s\n"), pAddress );
|
|
pAddress += _tcslen (pAddress) + 1;
|
|
}
|
|
}
|
|
|
|
dw1++;
|
|
pDeviceInfo++;
|
|
}
|
|
while ( dw1 < m_pDeviceInfoList->dwNumDeviceInfoEntries &&
|
|
pDeviceInfo->dwProviderID == dwProviderId );
|
|
}
|
|
return hr;
|
|
}
|
|
|
|
|