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

1251 lines
52 KiB
C

/*** MkMsg.c - Microsoft Make Messages Utility *******************************
*
* Copyright (c) 1986-1992, Microsoft Corporation. All Rights Reserved.
*
* Purpose:
* MkMsg.Exe is a utility that helps localize messages emitted by a utility by
* padding them and placing them in message segments.
* MkMsg also allows a programmer to associate symbols and Message Id to a
* message text. It does this by producing a C style header file with #defines
* or a Masm style include file with symbol definitions.
*
* Revision History:
* 02-Oct-1996 DS Remove special '\\' handling in RC output.
* 02-Feb-1994 HV Handle lines with only blansk.
* 25-Jan-1994 HV Add -err to generate external error file in the format:
* NUMBER\tMESSAGE
* 24-Jan-1994 HV Allow quoted message
* 21-Jan-1994 HV Nuke Lead Byte Table [[ ... ]], add !codepage directive
* 21-Jan-1994 HV Field separated by white spaces
* 19-Aug-1993 BW Add Lead Byte Table support
* 07-Apr-1993 BW Add -hex, -c options
* 13-Jul-1992 SB Add -msg option
* 16-Apr-1990 SB Add header
* 13-Apr-1990 SB Collated 6 different versions seperately SLM'ed on \\ODIN
* 19-Apr-1989 LN add "\xhh" recognition
* ??-Apr-1986 RN Created by Randy Nevin
*
* Syntax:
* MKMSG [-h cfile] [-x xcfile] [-inc afile] [-msg cfile] [-c cfile]
* [-asm srcfile [-def str] [-min|-max]] [-386] [-hex] txtfile
*
* Notes:
* The Utility takes as input a message file (a .txt file by convention) and
* produces an assembler source file. The lines in the message file have one
* of the formats which are instructions to MkMsg as follows -
*
* 1) "<<NMSG>>" -- use near message segment
* 2) "<<FMSG>>" -- use far message segment
* ----- Obsolete, see 8)
* 3) "[[XX, YY, ZZ]] - Use hex bytes XX, YY, ZZ as lead byte table.
* Lists are cumulative, empty list clears table.
* -----
* 4) "#Anything" -- comment line (ignored by mkmsg)
* 5) "" -- blank line (ignored by mkmsg)
* 6) "Handle<White space>Number<White space>MessageText" -- associate MessageText with
* message Id Number and the symbol Handle
* 7) "Number<White space>MessageText" -- associate MessageText with Id
* Id Number
* 8) "!codepage xxx" -- use this instead of [[ ... ]]
*
* Options:
* The options can be specified in any order. When no options are specified
* then the input is checked for syntactic validity.
*
*
* -h cfile: create a C style header file cfile with
* #define Handle IdNumber
* If -c is also defined, a declaration for __NMSG_TEXT
* and a mcros defining GET_MSG as __NMSG_TEXT is added.
* -msg cfile: create a C style header file cfile with
* array of struct having fields id and str
* -c cfile: create a file like -msg and add a definition
* of a retrieval function
* -x xcfile: create a C style header file xcfile with
* #define Handle MessageText
* -inc afile: create a MASM style include file afile with
* Handle = IdNumber
* -asm srcfile: create a MASM source file srcfile with standard segment
* definitions and messages in near segment (MSG) or far
* segment (FAR_MSG) depending on whether <<NMSG>> or
* <<FMSG>> is specified. The default is <<NMSG>>.
*
* If -min(-max) is specified then the minimum(maximum)
* 0-padding is calculated and placed in srcfile. This
* depends on the total length of the individual messages.
*
* If -def is specified the public symbol str gets declared.
* This is useful if the the object file produced from the
* MASM source produced is part of a library. The linker
* will link in that module from the library only if it
* resolves some external references.
*
* If -hex is specified and no Lead Byte Table is
* present, all MessageText is rendered in printf-style
* hexidecimal (e.g. "error" becomes
* x65\x72\x72\x6f\x72").
*
* if -hex is specified and a Lead Byte Table is present
* even if it is empty), then all Double Byte characters
* are rendering in hexidecimal.
*
* Future Directions:
* 1> Break up main() into smaller routines
* 2> Use mkmsg for handling messages
* 2> Allow use of spaces as separators, -- done. HV.
* 3> Allow specification of the separator
* 4> Provide -? and -help options.
*
*****************************************************************************/
// DEFINEs
// Standard INCLUDEs
#include <stdarg.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <ctype.h>
#include <locale.h>
#include <mbctype.h>
// Project INCLUDEs
// MACROs
// External PROTOTYPEs
// TYPEDEFs
typedef enum { FALSE = 0, TRUE = !0 } BOOL;
typedef struct status {
char *txt;
char *h;
char *rc;
char *x;
char *inc;
char *msg;
char *c;
char *asm;
char *def;
char *err;
int min;
int max;
int use32;
int hex;
} STATUS;
// Global PROTOTYPEs
void __cdecl Error(char *fmt, ...);
int __cdecl main(int argc, char **argv);
void parseCommandLine(unsigned argc, char **argv, STATUS *opt);
void msg_fputs (unsigned char * pch, int, FILE * fp);
int ReadLine (FILE * fp);
BOOL ParseLine(char **ppSymbol, char **ppNumber, char **ppMessage);
BOOL HandleDirectives(void);
void SetCodePage (const char *pszCodePage);
BOOL IsLeadByte(unsigned by);
// WARNING: the following CRT function is undocumented. This call can only
// be used in our internal product (like this one) only. Please
// contact CRT people for more information.
//void __cdecl __setmbctable(unsigned int);
//
// Lead Byte support
//
unsigned char bUseLeadByteTable = FALSE;
typedef struct
{
unsigned uCodePage; // Codepage number
unsigned byLead[12]; // Leadbyte ranges
} CPTABLE;
CPTABLE cpTable[] =
{
{932, {0x81, 0x9f, 0xe0, 0xfc, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00}},
{0, {0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00}} // trailing to end
};
CPTABLE * pCP = NULL;
// Local PROTOTYPEs
// External VARIABLEs
// Initialized VARIABLEs
int cError = 0;
char didnear = 0;
char didfar = 0;
char dir32[] = ".386\n";
char ex[] = "expected escape sequence: %s\n";
char f1[] = "FAR_HDR segment byte public \'FAR_MSG\'\nFAR_HDR ends\n";
char f2[] = "FAR_MSG segment byte public \'FAR_MSG\'\nFAR_MSG ends\n";
char f3[] = "FAR_PAD segment byte public \'FAR_MSG\'\nFAR_PAD ends\n";
char f4[] = "FAR_EPAD segment byte common \'FAR_MSG\'\nFAR_EPAD ends\n";
char f5[] = "FMGROUP group FAR_HDR,FAR_MSG,FAR_PAD,FAR_EPAD\n\n";
char n1[] = "HDR segment byte public \'MSG\'\nHDR ends\n";
char n2[] = "MSG segment byte public \'MSG\'\nMSG ends\n";
char n3[] = "PAD segment byte public \'MSG\'\nPAD ends\n";
char n4[] = "EPAD segment byte common \'MSG\'\nEPAD ends\n";
char n5[] = "DGROUP group HDR,MSG,PAD,EPAD\n\n";
char usage[] = "\nMicrosoft (R) Message Creation Utility Version %s"
"\nCopyright (c) Microsoft Corp %s. All rights reserved.\n"
"\nusage: MKMSG [-h cfile] [-rc rcfile] [-x xcfile] [-msg cfile]\n"
"\t[-c cfile] [-err errfile] [-inc afile]\n"
"\t[-asm srcfile [-def str] [-min|-max]] [-386] [-hex] txtfile\n";
char szVersionNo[] = "1.00.0011";
char szCopyRightYrs[] = "1986-1996";
// Local VARIABLEs
#define MSGSTYLE_COPY 0
#define MSGSTYLE_C_HEX 1
#define MSGSTYLE_ASM_BYTE 2
#define MSGSTYLE_ASM_TEXT 3
#define FALSE 0
#define TRUE (!FALSE)
#define INBUFSIZE 1024
//static char buf[INBUFSIZE]; // line buffer
char *buf; // The real buffer is in ReadLine()
//--------------------------------------------------------------------------
int __cdecl
main(int argc, char **argv)
{
FILE *f; // the input file
FILE *fh = NULL; // -h stream
FILE *frc = NULL; // -rc stream
FILE *fx = NULL; // -x stream
FILE *fc = NULL; // -c stream
FILE *finc = NULL; // -inc stream
FILE *fmsg = NULL; // -msg stream
FILE *fasm = NULL; // -asm stream
FILE *ferr = NULL; // -err stream
int asmstate = 0; // 0=nothing, 1=doing nmsg, 2=doing fmsg
int instring; // db "...
unsigned int nLineCur = 0; // Current line
unsigned int cchLine; // Chars in current line
char *p;
int npad = 0; // cumulative amount of near padding
int fpad = 0; // cumulative amount of far padding
int length;
double factor;
double result;
static STATUS opt; // Command line options
parseCommandLine(argc, argv, &opt);
if ((opt.def || opt.min || opt.max ) && !opt.asm) {
Error( "-def/-min/-max ignored; no -asm file\n" );
opt.min = opt.max = 0;
}
if (opt.min && opt.max) {
Error( "-min and -max are mutually exclusive; -min chosen\n" );
opt.max = 0;
}
if (!(f = fopen( opt.txt, "rb" ))) {
Error( "can't open txtfile %s for binary reading\n", opt.txt );
return( -1 );
}
if (opt.asm && !(fasm = fopen( opt.asm, "w" ))) {
Error( "can't open -asm file %s for writing\n", opt.asm );
return( -1 );
}
if (opt.h && !(fh = fopen( opt.h, "w" ))) {
Error( "can't open -h file %s for writing\n", opt.h );
return( -1 );
}
if (fh && opt.c) {
fprintf (fh, "char * __NMSG_TEXT (unsigned);\n\n");
}
if (opt.rc && !(frc = fopen(opt.rc, "w")))
{
Error("Can't open -rc file %s for writing\n", opt.rc);
return -1;
}
if (opt.rc) {
fprintf(frc, "STRINGTABLE\nBEGIN\n");
}
if (opt.x && !(fx = fopen( opt.x, "w" ))) {
Error( "can't open -x file %s for writing\n", opt.x );
return( -1 );
}
if (opt.inc && !(finc = fopen( opt.inc, "w" ))) {
Error( "can't open -inc file %s for writing\n", opt.inc );
return( -1 );
}
if (opt.msg && !(fmsg = fopen( opt.msg, "w" ))) {
Error( "can't open -msg file %s for writing\n", opt.msg );
return( -1 );
}
if (opt.err && !(ferr = fopen(opt.err, "w")))
{
Error("Can't open -err file %s for writing\n", opt.err);
return -1;
}
if (fmsg) {
fprintf(fmsg, "typedef struct _message {\n"
"\tunsigned\tid;\n"
"\tchar *str;\n"
"} MESSAGE;\n\n");
fprintf(fmsg, "MESSAGE __MSGTAB[] = {\n");
}
if (opt.c && !(fc = fopen( opt.c, "w" ))) {
Error( "can't open -c file %s for writing\n", opt.c );
return( -1 );
}
if (fc) {
fprintf(fc, "#include <stdio.h>\n\n");
fprintf(fc, "typedef struct _message {\n"
"\tunsigned int\tid;\n"
"\tchar *str;\n"
"} MESSAGE;\n\n");
fprintf(fc, "MESSAGE __MSGTAB[] = {\n");
}
while ((cchLine = ReadLine ( f )) != EOF) // process lines
{
nLineCur++;
if (buf[0] == '\0' || buf[0] == '#')
{
continue;
}
else if (HandleDirectives()) // Directive
{
continue;
}
else if (buf[0] == '[' && buf[1] == '[') // Old style leadbyte tbl
{
fprintf(stderr,
"WARNING: Ignore leadbyte table, use !codepage instead: %s\n",
buf);
continue;
}
else if (buf[0] == '<') // <<NMSG>> or <<FMSG>>
{
if (!strcmp( "<<NMSG>>", buf )) // near msgs follow
{
if (asmstate == 0)
{
if (fasm)
{
if (!didnear) {
didnear++;
if (opt.use32)
fprintf( fasm, dir32);
fprintf( fasm, n1 );
fprintf( fasm, n2 );
fprintf( fasm, n3 );
fprintf( fasm, n4 );
fprintf( fasm, n5 );
}
fprintf( fasm,
"MSG segment\n" );
if (opt.def)
{
fprintf( fasm, "\tpublic\t%s\n", opt.def );
fprintf( fasm, "%s\tequ\t$\n", opt.def );
}
asmstate = 1;
} // if (fasm)
} // if (asmstate == 0)
else if (asmstate == 1)
{
Error( "already in nmsg\n" );
}
else if (asmstate == 2 && !opt.use32)
{
if (fasm)
{
fprintf( fasm, "FAR_MSG ends\n\n" );
if (!didnear)
{
didnear++;
fprintf( fasm, n1 );
fprintf( fasm, n2 );
fprintf( fasm, n3 );
fprintf( fasm, n4 );
fprintf( fasm, n5 );
}
fprintf( fasm, "MSG segment\n" );
asmstate = 1;
} // if (fasm)
} // else if (asmstate == 2 ...)
else
{
Error( "internal error\n" );
return( -1 );
}
} // if near msg
else if (!strcmp( "<<FMSG>>", buf ))//far msgs follow
{
if (asmstate == 0)
{
if (fasm)
{
if (!didfar)
{
didfar++;
if (opt.use32)
{
fprintf( fasm, dir32);
fprintf( fasm, n1 );
fprintf( fasm, n2 );
fprintf( fasm, n3 );
fprintf( fasm, n4 );
fprintf( fasm, n5 );
fprintf( fasm, "MSG segment\n" );
}
else
{
fprintf( fasm, f1 );
fprintf( fasm, f2 );
fprintf( fasm, f3 );
fprintf( fasm, f4 );
fprintf( fasm, f5 );
fprintf( fasm, "FAR_MSG segment\n" );
}
} // if (!didfar)
if (opt.def)
{
fprintf( fasm, "\tpublic\t%s\n", opt.def );
fprintf( fasm, "%s\tequ\t$\n", opt.def );
}
asmstate = 2;
} // if (fasm)
} // if (asmstate == 0)
else if (asmstate == 1 && !opt.use32)
{
if (fasm)
{
fprintf( fasm, "MSG ends\n\n" );
if (!didfar)
{
didfar++;
fprintf( fasm, f1 );
fprintf( fasm, f2 );
fprintf( fasm, f3 );
fprintf( fasm, f4 );
fprintf( fasm, f5 );
}
fprintf( fasm, "FAR_MSG segment\n" );
asmstate = 2;
}
} // else if (asmstate == 1 ...)
else if (asmstate == 2)
{
Error( "already in far_msg\n" );
}
else
{
Error( "internal error\n" );
return( -1 );
}
} // far message
else // Not near, not far
{
Error( "ignoring bad line: %s\n", buf );
}
} // if (.. < ) near/far message
else if (buf[0] != '\r' && buf[0] != '\n') // something to do
{
char *pSymbol;
char *pNumber;
char *pMessage;
if (!ParseLine(&pSymbol, &pNumber, &pMessage))
{
fprintf( stderr, "%s(%d): error in line: \"%s\"\n", opt.txt, nLineCur, buf);
continue;
}
if (pSymbol && opt.h)
{
fprintf( fh, "#define\t%s\t%s\n", pSymbol, pNumber );
}
if (opt.rc)
{
fprintf(frc, "\t%s, \"%s\"\n", pNumber, pMessage);
}
if (pSymbol && opt.x)
{
fprintf( fx, "#define\t%s\t\"", pSymbol );
msg_fputs( pMessage
, opt.hex ? MSGSTYLE_C_HEX : MSGSTYLE_COPY
, fx );
putc( '\"', fx );
putc( '\n', fx );
}
if (pSymbol && opt.inc)
{
fprintf( finc, "%s\t=\t%s\n", pSymbol, pNumber);
}
if (opt.msg)
{
fprintf( fmsg, "{%s, \"", pNumber );
msg_fputs( pMessage, opt.hex ? MSGSTYLE_C_HEX : MSGSTYLE_COPY, fmsg );
fprintf( fmsg, "\"}," );
if (opt.hex)
fprintf( fmsg, " // \"%s\"", pMessage );
putc ( '\n', fmsg );
}
if (opt.c)
{
fprintf( fc, "{%s, \"", pNumber );
msg_fputs( pMessage, opt.hex ? MSGSTYLE_C_HEX : MSGSTYLE_COPY, fc );
fprintf( fc, "\"}," );
if (opt.hex)
fprintf( fc, " // \"%s\"", pMessage );
putc ( '\n', fc );
}
if (opt.err)
{
fprintf(ferr, "%s\t\"%s\"\n", pNumber, pMessage);
} // opt.err
if (fasm) // write asmfile
{
if (asmstate == 0)
{
if (!didnear)
{
didnear++;
if (opt.use32)
fprintf( fasm, dir32);
fprintf( fasm, n1 );
fprintf( fasm, n2 );
fprintf( fasm, n3 );
fprintf( fasm, n4 );
fprintf( fasm, n5 );
}
fprintf( fasm, "MSG segment\n" );
if (opt.def)
{
fprintf( fasm, "\tpublic\t%s\n", opt.def );
fprintf( fasm, "%s\tequ\t$\n", opt.def );
}
asmstate = 1;
} // if (asmstate == 0)
fprintf( fasm, "\tdw\t%s\n\tdb\t", pNumber );
instring = 0;
for (p = pMessage, length = 0; *p; p++, length++)
{
// allocate message
if (*p == '\\')
{
// C escape sequence
switch (*++p)
{
case 'r':
case 'n':
case 't':
case 'f':
case 'v':
case 'b':
case '\'':
case '"':
case '\\':
case 'x':
if (instring) {
putc( '"', fasm );
putc( ',', fasm );
instring = 0;
}
if (*p == 'x') {
p++;
if (*p && *(p+1))
fprintf ( fasm, "0%c%ch", *p, *(p+1));
else
puts ("Error in Hex Constant");
p++;
}
else if (*p == 'r')
fprintf( fasm, "13" );
else if (*p == 'n')
fprintf( fasm, "10" );
else if (*p == 't')
fprintf( fasm, "9" );
else if (*p == 'f')
fprintf( fasm, "12" );
else if (*p == 'v')
fprintf( fasm, "11" );
else if (*p == 'b')
fprintf( fasm, "8" );
else if (*p == '\'')
fprintf( fasm, "39" );
else if (*p == '"')
fprintf( fasm, "34" );
else if (*p == '\\')
fprintf( fasm, "92" );
putc( ',', fasm );
break;
case '\0':
//not an error, warning ...
fprintf(stderr, ex, buf);
p--;
break;
default:
if (!instring) {
putc( '"', fasm );
instring = 1;
}
putc( *p, fasm );
break;
} // switch
} //if (*p == '\\')
else if (instring) // keep building string
{
putc( *p, fasm );
if (IsLeadByte(*p))
putc( *++p, fasm );
}
else // start building string
{
putc( '"', fasm );
instring = 1;
putc( *p, fasm );
if (IsLeadByte(*p))
putc( *++p, fasm );
}
} // for
if (instring) // close string
{
putc( '"', fasm );
putc( ',', fasm );
}
putc( '0', fasm );
putc( '\n', fasm );
// calculate padding
// depends on msg length
if (opt.min || opt.max)
{
if (opt.min)
if (length <= 10)
factor = 1.01;
else if (length <= 20)
factor = 0.81;
else if (length <= 30)
factor = 0.61;
else if (length <= 50)
factor = 0.41;
else if (length <= 70)
factor = 0.31;
else
factor = 0.30;
else if (length <= 10)
factor = 2.00;
else if (length <= 20)
factor = 1.00;
else if (length <= 30)
factor = 0.80;
else if (length <= 50)
factor = 0.60;
else if (length <= 70)
factor = 0.40;
else
factor = 0.30;
result = (double)length * factor;
if (asmstate == 1 || opt.use32)
{
npad += (int)result;
if (result > (float)((int)result))
npad++;
}
else if (asmstate == 2)
{
fpad += (int)result;
if (result > (float)((int)result))
fpad++;
}
} // if (opt.min || opt.max)
} // if (fasm)...
} // Something to do
} // while read line
if (fmsg) { // finish up -msg stuff
fprintf(fmsg, "{0, NULL}\n};\n");
}
if (fc) // finish up -c stuff
{
fprintf(fc, "{0, NULL}\n};\n\n");
fprintf(fc, "char * __NMSG_TEXT(\n" );
fprintf(fc, "unsigned msgId\n" );
fprintf(fc, ") {\n" );
fprintf(fc, " MESSAGE *pMsg = __MSGTAB;\n" );
fprintf(fc, "\n" );
fprintf(fc, " for (;pMsg->id; pMsg++) {\n" );
fprintf(fc, " if (pMsg->id == msgId)\n" );
fprintf(fc, " break;\n" );
fprintf(fc, " }\n" );
fprintf(fc, " return pMsg->str;\n" );
fprintf(fc, "}\n" );
} // if (fc)
if (fasm) // finish up asm file
{
if (asmstate == 1 || opt.use32)
fprintf( fasm, "MSG ends\n\n");
else if (asmstate == 2)
fprintf( fasm, "FAR_MSG ends\n\n");
if (npad) { // add near padding
fprintf( fasm, "PAD segment\n\tdb\t%d dup(0)\n", npad );
fprintf( fasm, "PAD ends\n\n" );
}
if (fpad) { // add far padding
fprintf( fasm, "FAR_PAD segment\n\tdb\t%d dup(0)\n", fpad );
fprintf( fasm, "FAR_PAD ends\n\n" );
}
fprintf( fasm, "\tend\n" );
fclose( fasm );
} // if (fasm)
if (fh)
{
if (opt.c)
fprintf (fh, "\n#define GET_MSG(x) __NMSG_TEXT(x)\n");
fclose( fh );
}
if (frc) {
fprintf(frc, "END\n");
fclose( frc );
}
if (fx) fclose( fx );
if (finc) fclose( finc );
if (fmsg) fclose( fmsg );
if (fc) fclose( fc );
if (ferr) fclose( ferr );
fclose( f );
return(cError ? 1 : 0);
} // main()
void
parseCommandLine(
unsigned argc,
char **argv,
STATUS *opt
)
{
// skip argv[0]
argc--; argv++;
while (argc && **argv == '-') // process options
{
if (!strcmp("-err", *argv)) // Create .err file
{
argc--; argv++;
if (!argc)
Error("no -err file given\n");
else if (opt->err)
Error("extra -err for %s ignored\n", *argv);
else
{
opt->err = *argv;
argc--; argv++;
}
} // -err
else if (!strcmp( "-h", *argv )) { // create .h file
argc--; argv++;
if (!argc)
Error( "no -h file given\n" );
else if (opt->h) {
Error( "extra -h file %s ignored\n", *argv );
argc--; argv++;
}
else { // remember -h file
opt->h = *argv;
argc--; argv++;
}
}
else if (!strcmp("-rc", *argv)) { // Create .rc file
argc--; argv++;
if (!argc)
Error("no -rc file given\n");
else if (opt->rc)
Error("extra -rc for %s ignored\n", *argv);
else
{
opt->rc = *argv;
argc--; argv++;
}
}
else if (!strcmp( "-x", *argv )) { // create .h file
argc--; argv++;
if (!argc)
Error( "no -x file given\n" );
else if (opt->x) {
Error( "extra -x file %s ignored\n", *argv );
argc--; argv++;
}
else { // remember -x file
opt->x = *argv;
argc--; argv++;
}
}
else if (!strcmp( "-inc", *argv )) { // create .inc file
argc--; argv++;
if (!argc)
Error( "no -inc file given\n" );
else if (opt->inc) {
Error( "extra -inc file %s ignored\n", *argv );
argc--; argv++;
}
else { // remember -inc file
opt->inc = *argv;
argc--; argv++;
}
}
else if (!strcmp( "-msg", *argv )) { // create .h file with struct
argc--; argv++;
if (!argc)
Error( "no -msg file given\n" );
else if (opt->msg) {
Error( "extra -msg file %s ignored\n", *argv );
argc--; argv++;
}
else { // remember -msg file
opt->msg = *argv;
argc--; argv++;
}
}
else if (!strcmp( "-c", *argv )) { // create .c file with struct and function
argc--; argv++;
if (!argc)
Error( "no -c file given\n" );
else if (opt->c) {
Error( "extra -c file %s ignored\n", *argv );
argc--; argv++;
}
else { // remember -c file
opt->c = *argv;
argc--; argv++;
}
}
else if (!strcmp( "-asm", *argv )) { // create .asm file
argc--; argv++;
if (!argc)
Error( "no -asm file given\n" );
else if (opt->asm) {
Error( "extra -asm file %s ignored\n", *argv );
argc--;
argv++;
}
else { // remember -asm file
opt->asm = *argv;
argc--; argv++;
}
}
else if (!strcmp( "-def", *argv )) {
argc--; argv++;
if (!argc)
Error( "no -def string given\n" );
else {
opt->def = *argv;
argc--; argv++;
}
}
else if (!strcmp( "-min", *argv )) { // minimum padding
argc--; argv++;
if (opt->min)
Error( "redundant -min\n" );
opt->min = 1;
}
else if (!strcmp( "-max", *argv )) { // maximum padding
argc--; argv++;
if (opt->max)
Error( "redundant -max\n" );
opt->max = 1;
}
else if (!strcmp( "-386", *argv)) { // 32-bit segments
argc--; argv++;
if (opt->use32)
Error( "redundant -386\n" );
opt->use32 = 1;
}
else if (!strcmp( "-hex", *argv)) { // hex rendering of text
argc--; argv++;
if (opt->hex)
Error( "redundant -hex\n" );
opt->hex = 1;
}
else {
Error( "unknown option %s ignored\n", *argv );
argc--;
argv++;
}
} // while
if (!argc) { // no arguments
Error( usage, szVersionNo, szCopyRightYrs );
exit( -1 );
}
if (argc != 1) // extra arguments
Error( "ignoring extra arguments\n" );
opt->txt = *argv;
} // ParseCommandLine()
//
// Read One line into global buf
//
int
ReadLine (FILE * fp)
{
int i = 0;
int ch;
static char szBuffer[INBUFSIZE];
while ((ch = getc( fp )) != EOF && ch != '\r' && ch != '\n' && ch != '\x1A')
if (i < INBUFSIZE-1)
szBuffer[i++] = (char)ch;
if (ch == EOF && i == 0)
return EOF;
if (ch == '\r')
getc ( fp ); // Flush line feed
szBuffer[i] = '\0';
// Skip initial space
for (buf = szBuffer; *buf && isspace(*buf); buf++)
i--;
return i;
}
void
__cdecl
Error(
char *fmt,
...
) {
va_list args;
va_start (args, fmt);
vfprintf(stderr, fmt, args);
++cError;
}
///// msg_fputs
//
// Purpose:
//
// Send string to file in the given format.
//
//////////////////////////////////////////////////////////////////////////
void msg_fputs(
unsigned char * pch,
int style,
FILE * fp
) {
char chbuf[8];
unsigned char bInDBCS = FALSE;
static int bPrevWasHex = FALSE;
switch (style)
{
case MSGSTYLE_COPY:
fputs (pch, fp);
break;
case MSGSTYLE_C_HEX:
for (;*pch; pch++) {
// If a lead byte table was specified, we use hex
// only for double-byte characters, and for hex
// digits after hex output. This later is because
// hex constants terminate only when a non-hex digit
// (like a \) is encountered.
//
if (!bUseLeadByteTable
|| bInDBCS || IsLeadByte(*pch)
|| (bPrevWasHex && isxdigit(*pch))) {
sprintf (chbuf, "\\x%2.2x", *pch);
fputs (chbuf, fp);
bInDBCS = bInDBCS ? FALSE : IsLeadByte(*pch);
bPrevWasHex = TRUE;
}
else {
fputc(*pch, fp);
bPrevWasHex = FALSE;
}
}
break;
case MSGSTYLE_ASM_TEXT: // UNDONE
case MSGSTYLE_ASM_BYTE: // UNDONE
break;
}
}
///// SetCodePage
//
// Purpose:
//
// Switch to the specified codepage so that we can recognize a lead
// byte using IsLeadByte().
//
// Parameters:
// const char *pszCodePage: Points to a buffer containing the codepage
// in the the format of ".xxx" where xxx is
// the codepage number.
//
// Note:
// This function replaces the old SetLeadByteTable(), which fills
// the global lead byte table with values the user supplied in the
// message file.
//
///////////////////////////////////////////////////////////////////////////
void
SetCodePage(const char *pszCodePage)
{
unsigned i;
unsigned uCodePage;
if (!setlocale(LC_ALL, pszCodePage)) // Switch to new locale
{
//__setmbctable(atoi(pszCodePage+1)); // Failed, use undoc'ed call
// Failed, use internal codepage table
uCodePage = atoi(pszCodePage+1);
for (i = 0; cpTable[i].uCodePage; i++)
{
if (cpTable[i].uCodePage == uCodePage) // Found
{
pCP = &cpTable[i];
break;
}
} // for
if (0 == cpTable[i].uCodePage)
fprintf(stderr, "WARNING: unknown codepage: %s\n", pszCodePage+1);
}
bUseLeadByteTable = TRUE;
}
#ifdef VERBOSE
#define DB(x) x
#else
#define DB(x)
#endif
BOOL
IsLeadByte(unsigned by)
{
unsigned byIndex;
DB(printf("IsLeadByte(0x%02x) ==> ",by));
if (!bUseLeadByteTable)
{
DB(printf("FALSE\n"));
return FALSE;
}
if (!pCP)
#ifdef NT_BUILD
puts("Codepage support not implemented");
#else
return _ismbblead(by);
#endif
for (byIndex = 0; pCP->byLead[byIndex]; byIndex += 2)
{
if (pCP->byLead[byIndex] <= by && by <= pCP->byLead[byIndex+1])
{
DB(printf("TRUE\n"));
return TRUE;
}
}
DB(printf("FALSE\n"));
return FALSE;
}
///// ParseLine
//
// Purpose:
//
// Break the input line to 3 fields: symbol, number, and message.
// The symbol field is optional: if the first non-blank char in
// the line is a digit, then the symbol field is set to NULL.
//
// Assumption:
// We assume that the input line has the following formats:
// [<White space> SYMBOL] <White space> NUMBER <White space> MESSAGE
// where as the text between the square brackets ([]) is optional.
//
// Parameters:
// char **ppSymbol: Points to the buffer containing the symbol.
// *ppSymbol == NULL if there is no symbol.
// char **ppNumber: Points to the buffer containing the number.
// char **ppMessage: Points to the buffer containing the message.
//
// Use Global:
// buf Contains the line to examine.
//
// Return Value:
// TRUE: The line is a valid line.
// FALSE: The line is not a valid line.
//
///////////////////////////////////////////////////////////////////////////
BOOL
ParseLine(char **ppSymbol, char **ppNumber, char **ppMessage)
{
unsigned char *pBuf = buf;
#define SKIP_BLANKS() for ( ; *pBuf && isspace(*pBuf); pBuf++)
#define SKIP_TO_BLANKS() for ( ; *pBuf && !isspace(*pBuf); pBuf++)
#define CHECK_NULL() if (!*pBuf) return FALSE;
SKIP_BLANKS(); // Skip initial blanks
CHECK_NULL(); // Blank line?
if (!isdigit(*pBuf)) // Symbol?
{
*ppSymbol = pBuf;
SKIP_TO_BLANKS();
CHECK_NULL();
*pBuf++ = '\0';
SKIP_BLANKS();
CHECK_NULL();
}
else
*ppSymbol = NULL;
*ppNumber = pBuf;
SKIP_TO_BLANKS();
CHECK_NULL();
*pBuf++ = '\0';
SKIP_BLANKS();
CHECK_NULL();
*ppMessage = pBuf;
// Handle quoted message: if the message does not begin with quote, it is
// extended to end of line. If it begins with a double quote character,
// then it is extended to the closing quote, or end of line, whichever
// comes first. While scanning for closing quote, we ignore the '\"',
// which is the literal quote character.
if ('\"' == *pBuf) // quoted message
{
*ppMessage = ++pBuf;
while (*pBuf && '\"' != *pBuf)
{
if ('\\' == *pBuf || IsLeadByte(*pBuf))
pBuf++;
if (*pBuf)
pBuf++;
} // while
*pBuf = '\0';
} // if ... quoted message
return TRUE;
}
///// HandleDirectives
//
// Purpose:
//
// Determine if a line contains a directive, then carry out the
// directive's command.
//
// Use Global:
// buf Contains the line to examine.
//
// Return Value:
// TRUE: The line is a directive.
// FALSE: The line is not a directive
//
///////////////////////////////////////////////////////////////////////////
BOOL
HandleDirectives(void)
{
register unsigned char *pBuf = buf;
unsigned char *pEnd;
for ( ; *pBuf && isspace(*pBuf); pBuf++) // Skip leading spaces
;
if (!_strnicmp("!codepage", pBuf, 9)) // Change the codepage
{
for (pBuf += 9; *pBuf && isspace(*pBuf); pBuf++) // Skip spaces
;
*--pBuf = '.';
for (pEnd = pBuf + 1; *pEnd && isdigit(*pEnd); pEnd++)
;
*pEnd = '\0';
SetCodePage(pBuf);
return TRUE;
}
else if ('!' == *pBuf)
{
Error("Unrecognized directive: '%s'\n", pBuf);
return TRUE;
}
return FALSE;
} // HandleDirectives