/***************************************************************************** ** 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 #include #include #include #include #include #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 and 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 macro and simply advance past it else if ( ! _strnicmp(lpszValue,APPEND_MACRO, strlen(APPEND_MACRO)) ) lpszValue += strlen(APPEND_MACRO); // check for macro else if ( ! _strnicmp(lpszValue,IGNORE_MACRO, strlen(IGNORE_MACRO)) ) { bMatch = FULL_MATCH; break; } // check for 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. * , , and 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 * and 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); }