1535 lines
49 KiB
C
1535 lines
49 KiB
C
|
/*****************************************************************************
|
||
|
** Microsoft RAS Device INF Library wrapper **
|
||
|
** Copyright (C) 1992-93 Microsft Corporation. All rights reserved. **
|
||
|
** **
|
||
|
** File Name : msxwrap.c **
|
||
|
** **
|
||
|
** Revision History : **
|
||
|
** July 23, 1992 David Kays Created **
|
||
|
** Feb 22, 1993 Perryh Hannah Changed static routines to global to **
|
||
|
** ease degugging. **
|
||
|
** **
|
||
|
** Description : **
|
||
|
** RAS Device INF File Library wrapper above RASFILE Library for **
|
||
|
** modem/X.25/switch DLL (RASMXS). **
|
||
|
*****************************************************************************/
|
||
|
|
||
|
#define _CTYPE_DISABLE_MACROS
|
||
|
#include <stddef.h>
|
||
|
#include <stdlib.h>
|
||
|
#include <string.h>
|
||
|
#include <ctype.h>
|
||
|
#include <windef.h>
|
||
|
#include <winnt.h>
|
||
|
#include "rasfile.h"
|
||
|
#include "rasman.h" // RASMAN_DEVICEINFO, RAS_PARAMS struct, etc.
|
||
|
#include "raserror.h" // SUCCESS & ERROR_BUFFER_TOO_SMALL
|
||
|
#include "rasmxs.h" // Public rasmxs DLL error messages
|
||
|
#include "mxsint.h" // Internal rasmxs DLL error messages
|
||
|
#include "wrapint.h"
|
||
|
#include "mxswrap.h"
|
||
|
|
||
|
// local
|
||
|
BOOL rasDevGroupFunc( LPTSTR );
|
||
|
BOOL rasDevIsDecimalMacro ( LPTSTR );
|
||
|
void rasDevGetMacroValue ( LPTSTR *, DWORD *, LPTSTR );
|
||
|
void rasDevGetDecimalMacroValue ( LPTSTR *, DWORD *, LPTSTR );
|
||
|
BOOL rasDevExpandMacros( LPTSTR, LPTSTR, DWORD *, BYTE,
|
||
|
MACROXLATIONTABLE *);
|
||
|
BOOL rasDevLookupMacro( LPTSTR, LPTSTR *, MACROXLATIONTABLE *);
|
||
|
DWORD rasDevMacroInsert( MACRO *, WORD, MACROXLATIONTABLE *);
|
||
|
void rasDevExtractKey ( LPTSTR , LPTSTR );
|
||
|
void rasDevExtractValue ( LPTSTR , LPTSTR, DWORD, HRASFILE );
|
||
|
void rasDevSortParams( LPTSTR *, DWORD );
|
||
|
void rasDevCheckParams( LPTSTR *, DWORD *);
|
||
|
void rasDevCheckMacros( LPTSTR *, LPTSTR *, DWORD *);
|
||
|
BYTE ctox( char );
|
||
|
void GetMem(DWORD dSize, BYTE **ppMem);
|
||
|
|
||
|
/*
|
||
|
* RasDevEnumDevices :
|
||
|
* Returns in pBuffer an array of RASMAN_DEVICE structures which contain
|
||
|
* all the devices in the INF file (i.e. all header names).
|
||
|
*
|
||
|
* Arguments :
|
||
|
* lpszFileName (IN) - File path name of device file
|
||
|
* pNumEntries (OUT) - Number of devices found in the INF file
|
||
|
* pBuffer (OUT) - Buffer to contain the RASMAN_DEVICE structures
|
||
|
* pwSize (INOUT) - Size of pBuffer, this parameter filled with
|
||
|
* the needed size of pBuffer if too small
|
||
|
*
|
||
|
* Return Value :
|
||
|
* ERROR_BUFFER_TOO_SMALL if pBuffer not big enough to hold all of the
|
||
|
* RASMAN_DEVICE structs, SUCCESS otherwise.
|
||
|
*/
|
||
|
DWORD APIENTRY
|
||
|
RasDevEnumDevices( PTCH lpszFileName, DWORD *pNumEntries,
|
||
|
BYTE *pBuffer, DWORD *pdwSize )
|
||
|
{
|
||
|
HRASFILE hFile;
|
||
|
RASMAN_DEVICE DeviceName;
|
||
|
DWORD dwCurSize;
|
||
|
|
||
|
if ( (hFile = RasfileLoad(lpszFileName,RFM_ENUMSECTIONS,NULL,NULL)) == -1 )
|
||
|
return ERROR_FILE_COULD_NOT_BE_OPENED;
|
||
|
|
||
|
*pNumEntries = 0;
|
||
|
if ( ! RasfileFindFirstLine(hFile,RFL_SECTION,RFS_FILE) ) {
|
||
|
*pBuffer = '\0';
|
||
|
*pdwSize = 0;
|
||
|
RasfileClose(hFile);
|
||
|
return SUCCESS;
|
||
|
}
|
||
|
|
||
|
// copy RASMAN_DEVICE structs
|
||
|
dwCurSize = 0;
|
||
|
do {
|
||
|
// get the section name
|
||
|
RasfileGetSectionName(hFile,(LPTSTR) &DeviceName);
|
||
|
// ignore the Modem Responses section
|
||
|
if ( ! _stricmp(RESPONSES_SECTION_NAME,(LPTSTR) &DeviceName) )
|
||
|
continue;
|
||
|
|
||
|
dwCurSize += sizeof(RASMAN_DEVICE);
|
||
|
// if current size exceeds the size of the buffer then just
|
||
|
// continue counting the size needed
|
||
|
if ( dwCurSize > *pdwSize )
|
||
|
continue;
|
||
|
|
||
|
strcpy(pBuffer,(LPTSTR) &DeviceName);
|
||
|
pBuffer += sizeof(RASMAN_DEVICE);
|
||
|
(*pNumEntries)++;
|
||
|
} while ( RasfileFindNextLine(hFile,RFL_SECTION,RFS_FILE) );
|
||
|
|
||
|
RasfileClose(hFile);
|
||
|
if ( dwCurSize > *pdwSize ) {
|
||
|
*pdwSize = dwCurSize;
|
||
|
return ERROR_BUFFER_TOO_SMALL;
|
||
|
}
|
||
|
else
|
||
|
*pdwSize = dwCurSize;
|
||
|
|
||
|
return SUCCESS;
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
* RasDevOpen :
|
||
|
* Open an INF file for use by the device DLL.
|
||
|
*
|
||
|
* Arguments :
|
||
|
* lpszFileName (IN) - File path name of device file
|
||
|
* lpszSectionName (IN) - Section of device file to be loaded (by Rasfile)
|
||
|
* hFile (OUT) - File handle obtained from RasfileLoad()
|
||
|
*
|
||
|
* Return Value :
|
||
|
* ERROR_FILE_COULD_NOT_BE_OPENED if the file could not be found or opened.
|
||
|
* ERROR_DEVICENAME_NOT_FOUND if the section name was not found in the
|
||
|
* INF file.
|
||
|
* ERROR_DEVICENAME_TOO_LONG if the section name is too long.
|
||
|
* SUCCESS otherwise.
|
||
|
*/
|
||
|
DWORD APIENTRY
|
||
|
RasDevOpen( PTCH lpszFileName, PTCH lpszSectionName, HRASFILE *hFile )
|
||
|
{
|
||
|
HRASFILE hRasfile;
|
||
|
|
||
|
if ( strlen(lpszSectionName) > MAX_DEVICE_NAME )
|
||
|
return ERROR_DEVICENAME_TOO_LONG;
|
||
|
|
||
|
// send RasfileLoad() the rasDevGroupFunc() to identify command lines
|
||
|
// as group headers
|
||
|
if ( (hRasfile = RasfileLoad(lpszFileName,RFM_READONLY,
|
||
|
lpszSectionName,rasDevGroupFunc)) == -1 )
|
||
|
return ERROR_FILE_COULD_NOT_BE_OPENED;
|
||
|
|
||
|
// if there is no section header loaded then the device name is invalid
|
||
|
if ( ! RasfileFindFirstLine(hRasfile,RFL_SECTION,RFS_FILE) ) {
|
||
|
RasfileClose(hRasfile);
|
||
|
return ERROR_DEVICENAME_NOT_FOUND;
|
||
|
}
|
||
|
|
||
|
// check if this section has an ALIAS
|
||
|
// current Rasfile line is the section header
|
||
|
if ( RasfileFindNextKeyLine(hRasfile,"ALIAS",RFS_SECTION) ) {
|
||
|
TCHAR szSection[MAX_DEVICE_NAME + 1];
|
||
|
RasfileGetKeyValueFields(hRasfile,NULL,szSection);
|
||
|
RasfileClose(hRasfile);
|
||
|
if ( (hRasfile = RasfileLoad(lpszFileName,RFM_READONLY,
|
||
|
szSection,rasDevGroupFunc)) == -1 )
|
||
|
return ERROR_FILE_COULD_NOT_BE_OPENED;
|
||
|
if ( ! RasfileFindFirstLine(hRasfile,RFL_SECTION,RFS_FILE) ) {
|
||
|
RasfileClose(hRasfile);
|
||
|
return ERROR_DEVICENAME_NOT_FOUND;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// set the Rasfile current line to the first keyvalue line
|
||
|
RasfileFindFirstLine(hRasfile,RFL_KEYVALUE,RFS_SECTION);
|
||
|
|
||
|
*hFile = hRasfile;
|
||
|
return SUCCESS;
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
* RasDevClose :
|
||
|
* Close an INF file used by the device DLL.
|
||
|
*
|
||
|
* Arguments :
|
||
|
* hFile (IN) - the Rasfile handle of the file to close
|
||
|
*
|
||
|
* Return Value :
|
||
|
* ERROR_INVALID_HANDLE if hFile is invalid, SUCCESS otherwise.
|
||
|
*/
|
||
|
void APIENTRY
|
||
|
RasDevClose( HRASFILE hFile )
|
||
|
{
|
||
|
RasfileClose(hFile);
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
* RasDevGetParams :
|
||
|
* Returns in pBuffer a RASMAN_DEVICEINFO structure which contains all of the
|
||
|
* keyword=value pairs between the top of the section loaded and the
|
||
|
* first command.
|
||
|
*
|
||
|
* Assumptions:
|
||
|
* All strings read from INF files are zero terminated.
|
||
|
*
|
||
|
* Arguments :
|
||
|
* hFile (IN) - the Rasfile handle of the opened INF file
|
||
|
* pBuffer (OUT) - the buffer to hold the RASMAN_DEVICEINFO structure
|
||
|
* pdSize (INOUT) - the size of pBuffer, this is filled in with the needed
|
||
|
* buffer size to hold the RASMAN_DEVICEINFO struct if
|
||
|
* pBuffer is too small
|
||
|
*
|
||
|
* Return Value :
|
||
|
* ERROR_BUFFER_TOO_SMALL if pBuffer is too small to contain the
|
||
|
* RASMAN_DEVICEINFO structure, SUCCESS otherwise.
|
||
|
*/
|
||
|
DWORD APIENTRY
|
||
|
RasDevGetParams( HRASFILE hFile, BYTE *pBuffer, DWORD *pdSize )
|
||
|
{
|
||
|
RASMAN_DEVICEINFO *pDeviceInfo;
|
||
|
DWORD dParams, dCurrentSize, i, dValueLen;
|
||
|
LPTSTR *alpszLines, *alpszLinesSave, *lppszLine, *alpszMallocedLines;
|
||
|
BOOL bBufferTooSmall = FALSE;
|
||
|
TCHAR szString[RAS_MAXLINEBUFLEN];
|
||
|
|
||
|
if ( ! RasfileFindFirstLine(hFile,RFL_KEYVALUE,RFS_SECTION) ) {
|
||
|
|
||
|
if (*pdSize >= sizeof(DWORD)) {
|
||
|
*((DWORD *)pBuffer) = 0;
|
||
|
*pdSize = sizeof(DWORD);
|
||
|
return SUCCESS;
|
||
|
}
|
||
|
else {
|
||
|
*pdSize = sizeof(DWORD);
|
||
|
return ERROR_BUFFER_TOO_SMALL;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// count the number of keyvalue lines between the top of section and
|
||
|
// the first command, and the number of bytes to hold all of the lines
|
||
|
dParams = 0;
|
||
|
do {
|
||
|
if ( RasfileGetLineType(hFile) & RFL_GROUP )
|
||
|
break;
|
||
|
dParams++;
|
||
|
} while ( RasfileFindNextLine(hFile,RFL_KEYVALUE,RFS_SECTION) );
|
||
|
|
||
|
RasfileFindFirstLine(hFile,RFL_KEYVALUE,RFS_SECTION);
|
||
|
|
||
|
// malloc enough for two times as many lines as currently exist
|
||
|
lppszLine = alpszLines = malloc(2 * dParams * sizeof(LPTSTR));
|
||
|
alpszMallocedLines = malloc(dParams * sizeof(LPTSTR));
|
||
|
|
||
|
if( (NULL == lppszLine)
|
||
|
|| (NULL == alpszMallocedLines))
|
||
|
{
|
||
|
DWORD retcode = GetLastError();
|
||
|
|
||
|
if(NULL != lppszLine)
|
||
|
{
|
||
|
free(lppszLine);
|
||
|
}
|
||
|
|
||
|
if(NULL != alpszMallocedLines)
|
||
|
{
|
||
|
free(alpszMallocedLines);
|
||
|
}
|
||
|
|
||
|
return retcode;
|
||
|
}
|
||
|
|
||
|
// record all Rasfile keyvalue lines until a group header or end of
|
||
|
// section is found
|
||
|
do {
|
||
|
if ( RasfileGetLineType(hFile) & RFL_GROUP )
|
||
|
break;
|
||
|
*lppszLine++ = (LPTSTR)RasfileGetLine(hFile);
|
||
|
} while ( RasfileFindNextLine(hFile,RFL_KEYVALUE,RFS_SECTION) );
|
||
|
|
||
|
// sort the lines by key
|
||
|
rasDevSortParams( alpszLines, dParams );
|
||
|
// check for duplicate keys and remove any that are found
|
||
|
rasDevCheckParams( alpszLines, &dParams );
|
||
|
// insert missing _ON or _OFF macros into the list
|
||
|
rasDevCheckMacros( alpszLines, alpszMallocedLines, &dParams );
|
||
|
|
||
|
// check if given buffer is large enough
|
||
|
dCurrentSize = sizeof(RASMAN_DEVICEINFO)
|
||
|
+ ((dParams - 1) * sizeof(RAS_PARAMS));
|
||
|
if ( (NULL == pBuffer)
|
||
|
|| (dCurrentSize > *pdSize )) {
|
||
|
*pdSize = dCurrentSize;
|
||
|
lppszLine = alpszMallocedLines;
|
||
|
while ( *lppszLine != NULL )
|
||
|
free(*lppszLine++);
|
||
|
free(alpszMallocedLines);
|
||
|
free(alpszLines);
|
||
|
return ERROR_BUFFER_TOO_SMALL;
|
||
|
}
|
||
|
|
||
|
// fill in pBuffer with RASMAN_DEVICEINFO struct
|
||
|
pDeviceInfo = (RASMAN_DEVICEINFO *) pBuffer;
|
||
|
pDeviceInfo->DI_NumOfParams = (WORD) dParams;
|
||
|
|
||
|
for ( i = 0, alpszLinesSave = alpszLines; i < dParams; i++, alpszLines++) {
|
||
|
RAS_PARAMS *pParam;
|
||
|
pParam = &(pDeviceInfo->DI_Params[i]);
|
||
|
|
||
|
if (!bBufferTooSmall) {
|
||
|
// set the Type and Attributes field
|
||
|
pParam->P_Type = String;
|
||
|
if ( strcspn(*alpszLines,LMS) < strcspn(*alpszLines,"=") )
|
||
|
pParam->P_Attributes = 0;
|
||
|
else
|
||
|
pParam->P_Attributes = ATTRIB_VARIABLE;
|
||
|
|
||
|
// get the key
|
||
|
rasDevExtractKey(*alpszLines,pParam->P_Key);
|
||
|
|
||
|
// if there are continuation lines for this keyword=value pair,
|
||
|
// then set Rasfile line to the proper line
|
||
|
if ( strcspn(*alpszLines,"\\") < strlen(*alpszLines) ) {
|
||
|
TCHAR szFullKey[MAX_PARAM_KEY_SIZE];
|
||
|
|
||
|
if ( ! pParam->P_Attributes ) {
|
||
|
strcpy(szFullKey,LMS);
|
||
|
strcat(szFullKey,pParam->P_Key);
|
||
|
strcat(szFullKey,RMS);
|
||
|
}
|
||
|
else
|
||
|
strcpy(szFullKey,pParam->P_Key);
|
||
|
|
||
|
// find the last occurence of this key
|
||
|
RasfileFindFirstLine(hFile,RFL_KEYVALUE,RFS_SECTION);
|
||
|
while ( RasfileFindNextKeyLine(hFile,szFullKey,RFS_SECTION) )
|
||
|
;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// get the value string
|
||
|
rasDevExtractValue(*alpszLines,
|
||
|
szString,
|
||
|
sizeof(szString),
|
||
|
hFile);
|
||
|
|
||
|
dValueLen = strlen(szString);
|
||
|
pParam->P_Value.String.Length = dValueLen;
|
||
|
pParam->P_Value.String.Data = malloc(dValueLen + 1);
|
||
|
if(NULL != pParam->P_Value.String.Data)
|
||
|
{
|
||
|
strcpy(pParam->P_Value.String.Data, szString);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// free up all mallocs
|
||
|
lppszLine = alpszMallocedLines;
|
||
|
while ( *lppszLine != NULL )
|
||
|
free(*lppszLine++);
|
||
|
free(alpszMallocedLines);
|
||
|
free(alpszLinesSave);
|
||
|
|
||
|
return SUCCESS;
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
* RasDevGetCommand :
|
||
|
* Returns the next command line of the given type and advances
|
||
|
* the Rasfile file pointer to the first line following this command
|
||
|
* line.
|
||
|
*
|
||
|
* Arguments :
|
||
|
* hFile (IN) - the Rasfile file handle for the INF file
|
||
|
* pszCmdTypeSuffix (IN) - the type of command line to search for :
|
||
|
* GENERIC, INIT, DIAL, or LISTEN.
|
||
|
* pMacroXlations (IN) - the Macro Translation table used to expand
|
||
|
* all macros in the command line
|
||
|
* lpsCommand (OUT) - buffer to hold the value string of the found
|
||
|
* command line
|
||
|
* pdwCmdLen (OUT) - length of output string with expanded macros
|
||
|
*
|
||
|
* Return Value :
|
||
|
* ERROR_END_OF_SECTION if no command lines of the given type could
|
||
|
* be found.
|
||
|
* ERROR_MACRO_NOT_DEFINED if no entry in the given Macro Translation table
|
||
|
* for a macro found in the command line could be found.
|
||
|
* SUCCESS otherwise.
|
||
|
*/
|
||
|
DWORD APIENTRY
|
||
|
RasDevGetCommand( HRASFILE hFile, PTCH pszCmdTypeSuffix,
|
||
|
MACROXLATIONTABLE *pMacroXlations, PTCH lpsCommand,
|
||
|
DWORD *pdwCmdLen )
|
||
|
{
|
||
|
TCHAR szLineKey[MAX_PARAM_KEY_SIZE], sCommand[MAX_PARAM_KEY_SIZE];
|
||
|
TCHAR szValue[RAS_MAXLINEBUFLEN];
|
||
|
TCHAR sCommandValue[2*MAX_CMD_BUF_LEN];// WARNING : if we ever
|
||
|
// get a command line > this
|
||
|
// size msxwrap could bomb!
|
||
|
|
||
|
LPTSTR lpszLine;
|
||
|
|
||
|
if ( ! (RasfileGetLineType(hFile) & RFL_GROUP) ) {
|
||
|
if ( ! RasfileFindNextLine(hFile,RFL_GROUP,RFS_SECTION) )
|
||
|
return ERROR_END_OF_SECTION;
|
||
|
}
|
||
|
else if ( RasfileGetLineMark(hFile) == EOS_COOKIE ) {
|
||
|
RasfilePutLineMark(hFile,0);
|
||
|
return ERROR_END_OF_SECTION;
|
||
|
}
|
||
|
|
||
|
strcpy(sCommand,"command");
|
||
|
strcat(sCommand,pszCmdTypeSuffix);
|
||
|
|
||
|
for ( ;; ) {
|
||
|
|
||
|
lpszLine = (LPTSTR) RasfileGetLine(hFile);
|
||
|
|
||
|
if(NULL == lpszLine)
|
||
|
{
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
rasDevExtractKey(lpszLine,szLineKey);
|
||
|
if ( ! _stricmp(sCommand,szLineKey) ) {
|
||
|
// get the value string
|
||
|
lpszLine = (LPTSTR) RasfileGetLine(hFile);
|
||
|
if(!lpszLine)
|
||
|
return ERROR_END_OF_SECTION;
|
||
|
rasDevExtractValue((LPTSTR)lpszLine,szValue,
|
||
|
RAS_MAXLINEBUFLEN,hFile);
|
||
|
// expand all macros in the value string
|
||
|
if ( ! rasDevExpandMacros(szValue, sCommandValue, pdwCmdLen,
|
||
|
EXPAND_ALL, pMacroXlations) )
|
||
|
return ERROR_MACRO_NOT_DEFINED;
|
||
|
if ( *pdwCmdLen > MAX_CMD_BUF_LEN )
|
||
|
return ERROR_CMD_TOO_LONG;
|
||
|
else
|
||
|
memcpy(lpsCommand, sCommandValue, *pdwCmdLen);
|
||
|
break;
|
||
|
}
|
||
|
if ( ! RasfileFindNextLine(hFile,RFL_GROUP,RFS_SECTION) )
|
||
|
return ERROR_END_OF_SECTION;
|
||
|
}
|
||
|
|
||
|
// advance to the first response following the command or
|
||
|
// to the next command line; if no such line exists mark the
|
||
|
// current line as the end of the section
|
||
|
if ( ! RasfileFindNextLine(hFile,RFL_ANYACTIVE,RFS_SECTION) )
|
||
|
RasfilePutLineMark(hFile,EOS_COOKIE);
|
||
|
|
||
|
return SUCCESS;
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
* RasDevResetCommand :
|
||
|
* Moves the Rasfile file pointer to the first command of any type
|
||
|
* in the loaded section.
|
||
|
*
|
||
|
* Arguments :
|
||
|
* hFile (IN) - the Rasfile handle to the loaded file
|
||
|
*
|
||
|
* Return Value :
|
||
|
* ERROR_NO_COMMAND_FOUND if no command line could be found,
|
||
|
* SUCCESS otherwise.
|
||
|
*/
|
||
|
DWORD APIENTRY
|
||
|
RasDevResetCommand( HRASFILE hFile )
|
||
|
{
|
||
|
if ( ! RasfileFindFirstLine(hFile,RFL_GROUP,RFS_SECTION) )
|
||
|
return ERROR_NO_COMMAND_FOUND;
|
||
|
else
|
||
|
return SUCCESS;
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
* RasDevCheckResponse :
|
||
|
* Returns the keyword found in the line whose value string matches
|
||
|
* the string in lpszReceived. Any macros other than fixed macros
|
||
|
* which are found in the received string have their values copied
|
||
|
* into the Macro Translation table.
|
||
|
* All lines in a Command-Response Set are checked.
|
||
|
*
|
||
|
* Arguments :
|
||
|
* hFile (IN) - the Rasfile handle to the loaded file
|
||
|
* lpszReceived (IN) - the string received from the modem or X25 net
|
||
|
* dReceivedLength (IN) - length of the received string
|
||
|
* pMacroXlations (INOUT) - the Macro Translation table
|
||
|
* lpszResponse (OUT) - buffer to copy the found keyword into
|
||
|
*
|
||
|
* Return Value :
|
||
|
* ERROR_PARTIAL_RESPONSE if a line is matched up to the APPEND_MACRO.
|
||
|
* ERROR_MACRO_NOT_DEFINED if a value for "carrierbaud", "connectbaud",
|
||
|
* or "diagnotics" is found in the received string, but could
|
||
|
* not be found in the given Macro Translation table.
|
||
|
* ERROR_UNRECOGNIZED_RESPONSE if no matching reponse could be
|
||
|
* found.
|
||
|
* ERROR_NO_REPSONSES if when called, the Rasfile current line is a
|
||
|
* command, section header, or is invalid.
|
||
|
* SUCCESS otherwise.
|
||
|
*/
|
||
|
DWORD APIENTRY
|
||
|
RasDevCheckResponse( HRASFILE hFile, PTCH lpsReceived, DWORD dReceivedLength,
|
||
|
MACROXLATIONTABLE *pMacroXlations, PTCH lpszResponse )
|
||
|
{
|
||
|
LPTSTR lpszValue, lpsRec, lpszResponseLine;
|
||
|
TCHAR szValueString[RAS_MAXLINEBUFLEN], szValue[RAS_MAXLINEBUFLEN];
|
||
|
MACRO aszMacros[10];
|
||
|
DWORD dwRC, dRecLength, dwValueLen;
|
||
|
WORD wMacros;
|
||
|
BYTE bMatch;
|
||
|
|
||
|
// find the nearest previous COMMAND line (Modem section) or
|
||
|
// the section header (Modem Responses section)
|
||
|
|
||
|
if ( RasfileGetLineMark(hFile) != EOS_COOKIE ) {
|
||
|
RasfileFindPrevLine(hFile,RFL_ANYHEADER,RFS_SECTION);
|
||
|
// set Rasfile line to the first keyvalue line in the response set
|
||
|
RasfileFindNextLine(hFile,RFL_KEYVALUE,RFS_SECTION);
|
||
|
}
|
||
|
|
||
|
// else this line is a COMMAND line and the last line of the section
|
||
|
// and ERROR_NO_RESPONSES will be returned
|
||
|
|
||
|
if ( RasfileGetLine(hFile) == NULL ||
|
||
|
RasfileGetLineType(hFile) & RFL_ANYHEADER )
|
||
|
return ERROR_NO_RESPONSES;
|
||
|
|
||
|
for ( ;; ) {
|
||
|
lpszResponseLine = (LPTSTR)RasfileGetLine(hFile);
|
||
|
|
||
|
if(NULL == lpszResponseLine)
|
||
|
{
|
||
|
return ERROR_NO_RESPONSES;
|
||
|
}
|
||
|
|
||
|
rasDevExtractValue(lpszResponseLine,szValueString,
|
||
|
RAS_MAXLINEBUFLEN,hFile);
|
||
|
|
||
|
// expand <cr> and <lf> macros only
|
||
|
//*** Warning: this could expand line beyond array size!
|
||
|
|
||
|
if ( ! rasDevExpandMacros(szValueString, szValue, &dwValueLen,
|
||
|
EXPAND_FIXED_ONLY, NULL) )
|
||
|
return ERROR_MACRO_NOT_DEFINED;
|
||
|
|
||
|
lpsRec = lpsReceived;
|
||
|
dRecLength = dReceivedLength;
|
||
|
bMatch = 0;
|
||
|
wMacros = 0;
|
||
|
|
||
|
for ( lpszValue = szValue; *lpszValue != '\0' && dRecLength > 0; ) {
|
||
|
|
||
|
// check for a macro
|
||
|
if ( *lpszValue == LMSCH ) {
|
||
|
|
||
|
// check for <<
|
||
|
if (*(lpszValue + 1) == LMSCH) {
|
||
|
if (*lpsRec == LMSCH) {
|
||
|
lpszValue +=2;
|
||
|
lpsRec++;
|
||
|
dRecLength--;
|
||
|
}
|
||
|
else
|
||
|
break; // fond a mismatch
|
||
|
}
|
||
|
|
||
|
// check for <append> macro and simply advance past it
|
||
|
else if ( ! _strnicmp(lpszValue,APPEND_MACRO,
|
||
|
strlen(APPEND_MACRO)) )
|
||
|
lpszValue += strlen(APPEND_MACRO);
|
||
|
|
||
|
// check for <ignore> macro
|
||
|
else if ( ! _strnicmp(lpszValue,IGNORE_MACRO,
|
||
|
strlen(IGNORE_MACRO)) ) {
|
||
|
bMatch = FULL_MATCH;
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
// check for <match> macro
|
||
|
else if ( ! _strnicmp(lpszValue,MATCH_MACRO,
|
||
|
strlen(MATCH_MACRO)) ) {
|
||
|
TCHAR szSubString[RAS_MAXLINEBUFLEN];
|
||
|
memset(szSubString,0,RAS_MAXLINEBUFLEN);
|
||
|
// advance value string to first char in match string
|
||
|
lpszValue += strcspn(lpszValue,"\"") + 1;
|
||
|
// extract match string
|
||
|
strncpy(szSubString,lpszValue,strcspn(lpszValue,"\""));
|
||
|
if ( RasDevSubStr(lpsRec,
|
||
|
dRecLength,
|
||
|
szSubString,
|
||
|
strlen(szSubString)) != NULL ) {
|
||
|
rasDevExtractKey(lpszResponseLine,lpszResponse);
|
||
|
return SUCCESS;
|
||
|
}
|
||
|
else
|
||
|
break; // value string does not match
|
||
|
}
|
||
|
|
||
|
// check for hex macro
|
||
|
else if ( (lpszValue[1] == 'h' || lpszValue[1] == 'H') &&
|
||
|
isxdigit(lpszValue[2]) && isxdigit(lpszValue[3]) &&
|
||
|
lpszValue[4] == RMSCH ) {
|
||
|
char c;
|
||
|
c = (char) (ctox(lpszValue[2]) * 0x10 + ctox(lpszValue[3]));
|
||
|
if ( c == *lpsRec++ ) {
|
||
|
lpszValue += 5; // '<', 'h', two hex digits, and '>'
|
||
|
dRecLength--;
|
||
|
continue;
|
||
|
}
|
||
|
else // does not match
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
// check for wildcard character
|
||
|
else if ( ! _strnicmp(lpszValue,WILDCARD_MACRO,
|
||
|
strlen(WILDCARD_MACRO)) ) {
|
||
|
lpszValue += strlen(WILDCARD_MACRO);
|
||
|
lpsRec++; // advance Receive string one character
|
||
|
dRecLength--;
|
||
|
}
|
||
|
|
||
|
else { // get macro name and value
|
||
|
memset(aszMacros[wMacros].MacroName,0,MAX_PARAM_KEY_SIZE);
|
||
|
|
||
|
// copy macro name
|
||
|
strncpy(aszMacros[wMacros].MacroName, lpszValue + 1,
|
||
|
strcspn(lpszValue,RMS) - 1);
|
||
|
|
||
|
// advance the value string over the macro
|
||
|
lpszValue += strcspn(lpszValue,RMS) + 1 /* past RMS */;
|
||
|
|
||
|
// get macro value
|
||
|
if (rasDevIsDecimalMacro(aszMacros[wMacros].MacroName))
|
||
|
rasDevGetDecimalMacroValue(&lpsRec, &dRecLength,
|
||
|
aszMacros[wMacros++].MacroValue);
|
||
|
else
|
||
|
rasDevGetMacroValue(&lpsRec, &dRecLength,
|
||
|
aszMacros[wMacros++].MacroValue);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
else if ( *lpszValue == *lpsRec ) {
|
||
|
if (*lpszValue == RMSCH && *(lpszValue + 1) == RMSCH)
|
||
|
lpszValue++;
|
||
|
lpszValue++;
|
||
|
lpsRec++;
|
||
|
dRecLength--;
|
||
|
continue;
|
||
|
}
|
||
|
else // found a mismatch
|
||
|
break;
|
||
|
} // for
|
||
|
|
||
|
|
||
|
// If we already have a match break out pf outer loop now
|
||
|
|
||
|
if (bMatch != 0)
|
||
|
break;
|
||
|
|
||
|
// full match. When there is trailing line noise dRecLength will not
|
||
|
// be zero, so check for full match aganist length of expected
|
||
|
// response. Also make sure expected response is not empty.
|
||
|
|
||
|
if ( *lpszValue == '\0' && lpszValue != szValue) {
|
||
|
bMatch |= FULL_MATCH;
|
||
|
break;
|
||
|
}
|
||
|
// partial match
|
||
|
else if ( dRecLength == 0 &&
|
||
|
! _strnicmp(lpszValue,APPEND_MACRO,strlen(APPEND_MACRO)) ) {
|
||
|
bMatch |= PARTIAL_MATCH;
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
if ( ! RasfileFindNextLine(hFile,RFL_KEYVALUE,RFS_SECTION) )
|
||
|
return ERROR_UNRECOGNIZED_RESPONSE;
|
||
|
if ( RasfileGetLineType(hFile) & RFL_GROUP )
|
||
|
return ERROR_UNRECOGNIZED_RESPONSE;
|
||
|
} // for
|
||
|
|
||
|
// sanity check
|
||
|
if ( ! (bMatch & (FULL_MATCH | PARTIAL_MATCH)) )
|
||
|
return ERROR_UNRECOGNIZED_RESPONSE;
|
||
|
|
||
|
// only get this far if a full or partial match was made
|
||
|
|
||
|
// insert any macro values found in the received string
|
||
|
// into the macro translation table
|
||
|
if ((dwRC = rasDevMacroInsert(aszMacros,wMacros,pMacroXlations)) != SUCCESS)
|
||
|
return(dwRC);
|
||
|
|
||
|
// finally, copy the keyword string into lpszResponse string
|
||
|
rasDevExtractKey(lpszResponseLine,lpszResponse);
|
||
|
return ( bMatch & FULL_MATCH ) ? SUCCESS : ERROR_PARTIAL_RESPONSE;
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
* RasDevResponseExpected :
|
||
|
* Checks the INF for presence of reponses to the current command.
|
||
|
* If the key work "NoResponse" is found on the current line the
|
||
|
* function returns FALSE. Otherwise modems always expect responses.
|
||
|
*
|
||
|
* Arguments :
|
||
|
* hFile (IN) - Rasfile file handle for the INF file.
|
||
|
* eDevType (IN) - The type of the device. (Modem, PAD, or Switch)
|
||
|
*
|
||
|
* Return Value :
|
||
|
* FALSE if the current Rasfile line points to a command line or the
|
||
|
* current line starts with "NoResponse", TRUE otherwise. Except
|
||
|
* modems always return TRUE unless "NoResponse" key word is found.
|
||
|
* (See code.)
|
||
|
*/
|
||
|
BOOL APIENTRY
|
||
|
RasDevResponseExpected( HRASFILE hFile, DEVICETYPE eDevType )
|
||
|
{
|
||
|
TCHAR szLine[RAS_MAXLINEBUFLEN];
|
||
|
|
||
|
szLine[0] = TEXT('\0');
|
||
|
|
||
|
RasfileGetLineText( hFile, szLine );
|
||
|
if ( _strnicmp(szLine, MXS_NORESPONSE, strlen(MXS_NORESPONSE)) == 0 )
|
||
|
return( FALSE );
|
||
|
|
||
|
if (eDevType == DT_MODEM)
|
||
|
return( TRUE );
|
||
|
|
||
|
if ( RasfileGetLineType(hFile) & RFL_ANYHEADER )
|
||
|
return( FALSE );
|
||
|
else
|
||
|
return( TRUE );
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
* RasDevEchoExpected :
|
||
|
* Checks the current line of the INF file for the keyword NoEcho.
|
||
|
* If found the function returns FALSE. Otherwise, it returns TRUE.
|
||
|
*
|
||
|
* Arguments :
|
||
|
* hFile (IN) - Rasfile file handle for the INF file.
|
||
|
*
|
||
|
* Return Value :
|
||
|
* FALSE if the current line is "NoEcho", else TRUE.
|
||
|
*/
|
||
|
BOOL APIENTRY
|
||
|
RasDevEchoExpected( HRASFILE hFile )
|
||
|
{
|
||
|
TCHAR szLine[RAS_MAXLINEBUFLEN];
|
||
|
|
||
|
szLine[0] = TEXT('\0');
|
||
|
|
||
|
RasfileGetLineText( hFile, szLine );
|
||
|
return( ! (_strnicmp(szLine, MXS_NOECHO, strlen(MXS_NOECHO)) == 0) );
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
* RasDevIdFistCommand :
|
||
|
* Determines the type of the first command in the section.
|
||
|
*
|
||
|
* Arguments :
|
||
|
* hFile (IN) - Rasfile file handle for the INF file.
|
||
|
*
|
||
|
* Assumptions :
|
||
|
* RasDevGetParams has been called previously, that is, the current
|
||
|
* line is the first command.
|
||
|
*
|
||
|
* Return Value :
|
||
|
* FALSE if current line is not a command, otherwise TRUE.
|
||
|
*/
|
||
|
CMDTYPE APIENTRY
|
||
|
RasDevIdFirstCommand( HRASFILE hFile )
|
||
|
{
|
||
|
TCHAR szKey[MAX_PARAM_KEY_SIZE + 1];
|
||
|
|
||
|
|
||
|
// Find the first command
|
||
|
|
||
|
if ( ! RasfileFindFirstLine(hFile,RFL_GROUP,RFS_SECTION))
|
||
|
return(CT_UNKNOWN);
|
||
|
|
||
|
if ( ! RasfileGetKeyValueFields(hFile, szKey, NULL))
|
||
|
return(CT_UNKNOWN);
|
||
|
|
||
|
|
||
|
// Convert Key from the line into an enum
|
||
|
|
||
|
if (_stricmp(MXS_GENERIC_COMMAND, szKey) == 0)
|
||
|
return(CT_GENERIC);
|
||
|
|
||
|
else if (_stricmp(MXS_DIAL_COMMAND, szKey) == 0)
|
||
|
return(CT_DIAL);
|
||
|
|
||
|
else if (_stricmp(MXS_INIT_COMMAND, szKey) == 0)
|
||
|
return(CT_INIT);
|
||
|
|
||
|
else if (_stricmp(MXS_LISTEN_COMMAND, szKey) == 0)
|
||
|
return(CT_LISTEN);
|
||
|
|
||
|
else
|
||
|
return(CT_UNKNOWN);
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
* RasDevSubStr :
|
||
|
* Finds a substring and returns a pointer to it. This function works like
|
||
|
* the C runtime function strstr, but works in strings that contain zeros.
|
||
|
*
|
||
|
* Arguments :
|
||
|
* psStr (IN) - the string to be searched for a substring
|
||
|
* dwStrLen (IN) - length of the string to be searched
|
||
|
* psSubStr (IN) - the substring to search for
|
||
|
* dwSubStrLen (IN) - length of the substring
|
||
|
*
|
||
|
* Return Value :
|
||
|
* A pointer to the beginning of the substring, or NULL if the substring
|
||
|
* was not found.
|
||
|
*/
|
||
|
LPTSTR APIENTRY
|
||
|
RasDevSubStr( LPTSTR psStr,
|
||
|
DWORD dwStrLen,
|
||
|
LPTSTR psSubStr,
|
||
|
DWORD dwSubStrLen )
|
||
|
{
|
||
|
LPTSTR ps;
|
||
|
|
||
|
|
||
|
if (dwSubStrLen > dwStrLen)
|
||
|
return NULL;
|
||
|
|
||
|
for (ps = psStr; ps <= psStr + dwStrLen - dwSubStrLen; ps++)
|
||
|
|
||
|
if (memcmp(ps, psSubStr, dwSubStrLen) == 0)
|
||
|
return ps;
|
||
|
|
||
|
return NULL;
|
||
|
}
|
||
|
|
||
|
|
||
|
/*****************************************************************************
|
||
|
** Rasfile Wrapper internal routines **
|
||
|
****************************************************************************/
|
||
|
|
||
|
/*
|
||
|
* rasDevGroupFunc :
|
||
|
* The PFBISGROUP function passed to RasfileLoad().
|
||
|
*
|
||
|
* Arguments :
|
||
|
* lpszLine (IN) - a Rasfile line
|
||
|
*
|
||
|
* Return Value :
|
||
|
* TRUE if the line is a command line, FALSE otherwise.
|
||
|
*/
|
||
|
BOOL rasDevGroupFunc( LPTSTR lpszLine )
|
||
|
{
|
||
|
TCHAR szKey[MAX_PARAM_KEY_SIZE], *lpszKey;
|
||
|
|
||
|
if ( strcspn(lpszLine,"=") == strlen(lpszLine) )
|
||
|
return FALSE;
|
||
|
|
||
|
while ( *lpszLine == ' ' || *lpszLine == '\t' )
|
||
|
lpszLine++;
|
||
|
|
||
|
lpszKey = szKey;
|
||
|
while ( *lpszLine != ' ' && *lpszLine != '\t' && *lpszLine != '=' )
|
||
|
*lpszKey++ = *lpszLine++;
|
||
|
*lpszKey = '\0';
|
||
|
|
||
|
if ( ! _stricmp(szKey,"COMMAND") || ! _stricmp(szKey,"COMMAND_INIT") ||
|
||
|
! _stricmp(szKey,"COMMAND_DIAL") || ! _stricmp(szKey,"COMMAND_LISTEN") )
|
||
|
return TRUE;
|
||
|
else
|
||
|
return FALSE;
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
* rasDevIsDecimalMacro :
|
||
|
* Indicates whether or not a given macro must have only ascii
|
||
|
* decimal digits for its value.
|
||
|
*
|
||
|
* Arguments:
|
||
|
* lpszMacroName (IN) - macro name
|
||
|
*
|
||
|
* Return Value:
|
||
|
* TRUE if only digits are legal in the macro value; otherwise FALSE.
|
||
|
*
|
||
|
* Remarks:
|
||
|
* Called by API RasDevCheckResponse().
|
||
|
*/
|
||
|
BOOL rasDevIsDecimalMacro ( LPTSTR lpszMacroName )
|
||
|
{
|
||
|
if (_stricmp(lpszMacroName, MXS_CONNECTBPS_KEY) == 0 ||
|
||
|
_stricmp(lpszMacroName, MXS_CARRIERBPS_KEY) == 0)
|
||
|
return(TRUE);
|
||
|
else
|
||
|
return(FALSE);
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
* rasDevGetMacroValue :
|
||
|
* Extracts a macro value from string *lppszReceived and copies it
|
||
|
* to string lpszMacro. Also updates the string pointer of
|
||
|
* lppszValue and lppszReceived, and updates dRecLength.
|
||
|
*
|
||
|
* Arguments :
|
||
|
* lppszReceived (INOUT) - received string (from a modem)
|
||
|
* dRecLength (INOUT) - remaining length of the received string
|
||
|
* lpszMacro (OUT) - buffer to receive the macro value
|
||
|
*
|
||
|
* Return Value :
|
||
|
* None.
|
||
|
*
|
||
|
* Remarks :
|
||
|
* Called by API RasDevCheckResponse().
|
||
|
*/
|
||
|
void rasDevGetMacroValue ( LPTSTR *lppszReceived, DWORD *dRecLength,
|
||
|
LPTSTR lpszMacroValue )
|
||
|
{
|
||
|
while ( **lppszReceived != CR && **lppszReceived != '\0' ) {
|
||
|
*lpszMacroValue++ = *(*lppszReceived)++;
|
||
|
(*dRecLength)--;
|
||
|
}
|
||
|
*lpszMacroValue = '\0'; // Null terminate the Macro value string
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
* rasDevGetDecimalMacroValue :
|
||
|
* Extracts a macro value from string *lppszReceived and copies it
|
||
|
* to string lpszMacro. Also updates the string pointer of
|
||
|
* lppszReceived, and updates dRecLength.
|
||
|
* This functions only extracts characters which are ascii decimal
|
||
|
* digits.
|
||
|
*
|
||
|
* Arguments :
|
||
|
* lppszReceived (INOUT) - received string (from a modem)
|
||
|
* dRecLength (INOUT) - remaining length of the received string
|
||
|
* lpszMacro (OUT) - buffer to receive the macro value
|
||
|
*
|
||
|
* Return Value :
|
||
|
* None.
|
||
|
*
|
||
|
* Remarks :
|
||
|
* Called by API RasDevCheckResponse().
|
||
|
*/
|
||
|
void rasDevGetDecimalMacroValue ( LPTSTR *lppszReceived,
|
||
|
DWORD *dRecLength,
|
||
|
LPTSTR lpszMacroValue )
|
||
|
{
|
||
|
TCHAR szBuffer[16], *pBuf = szBuffer;
|
||
|
WORD wcRightHandDigits = 0;
|
||
|
BOOL bDpFound = FALSE;
|
||
|
ULONG lBps;
|
||
|
|
||
|
|
||
|
while ( isdigit(**lppszReceived) || **lppszReceived == '.' ) {
|
||
|
|
||
|
if (isdigit(**lppszReceived)) {
|
||
|
*pBuf++ = *(*lppszReceived)++;
|
||
|
(*dRecLength)--;
|
||
|
if (bDpFound)
|
||
|
wcRightHandDigits++;
|
||
|
}
|
||
|
else if (!bDpFound && **lppszReceived == '.') {
|
||
|
(*lppszReceived)++;
|
||
|
(*dRecLength)--;
|
||
|
bDpFound = TRUE;
|
||
|
}
|
||
|
else
|
||
|
break;
|
||
|
}
|
||
|
*pBuf = '\0'; // Null terminate the Macro value string
|
||
|
|
||
|
lBps = atol(szBuffer);
|
||
|
|
||
|
switch(wcRightHandDigits)
|
||
|
{
|
||
|
case 0: case 3:
|
||
|
break;
|
||
|
case 1:
|
||
|
lBps *= 100;
|
||
|
break;
|
||
|
case 2:
|
||
|
lBps *= 10;
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
_ltoa(lBps, lpszMacroValue, 10);
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
* rasDevExpandMacros :
|
||
|
* Takes the string lpszLine, and copies it to lpszVal, using
|
||
|
* Macro Translation table pMacroXlations to expand macros.
|
||
|
* <cr>, <lf>, and <hxx> macros are always expanded directly.
|
||
|
* If bFlag == EXPAND_ALL << and >> are converted to < and >.
|
||
|
* (A single > which is not at the end of a macro is simply copied.
|
||
|
* An error could be raised here for such a >, but it is left to
|
||
|
* be caught later when the device chokes on the unexpected >.
|
||
|
* This has the advantage that a > where a >> should be will work.)
|
||
|
*
|
||
|
* Assumptions:
|
||
|
* Expanded macros may contain zeros, therefore output command string
|
||
|
* may contain zeros.
|
||
|
*
|
||
|
* Arguments :
|
||
|
* lpszLine (IN) - a value string from a Rasfile keyword=value line
|
||
|
* lpsVal (OUT) - buffer to copied to with expanded macros
|
||
|
* pdwValLen (OUT) - length of output string with expanded macros
|
||
|
* bFlag (IN) - EXPAND_FIXED_ONLY if only the fixed macros <cr>
|
||
|
* and <lf> macros are to be expanded, and
|
||
|
* EXPAND_ALL if all macros should be expanded
|
||
|
* pMacroXlations (IN) - the Macro Translation table
|
||
|
*
|
||
|
* Return Value :
|
||
|
* FALSE if a needed macro translation could not be found in the
|
||
|
* pMacroXlations table, TRUE otherwise.
|
||
|
*
|
||
|
* Remarks :
|
||
|
* Called by APIs RasDevGetCommand() and RasDevCheckResponse().
|
||
|
*/
|
||
|
BOOL rasDevExpandMacros( LPTSTR lpszLine,
|
||
|
LPTSTR lpsVal,
|
||
|
DWORD *pdwValLen,
|
||
|
BYTE bFlag,
|
||
|
MACROXLATIONTABLE *pMacroXlations )
|
||
|
{
|
||
|
TCHAR szMacro[RAS_MAXLINEBUFLEN];
|
||
|
LPTSTR lpsValue;
|
||
|
|
||
|
|
||
|
lpsValue = lpsVal;
|
||
|
|
||
|
for ( ; *lpszLine != '\0'; ) {
|
||
|
// check for RMSCH
|
||
|
// if EXPAND_ALL convert double RMSCH to single RMSCH, and
|
||
|
// simply copy single RMSCH.
|
||
|
if ((bFlag & EXPAND_ALL) && *lpszLine == RMSCH) {
|
||
|
*lpsValue++ = *lpszLine++;
|
||
|
if (*lpszLine == RMSCH)
|
||
|
lpszLine++;
|
||
|
}
|
||
|
// check for a macro or double LMSCH
|
||
|
else if ( *lpszLine == LMSCH ) {
|
||
|
if ((bFlag & EXPAND_ALL) && *(lpszLine + 1) == LMSCH) {
|
||
|
*lpsValue++ = *lpszLine;
|
||
|
lpszLine += 2;
|
||
|
}
|
||
|
else if ( ! _strnicmp(lpszLine,CR_MACRO,4) ) {
|
||
|
*lpsValue++ = CR;
|
||
|
lpszLine += 4;
|
||
|
}
|
||
|
else if ( ! _strnicmp(lpszLine,LF_MACRO,4) ) {
|
||
|
*lpsValue++ = LF;
|
||
|
lpszLine += 4;
|
||
|
}
|
||
|
else if ( ! _strnicmp(lpszLine,APPEND_MACRO,8) &&
|
||
|
(bFlag & EXPAND_ALL) )
|
||
|
lpszLine += 8;
|
||
|
|
||
|
// Hex macro stuff
|
||
|
//
|
||
|
else if ((lpszLine[1] == 'h' || lpszLine[1] == 'H') &&
|
||
|
isxdigit(lpszLine[2]) && isxdigit(lpszLine[3]) &&
|
||
|
(lpszLine[4] == RMSCH) &&
|
||
|
( bFlag & EXPAND_ALL )) {
|
||
|
char c;
|
||
|
c = (char) (ctox(lpszLine[2]) * 0x10 + ctox(lpszLine[3]));
|
||
|
lpszLine += 5; // '<', 'h', two hex digits, and '>'
|
||
|
|
||
|
*lpsValue++ = c;
|
||
|
}
|
||
|
else if ( bFlag & EXPAND_ALL ) {
|
||
|
LPTSTR lpszStr;
|
||
|
char buf[256];
|
||
|
|
||
|
for ( lpszLine++, lpszStr = szMacro; *lpszLine != RMSCH; )
|
||
|
*lpszStr++ = *lpszLine++;
|
||
|
lpszLine++; // advance past RMSCH
|
||
|
*lpszStr = '\0'; // Null terminate szMacro string
|
||
|
|
||
|
if ( ! rasDevLookupMacro(szMacro,&lpsValue,pMacroXlations) )
|
||
|
return FALSE;
|
||
|
}
|
||
|
else {
|
||
|
// just copy the macro if EXPAND_ALL is not set
|
||
|
while ( *lpszLine != RMSCH )
|
||
|
*lpsValue++ = *lpszLine++;
|
||
|
*lpsValue++ = *lpszLine++;
|
||
|
}
|
||
|
}
|
||
|
else
|
||
|
*lpsValue++ = *lpszLine++;
|
||
|
} // for
|
||
|
|
||
|
*lpsValue = '\0';
|
||
|
*pdwValLen = (DWORD) (lpsValue - lpsVal);
|
||
|
|
||
|
return TRUE;
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
* rasDevLookupMacro :
|
||
|
* Lookup macro lpszMacro in the given Macro Translation table, and
|
||
|
* return it's value in *lppszExpanded if found.
|
||
|
*
|
||
|
* Arguments :
|
||
|
* lpszMacro (IN) - the macro whose value is sought
|
||
|
* lppszExpanded (OUT) - double pointer to increment and copy the
|
||
|
* macro's value to
|
||
|
* pMacroXlations (IN) - the Macro Translation table
|
||
|
*
|
||
|
* Return Value :
|
||
|
* FALSE if the macro could not be found in the given Macro Translation
|
||
|
* table, TRUE otherwise.
|
||
|
*
|
||
|
* Remarks :
|
||
|
* Called by internal function rasDevExpandMacros().
|
||
|
*/
|
||
|
BOOL rasDevLookupMacro( LPTSTR lpszMacro, LPTSTR *lppszExpanded,
|
||
|
MACROXLATIONTABLE *pMacroXlations )
|
||
|
{
|
||
|
WORD i;
|
||
|
LPTSTR lpszMacroValue;
|
||
|
|
||
|
for ( i = 0; i < pMacroXlations->MXT_NumOfEntries; i++ ) {
|
||
|
if ( ! _stricmp(pMacroXlations->MXT_Entry[i].E_MacroName, lpszMacro) ) {
|
||
|
lpszMacroValue =
|
||
|
pMacroXlations->MXT_Entry[i].E_Param->P_Value.String.Data;
|
||
|
|
||
|
while (*lpszMacroValue != 0) {
|
||
|
**lppszExpanded = *lpszMacroValue; // copy macro char by char
|
||
|
|
||
|
if ((*lpszMacroValue == LMSCH && *(lpszMacroValue+1) == LMSCH)
|
||
|
|| (*lpszMacroValue == RMSCH && *(lpszMacroValue+1) == RMSCH))
|
||
|
lpszMacroValue++; // skip one of double angle brackets
|
||
|
|
||
|
lpszMacroValue++;
|
||
|
(*lppszExpanded)++;
|
||
|
}
|
||
|
|
||
|
return TRUE;
|
||
|
}
|
||
|
}
|
||
|
return FALSE;
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
* rasDevMacroInsert :
|
||
|
* Updates the value of macro lpszMacro with new value lpszNewValue
|
||
|
* in the given Macro Translation table.
|
||
|
*
|
||
|
* Arguments :
|
||
|
* aszMacros (IN) - array of macro name and value pairs
|
||
|
* wMacros (IN) - number of elements of aszMacros array
|
||
|
* pMacroXlations (INOUT) - the Macro Translation table
|
||
|
*
|
||
|
* Return Value : SUCCESS
|
||
|
* ERROR_MACRO_NOT_DEFINED
|
||
|
*
|
||
|
* Remarks :
|
||
|
* Called by API RasDevCheckResponse().
|
||
|
*/
|
||
|
DWORD rasDevMacroInsert( MACRO *aszMacros, WORD wMacros,
|
||
|
MACROXLATIONTABLE *pMacroXlations )
|
||
|
{
|
||
|
int iMacros;
|
||
|
WORD iXlations;
|
||
|
DWORD dwRC;
|
||
|
|
||
|
for ( iMacros = (int)(wMacros - 1); iMacros >= 0; iMacros-- ) {
|
||
|
|
||
|
for ( iXlations = 0; iXlations < pMacroXlations->MXT_NumOfEntries;
|
||
|
iXlations++ ) {
|
||
|
|
||
|
if ( ! _stricmp(pMacroXlations->MXT_Entry[iXlations].E_MacroName,
|
||
|
aszMacros[iMacros].MacroName) ) {
|
||
|
|
||
|
dwRC = UpdateParamString(pMacroXlations->MXT_Entry[iXlations].E_Param,
|
||
|
aszMacros[iMacros].MacroValue,
|
||
|
strlen(aszMacros[iMacros].MacroValue));
|
||
|
if (dwRC != SUCCESS)
|
||
|
return dwRC;
|
||
|
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
if ( iXlations == pMacroXlations->MXT_NumOfEntries )
|
||
|
return ERROR_MACRO_NOT_DEFINED;
|
||
|
}
|
||
|
return SUCCESS;
|
||
|
}
|
||
|
|
||
|
|
||
|
/*
|
||
|
* rasDevExtractKey :
|
||
|
* Extracts the keyvalue from a Rasfile line.
|
||
|
*
|
||
|
* Arguments :
|
||
|
* lpszString (IN) - Rasfile line pointer.
|
||
|
* lpszKey (OUT) - buffer to hold the keyvalue
|
||
|
*
|
||
|
* Return Value :
|
||
|
* None.
|
||
|
*
|
||
|
* Remarks :
|
||
|
* Called by APIs RasDevGetParams(), RasDevGetCommand(), and
|
||
|
* RasDevCheckResponse(), and internal functions rasDevCheckParams()
|
||
|
* and rasDevCheckMacros().
|
||
|
*/
|
||
|
void rasDevExtractKey ( LPTSTR lpszString, LPTSTR lpszKey )
|
||
|
{
|
||
|
// skip to beginning of keyword (skip '<' if present)
|
||
|
while ( *lpszString == ' ' || *lpszString == '\t' ||
|
||
|
*lpszString == LMSCH )
|
||
|
lpszString++;
|
||
|
|
||
|
while ( *lpszString != RMSCH && *lpszString != '=' &&
|
||
|
*lpszString != ' ' && *lpszString != '\t' )
|
||
|
*lpszKey++ = *lpszString++;
|
||
|
*lpszKey = '\0'; // Null terminate keyword string
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
* rasDevExtractValue :
|
||
|
* Extracts the value string for a keyword=value string which
|
||
|
* begins on Rasfile line lpszString. This function recongizes a
|
||
|
* backslash \ as a line continuation character and a double
|
||
|
* backslash \\ as a backslash character.
|
||
|
*
|
||
|
* Assumptions: lpszValue output buffer is ALWAYS large enough.
|
||
|
*
|
||
|
* Arguments :
|
||
|
* lpszString (IN) - Rasfile line where the keyword=value string begins
|
||
|
* lpszValue (OUT) - buffer to hold the value string
|
||
|
* dSize (IN) - size of the lpszValue buffer
|
||
|
* hFile (IN) - Rasfile handle, the current line must be the line
|
||
|
* which lpszString points to
|
||
|
*
|
||
|
* Return Value :
|
||
|
* None.
|
||
|
*
|
||
|
* Remarks :
|
||
|
* Called by APIs RasDevGetParams(), RasDevGetCommand(), and
|
||
|
* RasDevCheckResponse().
|
||
|
*/
|
||
|
void rasDevExtractValue ( LPTSTR lpszString, LPTSTR lpszValue,
|
||
|
DWORD dSize, HRASFILE hFile )
|
||
|
{
|
||
|
LPTSTR lpszInputStr;
|
||
|
BOOL bLineContinues;
|
||
|
|
||
|
|
||
|
// skip to beginning of value string
|
||
|
for ( lpszString += strcspn(lpszString,"=") + 1;
|
||
|
*lpszString == ' ' || *lpszString == '\t'; lpszString++ )
|
||
|
;
|
||
|
|
||
|
// check for continuation lines
|
||
|
if ( strcspn(lpszString,"\\") == strlen(lpszString) )
|
||
|
strcpy(lpszValue,lpszString); // copy value string
|
||
|
|
||
|
else {
|
||
|
memset(lpszValue,0,dSize);
|
||
|
lpszInputStr = lpszString;
|
||
|
|
||
|
for (;;) {
|
||
|
// copy the current line
|
||
|
bLineContinues = FALSE;
|
||
|
|
||
|
while (*lpszInputStr != '\0') {
|
||
|
if (*lpszInputStr == '\\')
|
||
|
if (*(lpszInputStr + 1) == '\\') {
|
||
|
*lpszValue++ = *lpszInputStr; // copy one backslash
|
||
|
lpszInputStr += 2;
|
||
|
}
|
||
|
else {
|
||
|
bLineContinues = TRUE;
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
else
|
||
|
*lpszValue++ = *lpszInputStr++;
|
||
|
}
|
||
|
|
||
|
if ( ! bLineContinues)
|
||
|
break;
|
||
|
|
||
|
// get the next line
|
||
|
if ( ! RasfileFindNextLine(hFile,RFL_ANYACTIVE,RFS_SECTION) )
|
||
|
break;
|
||
|
lpszInputStr = (LPTSTR)RasfileGetLine(hFile);
|
||
|
}
|
||
|
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
* rasDevSortParams :
|
||
|
* Sorts an array of Rasfile lines by keyvalue.
|
||
|
*
|
||
|
* Arguments :
|
||
|
* alpszLines (INOUT) - the array of line pointers
|
||
|
* dParams (IN) - number of elements in the array
|
||
|
*
|
||
|
* Return Value :
|
||
|
* None.
|
||
|
*
|
||
|
* Remarks :
|
||
|
* Called by API RasDevGetParams().
|
||
|
*/
|
||
|
void rasDevSortParams( LPTSTR *alpszLines, DWORD dParams )
|
||
|
{
|
||
|
TCHAR szKey1[MAX_PARAM_KEY_SIZE], szKey2[MAX_PARAM_KEY_SIZE];
|
||
|
LPTSTR lpszTemp;
|
||
|
DWORD i,j;
|
||
|
BOOL changed;
|
||
|
|
||
|
// If there is nothing to sort, don't try
|
||
|
if (dParams < 2)
|
||
|
return;
|
||
|
|
||
|
/* Bubble sort - it's stable */
|
||
|
for ( i = dParams - 1; i > 0; i-- ) {
|
||
|
changed = FALSE;
|
||
|
for ( j = 0; j < i; j++ ) {
|
||
|
rasDevExtractKey(alpszLines[j],szKey1);
|
||
|
rasDevExtractKey(alpszLines[j+1],szKey2);
|
||
|
// sort by keyvalue
|
||
|
if ( _stricmp(szKey1,szKey2) > 0 ) {
|
||
|
lpszTemp = alpszLines[j];
|
||
|
alpszLines[j] = alpszLines[j+1];
|
||
|
alpszLines[j+1] = lpszTemp;
|
||
|
changed = TRUE;
|
||
|
}
|
||
|
}
|
||
|
if ( ! changed )
|
||
|
return;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
* rasDevCheckParams :
|
||
|
* Removes duplicate lines from the alpszLines array of lines.
|
||
|
* Duplicates lines are those whose keyvalue is identical. The
|
||
|
* line with the lesser index is removed.
|
||
|
*
|
||
|
* Arguments :
|
||
|
* alpszLines (INOUT) - the array of line pointers
|
||
|
* pdTotalParams (INOUT) - number of array entries, this is updated
|
||
|
* if duplicates are removed
|
||
|
*
|
||
|
* Return Value :
|
||
|
* None.
|
||
|
*
|
||
|
* Remarks :
|
||
|
* Called by API RasDevGetParams().
|
||
|
*/
|
||
|
void rasDevCheckParams( LPTSTR *alpszLines, DWORD *pdTotalParams )
|
||
|
{
|
||
|
TCHAR szKey1[MAX_PARAM_KEY_SIZE], szKey2[MAX_PARAM_KEY_SIZE];
|
||
|
DWORD dParams, i;
|
||
|
|
||
|
dParams = *pdTotalParams;
|
||
|
for ( i = 1; i < *pdTotalParams ; i++ ) {
|
||
|
rasDevExtractKey(alpszLines[i-1],szKey1);
|
||
|
rasDevExtractKey(alpszLines[i],szKey2);
|
||
|
if ( _stricmp(szKey1,szKey2) == 0 ) {
|
||
|
memcpy(&(alpszLines[i-1]),&(alpszLines[i]),
|
||
|
(*pdTotalParams - i) * sizeof(LPTSTR *));
|
||
|
(*pdTotalParams)--;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
* rasDevCheckMacros :
|
||
|
* Checks the array of lines for missing _ON or _OFF macros
|
||
|
* in binary macro pairs and inserts any such missing macro
|
||
|
* into the array of lines.
|
||
|
*
|
||
|
* Arguments :
|
||
|
* alpszLines (INOUT) - array of lines
|
||
|
* alpszMallocedLines (OUT) - array of newly malloced lines for
|
||
|
* this routine
|
||
|
* pdTotalParams (INOUT) - total number of elements in alpszLines
|
||
|
* array, this is updated if new entries are
|
||
|
* added
|
||
|
*
|
||
|
* Return Value :
|
||
|
* None.
|
||
|
*
|
||
|
* Remarks :
|
||
|
* Called by API RasDevGetParams().
|
||
|
*/
|
||
|
void rasDevCheckMacros( LPTSTR *alpszLines, LPTSTR *alpszMallocedLines,
|
||
|
DWORD *pdTotalParams )
|
||
|
{
|
||
|
TCHAR szKey1[MAX_PARAM_KEY_SIZE], szKey2[MAX_PARAM_KEY_SIZE];
|
||
|
DWORD i, j;
|
||
|
BYTE bMissing;
|
||
|
|
||
|
if(alpszLines == NULL)
|
||
|
{
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
// insert missing _ON and _OFF macros
|
||
|
for ( i = 0; i < *pdTotalParams; i++ ) {
|
||
|
if ( strcspn(alpszLines[i],LMS) > strcspn(alpszLines[i],"=") )
|
||
|
continue; // not a macro
|
||
|
|
||
|
bMissing = NONE;
|
||
|
rasDevExtractKey(alpszLines[i],szKey1);
|
||
|
|
||
|
// if current key is an _OFF macro, check for a missing _ON
|
||
|
if ( strstr(szKey1,"_OFF") != NULL || strstr(szKey1,"_off") != NULL ) {
|
||
|
if ( i+1 == *pdTotalParams ) // looking at last parameter
|
||
|
bMissing = ON;
|
||
|
// get next key
|
||
|
else {
|
||
|
rasDevExtractKey(alpszLines[i+1],szKey2);
|
||
|
if (_strnicmp(szKey1,szKey2,strlen(szKey1) - strlen("OFF")) != 0)
|
||
|
bMissing = ON;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// if current key is an _ON macro, check for a missing _OFF
|
||
|
if ( strstr(szKey1,"_ON") != NULL || strstr(szKey1,"_on") != NULL ) {
|
||
|
if ( i == 0 ) // looking at first parameter
|
||
|
bMissing = OFF;
|
||
|
// get previous key
|
||
|
else {
|
||
|
rasDevExtractKey(alpszLines[i-1],szKey2);
|
||
|
if (_strnicmp(szKey1,szKey2,strlen(szKey1) - strlen("ON")) != 0)
|
||
|
bMissing = OFF;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if ( bMissing != NONE ) {
|
||
|
// shift everything over one position
|
||
|
for ( j = *pdTotalParams - 1;
|
||
|
j >= i + ((bMissing == ON) ? 1 : 0); j-- )
|
||
|
alpszLines[j+1] = alpszLines[j];
|
||
|
|
||
|
// point j to the new empty array entry
|
||
|
j = (bMissing == OFF) ? i : i + 1;
|
||
|
|
||
|
alpszLines[j] = malloc(sizeof(TCHAR) * RAS_MAXLINEBUFLEN);
|
||
|
|
||
|
if(NULL == alpszLines[j])
|
||
|
{
|
||
|
*alpszMallocedLines = NULL;
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
*alpszMallocedLines++ = alpszLines[j];
|
||
|
|
||
|
memset(alpszLines[j],0,sizeof(TCHAR) * RAS_MAXLINEBUFLEN);
|
||
|
strcpy(alpszLines[j],LMS);
|
||
|
if ( bMissing == ON )
|
||
|
strncat(alpszLines[j],szKey1,
|
||
|
strlen(szKey1) - strlen(OFF_STR));
|
||
|
else // bMissing == OFF
|
||
|
strncat(alpszLines[j],szKey1,
|
||
|
strlen(szKey1) - strlen(ON_STR));
|
||
|
strcat(alpszLines[j], bMissing == ON ? ON_STR : OFF_STR );
|
||
|
strcat(alpszLines[j], RMS);
|
||
|
strcat(alpszLines[j], "=");
|
||
|
|
||
|
(*pdTotalParams)++;
|
||
|
i++; // increment i to compensate for the new entry
|
||
|
}
|
||
|
|
||
|
} // for
|
||
|
|
||
|
// Null terminate the Malloced Lines array
|
||
|
*alpszMallocedLines = NULL;
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
* ctox :
|
||
|
* Convert char hex digit to decimal number.
|
||
|
*/
|
||
|
BYTE ctox( char ch )
|
||
|
{
|
||
|
if ( isdigit(ch) )
|
||
|
return ch - '0';
|
||
|
else
|
||
|
return (tolower(ch) - 'a') + 10;
|
||
|
}
|
||
|
|
||
|
//* UpdateParamString ------------------------------------------------------
|
||
|
//
|
||
|
// Function: This function copys a new string into a PARAM.P_Value
|
||
|
// allocating new memory of the new string is longer than
|
||
|
// the old. The copied string is then zero terminated.
|
||
|
//
|
||
|
// NOTE: This function frees and allocates memory and is not
|
||
|
// suitable for copying into an existing buffer. Use with
|
||
|
// InfoTable and other RAS_PARAMS with 'unpacked' strings.
|
||
|
//
|
||
|
// Arguments:
|
||
|
// pParam OUT Pointer to Param to update
|
||
|
// psStr IN Input string
|
||
|
// dwStrLen IN Length of input string
|
||
|
//
|
||
|
// Returns: SUCCESS
|
||
|
// ERROR_ALLOCATING_MEMORY
|
||
|
//*
|
||
|
|
||
|
DWORD
|
||
|
UpdateParamString(RAS_PARAMS *pParam, TCHAR *psStr, DWORD dwStrLen)
|
||
|
{
|
||
|
|
||
|
if (dwStrLen > pParam->P_Value.String.Length)
|
||
|
{
|
||
|
free(pParam->P_Value.String.Data);
|
||
|
|
||
|
GetMem(dwStrLen + 1, &(pParam->P_Value.String.Data));
|
||
|
if (pParam->P_Value.String.Data == NULL)
|
||
|
return(ERROR_ALLOCATING_MEMORY);
|
||
|
}
|
||
|
pParam->P_Value.String.Length = dwStrLen;
|
||
|
|
||
|
memcpy(pParam->P_Value.String.Data, psStr, dwStrLen);
|
||
|
pParam->P_Value.String.Data[dwStrLen] = '\0'; //Zero Terminate
|
||
|
|
||
|
return(SUCCESS);
|
||
|
}
|
||
|
|
||
|
//* GetMem -----------------------------------------------------------------
|
||
|
//
|
||
|
// Function: Allocates memory. If the memory allocation fails the output
|
||
|
// parameter will be NULL.
|
||
|
//
|
||
|
// Returns: Nothing.
|
||
|
//
|
||
|
//*
|
||
|
|
||
|
void
|
||
|
GetMem(DWORD dSize, BYTE **ppMem)
|
||
|
{
|
||
|
|
||
|
*ppMem = (BYTE *) calloc(dSize, 1);
|
||
|
}
|