2020-09-30 16:53:55 +02:00

1110 lines
27 KiB
C

//==========================================================================;
//
// Copyright (c) 1991-1995 Microsoft Corporation
//
// You have a royalty-free right to use, modify, reproduce and
// distribute the Sample Files (and/or any modified version) in
// any way you find useful, provided that you agree that
// Microsoft has no warranty obligations or liability for any
// Sample Application Files which are modified.
//
//--------------------------------------------------------------------------;
//
// debug.c
//
// Description:
// This file contains code yanked from several places to provide debug
// support that works in win 16 and win 32.
//
// History:
// 11/23/92 cjp [curtisp]
//
//==========================================================================;
#ifdef DEBUG
#include <windows.h>
#include <windowsx.h>
#include <mmsystem.h>
#include <stdarg.h>
#include "debug.h"
#ifdef WIN32
#define BCODE
#else
#define BCODE __based(__segname("_CODE"))
#endif // End #ifdef WIN32
#ifdef WIN32
#define GlobalSmartPageLock(a) (TRUE)
#endif // End #ifdef WIN32
#define WSPRINTF_LIMIT 1024
typedef struct tagLOG
{
LPTSTR lpszQueue; // TCHAR Representation
UINT cchBuffer; // Size of Log in TCHAR's
UINT idxRead; // Read index
UINT idxWrite; // Write index
} LOG, FAR *LPLOG;
#define LOG_INCIDX(pl,x) ((++x >= pl->cchBuffer) ? x = 0 : x)
void FAR CDECL DbgVPrintF (LPTSTR szFmt, va_list va);
BOOL NEAR PASCAL LogInit (LPLOG lpLog, UINT ckBuffer);
void NEAR PASCAL LogWrite (LPLOG lpLog, LPTSTR lpstrEvent);
BOOL NEAR PASCAL LogRead (LPLOG lpLog, LPTSTR lpstrBuffer, UINT cchBuffer);
#ifdef ISRDEBUG
int wivsprintf (LPTSTR lpOut, LPCTSTR lpFmt, VOID FAR* lpParms) ;
LPCTSTR NEAR PASCAL SP_GetFmtValue (LPCTSTR lpch, UINT * lpw) ;
UINT NEAR PASCAL SP_PutNumber (LPTSTR lpb, DWORD n, UINT limit, UINT radix, UINT icase) ;
VOID NEAR PASCAL SP_Reverse (LPTSTR lpFirst, LPTSTR lpLast) ;
UINT NEAR PASCAL ilstrlen (LPTSTR lpstr) ;
VOID NEAR PASCAL ilstrcat (LPTSTR lpstrDest, LPTSTR lpstrSrc) ;
#endif
//
// Use interruptable versions of functions
//
#ifdef ISRDEBUG
#define wvsprintf wivsprintf
#define lstrcat ilstrcat
#define lstrlen ilstrlen
#endif
//
//
//
BOOL __gfDbgEnabled = TRUE; // master enable
UINT __guDbgLevel = 0; // current debug level
BOOL __gfLogging = 0; // Are we logging as well?
HWND ghWndCB = (HWND)NULL;
LOG gLog;
WORD wDebugLevel = 0;
//************************************************************************
//**
//** WinAssert();
//**
//** DESCRIPTION:
//**
//**
//** ARGUMENTS:
//** LPSTR lpstrExp
//** LPSTR lpstrFile
//** DWORD dwLine
//**
//** RETURNS:
//** void
//**
//** HISTORY:
//**
//************************************************************************
VOID WINAPI WinAssert(
LPSTR lpstrExp,
LPSTR lpstrFile,
DWORD dwLine)
{
static TCHAR szWork[256];
static TCHAR BCODE szFormat[] =
TEXT ("Assertion failed!\n\nFile:\t%s\nLine:\t%lu\n\n[%s]");
static TCHAR BCODE szOops[] =
DEBUG_MODULE_NAME TEXT (" is confused");
// Use regular wsprintf here; assert's can't be at interrupt time
// anyway.
//
#ifdef UNICODE
static TCHAR szFile[256];
static TCHAR szMsg[256];
// Convert File to UNICODE
INT cLen = lstrlenA (lpstrFile);
if (cLen >= 255)
cLen = 255;
MultiByteToWideChar (CP_ACP, MB_PRECOMPOSED,
lpstrFile, cLen, szFile, 256);
szFile[cLen] = 0;
// Convert Message to UNICODE
cLen = lstrlenA (lpstrExp);
if (cLen >= 255)
cLen = 255;
MultiByteToWideChar (CP_ACP, MB_PRECOMPOSED,
lpstrExp, cLen, szMsg, 256);
szMsg[cLen] = 0;
// Create Assert String
wsprintf (szWork, szFormat, szFile, dwLine, szMsg);
#else
wsprintf (szWork, szFormat, lpstrFile, dwLine, lpstrExp);
#endif
if (IDCANCEL == MessageBox(NULL, szWork, szOops, MB_OKCANCEL|MB_ICONEXCLAMATION))
DebugBreak();
}
//************************************************************************
//**
//** DbgVPrintF();
//**
//** DESCRIPTION:
//**
//**
//** ARGUMENTS:
//** LPSTR szFmt
//** LPSTR va
//**
//** RETURNS:
//** void
//**
//** HISTORY:
//**
//************************************************************************
void FAR CDECL DbgVPrintF(
LPTSTR szFmt,
va_list va)
{
TCHAR ach[DEBUG_MAX_LINE_LEN];
BOOL fDebugBreak = FALSE;
BOOL fPrefix = TRUE;
BOOL fCRLF = TRUE;
ach[0] = TEXT ('\0');
for (;;)
{
switch(*szFmt)
{
case '!':
fDebugBreak = TRUE;
szFmt++;
continue;
case '`':
fPrefix = FALSE;
szFmt++;
continue;
case '~':
fCRLF = FALSE;
szFmt++;
continue;
}
break;
}
if (fDebugBreak)
{
ach[0] = TEXT ('\007');
ach[1] = TEXT ('\0');
}
if (fPrefix)
lstrcat (ach, DEBUG_MODULE_NAME TEXT (": "));
wvsprintf (ach + lstrlen(ach), szFmt, va);
if (fCRLF)
lstrcat (ach, TEXT ("\r\n") );
if (__gfLogging)
{
LogWrite (&gLog, ach);
if (ghWndCB)
PostMessage (ghWndCB, WM_DEBUGUPDATE, 0, 0);
}
OutputDebugString (ach);
if (fDebugBreak)
DebugBreak();
} //** DbgVPrintF()
//************************************************************************
//**
//** dprintf();
//**
//** DESCRIPTION:
//** dprintf() is called by the DPF macro if DEBUG is defined at compile
//** time.
//**
//** The messages will be send to COM1: like any debug message. To
//** enable debug output, add the following to WIN.INI :
//**
//** [debug]
//** ICSAMPLE=1
//**
//**
//** ARGUMENTS:
//** UINT uDbgLevel
//** LPCSTR szFmt
//** ...
//**
//** RETURNS:
//** void
//**
//** HISTORY:
//** 06/12/93 [t-kyleb]
//**
//************************************************************************
void FAR CDECL dprintf(
UINT uDbgLevel,
LPTSTR szFmt,
...)
{
va_list va;
if (!__gfDbgEnabled || (__guDbgLevel < uDbgLevel))
return;
va_start (va, szFmt);
DbgVPrintF (szFmt, va);
va_end (va);
} //** dprintf()
//************************************************************************
//**
//** DbgEnable();
//**
//** DESCRIPTION:
//**
//**
//** ARGUMENTS:
//** BOOL fEnable
//**
//** RETURNS:
//** BOOL
//**
//** HISTORY:
//** 06/12/93 [t-kyleb]
//**
//************************************************************************
BOOL WINAPI DbgEnable(
BOOL fEnable)
{
BOOL fOldState;
fOldState = __gfDbgEnabled;
__gfDbgEnabled = fEnable;
return (fOldState);
} //** DbgEnable()
//************************************************************************
//**
//** DbgSetLevel();
//**
//** DESCRIPTION:
//**
//**
//** ARGUMENTS:
//** UINT uLevel
//**
//** RETURNS:
//** UINT
//**
//** HISTORY:
//** 06/12/93 [t-kyleb]
//**
//************************************************************************
UINT WINAPI DbgSetLevel(
UINT uLevel)
{
UINT uOldLevel;
uOldLevel = __guDbgLevel;
__guDbgLevel = wDebugLevel = uLevel;
return (uOldLevel);
} //** DbgSetLevel()
//--------------------------------------------------------------------------;
//
// UINT DbgInitialize(void)
//
// Description:
//
//
// Arguments:
//
// Return (UINT):
//
//
// History:
// 11/24/92 cjp [curtisp]
//
//--------------------------------------------------------------------------;
UINT WINAPI DbgInitialize(BOOL fEnable)
{
TCHAR szTemp[64];
LPTSTR pstr;
UINT uLevel;
UINT uLogMem;
GetProfileString (DEBUG_SECTION, DEBUG_MODULE_NAME, TEXT (""), szTemp, sizeof(szTemp));
pstr = szTemp;
uLevel = 0;
while (*pstr >= TEXT ('0') && *pstr <= TEXT ('9'))
{
uLevel = uLevel*10 + (UINT)(*pstr - TEXT ('0'));
pstr++;
}
__gfLogging = FALSE;
if (*pstr == TEXT (','))
{
pstr++;
uLogMem = 0;
while (*pstr >= TEXT ('0') && *pstr <= TEXT ('9'))
{
uLogMem = uLogMem*10 + (UINT)(*pstr - TEXT ('0'));
pstr++;
}
if (0 == uLogMem)
uLogMem = K_DEFAULT_LOGMEM;
if (uLogMem > K_MAX_LOGMEM)
uLogMem = K_MAX_LOGMEM;
__gfLogging = TRUE;
}
if (__gfLogging)
__gfLogging = LogInit(&gLog, uLogMem);
DbgSetLevel (GetProfileInt(DEBUG_SECTION, DEBUG_MODULE_NAME, 0));
DbgEnable (fEnable);
return (__guDbgLevel);
} // DbgInitialize()
void WINAPI DbgRegisterCallback (HWND hWnd)
{
ghWndCB = hWnd;
}
BOOL WINAPI DbgGetNextLogEntry (LPTSTR lpstrBuffer, UINT cchBuffer)
{
if (!__gfLogging)
return FALSE;
return LogRead (&gLog, lpstrBuffer, cchBuffer);
}
BOOL NEAR PASCAL LogInit (LPLOG lpLog, UINT ckMem)
{
DWORD cbMem = 1024L * ckMem;
LPTSTR lpszQueue = GlobalAllocPtr (GPTR, cbMem);
if (NULL == lpszQueue)
return FALSE;
if (! GlobalSmartPageLock (HIWORD(lpszQueue)))
{
GlobalFreePtr (lpszQueue);
return FALSE;
}
lpLog->lpszQueue = (LPTSTR)lpszQueue;
lpLog->cchBuffer = (UINT)cbMem/sizeof(TCHAR);
lpLog->idxRead = 0;
lpLog->idxWrite = 0;
return TRUE;
}
void NEAR PASCAL LogWrite (LPLOG lpLog, LPTSTR lpstrEvent)
{
if (!*lpstrEvent)
return;
while (*lpstrEvent)
{
lpLog->lpszQueue[lpLog->idxWrite] = *lpstrEvent++;
LOG_INCIDX (lpLog,lpLog->idxWrite);
}
lpLog->idxRead = lpLog->idxWrite;
while (lpLog->lpszQueue[lpLog->idxRead])
{
lpLog->lpszQueue[lpLog->idxRead] = TEXT ('\0');
LOG_INCIDX(lpLog,lpLog->idxRead);
}
LOG_INCIDX(lpLog,lpLog->idxRead);
LOG_INCIDX(lpLog,lpLog->idxWrite);
}
BOOL NEAR PASCAL LogRead(LPLOG lpLog, LPTSTR lpstrBuffer, UINT cchBuffer)
{
TCHAR ch;
UINT idx;
if (!cchBuffer)
return FALSE;
idx = lpLog->idxRead;
while (TEXT ('\0') == lpLog->lpszQueue[idx])
{
LOG_INCIDX(lpLog,idx);
if (idx == lpLog->idxRead)
return FALSE;
}
cchBuffer--;
while (0 != (ch = lpLog->lpszQueue[idx]))
{
if (cchBuffer)
{
*lpstrBuffer++ = ch;
cchBuffer--;
}
lpLog->lpszQueue[idx] = TEXT ('\0');
LOG_INCIDX(lpLog,idx);
}
*lpstrBuffer = TEXT ('\0');
LOG_INCIDX (lpLog,idx);
lpLog->idxRead = idx;
return TRUE;
}
//--------------------------------------------------------------------------;
//
// The rest of the code is only needed if we're in Win16 and need to be
// interrupt callable.
//
//--------------------------------------------------------------------------;
#ifdef ISRDEBUG
#define OUT(ch) if (--cchLimit) *lpOut++=(ch); else goto error_Out
//************************************************************************
//**
//** wivsprintf();
//**
//** DESCRIPTION:
//** Interrupt callable version of wvsprintf()
//**
//**
//** ARGUMENTS:
//** LPTSTR lpOut - Buffer to format into.
//** LPCTSTR lpFmt - Format string.
//** VOID FAR* lpParms - Points to the first of args
//** described by lpFmt.
//**
//** RETURNS:
//** int - Number of characters stored.
//**
//** HISTORY:
//** 3/28/93 jfg [jimge]
//**
//************************************************************************
int wivsprintf(
LPTSTR lpOut,
LPCTSTR lpFmt,
VOID FAR* lpParms)
{
int left ;
TCHAR prefix ;
int width ;
int prec ;
TCHAR fillch ;
int size ;
int sign ;
int radix ;
int upper ;
int cchLimit = WSPRINTF_LIMIT;
int cch ;
LPTSTR lpT ;
union
{
long l ;
unsigned long ul ;
TCHAR sz[sizeof(long)] ;
} val;
while (*lpFmt)
{
if (*lpFmt==TEXT ('%'))
{
//
// Read the format flags.
//
left = 0 ;
prefix = 0 ;
while (*++lpFmt)
{
if (*lpFmt==TEXT ('-'))
{
left++;
}
else if (*lpFmt==TEXT ('#'))
{
prefix++;
}
else
{
break;
}
}
//
// Find the fill character (either '0' or ' ')
//
if (*lpFmt==TEXT ('0'))
{
fillch = TEXT ('0') ;
lpFmt++ ;
}
else
{
fillch = TEXT (' ') ;
}
//
// Now parse [width[.precision]]
//
lpFmt = SP_GetFmtValue(lpFmt,&cch);
width = cch;
if (*lpFmt==TEXT ('.'))
{
lpFmt = SP_GetFmtValue(++lpFmt,&cch);
prec = cch;
}
else
{
prec = (UINT)-1 ;
}
//
// Get the operand size modifier
//
if (*lpFmt==TEXT ('l'))
{
size = 1 ;
lpFmt++ ;
}
else
{
size = 0 ;
if (*lpFmt==TEXT ('h'))
{
lpFmt++ ;
}
}
//
// We've gotten all the modifier; now format the output
// based on the type (which should now be pointed at
// by lpFmt).
//
upper = 0 ;
sign = 0 ;
radix = 10 ;
switch (*lpFmt)
{
case 0:
goto error_Out ;
case TEXT ('i') :
case TEXT ('d') :
sign++ ;
case TEXT ('u'):
//
// Don't show a prefix for decimal formats
//
prefix=0 ;
do_Numeric:
//
// Special cases to act like MSC v5.10
//
if (left || prec>=0)
{
fillch = TEXT (' ');
}
//
// Get value from parm list into val union
//
if (size)
{
val.l=*((long far *)lpParms)++;
}
else
{
if (sign)
{
val.l=(long)*((short far *)lpParms)++;
}
else
{
val.ul=(unsigned long)*((unsigned far *)lpParms)++;
}
}
//
// Save sign of val.l in sign and set val.l positive.
//
if (sign && val.l<0L)
{
val.l=-val.l;
}
else
{
sign=0;
}
//
// Save start of output stream for later reverse
//
lpT = lpOut;
//
// Blast the number backwards into the user buffer
//
cch = SP_PutNumber(lpOut,val.l,cchLimit,radix,upper) ;
if (!(cchLimit-=cch))
goto error_Out ;
lpOut += cch ;
width -= cch ;
prec -= cch ;
if (prec>0)
{
width -= prec ;
}
//
// Fill in up to precision
//
while (prec-- > 0)
{
OUT(TEXT ('0')) ;
}
if (width>0 && !left)
{
//
// If we're filling with spaces, put sign first
//
if (fillch != '0')
{
if (sign)
{
sign = 0 ;
OUT(TEXT ('-')) ;
width-- ;
}
if (prefix)
{
OUT(prefix) ;
OUT(TEXT ('0')) ;
prefix = 0 ;
}
}
if (sign)
{
width-- ;
}
//
// Now fill to width
//
while (width-- > 0)
{
OUT(fillch) ;
}
//
// Still have a sign?
//
if (sign)
{
OUT(TEXT ('-')) ;
}
if (prefix)
{
OUT(prefix) ;
OUT(TEXT ('0')) ;
}
//
// Now reverse the string in place
//
SP_Reverse(lpT,lpOut-1);
}
else
{
//
// Add the sign character
//
if (sign)
{
OUT(TEXT ('-')) ;
width-- ;
}
if (prefix)
{
OUT(prefix);
OUT(TEXT ('0'));
}
//
// Now reverse the string in place
//
SP_Reverse(lpT,lpOut-1);
//
// Pad to the right of the string in case left aligned
//
while (width-- > 0)
{
OUT(fillch) ;
}
}
break ;
case TEXT ('X'):
upper++ ;
//
// Falling through...
//
case TEXT ('x'):
radix=16 ;
if (prefix)
{
prefix = upper ? TEXT ('X') : TEXT ('x') ;
}
goto do_Numeric ;
case TEXT ('c'):
//
// Save as one character string and join common code
//
val.sz[0] = *((TCHAR far*)lpParms) ;
val.sz[1] = 0 ;
lpT = val.sz ;
cch = 1 ;
// Note: this may need to be fixed for UNICODE
(BYTE far*)lpParms += sizeof(WORD) ;
goto put_String ;
case 's':
lpT = *((LPTSTR FAR *)lpParms)++ ;
cch = ilstrlen(lpT) ;
put_String:
if (prec>=0 && cch>prec)
{
cch = prec ;
}
width -= cch ;
if (left)
{
while (cch--)
{
OUT(*lpT++) ;
}
while (width-->0)
{
OUT(fillch) ;
}
}
else
{
while (width-- > 0)
{
OUT(fillch) ;
}
while (cch--)
{
OUT(*lpT++) ;
}
}
break ;
default:
//
// An unsupported type character was given. We just
// print the character and go on.
//
OUT(*lpFmt) ;
break ;
} // switch(*lpfmt)
} // if (*lpfmt == '%')
else
{
//
// Normal not-format character
//
OUT(*lpFmt) ;
}
lpFmt++ ;
} // while (*lpFmt)
error_Out:
*lpOut = 0 ;
return WSPRINTF_LIMIT-cchLimit ;
} //** wivsprintf()
//************************************************************************
//**
//** SP_GetFmtValue();
//**
//** DESCRIPTION:
//** Parse a decimal integer forming part of a format string.
//**
//**
//** ARGUMENTS:
//** LPCSTR lpch - Points to the string to parse.
//** LPWORD lpw - Points to a word where the value will be
//** returned.
//**
//** RETURNS:
//** LPCSTR - Pointer of first character past the format value.
//**
//** HISTORY:
//** 3/28/93 jfg [jimge]
//**
//************************************************************************
LPCTSTR NEAR PASCAL SP_GetFmtValue(
LPCTSTR lpch,
UINT * lpw)
{
UINT i = 0 ;
while (*lpch>=TEXT ('0') && *lpch<=TEXT ('9'))
{
i *= 10;
i += (UINT)(*lpch++-TEXT ('0'));
}
*lpw = i;
return(lpch);
} //** SP_GetFmtValue()
//************************************************************************
//**
//** SP_PutNumber();
//**
//** DESCRIPTION:
//** Formats the given number in the given radix into the buffer
//** *backwards*. The entire string will be reversed after printf
//** has added sign, prefix, etc. to it.
//**
//**
//** ARGUMENTS:
//** LPSTR lpb - Points to the output buffer.
//** DWORD n - Number to convert.
//** UINT limit - Maximum number of characters to store.
//** UINT radix - Base to format in.
//** UINT icase - Non-zero if the string should be upper case (hex).
//**
//** RETURNS:
//** UINT - Number of characters output.
//**
//** HISTORY:
//**
//************************************************************************
UINT NEAR PASCAL SP_PutNumber(
LPTSTR lpb,
DWORD n,
UINT limit,
UINT radix,
UINT icase)
{
TCHAR bTemp;
UINT cchStored = 0;
//
// Set icase to the offset to add to the character if it
// represents a value > 10
//
icase = (icase ? TEXT ('A') : TEXT ('a')) - TEXT ('0') - 10 ;
while (limit--)
{
bTemp = TEXT ('0') + (TCHAR)(n%radix);
if (bTemp > TEXT ('9'))
{
bTemp += icase ;
}
*lpb++ = bTemp;
++cchStored;
n /= radix;
if (n == 0)
{
break ;
}
}
return cchStored ;
} //** SP_PutNumber()
//************************************************************************
//**
//** SP_Reverse();
//**
//** DESCRIPTION:
//** Reverse string in place.
//**
//** ARGUMENTS:
//** LPSTR pFirst
//** LPSTR pLast
//**
//** RETURNS:
//** VOID
//**
//** HISTORY:
//**
//************************************************************************
VOID NEAR PASCAL SP_Reverse(
LPTSTR pFirst,
LPTSTR pLast)
{
UINT uSwaps = (pLast - pFirst + sizeof(TCHAR)) / (2 * sizeof(TCHAR));
TCHAR bTemp;
while (uSwaps--)
{
bTemp = *pFirst;
*pFirst = *pLast;
*pLast = bTemp;
pFirst++, pLast--;
}
} //** SP_Reverse()
//************************************************************************
//**
//** ilstrlen();
//**
//** DESCRIPTION:
//** Interrupt callable version of strlen().
//**
//** ARGUMENTS:
//** LPSTR pstr
//**
//** RETURNS:
//** UINT
//**
//** HISTORY:
//**
//************************************************************************
UINT NEAR PASCAL ilstrlen(
LPTSTR pstr)
{
UINT cch = 0 ;
while (*pstr++)
++cch;
return(cch);
} //** ilstrlen()
//************************************************************************
//**
//** ilstrcat();
//**
//** DESCRIPTION:
//** Interrupt callable version of lstrcat().
//**
//** ARGUMENTS:
//** LPSTR pstrDest
//** LPSTR pstrSrc
//**
//** RETURNS:
//** VOID
//**
//** HISTORY:
//**
//************************************************************************
VOID NEAR PASCAL ilstrcat(
LPTSTR pstrDest,
LPTSTR pstrSrc)
{
while (*pstrDest)
pstrDest++;
while (*pstrDest++ = *pstrSrc++)
;
} //** ilstrcat()
#endif // #ifdef ISRDEBUG
#endif // #ifdef DEBUG