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

471 lines
17 KiB
C

// NMAKE.H -- main header file
//
// Copyright (c) 1988-1990, Microsoft Corporation. All rights reserved.
//
// Purpose:
// This file is the main header file for NMAKE and contains global typedef and
// macros. Global constants are also defined here. Global data is in global.h
//
// Revision History:
// 01-Feb-1994 HV Move messages to external file.
// 15-Nov-1993 JdR Major speed improvements
// 15-Oct-1993 HV Use tchar.h instead of mbstring.h directly, change STR*() to _ftcs*()
// 23-Jun-1993 HV Kill the near keyword
// 01-Jun-1993 HV Change #ifdef KANJI to _MBCS
// 10-May-1993 HV Add _MBCS version of the string library.
// 10-Jul-1992 SB Port to NTMIPS
// 08-Jun-1992 SS Port to DOSX32
// 02-Feb-1990 SB add definition of FILEOPEN
// 04-Dec-1989 SB Changed definition of PFV to have prototype of function
// which it dereferences
// 01-Dec-1989 SB Contains an hack #ifdef'ed for Overlayed version
// also defined REALLOC
// 22-Nov-1989 SB #define FREE
// 13-Nov-1989 SB Definitions CCHMAXPATH(COMP) conform with bsedos.h
// 02-Oct-1989 SB added support for dynamic inline files
// 14-Sep-1989 SB added inLines field to block
// 04-Sep-1989 SB Added M_COMMAND_LINE but not used
// 24-Aug-1989 SB Add A_DEPENDENT for NMAKE to know that it is to look for a
// dependent
// 16-May-1989 SB NOLOGO flag /L set up for passing to recursive builds
// 24-Apr-1989 SB added CCHMAXPATH & CCHMAXPATHCOMP for OS/2 ver 1.2 support
// & removed FILEINFO typedef's (not needed anymore)
// 22-Feb-1989 SB changed value of MAXCMDLINELENGTH to 2k
// 03-Feb-1989 SB Added struct for FILEINFO for OS2 ver 1.2
// 02-Feb-1989 SB Redefined SPAWNV(P) and SYSTEM as NMAKE was really not
// supporting KANJI
// 31-Jan-1989 SB Changed MAXNAME to 257 for OS2 Ver 1.2 support
// 21-Dec-1988 SB Added SCRIPTLIST and makeNewScriptListElement() to allow
// multiple script fileseach with its KEEP/NOKEEP action
// 06-Dec-1988 SB Updated Comment about bits corr to flags set
// 05-Dec-1988 SB Added #define CDECL; NMAKE now uses Pascal Calling
// Add SIG_IGN to handle compiler problem
// 30-Nov-1988 SB Added suppport for 'z' option in setFlags()
// 23-Nov-1988 SB Defined MAXCMDLINELENGTH for extmake syntax
// 10-Nov-1988 SB Changed BOOL as 'unsigned short' as in 'os2.h'
// 17-Aug-1988 RB Clean up.
// 14-Jul-1988 rj Added dateTime to BUILDBLOCK def to support multiple
// targets with the same command block.
// 07-Jul-1988 rj Added targetFlag parameter to findMacro, findTarget
// 15-Jun-1988 rj Add definition of EScapeCHaracter.
// 25-May-1988 rb Clean up definition of LOCAL.
// Better char-type defs for ECS.
// Include from the LANGAPI (shared components) project
#include <assert.h>
#include <direct.h>
#include <dos.h>
#include <errno.h>
#include <io.h>
#include <limits.h>
#include <malloc.h>
#include <mbctype.h>
#include <process.h>
#include <share.h>
#include <signal.h>
#include <stdarg.h>
#include <stdio.h>
#include <stdlib.h>
#include <tchar.h>
#include "getmsg.h"
#define STRICT
#define NOMINMAX // windef.h
#define NOGDI // wingdi.h
#define NOIME // ime.h
#define NOUSER // winuser.h
#define NOHELP
#define NOPROFILER
#define NOSYSPARAMSINFO
#define NONLS // winnls.h
#define NOSERVICE // winsvc.h
#include <windows.h>
#define FILEOPEN open_file
#define FREE_STRINGLIST free_stringlist
#define ALLOC_STRINGLIST(type) (STRINGLIST *)alloc_stringlist()
#define FREE free
#define REALLOC realloc
typedef char CHAR;
typedef unsigned char UCHAR;
typedef unsigned short USHORT;
typedef unsigned long ULONG;
#define NMHANDLE INT_PTR
typedef void (*PFV) (void); // pointer to void function
// size constants for buffers
//
// I impose an arbitrary limit of 128 bytes for a macro or target name.
// This should be much larger than any intelligent user would need.
// The value of a macro can be longer -- up to MAXBUF in length.
//
// The hash table for macros was only 64 pointers long. That seemed a
// reasonable number, since most makefiles have fewer than 64 variables
// defined. Measurements on real makefiles (P1) showed that we had almost
// 8 elements per chain, so it was increased to 256 on FLAT architectures.
// The hashing algorithm isn't perfect, but it should be good
// enough. Even if the macro we're looking up is in a bucket w/ more
// than one entry, we'll only have to chain one or two links down the
// list to find it. When we add macros a bucket's list we prepend them,
// so we don't have to chain at all there.
#define MAXNAME 257
#define MAXMACRO 256
#define MAXTARGET 128
#define MAXBUF 1024
#define MAXSEGMENT 65535
#define MAXARG MAXNAME / 2 - 1
#define CHUNKSIZE 8
#define MAXCMDLINELENGTH 8192
// constants used by error.c
//
// error numbers are of the form Unxxx, where the 'U' stands for utility,
// n is the 1st digit of one of the values below indicating the type of
// error, and xxx is the number of the error (error messages aren't in any
// order right now -- related ones should probably be grouped together
// when all error messages have been added)
#define FATAL_ERR 1000
#define NONFATAL_ERR 2000
#define WARNING_ERR 4000
// other constants
#define WRITE 2 // READ,WRITE,APPEND used by
#define APPEND 3 // redirect() in build.c
#define READ 4 // also value for access()
#define ESCH '^' // general escape character
typedef struct INCLUDEINFO
{
unsigned line;
char *name;
FILE *file;
} INCLUDEINFO;
#define MAXINCLUDE 16
// STRINGLIST structure is used to construct lists that keep track of the
// makefiles to read, the targets to update, the dependency list for each
// target, the list of build commands for each target, and the values
// for a macro.
typedef struct strlist {
struct strlist *next;
char *text;
} STRINGLIST;
typedef struct inlinelist {
struct inlinelist *next;
char *text;
char *name;
BOOL fKeep;
unsigned size;
} INLINELIST;
typedef struct scrptlist { // List used to handle multiple
struct scrptlist *next; // scriptfiles
char *sFile; // -- Script file name & its
BOOL fKeep; // -- keep status (default nokeep)
} SCRIPTLIST;
typedef struct BLOCK {
STRINGLIST *dependents; // dependents of the target
STRINGLIST *dependentMacros; //
STRINGLIST *buildCommands; // command list to build target
STRINGLIST *buildMacros;
UCHAR flags;
time_t dateTime;
} BUILDBLOCK;
typedef struct bldlist {
struct bldlist *next;
BUILDBLOCK *buildBlock;
} BUILDLIST;
typedef struct BATCH { // State info for deferred batch commands
struct BATCH *next;
struct RULE *pRule; // batch inference rule
UCHAR flags; // build flags
STRINGLIST *nameList; // list of names to be "touched" (nmake -t)
STRINGLIST *dollarLt; // list for constructing batch $<
} BATCHLIST;
typedef struct OBJECT {
struct OBJECT *next;
char *name;
UCHAR flags2;
UCHAR flags3;
time_t dateTime;
BUILDLIST *buildList;
BATCHLIST **ppBatch; // batch list that contains this object,
// if being built in batch-mode
} MAKEOBJECT;
typedef struct iobject { // used for dependents NOT in
struct object *next; // the makefile. We add them
char *name; // to the target table w/ a
UCHAR flags2; // flag that says "already
UCHAR flags3; // built" and then we never
long datetime; // have to time-stamp again
} IMPLIEDOBJECT;
typedef struct RULE {
struct RULE *next;
struct RULE *back; // doubly-link rules for ease
char *name; // in sorting later . . .
STRINGLIST *buildCommands; // (# of rules is mostly small
STRINGLIST *buildMacros; // so not much memory used
BOOL fBatch; // TRUE if batch rule (doublecolon)
} RULELIST;
typedef struct deplist {
struct deplist *next;
char *name;
time_t depTime;
} DEPLIST;
// Bits in flags/gFlags indicate which cmdline options are set
//
// -a sets FORCE_BUILD
// -c sets CRYPTIC_OUTPUT (only fatal errors get displayed)
// -d sets DISPLAY_FILE_DATES
// -e sets USE_ENVIRON_VARS
// -i sets IGNORE_EXIT_CODES
// -n sets NO_EXECUTE
// -p sets PRINT_INFORMATION
// -q sets QUESTION_STATUS
// -r sets IGNORE_EXTERN_RULES
// -s sets NO_ECHO
// -t sets TOUCH_TARGETS
// -z sets REVERSE_BATCH_FILE (Required by PWB)
// -l sets NO_LOGO (internally /l actually -nologo)
//
// Also included are bits for
//
// BUILDING_THIS_ONE - to detect cycles in dependencies
// DOUBLECOLON - to indicate type of separator found between
// the targets and dependencies (':' or '::')
// ALREADY_BUILT - to indicate that a target has been built
// OUT_OF_DATE - to indicate that this target is out of date
#define F1_PRINT_INFORMATION 0x01 // "global" flags that affect
#define F1_IGNORE_EXTERN_RULES 0x02 // all targets (it doesn't
#define F1_USE_ENVIRON_VARS 0x04 // make sense to allow the
#define F1_QUESTION_STATUS 0x08 // user to change these)
#define F1_TOUCH_TARGETS 0x10
#define F1_CRYPTIC_OUTPUT 0x20
#define F1_NO_BATCH 0x40 // disable batching functionality
#define F1_NO_LOGO 0x80
#define F2_DISPLAY_FILE_DATES 0x01 // these are resettable w/in
#define F2_IGNORE_EXIT_CODES 0x02 // the makefile
#define F2_NO_EXECUTE 0x04 // each target keeps own copy
#define F2_NO_ECHO 0x08
#define F2_FORCE_BUILD 0x10 // build even if up-to-date
#define F2_DOUBLECOLON 0x20 // indicates separator type
#define F2_DUMP_INLINE 0x40 // dump inline files
#define F3_BUILDING_THIS_ONE 0x01 // finds cyclical dependencies
#define F3_ALREADY_BUILT 0x02
#define F3_OUT_OF_DATE 0x04 // reuse :: bit after target
// has been built
#define F3_ERROR_IN_CHILD 0x08 // used to implement slash k
// ----------------------------------------------------------------------------
// MACRODEF structure is used to make a list of macro definitions from the
// commandline, makefile, TOOLS.INI file, and environment. It contains
// a flag which is set for macros defined in the command line so that
// a later definition of the same macro will be ignored. It also contains
// a flag that gets set when NMAKE is expanding the macro so that recursive
// definitions can be detected.
///
typedef struct macro {
struct macro *next;
char *name;
STRINGLIST *values; // can just be list of size 1
UCHAR flags;
} MACRODEF;
// Bits in flags field for macros. We really only need to know if a macro
// was defined on the commandline (in which case we ignore all redefinitions),
// or if we're currently expanding macros in its value (so when we look
// up a macro and that bit is set we can tell that the macro is defined
// recursively).
#define M_EXPANDING_THIS_ONE 0x01
#define M_NON_RESETTABLE 0x02
#define M_ENVIRONMENT_DEF 0x04
#define M_WARN_IF_RESET 0x08
#define M_UNDEFINED 0x10
#define M_COMMAND_LINE 0x20
#define M_LITERAL 0x40 // value contains no other macros
// treat $ literary
// macros to simplify dealing w/ bits in flags, allocating memory, and
// testing characters
#define SET(A,B) ((A) |= (UCHAR)(B)) // turn bit B on in A
#define CLEAR(A,B) ((A) &= (UCHAR)(~B)) // turn bit B off in A
#define ON(A,B) ((A) & (UCHAR)(B)) // is bit B on in A?
#define OFF(A,B) (!ON(A,B)) // is bit B off in A?
#define FLIP(A,B) (ON(A,B)) ? (CLEAR(A,B)) : (SET(A,B))
#define CANT_REDEFINE(A) (ON((A)->flags,M_NON_RESETTABLE) \
|| (ON(gFlags,F1_USE_ENVIRON_VARS) \
&& ON((A)->flags,M_ENVIRONMENT_DEF)))
#define ALLOCATE_OBJECT(type) ((type *) allocate(sizeof(type)))
#define makeNewStrListElement() ALLOC_STRINGLIST(STRINGLIST)
#define makeNewInlineListElement() ALLOCATE_OBJECT(INLINELIST)
#define makeNewScriptListElement() ALLOCATE_OBJECT(SCRIPTLIST)
#define makeNewMacro() ALLOCATE_OBJECT(MACRODEF)
#define makeNewObject() ALLOCATE_OBJECT(MAKEOBJECT)
#define makeNewImpliedObject() ALLOCATE_OBJECT(MAKEOBJECT)
#define makeNewBuildBlock() ALLOCATE_OBJECT(BUILDBLOCK)
#define makeNewBldListElement() ALLOCATE_OBJECT(BUILDLIST)
#define makeNewRule() ALLOCATE_OBJECT(RULELIST)
#define MakeNewDepListElement() ALLOCATE_OBJECT(DEPLIST)
#define makeNewBatchListElement() ALLOCATE_OBJECT(BATCHLIST)
#define WHITESPACE(A) ((A) == ' ' || (A) == '\t')
#if 1 //JdR see charmap.h
// #define MACRO_CHAR(A) IS_MACROCHAR(A)
// Modified MACRO_CHAR to fix handling of mbcs characters.
// 'A' may combine the bytes of the mbcs char in a single value and
// end up being >= 256. All values >=128 can be treated as
// valid macro characters [vc98 #9973 georgiop 9/19/97]
#define MACRO_CHAR(A) (IS_MACROCHAR(A) || ((unsigned)(A)) >= 128)
#else
#define MACRO_CHAR(A) ((A) == '_' || _istalnum(A) || ((unsigned)(A)) >= 128)
#endif
#define PATH_SEPARATOR(A) ((A) == '\\' || (A) == '/')
#define DYNAMIC_DEP(A) ((A)[2] == '(' \
&& (A)[3] == '@' \
&& (A)[5] == ')' \
&& (((A)[4] == 'F' \
|| (A)[4] == 'D' \
|| (A)[4] == 'B' \
|| (A)[4] == 'R')))
// values passed to getSpecialValue() to indicate which type of macro
// we're expanding
#define SPECIAL_MACRO 0x01 // $* $@ $? $< $**
#define DYNAMIC_MACRO 0x02 // $$@
#define X_SPECIAL_MACRO 0x03 // $(*F) $(@D) etc.
#define X_DYNAMIC_MACRO 0x04 // $$(@F) $$(@D)
#define DOLLAR_MACRO 0x05 // $$ -> $
// Bits in elements placed in the stack (ifStack) that keeps state
// information about if/else/endif directives. Here "if" directive
// includes if/ifdef/ifndef/if defined().
// -- used in routine lgetc() in ifexpr.c
#define NMIFELSE 0x01 // set for if/ifdef etc...reset for else
#define NMCONDITION 0x02 // set if condition part of if is true
#define NMIGNORE 0x04 // set if if/endif block is to be ignored/skipped
#define NMELSEIF 0x08 // set for else if/ifdef etc...reset for else
// Values to record which of if/ifdef/ifndef/etc was seen to decide
// the kind of processing to be done.
#define IF_TYPE 0x01
#define ELSE_TYPE 0x02
#define ELSE_IF_TYPE 0x03
#define ELSE_IFDEF_TYPE 0x04
#define ELSE_IFNDEF_TYPE 0x05
#define IFDEF_TYPE 0x06
#define IFNDEF_TYPE 0x07
#define ENDIF_TYPE 0x08
// Values to indicate if we are reading from the raw stream or thru'
// the routine lgetc() which preprocesses directives. These are used
// by a routine common to lgetc() module and the lexer.
#define FROMLOCAL 0x00
#define FROMSTREAM 0x01
// macros to simplify accessing hash tables
// find() returns a STRINGLIST pointer, which is then cast to a pointer
// of the appropriate structure type
#define findTarget(A) (MAKEOBJECT*) find(A, MAXTARGET, \
(STRINGLIST**)targetTable, \
(BOOL)TRUE)
// "action" flags for building target-table entries
// if any of the bits in A_SUFFIX to A_RULE is set, the action routines
// WON'T build a targetblock for the current target (really pseudotarget
// or rule)
// A_TARGET says expand names on input (dependent names get expanded when
// target is built) */
#define A_SUFFIX 0x01
#define A_SILENT 0x02
#define A_IGNORE 0x04
#define A_PRECIOUS 0x08
#define A_RULE 0x10
#define A_TARGET 0x20
#define A_STRING 0x40
#define A_DEPENDENT 0x80
// "build" flags used by recursive target-building function
#define B_COMMANDS 0x01
#define B_BUILD 0x02
#define B_INMAKEFILE 0x04
#define B_NOTARGET 0x08
#define B_ADDDEPENDENT 0x10
#define B_DOUBLECOLON 0x20
#define B_DEP_OUT_OF_DATE 0x40
// "command" flags used by doCommand function
#define C_SILENT 0x01
#define C_IGNORE 0x02
#define C_ITERATE 0x04
#define C_EXECUTE 0x08
#define C_EXPANDED 0x10
// keyword for better profiling, normally set to "static".
#ifndef LOCAL
#define LOCAL static
#endif
// GetTxtChr and UngetTxtChr are the MBCS counterparts of getc and ungetc.
#ifdef _MBCS
extern int GetTxtChr(FILE*);
extern int UngetTxtChr(int, FILE*);
#else
#define GetTxtChr(a) getc(a)
#define UngetTxtChr(c,f) ungetc(c,f)
#endif
#define strend(p) (p + _tcslen(p))
#include "charmap.h"