|
|
/***
*printf.h - print formatted * * Copyright (c) 1985-1991, Microsoft Corporation. All rights reserved. * *Purpose: * defines w4*printf() - print formatted data * defines w4v*printf() - print formatted output, get data from an * argument ptr instead of explicit args. * *Revision History: * 09-02-83 RN original sprintf * 06-17-85 TC rewrote to use new varargs macros, and to be vsprintf * 04-13-87 JCR added const to declaration * 11-07-87 JCR Multi-thread support * 12-11-87 JCR Added "_LOAD_DS" to declaration * 05-27-88 PHG Merged DLL and normal versions * 06-13-88 JCR Fake _iob entry is now static so that other routines * can assume _iob entries are in DGROUP. * 08-25-88 GJF Define MAXSTR to be INT_MAX (from LIMITS.H). * 06-06-89 JCR 386 mthread support * 08-18-89 GJF Clean up, now specific to OS/2 2.0 (i.e., 386 flat * model). Also fixed copyright and indents. * 02-16-90 GJF Fixed copyright * *******************************************************************************/
#include <stdarg.h>
#include <limits.h>
#include <windows.h>
#include "w4io.h"
#if defined(_W4PRINTF_)
static long fh; // extern long GetStdHandle(long);
// extern void WriteFile(long fh, char FAR *s, long cch, long FAR * pcchret, long);
# define _PRINTF_
#elif defined(_W4DPRINTF_)
# define _pwritechar _dwritechar
# define _pflushbuf _dflushbuf
# define w4printf w4dprintf
# define w4vprintf w4vdprintf
# define _PRINTF_
#elif defined(_W4SPRINTF_)
# define _pwritechar _swritechar
# define w4printf w4sprintf
# define w4vprintf w4vsprintf
#elif defined(_W4WCSPRINTF_)
# define _TCHAR_ wchar_t
# define _PBUF_ pwcbuf
# define _PSTART_ pwcstart
# define w4printf w4wcsprintf
# define w4vprintf w4vwcsprintf
# define _pwritechar _wwritechar
#else
# error configuration problem
#endif
#ifndef _TCHAR_
# define _TCHAR_ char
# define _PBUF_ pchbuf
# define _PSTART_ pchstart
#endif
#ifdef _PRINTF_
# ifndef FLAT
# define OutputDebugStringA OutputDebugString
# endif
int _cdecl _pflushbuf(struct w4io FAR *f); # define SPR(a)
# define MAXSTR 128
#else
# define SPR(a) a,
# define MAXSTR INT_MAX
#endif
void _cdecl _pwritechar(int ch, int num, struct w4io FAR *f, int FAR *pcchwritten); int _cdecl w4vprintf(SPR(_TCHAR_ FAR *string) const char FAR *format, va_list arglist);
#ifdef __cplusplus
extern "C" { #endif
DWORD FAR PASCAL CallProc32W(DWORD dw1, DWORD dw2, DWORD dw3, LPVOID pfn32, DWORD dwPtrTranslate, DWORD dwArgCount); #ifdef __cplusplus
} #endif
LPVOID lpThkCallOutputFunctionsProc;
/***
*int w4printf(format, ...) - print formatted data * *Purpose: * Prints formatted data using the format string to * format data and getting as many arguments as called for * Sets up a w4io so file i/o operations can be used. * w4iooutput does the real work here * *Entry: * char *format - format string to control data format/number * of arguments followed by list of arguments, number and type * controlled by format string * *Exit: * returns number of characters written * *Exceptions: * *******************************************************************************/
int _cdecl w4printf(SPR(_TCHAR_ FAR *string) const char FAR *format, ...) /*
* 'PRINT', 'F'ormatted */ { va_list arglist;
va_start(arglist, format); return(w4vprintf(SPR(string) format, arglist)); }
/***
*int w4vprintf(format, arglist) - print formatted data from arg ptr * *Purpose: * Prints formatted data, but gets data from an argument pointer. * Sets up a w4io so file i/o operations can be used, make string look * like a huge buffer to it, but _flsbuf will refuse to flush it if it * fills up. Appends '\0' to make it a true string. * * Multi-thread: (1) Since there is no stream, this routine must never try * to get the stream lock (i.e., there is no stream lock either). (2) * Also, since there is only one staticly allocated 'fake' iob, we must * lock/unlock to prevent collisions. * *Entry: * char *format - format string, describes format of data * va_list arglist - varargs argument pointer * *Exit: * returns number of characters written * *Exceptions: * *******************************************************************************/
int _cdecl w4vprintf(SPR(_TCHAR_ FAR *string) const char FAR *format, va_list arglist) /*
* 'V'ariable argument 'PRINT', 'F'ormatted */ { struct w4io outfile; register int retval; #ifdef _PRINTF_
char string[MAXSTR + 1]; // leave room for null termination
#else
int dummy; #endif
#ifdef _W4PRINTF_
long ldummy;
if (fh == 0 || fh == -1) { ldummy = -11; // C7 bug workaround
if ((fh = (long)GetStdHandle(ldummy)) == 0 || fh == -1) { OutputDebugStringA("GetStdHandle in " __FILE__ " failed\n"); return(-1); } } #endif
outfile._PBUF_ = outfile._PSTART_ = string; outfile.cchleft = MAXSTR; outfile.writechar = _pwritechar;
retval = w4iooutput(&outfile, format, arglist);
#ifdef _PRINTF_
if (_pflushbuf(&outfile) == -1) { return(-1); } #else
_pwritechar('\0', 1, &outfile, &dummy); #endif
return(retval); }
void _cdecl _pwritechar(int ch, int num, struct w4io FAR *f, int FAR *pcchwritten) { //printf(" char: ch=%c, cnt=%d, cch=%d\n", ch, num, *pcchwritten);
while (num-- > 0) { #ifdef _PRINTF_
if (f->cchleft < 2 && _pflushbuf(f) == -1) { *pcchwritten = -1; return; } #endif
#ifdef _W4DPRINTF_
# ifndef FLAT
if (ch == '\n') { *f->_PBUF_++ = '\r'; f->cchleft--; (*pcchwritten)++; } # endif
#endif
*f->_PBUF_++ = (char) ch; f->cchleft--; (*pcchwritten)++; } }
#ifdef _PRINTF_
int _cdecl _pflushbuf(struct w4io FAR *f) { int cch;
if (cch = (int) (f->pchbuf - f->pchstart)) { #ifdef _W4DPRINTF_
*f->pchbuf = '\0'; // null terminate
if (lpThkCallOutputFunctionsProc == NULL) { OutputDebugStringA(f->pchstart); } else { // note casting and dummy arguments to match other uses of it (see interop.hxx)
CallProc32W((DWORD) f->pchstart, 0, 0, lpThkCallOutputFunctionsProc, 0x00000004, 3); // thunk it to olethk32.dll
} #else
long cchret;
//*f->pchbuf = '\0'; // null terminate
//printf("%d chars: \"%s\"\n", cch, f->pchstart);
WriteFile((HANDLE)fh, f->pchstart, cch, &cchret, 0); if (cch != cchret) { OutputDebugString("WriteFile in " __FILE__ " failed\n"); return(-1); } #endif
f->pchbuf -= cch; // reset pointer
f->cchleft += cch; // reset count
} return(0); } #endif // _PRINTF_
|