|
|
/*
File devicedb.c
Definition of the device database for the ras dialup server.
Paul Mayfield, 10/2/97 */
#include "rassrv.h"
#include "precomp.h"
//
// Definitions
//
#define DEV_FLAG_DEVICE 0x1
#define DEV_FLAG_NULL_MODEM 0x2
#define DEV_FLAG_PORT 0x4
#define DEV_FLAG_FILTERED 0x10
#define DEV_FLAG_DIRTY 0x20
#define DEV_FLAG_ENABLED 0x40
// ===================================
// Definitions of the database objects
// ===================================
typedef struct _RASSRV_DEVICE { PWCHAR pszName; // Name of device
DWORD dwType; // Type of device
DWORD dwId; // Id of the item (for tapi properties)
DWORD dwEndpoints; // Number of enpoints item has
DWORD dwFlags; // see DEV_FLAG_XXX
WCHAR pszPort[MAX_PORT_NAME + 1]; struct _RASSRV_DEVICE * pModem; // Modem installed on a com port
// only valid if DEV_FLAG_PORT set
} RASSRV_DEVICE;
typedef struct _RASSRV_DEVICEDB { DWORD dwDeviceCount; RASSRV_DEVICE ** pDeviceList; BOOL bFlushOnClose; BOOL bVpnEnabled; // whether pptp or l2tp is enabled
BOOL bVpnEnabledOrig; // original value of vpn enabling
} RASSRV_DEVICEDB;
//
// Structure defines a node in a list of ports
//
typedef struct _RASSRV_PORT_NODE { PWCHAR pszName; WCHAR pszPort[MAX_PORT_NAME + 1]; struct _RASSRV_DEVICE * pModem; // modem installed on this port
struct _RASSRV_PORT_NODE * pNext; } RASSRV_PORT_NODE;
//
// Structure defines a list of ports
//
typedef struct _RASSRV_PORT_LIST { DWORD dwCount; RASSRV_PORT_NODE * pHead; WCHAR pszFormat[256]; DWORD dwFmtLen; } RASSRV_PORT_LIST;
// Reads device information from the system.
DWORD devGetSystemDeviceInfo( OUT RAS_DEVICE_INFO ** ppDevice, OUT LPDWORD lpdwCount) { DWORD dwErr, dwSize, dwVer, dwCount; // Calculate the size required to enumerate the ras devices
dwVer = RAS_VERSION; dwSize = 0; dwCount =0; dwErr = RasGetDeviceConfigInfo(NULL, &dwVer, &dwCount, &dwSize, NULL); if ((dwErr != ERROR_SUCCESS) && (dwErr != ERROR_BUFFER_TOO_SMALL)) { DbgOutputTrace( "devGetSysDevInfo: 0x%08x from RasGetDevCfgInfo (1)", dwErr); return dwErr; } *lpdwCount = dwCount; // If there is nothing to allocate, return with zero devices
if (dwSize == 0) { *lpdwCount = 0; return NO_ERROR; } // Allocate the buffer
if ((*ppDevice = RassrvAlloc(dwSize, FALSE)) == NULL) { return ERROR_NOT_ENOUGH_MEMORY; }
// Enumerate the devices
dwErr = RasGetDeviceConfigInfo( NULL, &dwVer, lpdwCount, &dwSize, (LPBYTE)(*ppDevice)); if (dwErr != ERROR_SUCCESS) { DbgOutputTrace( "devGetSysDevInfo: 0x%08x from RasGetDevCfgInfo (2)", dwErr); return dwErr; }
return NO_ERROR; }
//
// Frees the buffer returned by devGetSystemDeviceInfo
//
DWORD devFreeSystemDeviceInfo( IN RAS_DEVICE_INFO * pDevice) { if (pDevice) { RassrvFree(pDevice); } return NO_ERROR; }
//
// Returns whether the given device is a physical (i.e. not tunnel) device.
//
BOOL devIsPhysicalDevice( IN RAS_DEVICE_INFO * pDevice) { DWORD dwClass = RAS_DEVICE_CLASS (pDevice->eDeviceType);
return ((dwClass & RDT_Direct) || (dwClass & RDT_Null_Modem) || (dwClass == 0)); }
//
// Returns whether the given device is a tunneling device
//
BOOL devIsTunnelDevice( IN RAS_DEVICE_INFO * pDevice) { DWORD dwClass = RAS_DEVICE_CLASS (pDevice->eDeviceType);
return (dwClass & RDT_Tunnel); }
//
// Finds the device information associated with the given device
// based on its tapi line id
//
DWORD devFindDevice( IN RAS_DEVICE_INFO * pDevices, IN DWORD dwCount, OUT RAS_DEVICE_INFO ** ppDevice, IN DWORD dwTapiLineId) { DWORD i;
// Validate parameters
if (!pDevices || !ppDevice) return ERROR_INVALID_PARAMETER;
// Initialize
*ppDevice = NULL;
// Search the list
for (i = 0; i < dwCount; i++) { if (devIsPhysicalDevice(&pDevices[i])) { if (pDevices[i].dwTapiLineId == dwTapiLineId) { *ppDevice = &(pDevices[i]); break; } } }
// See if a match was found;
if (*ppDevice) { return NO_ERROR; } return ERROR_NOT_FOUND; }
//
// Determine the type of the device in question
//
DWORD devDeviceType( IN RAS_DEVICE_INFO * pDevice) { DWORD dwClass = RAS_DEVICE_CLASS (pDevice->eDeviceType); DWORD dwType = RAS_DEVICE_TYPE (pDevice->eDeviceType);
if ((dwClass & RDT_Direct) || (dwClass & RDT_Null_Modem) || (dwType == RDT_Irda) || (dwType == RDT_Parallel) ) { return INCOMING_TYPE_DIRECT; } else if (dwClass == RDT_Tunnel) { return INCOMING_TYPE_VPN; }
return INCOMING_TYPE_PHONE; }
//
// Returns the flags of a given device
//
DWORD devInitFlags( IN RAS_DEVICE_INFO * pDevice) { DWORD dwClass = RAS_DEVICE_CLASS (pDevice->eDeviceType), dwRet = 0;
// Set the device's enabling
if (pDevice->fRasEnabled) { dwRet |= DEV_FLAG_ENABLED; }
// Determine if it's a null modem or a device. It
// can't be a port, since those are not reported
// through ras.
if (dwClass & RDT_Null_Modem) { dwRet |= DEV_FLAG_NULL_MODEM; } else { dwRet |= DEV_FLAG_DEVICE; }
// Since the filtered and dirty flags are to be
// initialized to false, we're done.
return dwRet; }
//
// Generates the device name
//
PWCHAR devCopyDeviceName( IN RAS_DEVICE_INFO * pDevice, IN DWORD dwType) { PWCHAR pszReturn; DWORD dwSize; PWCHAR pszDccFmt = (PWCHAR) PszLoadString(Globals.hInstDll, SID_DEVICE_DccDeviceFormat); PWCHAR pszMultiFmt = (PWCHAR) PszLoadString(Globals.hInstDll, SID_DEVICE_MultiEndpointDeviceFormat); WCHAR pszPort[MAX_PORT_NAME + 1]; WCHAR pszDevice[MAX_DEVICE_NAME + 1];
// Sanity check the resources
//
if (!pszDccFmt || !pszMultiFmt) { return NULL; }
// Get the unicode versions of the port/device
//
pszPort[0] = L'\0'; pszDevice[0] = L'\0'; if (pDevice->szPortName) { StrCpyWFromAUsingAnsiEncoding( pszPort, pDevice->szPortName, strlen(pDevice->szPortName) + 1); } if (pDevice->szDeviceName) { #if 0
StrCpyWFromAUsingAnsiEncoding( pszDevice, pDevice->szDeviceName, strlen(pDevice->szDeviceName) + 1); #endif
wcsncpy( pszDevice, pDevice->wszDeviceName, MAX_DEVICE_NAME);
pszDevice[MAX_DEVICE_NAME] = L'\0'; }
// For direct connections, be sure to display the name of the com port
// in addition to the name of the null modem.
if (dwType == INCOMING_TYPE_DIRECT) { // Calculate the size
dwSize = wcslen(pszDevice) * sizeof(WCHAR) + // device
wcslen(pszPort) * sizeof(WCHAR) + // com port
wcslen(pszDccFmt) * sizeof(WCHAR) + // oth chars
sizeof(WCHAR); // null
// Allocate the new string
if ((pszReturn = RassrvAlloc (dwSize, FALSE)) == NULL) { return pszReturn; }
wsprintfW(pszReturn, pszDccFmt, pszDevice, pszPort); }
//
// If it's a modem device with multilple end points (such as isdn)
// display the endpoints in parenthesis
//
else if ((dwType == INCOMING_TYPE_PHONE) && (pDevice->dwNumEndPoints > 1)) { // Calculate the size
dwSize = wcslen(pszDevice) * sizeof(WCHAR) + // device
wcslen(pszMultiFmt) * sizeof(WCHAR) + // channels
20 * sizeof (WCHAR) + // oth chars
sizeof(WCHAR); // null
// Allocate the new string
if ((pszReturn = RassrvAlloc(dwSize, FALSE)) == NULL) { return pszReturn; }
wsprintfW( pszReturn, pszMultiFmt, pszDevice, pDevice->dwNumEndPoints); }
// Otherwise, this is a modem device with one endpoint.
// All that needs to be done here is to copy in the name.
//
else { // Calculate the size
dwSize = (wcslen(pszDevice) + 1) * sizeof(WCHAR); // Allocate the new string
if ((pszReturn = RassrvAlloc(dwSize, FALSE)) == NULL) { return pszReturn; }
wcscpy(pszReturn, pszDevice); }
return pszReturn; }
//
// Commits changes to device configuration to the system. The call
// to RasSetDeviceConfigInfo might fail if the device is in the process
// of being configured so we implement a retry
// mechanism here.
//
DWORD devCommitDeviceInfo( IN RAS_DEVICE_INFO * pDevice) { DWORD dwErr, dwTimeout = 10; do { // Try to commit the information
dwErr = RasSetDeviceConfigInfo( NULL, 1, sizeof(RAS_DEVICE_INFO), (LPBYTE)pDevice);
// If unable to complete, wait and try again
if (dwErr == ERROR_CAN_NOT_COMPLETE) { DbgOutputTrace( "devCommDevInfo: 0x%08x from RasSetDevCfgInfo (try again)", dwErr); Sleep(300); dwTimeout--; }
// If completed, return no error
else if (dwErr == NO_ERROR) { break; }
// Otherwise, return the error code.
else { DbgOutputTrace( "devCommDevInfo: can't commit %S, 0x%08x", pDevice->szDeviceName, dwErr); break; } } while (dwTimeout);
return dwErr; }
//
// Opens a handle to the database of general tab values
//
DWORD devOpenDatabase( IN HANDLE * hDevDatabase) { RASSRV_DEVICEDB * This; DWORD dwErr, i; if (!hDevDatabase) { return ERROR_INVALID_PARAMETER; }
// Allocate the database cache
if ((This = RassrvAlloc(sizeof(RASSRV_DEVICEDB), TRUE)) == NULL) { DbgOutputTrace("devOpenDatabase: can't allocate memory -- exiting"); return ERROR_NOT_ENOUGH_MEMORY; } // Initialize the values from the system
devReloadDatabase((HANDLE)This);
// Return the handle
*hDevDatabase = (HANDLE)This; This->bFlushOnClose = FALSE;
return NO_ERROR; }
//
// Closes the general database and flushes any changes
// to the system when bFlushOnClose is TRUE
//
DWORD devCloseDatabase( IN HANDLE hDevDatabase) { RASSRV_DEVICEDB * This = (RASSRV_DEVICEDB*)hDevDatabase; DWORD i; // Flush if requested
if (This->bFlushOnClose) devFlushDatabase(hDevDatabase); // Free all of the device names
for (i = 0; i < This->dwDeviceCount; i++) { if (This->pDeviceList[i]) { if (This->pDeviceList[i]->pszName) { RassrvFree(This->pDeviceList[i]->pszName); } RassrvFree(This->pDeviceList[i]); } }
// Free up the device list cache
if (This->pDeviceList) { RassrvFree (This->pDeviceList); }
// Free up the database cache
RassrvFree(This);
return NO_ERROR; }
//
// Commits the changes made to a particular device
//
DWORD devCommitDevice ( IN RASSRV_DEVICE * pDevice, IN RAS_DEVICE_INFO * pDevices, IN DWORD dwCount) { RAS_DEVICE_INFO *pDevInfo = NULL; devFindDevice(pDevices, dwCount, &pDevInfo, pDevice->dwId); if (pDevInfo) { pDevInfo->fWrite = TRUE; pDevInfo->fRasEnabled = !!(pDevice->dwFlags & DEV_FLAG_ENABLED); devCommitDeviceInfo(pDevInfo); }
// Mark the device as not dirty
pDevice->dwFlags &= ~DEV_FLAG_DIRTY;
return NO_ERROR; }
BOOL devIsVpnEnableChanged( IN HANDLE hDevDatabase) { RASSRV_DEVICEDB * pDevDb = (RASSRV_DEVICEDB*)hDevDatabase;
if ( pDevDb ) { return ( pDevDb->bVpnEnabled != pDevDb->bVpnEnabledOrig? TRUE:FALSE ); }
return FALSE;
}//devIsVpnEnableChanged()
//
// Commits any changes made to the general tab values
//
DWORD devFlushDatabase( IN HANDLE hDevDatabase) { RASSRV_DEVICEDB * This = (RASSRV_DEVICEDB*)hDevDatabase; DWORD dwErr = NO_ERROR, i, dwCount, dwTimeout; RAS_DEVICE_INFO * pDevices = NULL; RASSRV_DEVICE * pCur = NULL;
// Validate
if (!This) { return ERROR_INVALID_PARAMETER; }
// Get all of the system device information
dwErr = devGetSystemDeviceInfo(&pDevices, &dwCount); if (dwErr != NO_ERROR) { return dwErr; }
// Flush all changed settings to the system
for (i = 0; i < This->dwDeviceCount; i++) { pCur = This->pDeviceList[i]; // If this device needs to be flushed
if (pCur->dwFlags & DEV_FLAG_DIRTY) { // Reset the installed device's enabling if it still
// exists in the system
if ((pCur->dwFlags & DEV_FLAG_DEVICE) || (pCur->dwFlags & DEV_FLAG_NULL_MODEM)) { devCommitDevice(pCur, pDevices, dwCount); }
// If this is a com port, then we should enable that modem
// installed on the port if it exists or install a null modem
// over it if not.
else if (pCur->dwFlags & DEV_FLAG_PORT) { // If this port is associated with an already installed
// null modem, then set the enabling on this modem if
// it is different
if (pCur->pModem != NULL) { if ((pCur->dwFlags & DEV_FLAG_ENABLED) != (pCur->pModem->dwFlags & DEV_FLAG_ENABLED)) { // For whistler bug 499405 gangz
// devCommitDevice just flush out the device info as it
// is, so we have to change pCur->pModem->dwFlags
//
if( pCur->dwFlags & DEV_FLAG_ENABLED ) { pCur->pModem->dwFlags |= DEV_FLAG_ENABLED; } else { pCur->pModem->dwFlags &= ~DEV_FLAG_ENABLED; } devCommitDevice ( pCur->pModem, pDevices, dwCount); } }
// Otherwise, (if there is no null modem associated with
// this port) install a null modem over this port if
// it is set to enabled.
else if (pCur->dwFlags & DEV_FLAG_ENABLED) { dwErr = MdmInstallNullModem (pCur->pszPort); } } } }
// Flush all of the changed vpn settings
if (This->bVpnEnabled != This->bVpnEnabledOrig) { for (i = 0; i < dwCount; i++) { if (devIsTunnelDevice(&pDevices[i])) { pDevices[i].fWrite = TRUE; pDevices[i].fRasEnabled = This->bVpnEnabled; devCommitDeviceInfo(&pDevices[i]); } } This->bVpnEnabledOrig = This->bVpnEnabled; }
// Cleanup
if (pDevices) { devFreeSystemDeviceInfo(pDevices); }
return dwErr; }
//
// Rollsback any changes made to the general tab values
//
DWORD devRollbackDatabase( IN HANDLE hDevDatabase) { RASSRV_DEVICEDB * This = (RASSRV_DEVICEDB*)hDevDatabase; if (This == NULL) { return ERROR_INVALID_PARAMETER; } This->bFlushOnClose = FALSE; return NO_ERROR; }
//
// Reloads the device database
//
DWORD devReloadDatabase( IN HANDLE hDevDatabase) { RASSRV_DEVICEDB * This = (RASSRV_DEVICEDB*)hDevDatabase; DWORD dwErr = NO_ERROR, i, j = 0, dwSize; RAS_DEVICE_INFO * pRasDevices; RASSRV_DEVICE * pTempList, *pDevice;
// Validate
if (!This) { return ERROR_INVALID_PARAMETER; }
// Initialize vpn status
This->bVpnEnabled = FALSE; // Get the device information from rasman
pRasDevices = NULL; dwErr = devGetSystemDeviceInfo(&pRasDevices, &This->dwDeviceCount); if (dwErr != NO_ERROR) { return dwErr; } do { // Initialize the incoming ras capable devices list
if (This->dwDeviceCount) { dwSize = sizeof(RASSRV_DEVICE*) * This->dwDeviceCount; This->pDeviceList = RassrvAlloc(dwSize, TRUE); if (!This->pDeviceList) { dwErr = ERROR_NOT_ENOUGH_MEMORY; break; }
// Build the device array accordingly
j = 0; for (i = 0; i < This->dwDeviceCount; i++) { // If it's a physical device, fill in the appropriate
// fields.
if (devIsPhysicalDevice(&pRasDevices[i])) { // Allocate the new device
pDevice = RassrvAlloc(sizeof(RASSRV_DEVICE), TRUE); if (pDevice == NULL) { continue; }
// Assign its values
pDevice->dwType = devDeviceType(&pRasDevices[i]); pDevice->dwId = pRasDevices[i].dwTapiLineId; pDevice->pszName = devCopyDeviceName( &pRasDevices[i], pDevice->dwType); pDevice->dwEndpoints = pRasDevices[i].dwNumEndPoints; pDevice->dwFlags = devInitFlags(&pRasDevices[i]); StrCpyWFromA( pDevice->pszPort, pRasDevices[i].szPortName, MAX_PORT_NAME + 1); This->pDeviceList[j] = pDevice; j++; }
// If any tunneling protocol is enabled, we consider all
// to be
else if (devIsTunnelDevice(&pRasDevices[i])) { This->bVpnEnabled |= pRasDevices[i].fRasEnabled; This->bVpnEnabledOrig = This->bVpnEnabled; } }
// Set the actual size of phyiscal adapters buffer.
This->dwDeviceCount = j; }
} while (FALSE); // Cleanup
{ devFreeSystemDeviceInfo(pRasDevices); }
return dwErr; }
//
// Filters out all devices in the database except those that
// meet the given type description (can be ||'d).
//
DWORD devFilterDevices( IN HANDLE hDevDatabase, DWORD dwType) { RASSRV_DEVICEDB * This = (RASSRV_DEVICEDB*)hDevDatabase; RASSRV_DEVICE * pDevice; DWORD i; if (!This) { return ERROR_INVALID_PARAMETER; }
// Go through the list of marking out devices to be filtered
for (i = 0; i < This->dwDeviceCount; i++) { pDevice = This->pDeviceList[i]; if (pDevice == NULL) { continue; } if (pDevice->dwType & dwType) { pDevice->dwFlags &= ~DEV_FLAG_FILTERED; } else { pDevice->dwFlags |= DEV_FLAG_FILTERED; } }
return NO_ERROR; }
//
// Device enumeration function. Returns TRUE to stop enumeration,
// FALSE to continue.
//
BOOL devAddPortToList ( IN PWCHAR pszPort, IN HANDLE hData) { RASSRV_PORT_LIST * pList = (RASSRV_PORT_LIST*)hData; RASSRV_PORT_NODE * pNode = NULL; DWORD dwSize;
// Create the new node
pNode = (RASSRV_PORT_NODE *) RassrvAlloc(sizeof(RASSRV_PORT_NODE), TRUE); if (pNode == NULL) { return FALSE; }
// Add it to the head
pNode->pNext = pList->pHead; pList->pHead = pNode; pList->dwCount++;
// Add the names of the port
if (pszPort) { dwSize = (wcslen(pszPort) + pList->dwFmtLen + 1) * sizeof(WCHAR); pNode->pszName = (PWCHAR) RassrvAlloc (dwSize, FALSE); if (pNode->pszName == NULL) { return TRUE; } wsprintfW (pNode->pszName, pList->pszFormat, pszPort); lstrcpynW(pNode->pszPort, pszPort, sizeof(pNode->pszPort) / sizeof(WCHAR)); }
return FALSE; }
//
// Cleans up the resources used in a device list
//
DWORD devCleanupPortList( IN RASSRV_PORT_LIST * pList) { RASSRV_PORT_NODE * pCur = NULL, * pNext = NULL;
pCur = pList->pHead; while (pCur) { pNext = pCur->pNext; RassrvFree(pCur); pCur = pNext; }
return NO_ERROR; }
//
// Removes all ports from the list for which there are already
// devices installed in the database.
//
DWORD devFilterPortsInUse ( IN RASSRV_DEVICEDB *This, RASSRV_PORT_LIST *pList) { RASSRV_PORT_LIST PortList, *pDelete = &PortList; RASSRV_PORT_NODE * pCur = NULL, * pPrev = NULL; RASSRV_DEVICE * pDevice; DWORD i; BOOL bDone; INT iCmp;
// If the list is empty, return
if (pList->dwCount == 0) { return NO_ERROR; }
// Initailize
ZeroMemory(pDelete, sizeof(RASSRV_PORT_LIST)); // Compare all of the enumerated ports to the ports
// in use in the device list.
for (i = 0; i < This->dwDeviceCount; i++) { // Point to the current device
pDevice = This->pDeviceList[i]; // Initialize the current and previous and break if the
// list is now empty
pCur = pList->pHead; if (pCur == NULL) { break; }
// Remove the head node until it doesn't match
bDone = FALSE; while ((pList->pHead != NULL) && (bDone == FALSE)) { iCmp = lstrcmpi (pDevice->pszPort, pList->pHead->pszPort); // If a device is already using this com port
// then remove the com port from the list since it
// isn't available
if ((pDevice->dwFlags & DEV_FLAG_DEVICE) && (iCmp == 0)) { pCur = pList->pHead->pNext; RassrvFree(pList->pHead); pList->pHead = pCur; pList->dwCount -= 1; } else { // If the device is a null modem, then we filter
// it out of the list of available devices and we
// reference it in the com port so that we can
// enable/disable it later if we need to.
if (iCmp == 0) { pDevice->dwFlags |= DEV_FLAG_FILTERED; pList->pHead->pModem = pDevice; } bDone = TRUE; } }
// If we've elimated everyone, return
if (pList->dwCount == 0) { return NO_ERROR; }
// Loop through all of the past the head removing those
// that are in use by the current ras device.
pPrev = pList->pHead; pCur = pPrev->pNext; while (pCur) { iCmp = lstrcmpi (pDevice->pszPort, pCur->pszPort); // If a device is already using this com port
// that remove the com port from the list since it
// isn't available
if ((pDevice->dwFlags & DEV_FLAG_DEVICE) && (iCmp == 0)) { pPrev->pNext = pCur->pNext; RassrvFree(pCur); pCur = pPrev->pNext; pList->dwCount -= 1; } else { // If the device is a null modem, then we filter
// it out of the list of available devices and we
// reference it in the com port so that we can
// enable/disable it later if we need to.
if (iCmp == 0) { pDevice->dwFlags |= DEV_FLAG_FILTERED; pCur->pModem = pDevice; } pCur = pCur->pNext; pPrev = pPrev->pNext; } } }
return NO_ERROR; }
//
// Adds com ports as uninstalled devices in the device database
//
DWORD devAddComPorts( IN HANDLE hDevDatabase) { RASSRV_DEVICEDB * This = (RASSRV_DEVICEDB*)hDevDatabase; RASSRV_PORT_LIST PortList, *pList = &PortList; RASSRV_PORT_NODE * pNode = NULL; RASSRV_DEVICE ** ppDevices; DWORD dwErr = NO_ERROR, i; if (!This) { return ERROR_INVALID_PARAMETER; }
// Initialize the port list
//
ZeroMemory (pList, sizeof(RASSRV_PORT_LIST)); pList->dwFmtLen = LoadStringW ( Globals.hInstDll, SID_COMPORT_FORMAT, pList->pszFormat, sizeof(pList->pszFormat) / sizeof(WCHAR));
do { // Create the list of com ports
dwErr = MdmEnumComPorts(devAddPortToList, (HANDLE)pList); if (dwErr != NO_ERROR) { break; }
// Remove any ports that are currently in use
if ((dwErr = devFilterPortsInUse (This, pList)) != NO_ERROR) { break; } // If there aren't any ports, return
if (pList->dwCount == 0) { break; }
// Resize the list of ports to include the com ports
ppDevices = RassrvAlloc( sizeof(RASSRV_DEVICE*) * (This->dwDeviceCount + pList->dwCount), TRUE); if (ppDevices == NULL) { dwErr = ERROR_NOT_ENOUGH_MEMORY; break; }
// Copy over the current device information
CopyMemory( ppDevices, This->pDeviceList, This->dwDeviceCount * sizeof(RASSRV_DEVICE*));
// Delete the old device list and set to the new one
RassrvFree(This->pDeviceList); This->pDeviceList = ppDevices;
// Add the ports
pNode = pList->pHead; i = This->dwDeviceCount; while (pNode) { // Allocate the new device
ppDevices[i] = RassrvAlloc(sizeof(RASSRV_DEVICE), TRUE); if (!ppDevices[i]) { dwErr = ERROR_NOT_ENOUGH_MEMORY; break; } // Set all the non-zero values
ppDevices[i]->dwType = INCOMING_TYPE_DIRECT; ppDevices[i]->pszName = pNode->pszName; ppDevices[i]->pModem = pNode->pModem; ppDevices[i]->dwFlags = DEV_FLAG_PORT; lstrcpynW( ppDevices[i]->pszPort, pNode->pszPort, sizeof(ppDevices[i]->pszPort) / sizeof(WCHAR));
// Initialize the enabling of the com port
if (ppDevices[i]->pModem) { ppDevices[i]->dwFlags |= (ppDevices[i]->pModem->dwFlags & DEV_FLAG_ENABLED); }
// Increment
i++; pNode = pNode->pNext; } This->dwDeviceCount = i; } while (FALSE);
// Cleanup
{ devCleanupPortList(pList); }
return dwErr; }
//
// Returns whether the given index lies within the bounds of the
// list of devices store in This.
//
BOOL devBoundsCheck( IN RASSRV_DEVICEDB * This, IN DWORD dwIndex) { if (This->dwDeviceCount <= dwIndex) { DbgOutputTrace("devBoundsCheck: failed for index %d", dwIndex); return FALSE; } return TRUE; }
// Gets a handle to a device to be displayed in the general tab
DWORD devGetDeviceHandle( IN HANDLE hDevDatabase, IN DWORD dwIndex, OUT HANDLE * hDevice) { RASSRV_DEVICEDB * This = (RASSRV_DEVICEDB*)hDevDatabase; if (!This || !hDevice) { return ERROR_INVALID_PARAMETER; }
if (!devBoundsCheck(This, dwIndex)) { return ERROR_INVALID_INDEX; }
// Return nothing if device is filtered
if (This->pDeviceList[dwIndex]->dwFlags & DEV_FLAG_FILTERED) { *hDevice = NULL; return ERROR_DEVICE_NOT_AVAILABLE; }
// Otherwise, return the device
else { *hDevice = (HANDLE)(This->pDeviceList[dwIndex]); } return NO_ERROR; }
//
// Returns a count of devices to be displayed in the general tab
//
DWORD devGetDeviceCount( IN HANDLE hDevDatabase, OUT LPDWORD lpdwCount) { RASSRV_DEVICEDB * This = (RASSRV_DEVICEDB*)hDevDatabase; if (!This || !lpdwCount) { return ERROR_INVALID_PARAMETER; }
*lpdwCount = This->dwDeviceCount; return NO_ERROR; }
//
// Returns the count of enabled devices
//
DWORD devGetEndpointEnableCount( IN HANDLE hDevDatabase, OUT LPDWORD lpdwCount) { RASSRV_DEVICEDB * This = (RASSRV_DEVICEDB*)hDevDatabase; DWORD i; if (!This || !lpdwCount) { return ERROR_INVALID_PARAMETER; }
*lpdwCount = 0;
for (i = 0; i < This->dwDeviceCount; i++) { if (This->pDeviceList[i]->dwFlags & DEV_FLAG_ENABLED) { (*lpdwCount) += This->pDeviceList[i]->dwEndpoints; } } return NO_ERROR; }
//
// Loads the vpn enable status
//
DWORD devGetVpnEnable( IN HANDLE hDevDatabase, IN BOOL * pbEnabled) { RASSRV_DEVICEDB * This = (RASSRV_DEVICEDB*)hDevDatabase;
if (!This || !pbEnabled) { return ERROR_INVALID_PARAMETER; }
*pbEnabled = This->bVpnEnabled; return NO_ERROR; }
//
// Saves the vpn enable status
//
DWORD devSetVpnEnable( IN HANDLE hDevDatabase, IN BOOL bEnable) { RASSRV_DEVICEDB * This = (RASSRV_DEVICEDB*)hDevDatabase;
if (!This) { return ERROR_INVALID_PARAMETER; } This->bVpnEnabled = bEnable; return NO_ERROR; }
// Saves the vpn Original value enable status
//
DWORD devSetVpnOrigEnable( IN HANDLE hDevDatabase, IN BOOL bEnable) { RASSRV_DEVICEDB * This = (RASSRV_DEVICEDB*)hDevDatabase;
if (!This) { return ERROR_INVALID_PARAMETER; } This->bVpnEnabledOrig = bEnable; return NO_ERROR; }
//
// Returns a pointer to the name of a device
//
DWORD devGetDeviceName( IN HANDLE hDevice, OUT PWCHAR * pszDeviceName) { RASSRV_DEVICE* This = (RASSRV_DEVICE*)hDevice; if (!This || !pszDeviceName) { return ERROR_INVALID_PARAMETER; }
*pszDeviceName = This->pszName;
return NO_ERROR; }
//
// Returns the type of a device
//
DWORD devGetDeviceType( IN HANDLE hDevice, OUT LPDWORD lpdwType) { RASSRV_DEVICE* This = (RASSRV_DEVICE*)hDevice; if (!This || !lpdwType) { return ERROR_INVALID_PARAMETER; }
*lpdwType = This->dwType;
return NO_ERROR; }
//
// Returns an identifier of the device that can be used in
// tapi calls.
//
DWORD devGetDeviceId( IN HANDLE hDevice, OUT LPDWORD lpdwId) { RASSRV_DEVICE* This = (RASSRV_DEVICE*)hDevice; if (!This || !lpdwId) { return ERROR_INVALID_PARAMETER; }
*lpdwId = This->dwId;
//
// If this is a com port referencing a null modem,
// then return the tapi id of the null modem
//
if ((This->dwFlags & DEV_FLAG_PORT) && (This->pModem)) { *lpdwId = This->pModem->dwId; }
return NO_ERROR; }
//
// Returns the enable status of a device for dialin
//
DWORD devGetDeviceEnable( IN HANDLE hDevice, OUT BOOL * pbEnabled) { RASSRV_DEVICE* This = (RASSRV_DEVICE*)hDevice; if (!This || !pbEnabled) { return ERROR_INVALID_PARAMETER; }
*pbEnabled = !!(This->dwFlags & DEV_FLAG_ENABLED);
return NO_ERROR; }
//
// Sets the enable status of a device for dialin
//
DWORD devSetDeviceEnable( IN HANDLE hDevice, IN BOOL bEnable) { RASSRV_DEVICE* This = (RASSRV_DEVICE*)hDevice; if (!This) { return ERROR_INVALID_PARAMETER; }
// Mark the enabling and mark the device as dirty
if (bEnable) { This->dwFlags |= DEV_FLAG_ENABLED; } else { This->dwFlags &= ~DEV_FLAG_ENABLED; } This->dwFlags |= DEV_FLAG_DIRTY;
return NO_ERROR; }
//
// Returns whether the given device is a com port as added
// by devAddComPorts
//
DWORD devDeviceIsComPort( IN HANDLE hDevice, OUT PBOOL pbIsComPort) { RASSRV_DEVICE* This = (RASSRV_DEVICE*)hDevice; if (!This) { return ERROR_INVALID_PARAMETER; }
// This is a com port if it was added by
// devAddComPorts and if it has no null
// modem associated with it.
//
if ((This->dwFlags & DEV_FLAG_PORT) && (This->pModem == NULL) ) { *pbIsComPort = TRUE; } else { *pbIsComPort = FALSE; } return NO_ERROR; }
|