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

478 lines
13 KiB
C++

/*++
Copyright (c) 1997-2000 Microsoft Corporation
Module Name:
tpl.cpp
Abstract:
template file interpreter for tracewpp.exe
Author:
Gor Nishanov (gorn) 03-Apr-1999
Revision History:
Gor Nishanov (gorn) 03-Apr-1999 -- hacked together to prove that this can work
GorN: 29-Sep-2000 - fix WHERE clause handling
ToDo:
Clean it up
--*/
#define UNICODE
#include <stdio.h>
#include <windows.h>
#pragma warning(disable: 4786)
#pragma warning(disable: 4503) // decorated length
#pragma warning(disable: 4512) // cannot generate assignment
#pragma warning(disable: 4100) // '_P' : unreferenced formal parameter
#pragma warning(disable: 4018) // signed/unsigned mismatch
#pragma warning(disable: 4267) // 'return' : conversion from 'size_t' to 'int'
#include <xmemory>
#include <xstring>
#include <set>
#include <map>
#pragma warning(disable: 4663 4018)
#include <vector>
//#pragma warning(default: 4018 4663) // signed/unsigned mismatch
#pragma warning(default: 4100)
#include "ezparse.h"
#include "fieldtable.h"
#include "tpl.h"
LPCSTR FieldNames[] = {
#define FIELD_NAME(f) #f,
INSERT_FIELD_NAMES
#undef FIELD_NAME
};
OBJECT_MAP ObjectMap;
typedef std::map<std::string, FieldId, strless> FIELD_MAP;
FIELD_MAP FieldMap;
void PopulateFieldMap() {
#define FIELD_NAME(_name_) FieldMap[#_name_] = fid_ ## _name_;
INSERT_FIELD_NAMES
#undef FIELD_NAME
FIELD_MAP::iterator i;
}
////////////////////////////////////////////////////////////////////////////////////////////
struct LoopVar : FieldHolder {
Enumerator * Enum;
std::string Name;
LoopVar() {}
DWORD PrintField(int fieldId, FILE* f, const Enumerator** pEnum) const {
return Enum->GetData()->PrintField(fieldId, f, pEnum);
}
};
///////////////////////////////////////////////////////////////////////////////////////////
char Delimiter = '`';
std::string COMMENT("*");
std::string FORALL("FORALL");
std::string ENDFOR("ENDFOR");
std::string IF("IF");
std::string ENDIF("ENDIF");
std::string DELIMITER("DELIMITER");
std::string INCLUDE("INCLUDE");
std::string ENV("ENV");
typedef enum Action {
actText,
actVar,
actLoop,
actIf,
actInclude,
actLiteralString,
} Action;
#pragma warning(disable: 4201) // nonstandard extension used : nameless struct/union
struct Chunk {
struct {
Action action : 8;
UCHAR level : 8;
SHORT loopEnd:16;
};
union {
struct {
LPCSTR textBeg;
LPCSTR textEnd;
};
struct {
FieldHolder * p;
FieldId FieldNo;
std::string Filter;
};
};
Enumerator* getEnum() {
const Enumerator* Enum; p->PrintField(FieldNo, 0, &Enum); return (Enumerator*)Enum; }
void printField(FILE* out) const {
p->PrintField(FieldNo, out, 0); }
Chunk(){} // to make vector happy
Chunk (Action Act, FieldHolder* fh, FieldId fid, int lvl, const std::string& filter):
action(Act),FieldNo(fid),p(fh),level((UCHAR)lvl),Filter(filter) {}
Chunk (FieldHolder* fh, FieldId fid):action(actVar),FieldNo(fid),p(fh) {}
Chunk (LPCSTR b, LPCSTR e):action(actText),textBeg(b),textEnd(e) {}
Chunk (Action act, LPCSTR b, LPCSTR e):action(act),textBeg(b),textEnd(e) {}
explicit Chunk(std::string const& Text):action(actLiteralString),Filter(Text) {}
};
#define MAX_LOOP_LEVEL 127
struct TemplateProcessor {
LoopVar Loop[MAX_LOOP_LEVEL];
std::vector<Chunk> Chunks;
void RunIt(int beg, int end, FILE* out) {
for(int i = beg; i < end; ) {
switch(Chunks[i].action) {
case actLiteralString:
{
fwrite(Chunks[i].Filter.c_str(), Chunks[i].Filter.length(), 1, out );
++i;
break;
}
case actText:
{
for(LPCSTR p = Chunks[i].textBeg; p < Chunks[i].textEnd; ++p) {
if (*p != '\r') putc(*p, out);
}
++i;
break;
}
case actVar:
{
Chunks[i].printField(out);
++i;
break;
}
case actIf:
{
if (!Chunks[i].p->Hidden(Chunks[i].Filter)) {
RunIt(i+1, Chunks[i].loopEnd, out);
}
i = Chunks[i].loopEnd;
break;
}
case actLoop:
{
Enumerator * Enum = Chunks[i].getEnum();
Loop[Chunks[i].level].Enum = Enum;
for(Enum->Reset(Chunks[i].Filter); Enum->Valid(); Enum->Next(Chunks[i].Filter) ) {
RunIt(i+1, Chunks[i].loopEnd, out);
}
delete Enum;
i = Chunks[i].loopEnd;
break;
}
case actInclude:
{
ProcessTemplate(Chunks[i].textBeg, Chunks[i].textEnd, out);
++i;
break;
}
}
}
}
void DoId(LPCSTR q, LPCSTR p, FieldId& fid, FieldHolder*& fh)
{
LPCSTR dot = q;
while (q < p && isspace(*q)) ++q;
while (q < p && isspace(p[-1])) --p;
while (dot < p && *dot != '.') ++dot;
std::string ObjectName(q, dot);
OBJECT_MAP::iterator it = ObjectMap.find( ObjectName );
if (it == ObjectMap.end()) {
ReportError("Var not found: %s\n", ObjectName.c_str() );
exit(1);
} else {
std::string FieldName;
if (dot == p) {
fid = (FieldId)fid___default__;
FieldName.assign("__default__");
} else {
++dot;
while (p < dot && isspace(*dot)) ++dot;
FieldName.assign(dot,p);
FIELD_MAP::iterator fit = FieldMap.find( FieldName.c_str() );
if (fit == FieldMap.end()) {
ReportError("FieldNotFound: %s.%s\n", ObjectName.c_str(), FieldName.c_str() );
exit(1);
} else {
fid = fit->second;
}
}
}
fh = it->second;
}
void DoVar(LPCSTR q, LPCSTR p) {
FieldHolder* fh;
FieldId fid;
DoId(q,p, fid, fh);
Chunks.push_back( Chunk(fh, fid) );
}
void DoLoop(int loopLevel, LPCSTR beg, LPCSTR end) {
FieldHolder* fh;
FieldId fid;
std::string LoopVar;
std::string LoopSet;
std::string Filter;
LPCSTR p,q;
p = beg+6; while (p < end && isspace(*p)) ++p;
q = p; while(p < end && !isspace(*p)) ++p;
LoopVar.assign(q,p);
Loop[loopLevel].Name = LoopVar;
p += 4; while (p < end && isspace(*p)) ++p;
q = p; while(p < end && !isspace(*p)) ++p;
DoId(q,p, fid, fh);
LoopSet.assign(q, p);
p += 7; while (p < end && isspace(*p)) ++p;
q = p;
if (q < end) {
p = end; while(p > q && isspace(*--p));
if (p < end && !isspace(*p)) ++p;
}
Filter.assign(q,p);
Flood("FORALL %s IN %s WHERE %s\n", LoopVar.c_str(), LoopSet.c_str(),
Filter.c_str());
ObjectMap[LoopVar] = &Loop[loopLevel];
Chunks.push_back( Chunk(actLoop, fh, fid, loopLevel, Filter) );
}
void DoIf(int loopLevel, LPCSTR beg, LPCSTR end) {
FieldHolder* fh;
FieldId fid;
std::string Object;
std::string Filter;
LPCSTR p,q;
p = beg+3; while (p < end && isspace(*p)) ++p;
q = p; while(p < end && !isspace(*p)) ++p;
DoId(q,p, fid, fh); // Split id //
Object.assign(q, p);
while (p < end && isspace(*p)) ++p;
while (p < end && isspace(end[-1]) ) --end;
Filter.assign(p,end);
Flood("IF %s %s\n", Object.c_str(), Filter.c_str() );
Chunks.push_back( Chunk(actIf, fh, fid, loopLevel, Filter) );
}
DWORD
CompileAndRun(
IN LPCSTR begin,
IN LPCSTR end,
IN FILE* out
)
{
LPCSTR p = begin, PlainText = begin;
int loop = -1;
int loopLevel = -1;
bool comment;
Chunks.erase(Chunks.begin(), Chunks.end());
Chunks.reserve(128);
for(;;) {
LPCSTR q;
for(;;) {
if (p == end) {
Chunks.push_back( Chunk(PlainText, p) );
goto done;
}
if (*p == Delimiter)
break;
++p;
}
q = ++p;
comment = (p < end && *p == '*');
for(;;) {
if (p == end) {
ReportError("Unmatched delimiters\n");
exit(1);
}
if (*p == Delimiter) {
if (comment) {
if (p[-1] == '*') break;
} else {
break;
}
}
++p;
}
if (q-1 > PlainText) {
Chunks.push_back( Chunk(PlainText, q-1) );
}
if (p == q) {
// PERFPERF If the previous chunk was a text, we can extend it
Chunks.push_back( Chunk(q-1, p) );
} else {
std::string x(q,p);
if (x.compare(0, IF.size(), IF) == 0) {
int previous = loop;
// KLUDGE merge with FORALL
if (loopLevel == MAX_LOOP_LEVEL) {
ReportError("Too many nested blocks!\n");
exit(1);
}
++loopLevel;
loop = static_cast<int>( Chunks.size() );
DoIf(loopLevel, q,p);
if (previous >= 32765) {
ReportError("Too many chunks. Make loopEnd a UINT, %d\n", previous);
exit(1);
}
Chunks.back().loopEnd = (SHORT)previous;
while (p+1 < end && (p[1] == '\n' || p[1] == '\r') ) ++p;
} else if (x.compare(0, FORALL.size(), FORALL) == 0) {
int previous = loop;
if (loopLevel == MAX_LOOP_LEVEL) {
ReportError("Too many nested loops!\n");
exit(1);
}
++loopLevel;
loop = static_cast<int>( Chunks.size() );
DoLoop(loopLevel, q,p);
if (previous >= 32765) {
ReportError("Too many chunks. Make loopEnd a UINT, %d\n", previous);
exit(1);
}
Chunks.back().loopEnd = (SHORT)previous;
while (p+1 < end && (p[1] == '\n' || p[1] == '\r') ) ++p;
} else if (x.compare(0, DELIMITER.size(), DELIMITER) == 0 && x.size() > 10) {
Delimiter = x[10];
} else if (x.compare(0, ENV.size(), ENV) == 0) {
// we need to replace this field with
// the value of the specified env variable
LPCSTR val = getenv( std::string(q+4,p).c_str() );
if (val != NULL) {
Chunks.push_back( Chunk(std::string(val)) );
}
} else if (x.compare(0, COMMENT.size(), COMMENT) == 0) {
// eat away the whitespace
while (p+1 < end && (p[1] == '\n' || p[1] == '\r') ) ++p;
} else if (x.compare(0, INCLUDE.size(), INCLUDE) == 0) { // Doesn't work
Chunks.push_back(
Chunk(actInclude, q + INCLUDE.size() + 1, p) );
} else if ((x.compare(0, ENDIF.size(), ENDIF) == 0)
|| (x.compare(0, ENDFOR.size(), ENDFOR) == 0)) {
// KLUDGE make them separate or rename both to simply END
// End will be set in ENDFOR //
if (loop == -1) {
ReportError("ENDFOR without FORALL\n");
exit(1);
}
ObjectMap.erase( Loop[loopLevel].Name );
int previous = Chunks[loop].loopEnd;
// BUGBUG have a check that confirms that we didn't run out of space
Chunks[loop].loopEnd = (SHORT)Chunks.size();
loop = previous;
--loopLevel;
while (p+1 < end && (p[1] == '\n' || p[1] == '\r') ) ++p;
} else {
DoVar(q, p);
}
}
PlainText = ++p;
}
done:;
if (loop != -1) {
ReportError("No ENDFOR for loop, %d\n", loop);
exit(1);
}
RunIt(0, static_cast<int>( Chunks.size() ), out);
return 0;
}
};
DWORD
processTemplate(
IN LPCSTR begin,
IN LPCSTR end,
IN EZPARSE_CALLBACK,
IN PVOID Context,
IN PEZPARSE_CONTEXT
)
{
FILE *out = (FILE*)Context;
TemplateProcessor tpl;
return tpl.CompileAndRun(begin,end,out);
}