2330 lines
75 KiB
C
Raw Normal View History

2001-01-01 00:00:00 +01:00
//+---------------------------------------------------------------------------
//
// Microsoft Windows
// Copyright (C) Microsoft Corporation, 1992 - 1993.
//
// File: buildsrc.c
//
// Contents: Functions used to process SOURCES and DIRS files
//
// History: 16-May-89 SteveWo Created
// 26-Jul-94 LyleC Split out from buildmak.c
// 05-Dec-00 sbonev See SD changelist 2317
//
//----------------------------------------------------------------------------
#include "build.h"
//
// Definitions used by the macro functions
//
#define LPAREN '('
#define RPAREN ')'
// Legal character for a macro name.
#define MACRO_CHAR(ch) iscsym(ch)
#define CMACROMAX 1000 // maximum unique macros per sources/dirs file
typedef struct _MACRO {
LPSTR pszValue;
UCHAR szName[1];
} MACRO;
MACRO *apMacro[CMACROMAX];
UINT cMacro = 0;
LPSTR *ppCurrentDirsFileName;
DWORD StartTime;
DWORD DirectoryStartTime;
#define SOURCES_TARGETNAME 0
#define SOURCES_TARGETPATH 1
#define SOURCES_TARGETPATHLIB 2
#define SOURCES_TARGETTYPE 3
#define SOURCES_TARGETEXT 4
#define SOURCES_INCLUDES 5
#define SOURCES_NTTEST 6
#define SOURCES_UMTYPE 7
#define SOURCES_UMTEST 8
#define SOURCES_OPTIONAL_UMTEST 9
#define SOURCES_UMAPPL 10
#define SOURCES_UMAPPLEXT 11
#define SOURCES_NTTARGETFILE0 12
#define SOURCES_NTTARGETFILES 13
#define SOURCES_PRECOMPILED_INCLUDE 14
#define SOURCES_PRECOMPILED_PCH 15
#define SOURCES_PRECOMPILED_OBJ 16
#define SOURCES_PRECOMPILED_TARGET 17
#define SOURCES_CHICAGO_PRODUCT 18
#define SOURCES_CONDITIONAL_INCLUDES 19
#define SOURCES_SYNCHRONIZE_BLOCK 20
#define SOURCES_SYNCHRONIZE_DRAIN 21
#define SOURCES_PASS0_SOURCEDIR 22
#define SOURCES_PASS0_HEADERDIR 23
#define SOURCES_PASS0_UUIDDIR 24
#define SOURCES_PASS0_CLIENTDIR 25
#define SOURCES_PASS0_SERVERDIR 26
#define SOURCES_IDLTYPE 27
#define SOURCES_SOURCES_OPTIONS 28
#define SOURCES_MFC_INCLUDES 29
#define SOURCES_SDK_LIB_DEST 30
#define SOURCES_DDK_LIB_DEST 31
#define SOURCES_SDK_INC_PATH 32
#define SOURCES_CRT_INC_PATH 33
#define SOURCES_OAK_INC_PATH 34
#define SOURCES_DDK_INC_PATH 35
#define SOURCES_WDM_INC_PATH 36
#define SOURCES_PRIVATE_INC_PATH 37
#define SOURCES_CHECKED_ALT_DIR 38
#define SOURCES_PROJECT_NAME 39
#define SOURCES_PASS0_PUBLISH 40
#define SOURCES_USER_INCLUDES 41
#define SOURCES_LAST_INCLUDES 42
#define SOURCES_MIDL_UUIDDIR 43
#define SOURCES_SYNCHRONIZE_PASS2_BLOCK 44
#define SOURCES_SYNCHRONIZE_PASS2_DRAIN 45
#define SOURCES_MANAGED_CODE 46
#define SOURCES_SYNC_PRODUCES 47
#define SOURCES_SYNC_CONSUMES 48
#define SOURCES_MANAGED_VB 49
#define SOURCES_PASS0_BINPLACE 50
#define SOURCES_PASS1_LINK 51
LPSTR RelevantSourcesMacros[] = {
"TARGETNAME",
"TARGETPATH",
"TARGETPATHLIB",
"TARGETTYPE",
"TARGETEXT",
"INCLUDES",
"NTTEST",
"UMTYPE",
"UMTEST",
"OPTIONAL_UMTEST",
"UMAPPL",
"UMAPPLEXT",
"NTTARGETFILE0",
"NTTARGETFILES",
"PRECOMPILED_INCLUDE",
"PRECOMPILED_PCH",
"PRECOMPILED_OBJ",
"PRECOMPILED_TARGET",
"CHICAGO_PRODUCT",
"CONDITIONAL_INCLUDES",
"SYNCHRONIZE_BLOCK",
"SYNCHRONIZE_DRAIN",
"PASS0_SOURCEDIR",
"PASS0_HEADERDIR",
"PASS0_UUIDDIR",
"PASS0_CLIENTDIR",
"PASS0_SERVERDIR",
"IDL_TYPE",
"SOURCES_OPTIONS",
"MFC_INCLUDES",
"SDK_LIB_DEST",
"DDK_LIB_DEST",
"SDK_INC_PATH",
"CRT_INC_PATH",
"OAK_INC_PATH",
"DDK_INC_PATH",
"WDM_INC_PATH",
"PRIVATE_INC_PATH",
"CHECKED_ALT_DIR",
"_PROJECT_",
"PASS0_PUBLISH",
"USER_INCLUDES",
"LAST_INCLUDES",
"MIDL_UUIDDIR",
"SYNCHRONIZE_PASS2_BLOCK",
"SYNCHRONIZE_PASS2_DRAIN",
"MANAGED_CODE",
"BUILD_PRODUCES",
"BUILD_CONSUMES",
"MANAGED_VB",
"PASS0_BINPLACE",
"PASS1_LINK",
NULL
};
#define SOURCES_MAX (ARRAY_SIZE(RelevantSourcesMacros) - 1)
VOID
MarkDirNames(PDIRREC DirDB, LPSTR TextLine, BOOL Required);
//+---------------------------------------------------------------------------
//
// Function: CompressBlanks
//
// Synopsis: Compress multiple blank characters out of macro value, in
// place.
//
// Arguments: [psrc] -- String to compress
//
// Notes: Note that tabs, CRs, continuation lines (and their line
// breaks) have already been replaced with blanks.
//
//----------------------------------------------------------------------------
VOID
CompressBlanks(LPSTR psrc)
{
LPSTR pdst = psrc;
while (*psrc == ' ') {
psrc++; // skip leading macro value blanks
}
while (*psrc != '\0') {
if (*psrc == '#') { // stop at comment
break;
}
if ((*pdst++ = *psrc++) == ' ') {
while (*psrc == ' ') {
psrc++; // skip multiple blanks
}
}
}
*pdst = '\0'; // terminate the compressed copy
if (*--pdst == ' ') {
*pdst = '\0'; // trim trailing macro value blanks
}
}
//+---------------------------------------------------------------------------
//
// Function: GetBaseDir
//
// Synopsis: Return the value of BASEDIR, the base NT directory, if
// appropriate.
//
// Arguments: [pname] -- path to split
//
//----------------------------------------------------------------------------
LPSTR
GetBaseDir(LPSTR pname)
{
if (_stricmp("BASEDIR", pname) == 0) {
return (NtRoot);
}
return (NULL);
}
//+---------------------------------------------------------------------------
//
// Function: FindMacro
//
// Synopsis: Returns the value of a given macro by name.
//
// Arguments: [pszName] -- Name of macro who's value is desired.
//
// Returns: String containing the value of the macro
//
//----------------------------------------------------------------------------
LPSTR
FindMacro(LPSTR pszName)
{
MACRO **ppm;
for (ppm = apMacro; ppm < &apMacro[cMacro]; ppm++) {
if (_stricmp(pszName, (*ppm)->szName) == 0) {
return ((*ppm)->pszValue);
}
}
return (NULL);
}
//+---------------------------------------------------------------------------
//
// Function: SaveMacro
//
// Synopsis: Save the value of a macro
//
// Arguments: [pszName] -- Name of macro to save
// [pszValue] -- Value of macro
//
// Notes: A new string must be allocated and initialized prior to
// freeing the old string when updating a macro value.
//
//----------------------------------------------------------------------------
VOID
SaveMacro(LPSTR pszName, LPSTR pszValue)
{
MACRO **ppm;
for (ppm = apMacro; ppm < &apMacro[cMacro]; ppm++) {
if (_stricmp(pszName, (*ppm)->szName) == 0) {
break;
}
}
if (ppm == &apMacro[CMACROMAX]) {
BuildError("Macro table full, ignoring: %s = %s\r\n", pszName, pszValue);
return;
}
if (ppm == &apMacro[cMacro]) {
cMacro++;
AllocMem(sizeof(MACRO) + strlen(pszName), ppm, MT_MACRO);
strcpy((*ppm)->szName, pszName);
(*ppm)->pszValue = NULL;
}
MakeMacroString(&(*ppm)->pszValue, pszValue);
if (DEBUG_1) {
BuildMsg(
"SaveMacro(%s = %s)\r\n",
(*ppm)->szName,
(*ppm)->pszValue == NULL? "NULL" : (*ppm)->pszValue);
}
if ((*ppm)->pszValue == NULL) {
FreeMem(ppm, MT_MACRO);
*ppm = apMacro[--cMacro];
}
}
//+---------------------------------------------------------------------------
//
// Function: FreeMacros
//
// Synopsis: Free all macros
//
// Arguments: (none)
//
//----------------------------------------------------------------------------
VOID
FreeMacros(VOID)
{
MACRO **ppm;
for (ppm = apMacro; ppm < &apMacro[cMacro]; ppm++) {
FreeString(&(*ppm)->pszValue, MT_DIRSTRING);
FreeMem(ppm, MT_MACRO);
assert(*ppm == NULL);
}
cMacro = 0;
}
//+---------------------------------------------------------------------------
//
// Function: SplitMacro
//
// Synopsis: Take a string containing "MACRONAME = VALUE" and return
// the target and value.
//
// Arguments: [pline] -- String to split and target return.
//
// Returns: Value of macro.
//
//----------------------------------------------------------------------------
LPSTR
SplitMacro(LPSTR *pszTarget)
{
LPSTR pvalue, p, pline;
pvalue = NULL;
pline = *pszTarget;
// Quickly reject comments and ! directives.
if (*pline == '#' || *pline == '!') {
return NULL;
}
if ((p = strchr(pline, '=')) != NULL) {
pvalue = p + 1; // point past old '='
while (p > pline && p[-1] == ' ') {
p--; // point to start of trailing blanks
}
// Check for missing target.
if (p == pline) {
return NULL;
}
*p = '\0'; // trim trailing blanks & '='
// Perform macro substitution on target.
*pszTarget = NULL;
if (!MakeMacroString(pszTarget, pline)) {
return NULL;
}
// Validate target name. If must be a non-empty string of
// valid macro name characters.
if (**pszTarget == 0) {
FreeString(pszTarget, MT_DIRSTRING);
return NULL;
}
for (p = *pszTarget; *p != 0; p++) {
if (!MACRO_CHAR(*p)) {
FreeString(pszTarget, MT_DIRSTRING);
return NULL;
}
}
CompressBlanks(pvalue);
}
return (pvalue);
}
//+---------------------------------------------------------------------------
//
// Function: SubstituteString
//
// Synopsis: Perform any macro substitution. This code was copied from the
// nmake source.
//
// Arguments:
//
// Returns:
//
// Notes:
//
//----------------------------------------------------------------------------
void
SubstituteString(
char **result,
char **name,
char **dest,
char **end,
char *source,
unsigned *length
)
{
#define ESCH '^'
PVOID pReallocResult;
char *oldString, *newString;
char *pEq, *pPar, *t;
char *s;
unsigned i;
++*name;
for (pEq = *name; *pEq && *pEq != '='; pEq++)
if (*pEq == ESCH)
pEq++;
// Did we find the '=' sign?
if (*pEq != '=')
printf("Error1\n");
// Did the user forget the initial string?
if (pEq == *name)
printf("Error2\n");
for (pPar = pEq; *pPar && *pPar != ')'; pPar++)
if (*pPar == ESCH)
pPar++;
if (*pPar != ')')
printf("Error3\n");
oldString = (char *)malloc((UINT)((pEq - *name) + 1));
if (!oldString) {
BuildError("(Fatal Error) Out Of Memory: SubstituteString()\r\n");
exit(16);
}
for (s = oldString, t = *name; *t != '='; *s++ = *t++)
if (*t == ESCH)
++t;
*s = '\0';
i = strlen(oldString);
newString = (char *)malloc((UINT)(pPar - pEq));
if (!newString) {
BuildError("(Fatal Error) Out Of Memory: SubstituteString()\r\n");
exit(16);
}
for (s = newString, t++; *t != ')'; *s++ = *t++)
if (*t == ESCH)
++t;
*s = '\0';
*name = pPar + 1;
while (*source) {
if ((*source == *oldString) // check for match
&& !strncmp(source, oldString, i)) { // copy new in for
for (s = newString; *s; *(*dest)++ = *s++) // old string
if (*dest == *end) {
pReallocResult = realloc(*result, *length+100);
if (!pReallocResult) {
BuildError("(Fatal Error) Out Of Memory: SubstituteString()\r\n");
exit(16);
}
*result = pReallocResult;
*dest = *result + *length;
*length += 100;
*end = *result + *length;
}
source += i;
continue;
}
if (*dest == *end) {
pReallocResult = realloc(*result, *length+100);
if (!pReallocResult) {
BuildError("(Fatal Error) Out Of Memory: SubstituteString()\r\n");
exit(16);
}
*result = pReallocResult;
*dest = *result + *length;
*length += 100;
*end = *result + *length;
}
*(*dest)++ = *source++; // else copy 1 char
}
free(oldString);
free(newString);
}
//+---------------------------------------------------------------------------
//
// Function: MakeMacroString
//
// Synopsis: Take a string, and expand any macros in it. (e.g.
// "$(BASEDIR)\foobar\myfile.lib" is expanded to
// "f:\nt\private\foobar\myfile.lib" if $(BASEDIR) has a value of
// "f:\nt\private".
//
// Arguments: [pp] -- Output string
// [psrc] -- Input string
//
// Returns:
//
// Notes: Any previous string value in [pp] is freed before updating it.
//
//----------------------------------------------------------------------------
char MMSBuffer[64*1024];
BOOL
MakeMacroString(LPSTR *pp, LPSTR psrc)
{
LPSTR pname, p2, pdst, p3;
int cb;
char chTerminator;
int cNameChars;
int cChars;
pdst = MMSBuffer;
cb = strlen(psrc);
if (cb > sizeof(MMSBuffer) - 1) {
BuildError(
"(Fatal Error) Buffer overflow: MakeMacroString(%s)\r\n",
psrc);
exit(16);
}
while ((pname = strchr(psrc, '$')) != NULL &&
((pname[1] == LPAREN &&
(p2 = strchr(pname, RPAREN)) != NULL) ||
(MACRO_CHAR(pname[1]) &&
!MACRO_CHAR(pname[2])))) {
LPSTR pszvalue;
// Handle one-character non-paren macro usage.
if (pname[1] == LPAREN) {
// Initialize cNameChars with the number of chars to
// skip to get to the first name character.
cNameChars = 2;
} else {
p2 = pname + 2;
cNameChars = 1;
}
chTerminator = *p2;
*pname = *p2 = '\0';
// copy up to macro name
cChars = strlen(psrc);
memcpy(pdst, psrc, cChars + 1);
psrc += cChars;
pdst += cChars;
*pname = '$';
pname += cNameChars;
cNameChars += strlen(pname) + (chTerminator == RPAREN ? 1 : 0);
p3 = NULL;
if (chTerminator == RPAREN &&
(p3 = strchr(pname, ':')) != NULL) {
// macro substitution exists. ie: $(foo:old=new)
*p3 = '\0';
}
if ((pszvalue = FindMacro(pname)) == NULL &&
(pszvalue = getenv(pname)) == NULL &&
(pszvalue = GetBaseDir(pname)) == NULL) {
pszvalue = ""; // can't find macro name -- ignore it
}
if (p3) {
char *pNew = malloc(10);
char *pResult = pNew;
char *pEnd = pNew+10;
unsigned Len = 10;
if (!pNew) {
BuildError("(Fatal Error) Internal buffer overflow: MakeMacroString(%s[%s = %s]%s)\r\n",
MMSBuffer,
pname,
pszvalue,
p2 + 1);
exit(16);
}
*p3 = ':';
*p2=RPAREN;
SubstituteString(&pResult, &p3, &pNew, &pEnd, pszvalue, &Len);
*pNew = '\0';
*p2='\0';
pszvalue = pResult;
}
cb += strlen(pszvalue) - cNameChars;
assert(cb >= 0);
if (cb > sizeof(MMSBuffer) - 1) {
BuildError(
"(Fatal Error) Internal buffer overflow: MakeMacroString(%s[%s = %s]%s)\r\n",
MMSBuffer,
pname,
pszvalue,
p2 + 1);
exit(16);
}
strcpy(pdst, pszvalue); // copy expanded value
if (p3) {
free(pszvalue);
}
pdst += strlen(pdst);
*p2 = chTerminator;
psrc += cNameChars;
}
strcpy(pdst, psrc); // copy rest of string
if (pdst != MMSBuffer) {
CompressBlanks(MMSBuffer);
}
p2 = *pp;
*pp = NULL;
if (MMSBuffer[0] != '\0') {
MakeString(pp, MMSBuffer, TRUE, MT_DIRSTRING);
}
if (p2 != NULL) {
FreeMem(&p2, MT_DIRSTRING);
}
return (MMSBuffer[0] != '\0');
}
//+---------------------------------------------------------------------------
//
// Function: SetMacroString
//
// Synopsis: If the two macro names are the same, store the value in that
// macro
//
// Arguments: [pMacro1] -- Name of first macro
// [pMacro2] -- Name of second macro
// [pValue] -- Unexpanded value to store.
// [ppValue] -- Expanded value of macro.
//
// Returns: BOOL
//
//----------------------------------------------------------------------------
BOOL
SetMacroString(LPSTR pMacro1, LPSTR pMacro2, LPSTR pValue, LPSTR *ppValue)
{
if (_stricmp(pMacro1, pMacro2) == 0) {
MakeMacroString(ppValue, pValue);
return (TRUE); // return TRUE even if MakeMacroString stored a NULL
}
return (FALSE);
}
//+---------------------------------------------------------------------------
//
// Function: SplitToken
//
// Synopsis: Split the string at the given separator character or space.
//
// Arguments: [pbuf] -- First part of split string returned here.
// [chsep] -- Separator character.
// [ppstr] -- Source string to split. Becomes the second half.
//
// Returns: TRUE if the split was successful. FALSE if it wasn't split.
//
// Notes: If *ppstr = "path\filename" and chsep = '\' on input, then
// pbuf = "path" and *ppstr = "\filename" on output.
//
//----------------------------------------------------------------------------
BOOL
SplitToken(LPSTR pbuf, char chsep, LPSTR *ppstr)
{
LPSTR psrc, pdst;
psrc = *ppstr;
pdst = pbuf;
//BuildError("SplitToken('%c', '%s') ==> ", chsep, psrc);
while (*psrc == chsep || *psrc == ' ') {
psrc++;
}
while (*psrc != '\0' && *psrc != chsep && *psrc != ' ') {
*pdst = *psrc++;
if (*pdst == '/') {
*pdst = '\\';
}
pdst++;
}
*pdst = '\0';
*ppstr = psrc;
//BuildErrorRaw("('%s', '%s')\r\n", psrc, pbuf);
return (pdst != pbuf);
}
//+---------------------------------------------------------------------------
//
// Function: CrackSources
//
// Synopsis: Parse the SOURCES= line in a sources file and adds those source
// files to the list of sources in the DIRREC struct.
//
// Arguments: [pdr] -- Directory record
// [pds] -- Supplemental directory information
// [i] -- Which platform we're parsing
//
//----------------------------------------------------------------------------
VOID
CrackSources(
DIRREC *pdr,
DIRSUP *pds,
int i)
{
LPSTR pszsubdir, plist;
LPSTR pszfile, pszpath;
FILEREC *pfr;
DIRREC *pdrAssociate;
DIRREC *pdrParent;
DIRREC *pdrMachine;
DIRREC *pdrParentMachine;
DIRREC *pdrTarget;
DIRREC **ppdr;
LPSTR pszSources;
char path[DB_MAX_PATH_LENGTH];
TARGET_MACHINE_INFO *pMachine;
switch (i) {
case 0:
pMachine = TargetMachines[0];
pszSources = "SOURCES";
break;
case 1:
pMachine = TargetMachines[0];
pszSources = "OBJLIBFILES";
break;
default:
pMachine = PossibleTargetMachines[i - 2];
pszSources = pMachine->SourceVariable;
break;
}
pdrAssociate = pdrParent = pdrMachine = pdrParentMachine = pdrTarget = NULL;
plist = pds->SourcesVariables[i];
while (SplitToken(path, ' ', &plist)) {
UCHAR SubDirMask, SrcFlags;
SubDirMask = 0;
ppdr = &pdr; // assume current directory
pszsubdir = path;
if (pszsubdir[0] == '.' && pszsubdir[1] == '\\') {
BuildError(
"%s: Ignoring current directory prefix in %s= entry: %s\r\n",
pdr->Name,
pszSources,
path);
pszsubdir += 2;
}
if (pszsubdir[0] == '.' &&
pszsubdir[1] == '.' &&
pszsubdir[2] == '\\') {
SubDirMask = TMIDIR_PARENT;
ppdr = &pdrParent; // assume parent directory
pszsubdir += 3;
}
pszpath = path;
pszfile = strchr(pszsubdir, '\\');
if (pszfile == NULL) {
pszfile = pszsubdir;
} else {
LPSTR pszSecondSlash;
LPSTR pszAssociateDir;
LPSTR pszMachineDir;
// Check for second slash and handle $O\. If there is
// no second slash, check for a machine specific directory name.
// Second slashes are not legal if there's already been
// a '..'.
if ((SubDirMask & TMIDIR_PARENT) == 0) {
pszSecondSlash = strchr(pszfile + 1, '\\');
if (pszSecondSlash != NULL) {
pszfile = pszSecondSlash;
}
} else {
pszSecondSlash = NULL;
}
*pszfile = '\0';
if (pszSecondSlash != NULL) {
pszMachineDir = pMachine->ObjectDirectory[iObjectDir];
pszAssociateDir = pszMachineDir;
} else {
pszMachineDir = pMachine->SourceDirectory;
pszAssociateDir = pMachine->AssociateDirectory;
}
if (((_stricmp(pszsubdir, pszAssociateDir) != 0) &&
(_stricmp(pszsubdir, pszMachineDir) != 0)) ||
strchr(pszfile + 1, '\\') != NULL) {
*pszfile = '\\';
/*
* Managed code such as C# and VB.NET builds fine with a
* "prefix" directory, so don't complain in that case.
*/
if (!(pdr->DirFlags & DIRDB_MANAGED_CODE) && strcmp(pszSources, "OBJLIBFILES")) {
BuildError(
"%s: Ignoring invalid directory prefix in %s= entry: %s\r\n",
pdr->Name,
pszSources,
path);
}
pszpath = strrchr(path, '\\');
assert(pszpath != NULL);
pszpath++;
SubDirMask = 0;
ppdr = &pdr; // default to current direcory
} else {
SubDirMask |= pMachine->SourceSubDirMask;
*pszfile++ = '\\';
if (SubDirMask & TMIDIR_PARENT) {
ppdr = &pdrParentMachine;
} else if (pszSecondSlash != NULL) {
// Must have matched $O.
ppdr = &pdrTarget;
} else {
if (_stricmp(pszsubdir, pszMachineDir) != 0) {
ppdr = &pdrMachine;
} else {
ppdr = &pdrAssociate;
}
}
}
}
NewDirectory:
if (*ppdr == NULL) {
pfr = FindSourceFileDB(pdr, pszpath, ppdr);
} else {
pfr = LookupFileDB(*ppdr, pszfile);
}
SrcFlags = SOURCEDB_SOURCES_LIST;
if ((pfr == NULL) && !fPassZero) {
if (fDebug) {
BuildError("%s: Missing source file: %s\r\n", pdr->Name, path);
}
if (*ppdr == NULL) {
if (fDebug || pszpath == path) {
BuildError(
"%s: Directory does not exist: %s\r\n",
pdr->Name,
path);
}
// Probably an error in the subordinate sources file.
// since old versions of build managed to get these entries
// into the objects lists, we have to do the same...
//
// If ..\ prefix exists, strip it off and try again.
// Else try again with the current directory.
if (SubDirMask & TMIDIR_PARENT) {
SubDirMask &= ~TMIDIR_PARENT; // strip off "..\\"
} else {
SubDirMask = 0; // use current direcory
}
if (SubDirMask == 0) {
ppdr = &pdr; // current direcory
pszpath = pszfile;
} else {
ppdr = &pdrMachine; // machine sub dir
pszpath = pszsubdir;
}
goto NewDirectory;
}
pfr = InsertFileDB(*ppdr, pszfile, 0, 0, FILEDB_FILE_MISSING);
if (pfr == NULL && strcmp(pszSources, "OBJLIBFILES")) {
BuildError(
"%s: Ignoring invalid %s= entry: %s\r\n",
pdr->Name,
pszSources,
path);
}
}
if (pfr != NULL) {
AssertFile(pfr);
if (SubDirMask == 0) {
pfr->FileFlags |= FILEDB_OBJECTS_LIST;
}
if (pfr->FileFlags & FILEDB_FILE_MISSING) {
SrcFlags |= SOURCEDB_FILE_MISSING;
}
InsertSourceDB(&pds->psrSourcesList[i], pfr, SubDirMask, SrcFlags);
}
}
}
//+---------------------------------------------------------------------------
//
// Function: SaveUserTests
//
// Synopsis: Save the value of the UMTEST macro into the DIRREC struct.
//
// Arguments: [DirDB] -- Dir struct to save into
// [TextLine] -- String from UMTEST= line in sources file
//
//----------------------------------------------------------------------------
VOID
SaveUserTests(
PDIRREC DirDB,
LPSTR TextLine)
{
UINT i;
BOOL fSave = FALSE;
char name[DB_MAX_PATH_LENGTH];
char buf[512];
buf[0] = '\0';
if (DirDB->UserTests != NULL) {
strcpy(buf, DirDB->UserTests);
}
CopyString(TextLine, TextLine, TRUE);
while (SplitToken(name, '*', &TextLine)) {
for (i = 0; i < CountOptionalDirs; i++) {
if (!strcmp(name, OptionalDirs[i])) {
if (buf[0] != '\0') {
strcat(buf, "*");
DirDB->DirFlags |= DIRDB_FORCELINK; // multiple targets
}
strcat(buf, name);
fSave = TRUE;
break;
}
}
}
if (fSave) {
MakeMacroString(&DirDB->UserTests, buf);
DirDB->DirFlags |= DIRDB_LINKNEEDED;
}
}
void
AddProduceDependency(
DIRREC *DirDB,
LPSTR Value
)
{
char produces[DB_MAX_PATH_LENGTH];
PDEPENDENCY Dependency;
while (SplitToken(produces, ' ',&Value)) {
AllocMem(sizeof(DEPENDENCY)+strlen(produces), &Dependency, MT_DEPENDENCY);
InitializeListHead(&Dependency->WaitList);
InsertTailList(&DirDB->Produces, &Dependency->DependencyList);
Dependency->Producer = DirDB;
strcpy(Dependency->Name, produces);
Dependency->CheckSum = CheckSum(Dependency->Name);
Dependency->Next = AllDependencies;
Dependency->Done = FALSE;
Dependency->hEvent = CreateEvent(NULL, TRUE, FALSE, NULL);
if (Dependency->hEvent == NULL) {
BuildMsg("Failed to create dependency hEvent for dependency %s in directory %s\n",
Dependency, DirDB->Name);
}
AllDependencies = Dependency;
}
}
void
AddConsumeDependency(
DIRREC *DirDB,
LPSTR Value
)
{
char consumes[DB_MAX_PATH_LENGTH];
PDEPENDENCY Dependency;
PDEPENDENCY_WAIT Wait;
USHORT sum;
while (SplitToken(consumes, ' ',&Value)) {
//
// Find the dependency, which must have already been created by the
// producer.
//
sum = CheckSum(consumes);
Dependency = AllDependencies;
while (Dependency) {
if ((Dependency->CheckSum == sum) &&
(strcmp(Dependency->Name, consumes)==0)) {
break;
}
Dependency = Dependency->Next;
}
if (Dependency != NULL) {
AllocMem(sizeof(DEPENDENCY_WAIT), &Wait, MT_DEPENDENCY_WAIT);
InsertTailList(&DirDB->Consumes, &Wait->ListEntry);
Wait->Consumer = DirDB;
Wait->Dependency = Dependency;
}
}
}
//+---------------------------------------------------------------------------
//
// Function: ProcessSourcesFileLine
//
// Synopsis: Given a line from a sources file, do the right thing.
//
// Arguments: [DirDB] -- Directory containing sources file
// [pds] -- Supplementary info on directory
// [TextLine] -- Line to process
//
// Returns: void
//
//----------------------------------------------------------------------------
void
ProcessSourcesFileLine(
DIRREC *DirDB, // Current Directory record
DIRSUP *pds, // Supplemental Directory record
LPSTR TextLine, // Line to process
int iTarget // Index into target machine array.
)
{
LPSTR MacroName, p1;
UINT i, iMacro;
char path[DB_MAX_PATH_LENGTH];
BOOL fCleanNTTargetFile0 = FALSE;
LPSTR pValue; // right side of equal sign
LPSTR pszTemp=NULL;
pValue = SplitMacro(&TextLine);
if (pValue == NULL) {
return;
}
// Note: TextLine is now the left side of the equal sign. See if it's interesting.
//
// This sets pds->SourcesVariables[0] to the value of SOURCES= if
// the current line is SOURCES=...
//
if (SetMacroString(
"SOURCES",
TextLine,
pValue,
&pds->SourcesVariables[0])) {
DirDB->DirFlags |= DIRDB_SOURCES_SET;
goto SaveAndFreeMacro;
} else if (SetMacroString(
"OBJLIBFILES",
TextLine,
pValue,
&pds->SourcesVariables[1])) {
DirDB->DirFlags |= DIRDB_SOURCES_SET;
goto SaveAndFreeMacro;
} else {
for (i = 0; i < MAX_TARGET_MACHINES; i++) {
//
// This sets pds->SourcesVariables[0] to the value of
// PLAT_SOURCES= if the current line is PLAT_SOURCES=...
//
if (SetMacroString(
PossibleTargetMachines[i]->SourceVariable,
TextLine,
pValue,
&pds->SourcesVariables[i + 2])) {
DirDB->DirFlags |= DIRDB_SOURCES_SET;
goto SaveAndFreeMacro;
}
}
}
// Not a SOURCES or xxx_SOURCES macro, check against all the other interesting
// macro names.
iMacro = 0;
while ((MacroName = RelevantSourcesMacros[iMacro]) != NULL) {
if (_stricmp(TextLine, MacroName) == 0) {
break;
}
iMacro++;
}
if (MacroName != NULL) { // if macro name found in list
switch (iMacro) {
case SOURCES_TARGETNAME:
MakeMacroString(&DirDB->TargetName, pValue);
break;
case SOURCES_TARGETPATH:
if (strcmp(pValue, "obj") == 0) {
pValue = pszObjDir;
}
MakeMacroString(&DirDB->TargetPath, pValue);
if (DirDB->TargetPath != NULL) {
CreateBuildDirectory(DirDB->TargetPath);
for (i = 0; i < CountTargetMachines; i++) {
p1 = TargetMachines[i]->ObjectDirectory[iObjectDir];
assert(strncmp(pszObjDirSlash, p1, strlen(pszObjDirSlash)) == 0);
p1 += strlen(pszObjDirSlash);
sprintf(path, "%s\\%s", DirDB->TargetPath, p1);
CreateBuildDirectory(path);
}
}
break;
case SOURCES_TARGETPATHLIB:
if (strcmp(pValue, "obj") == 0) {
pValue = pszObjDir;
}
MakeMacroString(&DirDB->TargetPathLib, pValue);
if (DirDB->TargetPathLib != NULL) {
CreateBuildDirectory(DirDB->TargetPathLib);
for (i = 0; i < CountTargetMachines; i++) {
p1 = TargetMachines[i]->ObjectDirectory[iObjectDir];
assert(strncmp(pszObjDirSlash, p1, strlen(pszObjDirSlash)) == 0);
p1 += strlen(pszObjDirSlash);
sprintf(path, "%s\\%s", DirDB->TargetPathLib, p1);
CreateBuildDirectory(path);
}
}
break;
case SOURCES_TARGETTYPE:
if (!_stricmp(pValue, "PROGRAM") || !_stricmp(pValue, "PROGLIB")) {
MakeMacroString(&DirDB->TargetExt, ".exe");
DirDB->DirFlags |= DIRDB_LINKNEEDED;
} else if (!_stricmp(pValue, "OBJLIB")) {
MakeMacroString(&DirDB->TargetExt, ".olb");
DirDB->DirFlags |= DIRDB_LINKNEEDED;
} else if (!_stricmp(pValue, "DRIVER") || !_stricmp(pValue, "MINIPORT")) {
MakeMacroString(&DirDB->TargetExt, ".sys");
DirDB->DirFlags |= DIRDB_LINKNEEDED;
} else if (!_stricmp(pValue, "GDI_DRIVER")) {
MakeMacroString(&DirDB->TargetExt, ".dll");
DirDB->DirFlags |= DIRDB_LINKNEEDED;
} else if (!_stricmp(pValue, "EXPORT_DRIVER")) {
MakeMacroString(&DirDB->TargetExt, ".sys");
DirDB->DirFlags |= DIRDB_LINKNEEDED;
DirDB->DirFlags |= DIRDB_DLLTARGET;
} else if (!_stricmp(pValue, "DYNLINK") || !_stricmp(pValue, "HAL")) {
MakeMacroString(&DirDB->TargetExt, ".dll");
DirDB->DirFlags |= DIRDB_LINKNEEDED;
DirDB->DirFlags |= DIRDB_DLLTARGET;
} else if ((!_stricmp(pValue, "LIBRARY")) || (!_stricmp(pValue, "DRIVER_LIBRARY"))) {
MakeMacroString(&DirDB->TargetExt, ".lib");
DirDB->DirFlags &= ~DIRDB_LINKNEEDED;
} else if (!_stricmp(pValue, "UMAPPL_NOLIB")) {
DirDB->DirFlags &= ~DIRDB_LINKNEEDED;
} else if (!_stricmp(pValue, "NOTARGET")) {
//
// Used to indicate no target for a directory,
// e.g. if only pass0 files are generated
pds->fNoTarget = TRUE;
if (!fQuicky || (fQuickZero && fFirstScan)) {
DirDB->DirFlags |= DIRDB_PASS0NEEDED;
}
} else {
BuildError( "Unsupported TARGETTYPE value - %s\r\n", pValue);
}
break;
case SOURCES_TARGETEXT:
{
char TargetExt[_MAX_EXT] = ".";
strcat(TargetExt, pValue);
MakeMacroString(&DirDB->TargetExt, TargetExt);
}
break;
case SOURCES_INCLUDES:
MakeMacroString(&pds->LocalIncludePath, pValue);
if (DEBUG_1) {
BuildMsg(
" Found local INCLUDES=%s\r\n",
pds->LocalIncludePath);
}
break;
case SOURCES_USER_INCLUDES:
MakeMacroString(&pds->UserIncludePath, pValue);
if (DEBUG_1) {
BuildMsg(
" Found local USER_INCLUDES=%s\r\n",
pds->UserIncludePath);
}
break;
case SOURCES_LAST_INCLUDES:
MakeMacroString(&pds->LastIncludePath, pValue);
if (DEBUG_1) {
BuildMsg(
" Found local LAST_INCLUDES=%s\r\n",
pds->LastIncludePath);
}
break;
case SOURCES_MFC_INCLUDES:
// MFC_INCLUDES/SDK_INC/CRT_INC/OAK_INC really can't be changed
// in the sources file (yet) since we've already processed the
// system includes. Lay the groundwork for now.
MakeMacroString((char **)&pszIncMfc, pValue);
break;
case SOURCES_SDK_LIB_DEST:
MakeMacroString((char **)&pszSdkLibDest, pValue);
break;
case SOURCES_DDK_LIB_DEST:
MakeMacroString((char **)&pszDdkLibDest, pValue);
break;
case SOURCES_SDK_INC_PATH:
MakeMacroString((char **)&pszIncSdk, pValue);
break;
case SOURCES_CRT_INC_PATH:
MakeMacroString((char **)&pszIncCrt, pValue);
break;
case SOURCES_OAK_INC_PATH:
MakeMacroString((char **)&pszIncOak, pValue);
break;
case SOURCES_DDK_INC_PATH:
MakeMacroString((char **)&pszIncDdk, pValue);
break;
case SOURCES_WDM_INC_PATH:
MakeMacroString((char **)&pszIncWdm, pValue);
break;
case SOURCES_PRIVATE_INC_PATH:
MakeMacroString((char **)&pszIncPri, pValue);
break;
case SOURCES_PRECOMPILED_PCH:
MakeMacroString(&DirDB->Pch, pValue);
break;
case SOURCES_PRECOMPILED_OBJ:
MakeMacroString(&DirDB->PchObj, pValue);
break;
case SOURCES_PRECOMPILED_INCLUDE:
case SOURCES_PRECOMPILED_TARGET:
{
LPSTR *ppszPath, *ppszFile, p;
if (iMacro == SOURCES_PRECOMPILED_INCLUDE) {
ppszPath = &pds->PchIncludeDir;
ppszFile = &pds->PchInclude;
} else {
ppszPath = &pds->PchTargetDir;
ppszFile = &pds->PchTarget;
}
MakeMacroString(ppszPath, ""); // free old string
if (!MakeMacroString(ppszFile, pValue)) {
break;
}
p = *ppszFile + strlen(*ppszFile);
while (p > *ppszFile && *--p != '\\')
;
if (p > *ppszFile) {
*p = '\0';
MakeMacroString(ppszPath, *ppszFile);
MakeMacroString(ppszFile, p + 1);
}
if (DEBUG_1) {
BuildMsg(
"Precompiled header%s is %s in directory %s\r\n",
iMacro == SOURCES_PRECOMPILED_INCLUDE?
"" : " target",
*ppszFile,
*ppszPath != NULL?
*ppszPath : "'.'");
}
}
if (iMacro == SOURCES_PRECOMPILED_INCLUDE ||
pds->PchTargetDir == NULL) {
break;
}
EnsureDirectoriesExist(pds->PchTargetDir);
break;
case SOURCES_PASS0_HEADERDIR:
MakeMacroString(&pds->PassZeroHdrDir, pValue);
EnsureDirectoriesExist(pds->PassZeroHdrDir);
if (DEBUG_1) {
BuildMsg("Pass Zero Header Directory is '%s'\r\n",
pds->PassZeroHdrDir);
}
break;
case SOURCES_PASS0_SOURCEDIR:
// SOURCES_PASS0_SOURCEDIR and SOURCES_PASS0_CLIENTDIR
// are mutually exclusive - enforced by makefile.def
DirDB->DirFlags &= ~DIRDB_IDLTYPERPC;
MakeMacroString(&pds->PassZeroSrcDir1, pValue);
EnsureDirectoriesExist(pds->PassZeroSrcDir1);
if (DEBUG_1) {
BuildMsg("Pass Zero Source Directory is '%s'\r\n",
pds->PassZeroSrcDir1);
}
break;
case SOURCES_PASS0_CLIENTDIR:
// SOURCES_PASS0_SOURCEDIR and SOURCES_PASS0_CLIENTDIR
// are mutually exclusive - enforced by makefile.def
DirDB->DirFlags |= DIRDB_IDLTYPERPC;
MakeMacroString(&pds->PassZeroSrcDir1, pValue);
EnsureDirectoriesExist(pds->PassZeroSrcDir1);
if (DEBUG_1) {
BuildMsg("Pass Zero Client Directory is '%s'\r\n",
pds->PassZeroSrcDir1);
}
break;
case SOURCES_MIDL_UUIDDIR:
case SOURCES_PASS0_UUIDDIR:
// SOURCES_PASS0_UUIDDIR and SOURCES_PASS0_SERVERDIR
// are mutually exclusive - enforced by makefile.def
DirDB->DirFlags &= ~DIRDB_IDLTYPERPC;
MakeMacroString(&pds->PassZeroSrcDir2, pValue);
EnsureDirectoriesExist(pds->PassZeroSrcDir2);
if (DEBUG_1) {
BuildMsg("Pass Zero UUID Source Directory is '%s'\r\n",
pds->PassZeroSrcDir2);
}
break;
case SOURCES_PASS0_SERVERDIR:
// SOURCES_PASS0_UUIDDIR and SOURCES_PASS0_SERVERDIR
// are mutually exclusive - enforced by makefile.def
DirDB->DirFlags |= DIRDB_IDLTYPERPC;
MakeMacroString(&pds->PassZeroSrcDir2, pValue);
EnsureDirectoriesExist(pds->PassZeroSrcDir2);
if (DEBUG_1) {
BuildMsg("Pass Zero Server Directory is '%s'\r\n",
pds->PassZeroSrcDir2);
}
break;
case SOURCES_NTTEST:
if (MakeMacroString(&DirDB->KernelTest, pValue)) {
DirDB->DirFlags |= DIRDB_LINKNEEDED;
}
break;
case SOURCES_UMTYPE:
MakeMacroString(&pds->TestType, pValue);
if (DEBUG_1) {
BuildMsg(
" Found UMTYPE=%s\r\n",
pds->TestType);
}
break;
case SOURCES_UMTEST:
case SOURCES_OPTIONAL_UMTEST:
SaveUserTests(DirDB, pValue);
break;
case SOURCES_UMAPPL:
if (MakeMacroString(&DirDB->UserAppls, pValue)) {
DirDB->DirFlags |= DIRDB_LINKNEEDED;
}
break;
case SOURCES_UMAPPLEXT:
if (!_stricmp(pValue, ".exe")) {
MakeMacroString(&DirDB->TargetExt, ".exe");
} else
if (!_stricmp(pValue, ".com")) {
MakeMacroString(&DirDB->TargetExt, ".com");
} else
if (!_stricmp(pValue, ".scr")) {
MakeMacroString(&DirDB->TargetExt, ".scr");
} else {
BuildError(
"Unsupported UMAPPLEXT value - %s\r\n",
pValue);
}
break;
case SOURCES_IDLTYPE:
if (!_stricmp(pValue, "ole")) {
pds->IdlType = 0;
} else
if (!_stricmp(pValue, "rpc")) {
pds->IdlType = 1;
} else {
BuildError(
"Unsupported IDL_TYPE value - %s\r\n",
pValue);
}
break;
case SOURCES_SOURCES_OPTIONS:
fCleanNTTargetFile0 = fClean && strstr(pValue, "-c0");
break;
case SOURCES_NTTARGETFILE0:
DirDB->DirFlags |= DIRDB_TARGETFILE0;
if (fCleanNTTargetFile0) {
MakeMacroString(&DirDB->NTTargetFile0, pValue);
}
break;
case SOURCES_NTTARGETFILES:
DirDB->DirFlags |= DIRDB_TARGETFILES;
break;
case SOURCES_CHICAGO_PRODUCT:
DirDB->DirFlags |= DIRDB_CHICAGO_INCLUDES;
break;
case SOURCES_CONDITIONAL_INCLUDES:
MakeMacroString(&pds->ConditionalIncludes, pValue);
break;
case SOURCES_SYNCHRONIZE_BLOCK:
//if ((!fIgnoreSync) && !(DirDB->DirFlags & (DIRDB_SYNC_PRODUCES | DIRDB_SYNC_CONSUMES))) {
if (!fIgnoreSync) {
DirDB->DirFlags |= DIRDB_SYNCHRONIZE_BLOCK;
}
break;
case SOURCES_SYNCHRONIZE_PASS2_BLOCK:
DirDB->DirFlags |= DIRDB_SYNCHRONIZE_PASS2_BLOCK;
break;
case SOURCES_SYNCHRONIZE_DRAIN:
// if ((!fIgnoreSync) && !(DirDB->DirFlags & (DIRDB_SYNC_PRODUCES | DIRDB_SYNC_CONSUMES))) {
if ((!fIgnoreSync) && !(DirDB->DirFlags & DIRDB_SYNC_CONSUMES)) {
DirDB->DirFlags |= DIRDB_SYNCHRONIZE_DRAIN;
}
break;
case SOURCES_SYNCHRONIZE_PASS2_DRAIN:
DirDB->DirFlags |= DIRDB_SYNCHRONIZE_PASS2_DRAIN;
break;
case SOURCES_CHECKED_ALT_DIR:
if (!fBuildAltDirSet) {
DirDB->DirFlags |= DIRDB_CHECKED_ALT_DIR;
if (DEBUG_1) {
BuildMsg("Found CHECKED_ALT_DIR\r\n");
}
SetObjDir(TRUE);
if (fCheckedBuild) {
SaveMacro("_OBJ_DIR", pszObjDir);
if (iTarget < 0) {
SaveMacro("O", TargetMachines[0]->
ObjectDirectory[iObjectDir]);
} else {
SaveMacro("O",
PossibleTargetMachines[iTarget/2]->
ObjectDirectory[iObjectDir]);
}
}
}
break;
case SOURCES_PROJECT_NAME:
sprintf(path, "%s\\%s", NtRoot, pValue);
SaveMacro("PROJECT_ROOT", path);
break;
case SOURCES_PASS0_PUBLISH:
DirDB->DirFlags |= DIRDB_PASS0NEEDED;
break;
case SOURCES_MANAGED_CODE:
case SOURCES_MANAGED_VB:
DirDB->DirFlags |= DIRDB_MANAGED_CODE;
break;
case SOURCES_SYNC_PRODUCES:
DirDB->DirFlags |= DIRDB_SYNC_PRODUCES;
//A producer directory can be a BLOCK or a DRAIN.
//DirDB->DirFlags &= ~(DIRDB_SYNCHRONIZE_BLOCK | DIRDB_SYNCHRONIZE_DRAIN);
MakeMacroString(&pszTemp, pValue);
AddProduceDependency(DirDB, pszTemp);
break;
case SOURCES_SYNC_CONSUMES:
DirDB->DirFlags |= DIRDB_SYNC_CONSUMES;
//Changed so that the same directory can be the consumer as well as a block...
//DirDB->DirFlags &= ~(DIRDB_SYNCHRONIZE_BLOCK | DIRDB_SYNCHRONIZE_DRAIN);
DirDB->DirFlags &= ~(DIRDB_SYNCHRONIZE_DRAIN);
MakeMacroString(&pszTemp, pValue);
AddConsumeDependency(DirDB, pszTemp);
break;
case SOURCES_PASS0_BINPLACE:
DirDB->DirFlags |= DIRDB_PASS0NEEDED;
break;
case SOURCES_PASS1_LINK:
DirDB->DirFlags &= ~DIRDB_LINKNEEDED;
break;
}
}
SaveAndFreeMacro:
SaveMacro(TextLine, pValue);
// Make sure we cleanup from the SplitMacro call at the top.
FreeString(&TextLine, MT_DIRSTRING);
}
void
ReadProjectsInfo(
DIRREC *DirDB,
DIRSUP *pds
)
{
FILE *FileHandle;
LPSTR TextLine, pBackSlash, pszProject;
BOOL Found;
char pszProjectMkPath[DB_MAX_PATH_LENGTH];
char path[DB_MAX_PATH_LENGTH];
// First load project.mk for this project.
strcpy(pszProjectMkPath, DirDB->Name);
strcpy(path, pszProjectMkPath);
strcat(path, "\\project.mk");
Found = !_access(path, 0);
while (!Found && strlen(pszProjectMkPath)) {
pBackSlash=strrchr(pszProjectMkPath, '\\');
if (pBackSlash) {
*pBackSlash = '\0';
} else {
return;
}
strcpy(path, pszProjectMkPath);
strcat(path, "\\project.mk");
Found = !_access(path, 0);
}
if (!Found) {
return;
}
if (!OpenFilePush(pszProjectMkPath, "project.mk", "#", &FileHandle)) {
return;
}
SaveMacro("_PROJECT_MK_PATH", pszProjectMkPath);
while ((TextLine = ReadLine(FileHandle)) != NULL) {
ProcessSourcesFileLine(DirDB, pds, TextLine, -1);
}
CloseReadFile(NULL);
// Load the optional myproject.mk
if (OpenFilePush(pszProjectMkPath, "myproject.mk", "#", &FileHandle)) {
while ((TextLine = ReadLine(FileHandle)) != NULL) {
ProcessSourcesFileLine(DirDB, pds, TextLine, -1);
}
CloseReadFile(NULL);
}
// Then load ntmakeenv\projects.inc to get the other magic macro names.
if (!OpenFilePush(getenv("NTMAKEENV"), "projects.inc", "#", &FileHandle)) {
return;
}
while ((TextLine = ReadLine(FileHandle)) != NULL) {
ProcessSourcesFileLine(DirDB, pds, TextLine, -1);
}
CloseReadFile(NULL);
}
//+---------------------------------------------------------------------------
//
// Function: ReadSourcesFile
//
// Synopsis: Parses the sources files (common and platform specific)
//
// Arguments: [DirDB] -- Directory containing sources file
// [pds] -- Supplementary info on directory
// [pDateTimeSources] -- Timestamp of Sources file
//
// Returns: TRUE if read successfully
//
//----------------------------------------------------------------------------
#define SAVE_STATIC_MACRO(MacroName, InitialValue) \
{ \
static char sz[] = InitialValue; \
SaveMacro(MacroName, sz); \
} \
BOOL
ReadSourcesFile(DIRREC *DirDB, DIRSUP *pds, ULONG *pDateTimeSources)
{
FILE *InFileHandle;
LPSTR p, TextLine;
UINT i;
int iTarget;
ULONG DateTime;
char path[DB_MAX_PATH_LENGTH];
char temp[DB_MAX_PATH_LENGTH];
memset(pds, 0, sizeof(*pds));
pds->fNoTarget=FALSE;
assert(DirDB->TargetPath == NULL);
assert(DirDB->TargetPathLib == NULL);
assert(DirDB->TargetName == NULL);
assert(DirDB->TargetExt == NULL);
assert(DirDB->KernelTest == NULL);
assert(DirDB->UserAppls == NULL);
assert(DirDB->UserTests == NULL);
assert(DirDB->NTTargetFile0 == NULL);
assert(DirDB->Pch == NULL);
assert(DirDB->PchObj == NULL);
assert(cMacro == 0);
*pDateTimeSources = 0;
//
// Read the information in each of the target specific directories
// and simulate concatenation of all of the sources files.
//
// Possible sources files are read from DirDB->Name | target-source
// and DirDb->Name | ..\target-source.
//
// iTarget values, and the corresponding files processed are:
// -1 sources.
// 0 PossibleTargetMachines[0]\sources.
// 1 ..\PossibleTargetMachines[0]\sources.
// 2 PossibleTargetMachines[1]\sources.
// 3 ..\PossibleTargetMachines[1]\sources.
// 4 PossibleTargetMachines[2]\sources.
// 5 ..\PossibleTargetMachines[2]\sources.
SaveMacro("MAKEDIR", DirDB->Name);
SaveMacro("SDK_LIB_DEST", pszSdkLibDest);
SaveMacro("DDK_LIB_DEST", pszDdkLibDest);
SaveMacro("PUBLIC_INTERNAL_PATH", pszPublicInternalPath);
// Use the default architecture for now
SaveMacro("TARGET_DIRECTORY", TargetMachines[0]->SourceDirectory);
SetObjDir(FALSE);
SaveMacro("_OBJ_DIR", pszObjDir);
// Define a default CONDITIONAL_INCLUDES line to deal with the mac hdrs in windows/rpc/ole32.h.
MakeMacroString(&pds->ConditionalIncludes, "winwlm.h rpcmac.h rpcerr.h macapi.h macname1.h macname2.h macocidl.h macpub.h macwin32.h");
// Before processing the sources file, see if there's a projects.mk in the tree.
ReadProjectsInfo(DirDB, pds);
SAVE_STATIC_MACRO("PROJECT_PUBLIC_PATH", "$(PUBLIC_INTERNAL_PATH)\\$(_PROJECT_)");
SAVE_STATIC_MACRO("PROJECT_INC_PATH", "$(PROJECT_PUBLIC_PATH)\\inc");
strcpy(temp,"$(PROJECT_ROOT)\\inc;$(PROJECT_ROOT)\\inc\\$(O);$(PROJECT_INC_PATH)");
MakeMacroString(&pds->NTIncludePath, temp);
for (iTarget = -1; iTarget < 2*MAX_TARGET_MACHINES; iTarget++) {
path[0] = '\0';
if (iTarget >= 0) {
if (iTarget & 1) {
strcat(path, "..\\");
}
strcat(path, PossibleTargetMachines[iTarget/2]->SourceDirectory);
strcat(path, "\\");
}
strcat(path, "sources");
if (!OpenFilePush(DirDB->Name, path, "#", &InFileHandle)) {
if (iTarget == -1) {
FreeMacros();
return (FALSE);
}
continue; // skip non-existent subordinate sources files
}
if (DEBUG_1) {
BuildMsg(
" Scanning%s file %s\r\n",
iTarget >= 0 ? " subordinate" : "",
FormatPathName(DirDB->Name, path));
}
// Update per-target macros.
if (iTarget < 0) {
SaveMacro("TARGET_DIRECTORY",
TargetMachines[0]->SourceDirectory);
SaveMacro("O", TargetMachines[0]->
ObjectDirectory[iObjectDir]);
} else {
SaveMacro("TARGET_DIRECTORY",
PossibleTargetMachines[iTarget/2]->SourceDirectory);
SaveMacro("O", PossibleTargetMachines[iTarget/2]->
ObjectDirectory[iObjectDir]);
}
SAVE_STATIC_MACRO("SDK_LIB_PATH", "$(SDK_LIB_DEST)\\$(TARGET_DIRECTORY)");
SAVE_STATIC_MACRO("PROJECT_LIB_PATH", "$(PROJECT_PUBLIC_PATH)\\lib\\$(TARGET_DIRECTORY)");
DirDB->DirFlags |= DIRDB_SOURCESREAD;
while ((TextLine = ReadLine(InFileHandle)) != NULL) {
ProcessSourcesFileLine(DirDB, pds, TextLine, iTarget);
}
// Subordinate files close themselves at EOF. Timestamps
// are propagated in CloseReadFile so the primary
// file's timestamp is automatically updated.
}
// Close the primary file.
DateTime = CloseReadFile(NULL);
if (*pDateTimeSources < DateTime) {
*pDateTimeSources = DateTime; // keep newest timestamp
}
if (!pds->fNoTarget && (DirDB->TargetPath == NULL)) {
strcpy(path, "sources");
SetupReadFile(DirDB->Name, path, "#", &InFileHandle);
BuildError(
"Unknown TARGETPATH value\r\n",
NULL);
CloseReadFile(NULL);
}
FreeMacros();
if (fChicagoProduct) {
DirDB->DirFlags |= DIRDB_CHICAGO_INCLUDES;
}
//
// Directory has pass0 files in it (.idl, .mc, .asn, etc), check to make
// sure they specified where the generated files should go. Default to the
// obj subdirectories if they didn't. These always need to be non-null.
//
if (!pds->PassZeroHdrDir) {
MakeString(&pds->PassZeroHdrDir, ".", TRUE, MT_DIRSTRING);
}
if (!pds->PassZeroSrcDir1) {
MakeString(&pds->PassZeroSrcDir1, ".", TRUE, MT_DIRSTRING);
}
if (!pds->PassZeroSrcDir2)
MakeString(&pds->PassZeroSrcDir2, pds->PassZeroSrcDir1, TRUE, MT_DIRSTRING);
if (DirDB->UserTests != NULL) {
_strlwr(DirDB->UserTests);
}
if (DirDB->UserAppls != NULL) {
if (DirDB->UserTests != NULL || strchr(DirDB->UserAppls, '*') != NULL) {
DirDB->DirFlags |= DIRDB_FORCELINK; // multiple targets
}
}
PostProcessSources(DirDB, pds);
if (DEBUG_1) {
PrintDirDB(DirDB, 1|2);
PrintDirSupData(pds);
PrintDirDB(DirDB, 4);
}
pds->DateTimeSources = *pDateTimeSources;
return (TRUE);
}
#undef SAVE_STATIC_MACRO
//+---------------------------------------------------------------------------
//
// Function: PostProcessSources
//
// Synopsis: Scan the files in the given directory and add files to the
// directory's list of source files (SOURCEREC), including PCH
// files, UMTEST files, etc.
//
// Arguments: [pdr] -- Directory to process
// [pds] -- Directory supplemental information
//
//----------------------------------------------------------------------------
void
PostProcessSources(DIRREC *pdr, DIRSUP *pds)
{
PFILEREC FileDB, *FileDBNext;
char path[DB_MAX_PATH_LENGTH];
LPSTR p, p1;
UINT i;
for (i = 0; i < MAX_TARGET_MACHINES + 2; i++) {
if (pds->SourcesVariables[i] != NULL) {
CrackSources(pdr, pds, i);
}
}
FileDBNext = &pdr->Files;
while (FileDB = *FileDBNext) {
if (pds->PchInclude && strcmp(FileDB->Name, pds->PchInclude) == 0) {
InsertSourceDB(&pds->psrSourcesList[0], FileDB, 0, SOURCEDB_PCH);
if (DEBUG_1) {
BuildMsg("Adding PCH file to Sources List: %s.\r\n", FileDB->Name);
}
}
if ((FileDB->FileFlags & (FILEDB_SOURCE | FILEDB_OBJECTS_LIST)) ==
FILEDB_SOURCE) {
p = FileDB->Name;
p1 = path;
while (*p != '\0' && *p != '.') {
*p1++ = *p++;
}
*p1 = '\0';
_strlwr(path);
if (pdr->KernelTest != NULL &&
!strcmp(path, pdr->KernelTest)) {
FileDB->FileFlags |= FILEDB_OBJECTS_LIST;
} else
if (pdr->UserAppls != NULL &&
(p = strstr(pdr->UserAppls, path)) &&
(p == pdr->UserAppls || p[-1] == '*' || p[-1] == ' ')) {
FileDB->FileFlags |= FILEDB_OBJECTS_LIST;
} else
if (pdr->UserTests != NULL &&
(p = strstr(pdr->UserTests, path)) &&
(p == pdr->UserTests || p[-1] == '*' || p[-1] == ' ')) {
FileDB->FileFlags |= FILEDB_OBJECTS_LIST;
}
if (FileDB->FileFlags & FILEDB_OBJECTS_LIST) {
InsertSourceDB(&pds->psrSourcesList[0], FileDB, 0, 0);
}
}
FileDBNext = &FileDB->Next;
}
return;
}
//+---------------------------------------------------------------------------
//
// Function: ReadDirsFile
//
// Synopsis: Parse the DIRS file
//
// Arguments: [DirDB] -- Directory to look in
//
// Returns: TRUE if parsed
//
// Notes: The existence of a file named 'mydirs' or the name of the
// target specific dirs will override the normal 'dirs' file.
//
//----------------------------------------------------------------------------
BOOL
ReadDirsFile(
PDIRREC DirDB
)
{
FILE *InFileHandle;
LPSTR TextLine, pValue;
LPSTR apszDirs[] = { "mydirs", NULL, "dirs", NULL};
CHAR TargetName[16];
strcpy(&TargetName[0], pszTargetDirs);
strcat(&TargetName[0], ".");
apszDirs[1] = &TargetName[0];
for (ppCurrentDirsFileName = apszDirs;
*ppCurrentDirsFileName != NULL;
ppCurrentDirsFileName++) {
if (SetupReadFile(DirDB->Name, *ppCurrentDirsFileName, "#", &InFileHandle)) {
break;
}
}
if (*ppCurrentDirsFileName == NULL || !InFileHandle) {
FreeMacros();
return (FALSE);
}
if (fFirstScan && (ppCurrentDirsFileName <= &apszDirs[1])) {
BuildMsg("Using .\\%s instead of DIRS...\r\n",
FormatPathName(DirDB->Name, *ppCurrentDirsFileName));
}
if (DEBUG_1) {
BuildMsg(
" Scanning file %s\r\n",
FormatPathName(DirDB->Name, *ppCurrentDirsFileName));
}
assert(cMacro == 0);
while ((TextLine = ReadLine(InFileHandle)) != NULL) {
if ((pValue = SplitMacro(&TextLine)) != NULL) {
SaveMacro(TextLine, pValue);
FreeString(&TextLine, MT_DIRSTRING);
}
}
CloseReadFile(NULL);
if ((pValue = FindMacro("DIRS")) != NULL) {
MarkDirNames(DirDB, pValue, TRUE);
}
if ((pValue = FindMacro("OPTIONAL_DIRS")) != NULL) {
MarkDirNames(DirDB, pValue, BuildAllOptionalDirs);
}
if ((FindMacro("SYNCHRONIZE_DRAIN")) != NULL) {
if (!fIgnoreSync) {
DirDB->DirFlags |= DIRDB_SYNCHRONIZE_DRAIN;
}
}
FreeMacros();
return ( TRUE );
}
//
// Debugging and Utility Functions
//
VOID
PrintDirSupData(DIRSUP *pds)
{
int i;
if (pds->LocalIncludePath != NULL) {
BuildMsgRaw(" LocalIncludePath: %s\r\n", pds->LocalIncludePath);
}
if (pds->UserIncludePath != NULL) {
BuildMsgRaw(" UserIncludePath: %s\r\n", pds->UserIncludePath);
}
if (pds->LastIncludePath != NULL) {
BuildMsgRaw(" LastIncludePath: %s\r\n", pds->LastIncludePath);
}
if (pds->TestType != NULL) {
BuildMsgRaw(" TestType: %s\r\n", pds->TestType);
}
if (pds->PchIncludeDir != NULL) {
BuildMsgRaw(" PchIncludeDir: %s\r\n", pds->PchIncludeDir);
}
if (pds->PchInclude != NULL) {
BuildMsgRaw(" PchInclude: %s\r\n", pds->PchInclude);
}
if (pds->PchTargetDir != NULL) {
BuildMsgRaw(" PchTargetDir: %s\r\n", pds->PchTargetDir);
}
if (pds->PchTarget != NULL) {
BuildMsgRaw(" PchTarget: %s\r\n", pds->PchTarget);
}
if (pds->ConditionalIncludes != NULL) {
BuildMsgRaw(" ConditionalIncludes: %s\r\n", pds->ConditionalIncludes);
}
for (i = 0; i < MAX_TARGET_MACHINES + 2; i++) {
if (pds->SourcesVariables[i] != NULL) {
BuildMsgRaw(
" SourcesVariables[%d]: %s\r\n",
i,
pds->SourcesVariables[i]);
}
if (pds->psrSourcesList[i] != NULL) {
BuildMsgRaw(" SourcesList[%d]:\r\n", i);
PrintSourceDBList(pds->psrSourcesList[i], i);
}
}
}
VOID
FreeDirSupData(DIRSUP *pds)
{
int i;
if (pds->LocalIncludePath != NULL) {
FreeMem(&pds->LocalIncludePath, MT_DIRSTRING);
}
if (pds->UserIncludePath != NULL) {
FreeMem(&pds->UserIncludePath, MT_DIRSTRING);
}
if (pds->LastIncludePath != NULL) {
FreeMem(&pds->LastIncludePath, MT_DIRSTRING);
}
if (pds->NTIncludePath != NULL) {
FreeMem(&pds->NTIncludePath, MT_DIRSTRING);
}
if (pds->TestType != NULL) {
FreeMem(&pds->TestType, MT_DIRSTRING);
}
if (pds->PchInclude != NULL) {
FreeMem(&pds->PchInclude, MT_DIRSTRING);
}
if (pds->PchIncludeDir != NULL) {
FreeMem(&pds->PchIncludeDir, MT_DIRSTRING);
}
if (pds->PchTargetDir != NULL) {
FreeMem(&pds->PchTargetDir, MT_DIRSTRING);
}
if (pds->PchTarget != NULL) {
FreeMem(&pds->PchTarget, MT_DIRSTRING);
}
if (pds->ConditionalIncludes != NULL) {
FreeMem(&pds->ConditionalIncludes, MT_DIRSTRING);
}
if (pds->PassZeroHdrDir != NULL) {
FreeMem(&pds->PassZeroHdrDir, MT_DIRSTRING);
}
if (pds->PassZeroSrcDir1 != NULL) {
FreeMem(&pds->PassZeroSrcDir1, MT_DIRSTRING);
}
if (pds->PassZeroSrcDir2 != NULL) {
FreeMem(&pds->PassZeroSrcDir2, MT_DIRSTRING);
}
for (i = 0; i < MAX_TARGET_MACHINES + 2; i++) {
if (pds->SourcesVariables[i] != NULL) {
FreeMem(&pds->SourcesVariables[i], MT_DIRSTRING);
}
while (pds->psrSourcesList[i] != NULL) {
FreeSourceDB(&pds->psrSourcesList[i]);
}
}
}
VOID
FreeDirData(DIRREC *pdr)
{
if (pdr->TargetPath != NULL) {
FreeMem(&pdr->TargetPath, MT_DIRSTRING);
}
if (pdr->TargetPathLib != NULL) {
FreeMem(&pdr->TargetPathLib, MT_DIRSTRING);
}
if (pdr->TargetName != NULL) {
FreeMem(&pdr->TargetName, MT_DIRSTRING);
}
if (pdr->TargetExt != NULL) {
FreeMem(&pdr->TargetExt, MT_DIRSTRING);
}
if (pdr->KernelTest != NULL) {
FreeMem(&pdr->KernelTest, MT_DIRSTRING);
}
if (pdr->UserAppls != NULL) {
FreeMem(&pdr->UserAppls, MT_DIRSTRING);
}
if (pdr->UserTests != NULL) {
FreeMem(&pdr->UserTests, MT_DIRSTRING);
}
if (pdr->NTTargetFile0 != NULL) {
FreeMem(&pdr->NTTargetFile0, MT_DIRSTRING);
}
if (pdr->Pch != NULL) {
FreeMem(&pdr->Pch, MT_DIRSTRING);
}
if (pdr->PchObj != NULL) {
FreeMem(&pdr->PchObj, MT_DIRSTRING);
}
if (pdr->pds != NULL) {
FreeDirSupData(pdr->pds);
FreeMem(&pdr->pds, MT_DIRSUP);
pdr->pds = NULL;
}
}
//+---------------------------------------------------------------------------
//
// Function: MarkDirNames
//
// Synopsis: Parses a DIRS= or OPTIONAL_DIRS line and marks the directories
// appropriately.
//
// Arguments: [DirDB] -- Directory containing DIRS file
// [TextLine] -- DIRS= or OPTIONAL_DIRS= line
// [Required] -- Indicates if directories are optional or not.
//
//----------------------------------------------------------------------------
VOID
MarkDirNames(PDIRREC DirDB, LPSTR TextLine, BOOL Required)
{
UINT i;
LPSTR p, token;
PFILEREC FileDB, *FileDBNext;
char dirbuf[DB_MAX_PATH_LENGTH];
ULONG DirInclude;
AssertPathString(TextLine);
while (SplitToken(dirbuf, '*', &TextLine)) {
// Assume all platforms are included for this dir.
DirInclude = DIR_INCLUDE_ALL;
for (p = dirbuf; *p != '\0'; p++) {
if ( dirbuf != p && *p == '{' ) {
// An explicit include path was listed.
DirInclude = DIR_INCLUDE_NONE;
*p = '\0';
token = strtok(p+1, ",}");
while (token) {
if (!_stricmp(token, "X86") ||
!_stricmp(token, "I386") ||
!_stricmp(token, "386")) {
DirInclude |= DIR_INCLUDE_X86;
} else if (!_stricmp(token, "32") ||
!_stricmp(token, "Win32")) {
DirInclude |= DIR_INCLUDE_WIN32;
} else if (!_stricmp(token, "64") ||
!_stricmp(token, "Win64")) {
DirInclude |= DIR_INCLUDE_WIN64;
} else if (!_stricmp(token, "IA64")) {
DirInclude |= DIR_INCLUDE_IA64;
} else if (!_stricmp(token, "AMD64")) {
DirInclude |= DIR_INCLUDE_AMD64;
} else if (!_stricmp(token, "RISC")) {
DirInclude |= DIR_INCLUDE_RISC;
} else if (!_stricmp(token, "ARM")) {
DirInclude |= DIR_INCLUDE_ARM;
}
token = strtok(NULL, ",}");
}
break;
} else {
if (!iscsym(*p) && *p != '.' && *p != '-' ) {
BuildError(
"%s: ignoring bad subdirectory: %s\r\n",
DirDB->Name,
dirbuf);
p = NULL;
break;
}
}
}
if (!(DirInclude & TargetMachines[0]->DirIncludeMask)) {
continue;
}
if (p != NULL) {
if (!Required) {
for (i = 0; i < CountOptionalDirs; i++) {
if (!strcmp(dirbuf, OptionalDirs[i])) {
OptionalDirsUsed[i] = TRUE;
break;
}
}
if (i >= CountOptionalDirs) {
p = NULL;
}
} else {
for (i = 0; i < CountExcludeDirs; i++) {
if (!strcmp(dirbuf, ExcludeDirs[i])) {
ExcludeDirsUsed[i] = TRUE;
p = NULL;
break;
}
}
}
}
if (p != NULL) {
if ((fQuicky || fSemiQuicky) && (!fQuickZero)) {
FileDB = InsertFileDB(
DirDB,
dirbuf,
0,
FILE_ATTRIBUTE_DIRECTORY,
0);
if (FileDB != NULL) {
FileDB->SubDirIndex = ++DirDB->CountSubDirs;
}
} else {
FileDBNext = &DirDB->Files;
while (FileDB = *FileDBNext) {
if (FileDB->FileFlags & FILEDB_DIR) {
if (!strcmp(dirbuf, FileDB->Name)) {
FileDB->SubDirIndex = ++DirDB->CountSubDirs;
break;
}
}
FileDBNext = &FileDB->Next;
}
if (FileDB == NULL) {
BuildError(
"%s found in %s, is not a subdirectory of %s\r\n",
dirbuf,
FormatPathName(DirDB->Name, *ppCurrentDirsFileName),
DirDB->Name);
}
}
}
}
}
VOID
StartElapsedTime(VOID)
{
// we don't want to check for fElapsedTime since if we want XML file, we need it anyway
// and single GetTickCount() call is not a perf hit
if (StartTime == 0) {
StartTime = GetTickCount();
}
}
VOID
StartDirectoryElapsedTime(VOID)
{
DirectoryStartTime = GetTickCount();
}
VOID
PrintElapsedTime(VOID)
{
DWORD ElapsedTime;
DWORD ElapsedHours;
DWORD ElapsedMinutes;
DWORD ElapsedSeconds;
DWORD ElapsedMilliseconds;
CONSOLE_SCREEN_BUFFER_INFO csbi;
GetConsoleScreenBufferInfo(GetStdHandle(STD_ERROR_HANDLE), &csbi);
if (fPrintElapsed) {
ElapsedTime = GetTickCount() - StartTime;
ElapsedHours = ElapsedTime/(1000 * 60 * 60);
ElapsedTime = ElapsedTime % (1000 * 60 * 60);
ElapsedMinutes = ElapsedTime/(1000 * 60);
ElapsedTime = ElapsedTime % (1000 * 60);
ElapsedSeconds = ElapsedTime/1000;
ElapsedMilliseconds = ElapsedTime % 1000;
BuildColorMsg(
COLOR_STATUS,
"Elapsed time [%d:%02d:%02d.%03d]\r\n",
ElapsedHours,
ElapsedMinutes,
ElapsedSeconds,
ElapsedMilliseconds);
LogMsg(
"Elapsed time [%d:%02d:%02d.%03d]%s\r\n",
ElapsedHours,
ElapsedMinutes,
ElapsedSeconds,
ElapsedMilliseconds,
szAsterisks);
}
}
LPSTR
FormatElapsedTime(DWORD dwStartTime)
{
static char FormatElapsedTimeBuffer[16];
DWORD ElapsedTime;
DWORD ElapsedHours;
DWORD ElapsedMinutes;
DWORD ElapsedSeconds;
DWORD ElapsedMilliseconds;
ElapsedTime = GetTickCount() - dwStartTime;
ElapsedHours = ElapsedTime/(1000 * 60 * 60);
ElapsedTime = ElapsedTime % (1000 * 60 * 60);
ElapsedMinutes = ElapsedTime/(1000 * 60);
ElapsedTime = ElapsedTime % (1000 * 60);
ElapsedSeconds = ElapsedTime/1000;
ElapsedMilliseconds = ElapsedTime % 1000;
sprintf(
FormatElapsedTimeBuffer,
"%d:%02d:%02d.%03d",
ElapsedHours,
ElapsedMinutes,
ElapsedSeconds,
ElapsedMilliseconds);
return ( FormatElapsedTimeBuffer );
}
LPSTR
FormatCurrentDateTime()
{
static char FormatCurrentDateTimeBuffer[18];
SYSTEMTIME st;
GetLocalTime(&st);
sprintf(
FormatCurrentDateTimeBuffer,
"%04d%02d%02d%02d%02d%02d%03d",
st.wYear, st.wMonth, st.wDay,
st.wHour, st.wMinute, st.wSecond, st.wMilliseconds);
return ( FormatCurrentDateTimeBuffer );
}