/*++ Copyright (c) 1992-2002 Microsoft Corporation Purpose: Formatting functions. --*/ #include "precomp.hxx" #pragma hdrstop #define MAXNESTING (50) static TCHAR rgchOpenQuote[] = { _T('\"'), _T('\''), _T('('), _T('{'), _T('[') }; static TCHAR rgchCloseQuote[] = { _T('\"'), _T('\''), _T(')'), _T('}'), _T(']') }; #define MAXQUOTE (_tsizeof(rgchOpenQuote) / _tsizeof(rgchOpenQuote[0])) static TCHAR rgchDelim[] = { _T(' '), _T('\t'), _T(',') }; #define MAXDELIM (_tsizeof(rgchDelim) / _tsizeof(rgchDelim[0])) //extern LPSHF Lpshf; int CPCopyString( PTSTR * lplps, PTSTR lpT, TCHAR chEscape, BOOL fQuote ) /*++ Routine Description: Scan and copy an optionally quoted C-style string. If the first character is a quote, a matching quote will terminate the string, otherwise the scanning will stop at the first whitespace encountered. The target string will be null terminated if any characters are copied. Arguments: lplps - Supplies a pointer to a pointer to the source string lpt - Supplies a pointer to the target string chEscape - Supplies the escape character (typically '\\') fQuote - Supplies a flag indicating whether the first character is a quote Return Value: The number of characters copied into lpt[]. If an error occurs, -1 is returned. --*/ { PTSTR lps = *lplps; PTSTR lpt = lpT; int i; int n; int err = 0; TCHAR cQuote = _T('\0'); if (fQuote) { if (*lps) { cQuote = *lps++; } } while (!err) { if (*lps == 0) { if (fQuote) { err = 1; } else { *lpt = _T('\0'); } break; } else if (fQuote && *lps == cQuote) { *lpt = _T('\0'); // eat the quote lps++; break; } else if (!fQuote && (!*lps || *lps == _T(' ') || *lps == _T('\t') || *lps == _T('\r') || *lps == _T('\n'))) { *lpt = _T('\0'); break; } else if (*lps != chEscape) { *lpt++ = *lps++; } else { switch (*++lps) { case 0: err = 1; --lps; break; default: // any char - usually escape or quote *lpt++ = *lps; break; case _T('b'): // backspace *lpt++ = _T('\b'); break; case _T('f'): // formfeed *lpt++ = _T('\f'); break; case _T('n'): // newline *lpt++ = _T('\n'); break; case _T('r'): // return *lpt++ = _T('\r'); break; case _T('s'): // space *lpt++ = _T(' '); break; case _T('t'): // tab *lpt++ = _T('\t'); break; case _T('0'): // octal escape for (n = 0, i = 0; i < 3; i++) { ++lps; if (*lps < _T('0') || *lps > _T('7')) { --lps; break; } n = (n<<3) + *lps - _T('0'); } *lpt++ = (UCHAR)(n & 0xff); break; } lps++; // skip char from switch } } // while if (err) { return -1; } else { *lplps = lps; return (int) (lpt - lpT); } } BOOL CPFormatMemory( LPCH lpchTarget, DWORD cchTarget, LPBYTE lpbSource, DWORD cBits, FMTTYPE fmtType, DWORD radix ) /*++ Routine Description: CPFormatMemory. formats a value by template Arguments: lpchTarget - Destination buffer. cchTarget - Size of destination buffer. lpbSource - Data to be formatted. cBits - Number of bits in the data. fmtType - Determines how the data will be treated?? UINT, float, real, ... radix - Radix to use when formatting. Return Value: TRUE - Success FALSE - Bad things happened --*/ { LONG64 l; long cb; ULONG64 ul = 0; TCHAR rgch[512] = {0}; Assert (radix == 2 || radix == 8 || radix == 10 || radix == 16 || (fmtType & fmtBasis) == fmtAscii || (fmtType & fmtBasis) == fmtUnicode); Assert (cBits != 0); Assert (cchTarget <= _tsizeof(rgch)); switch (fmtType & fmtBasis) { // // Format from memory bytes into an integer format number // case fmtInt: if (radix == 10) { switch( (cBits + 7)/8 ) { case 1: l = *(signed char *)lpbSource; if (fmtType & fmtZeroPad) { _stprintf(rgch, _T("%0*I64d"), cchTarget-1, l); } else if (fmtType & fmtSpacePad) { _stprintf(rgch, _T("% *I64d"), cchTarget-1, l); } else { _stprintf(rgch, _T("% I64d"), l); } break; case 2: l = *(short *)lpbSource; if (fmtType & fmtZeroPad) { _stprintf(rgch, _T("%0*I64d"), cchTarget-1, l); } else if (fmtType & fmtSpacePad) { _stprintf(rgch, _T("% *I64d"), cchTarget-1, l); } else { _stprintf(rgch, _T("% I64d"), l); } break; case 4: l = *(long *)lpbSource; if (fmtType & fmtZeroPad) { _stprintf(rgch, _T("%0*I64d"), cchTarget-1, l); } else if (fmtType & fmtSpacePad) { _stprintf(rgch, _T("% *I64d"), cchTarget-1, l); } else { _stprintf(rgch, _T("% I64d"), l); } break; case 8: l = *(LONG64 *)lpbSource; if (fmtType & fmtZeroPad) { _stprintf(rgch, _T("%0*I64d"), cchTarget-1, l); } else if (fmtType & fmtSpacePad) { _stprintf(rgch, _T("% *I64d"), cchTarget-1, l); } else { _stprintf(rgch, _T("% I64d"), l); } break; default: return FALSE; // Bad format } if (_tcslen(rgch) >= cchTarget) { return FALSE; // Overrun } _tcscpy(lpchTarget, rgch); break; } // // then we should handle this as UInt // case fmtUInt: cb = (cBits + 7)/8; switch( cb ) { case 1: ul = *(BYTE *) lpbSource; break; case 2: ul = *(USHORT *) lpbSource; break; case 4: ul = *(ULONG *) lpbSource; break; // // MBH - bugbug - CENTAUR bug; // putting contents of instead of address of structure // for return value in a0. // case 8: ul = *(ULONG64 *) lpbSource; break; default: if (radix != 16 || (fmtType & fmtZeroPad) == 0) { return FALSE; // Bad format } } if (fmtType & fmtZeroPad) { switch (radix) { case 8: _stprintf(rgch, _T("%0*.*I64o"), cchTarget-1, cchTarget-1, ul); break; case 10: _stprintf(rgch, _T("%0*.*I64u"), cchTarget-1, cchTarget-1, ul); break; case 16: if (cb <= 8) { _stprintf(rgch, _T("%0*.*I64x"), cchTarget-1, cchTarget-1, ul); } else { // handle any size: // NOTENOTE a-kentf this is dependent on byte order for (l = 0; l < cb; l++) { _stprintf(rgch+l+l, _T("%02.2x"), lpbSource[cb - l - 1]); } //_stprintf(rgch, _T("%0*.*x"), cchTarget-1, cchTarget-1, ul); } break; } } else if (fmtType & fmtSpacePad) { switch (radix) { case 8: _stprintf(rgch, _T("% *.*I64o"), cchTarget-1, cchTarget-1, ul); break; case 10: _stprintf(rgch, _T("% *.*I64u"), cchTarget-1, cchTarget-1, ul); break; case 16: if (cb <= 8) { _stprintf(rgch, _T("% *.*I64x"), cchTarget-1, cchTarget-1, ul); } else { // handle any size: // NOTENOTE a-kentf this is dependent on byte order /*for (l = 0; l < cb; l++) { _stprintf(rgch+l+l, _T("% 2.2x"), lpbSource[cb - l - 1]); }*/ _stprintf(rgch, _T("% *.*I64x"), cchTarget-1, cchTarget-1, ul); } break; } } else { switch (radix) { case 8: _stprintf(rgch, _T("%I64o"), ul); break; case 10: _stprintf(rgch, _T("%I64u"), ul); break; case 16: _stprintf(rgch, _T("%I64x"), ul); break; } } if (_tcslen(rgch) >= cchTarget) { return FALSE; // Overrun } _tcscpy(lpchTarget, rgch); break; case fmtAscii: if ( cBits != 8 ) { return FALSE; // Bad format } lpchTarget[0] = *(BYTE *) lpbSource; if ((lpchTarget[0] < _T(' ')) || (lpchTarget[0] > 0x7e)) { lpchTarget[0] = _T('.'); } lpchTarget[1] = 0; return TRUE; // success case fmtUnicode: if (cBits != 16) { return FALSE; // Bad format } // This 2 should really be MB_CUR_MAX but that caused // a compatibility problem with Win2K as its definition // changed to a function. Assert(2 <= cchTarget); if ((wctomb(lpchTarget, *(LPWCH)lpbSource) == -1) || (lpchTarget[0] < _T(' ')) || (lpchTarget[0] > 0x7e)) { lpchTarget[0] = _T('.'); } lpchTarget[1] = 0; return TRUE; // success case fmtFloat: switch ( cBits ) { case 4*8: _stprintf(rgch, _T("% 12.6e"),*((float *) lpbSource)); break; case 8*8: // _stprintf(rgch, _T("% 17.11le"), *((double *) lpbSource)); _stprintf(rgch, _T("% 21.14le"), *((double *) lpbSource)); break; case 10*8: if (_uldtoa((_ULDOUBLE *)lpbSource, 25, rgch) == NULL) { return FALSE; // Bad format } break; case 16*8: // v-vadimp this is an IA64 float - may have to rethink the format here // what we are getting here is really FLOAT128 if (_uldtoa((_ULDOUBLE *)(lpbSource), 30, rgch) == NULL) { return FALSE; // Bad format } break; default: return FALSE; // Bad format } if (_tcslen(rgch) >= cchTarget) { return FALSE; // Overrun } _tcsncpy(lpchTarget, rgch, cchTarget-1); lpchTarget[cchTarget-1] = 0; return TRUE; // success case fmtBit: { WORD i,j,shift=0; //shift will allow for a blank after each 8 bits for (i=0;i<(cBits+7)/8;i++) { for(j=0;j<8;j++) { if((lpbSource[i]>> (7-j)) & 0x1) { rgch[i*8+j+shift]=_T('1'); } else { rgch[i*8+j+shift]=_T('0'); } } rgch[(i+1)*8+shift]=_T(' '); shift++; } rgch[cBits+shift-1]=_T('\0'); if (_tcslen(rgch) >= cchTarget) { return FALSE; // Overrun } _tcscpy(lpchTarget,rgch); } return TRUE; // success default: return FALSE; // Bad format } return TRUE; // success } /* CPFormatMemory() */