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

3255 lines
101 KiB
C++

/*++
Copyright (c) 1997-2000 Microsoft Corporation
Module Name:
tracewpp.c
Abstract:
Parameter processing and main entry point for tracewpp.exe
Author:
Gor Nishanov (gorn) 03-Apr-1999
Revision History:
gorn 03-Apr-1999 -- hacked together to prove that this can work
GorN: 29-Sep-2000 - do not scan files that didn't change
GorN: 29-Sep-2000 - support -notimechk -nohashchk -dll switches
GorN: 29-Sep-2000 - add support for KdPrintEx((level,comp,msg,...)) like functions
GorN: 09-Oct-2000 - fix hashing; added NULL arg
GorN: 07-Mar-2001 - add arglimit option
GorN: 05-Sep-2001 - add preserveext option (do not strip specified extensions)
GorN: 05-Sep-2001 - add Func field in message (to be used from templates)
GorN: 05-Sep-2001 - hash full path, in addition to all the messages
BassamT: 01-Oct-2001 - Added NoMsg option + filter for functions that do not have MSG,...
GorN: 15-May-2002 - handle assumed MSG properly
ToDo:
NullArg Remove limitation of having only one NULL arg in a func
all Clean it up
parseConfigCheck for premature termination Ex: CUSTOM_TYPE(x, ItemListLong(dfdf))
cool Automatic fill of the arguments if ...
nice better error chk
types to mof it stinks to generate long enums multiple times
later guidless
bug check for types with dup names
bug tpl: check for unterminated keyword
later detect WPP_CLEANUP -- complain
print in macros what to do about prints in macro?
cmdline get the flags from the env variable
!!! Don't assume that whatever is not recognized is func
bug updated CRC computations to properly work with consts
bug enum doesn't work
bug %10!..! doesnt' work
bug %% is not handled
mess rename hidden => visible
mess remove ugly MSGTYPBASE. Make FmtStr 0 based
mess let message share some of the func field handlers
traceprt think how to make the level indenting
ezparse report unmatched end_wpp
now get cmdline options from a file or env.var
later handle "str" STRMACRO "another string" as a MSG arg
strange output
..\logsup.c(4180) : error : Unterminated Format SpecifierHere is the signature: (0)
..\logsup.c(4180) : error : Extra argument. Only 0 are specified in the string
why we didn't catch the error
CsDbgPrint(LOG_NOISE,
("[FM] FmpSetGroupEnumOwner: Group %1!ws! not found\n"));
CsDbgPrint(LOG_UNUSUAL, (
"[NMJOIN] Cannot add node '%1!ws!' to the cluster because "
"no slots are available in the node table.\n"
));
printf("Port no in network order %1!port!, regular order %1!u!, 0x0102);
NetDeviceTrace( NETDEV_DBG_EXIT | NETDEV_DBG_INFO,
"ClientMdlSetup Client[%10!d!], RCB[%11!d!]: "
"Exit w/ Func 0x%12!04X!, Send 0x%14!04X!, Recv 0x%15!04X!",
deviceExtension->Ordinal, // LOGULONG
RequestCB->RequestIdx, // LOGUSHORT
ios->MajorFunction, // LOGUCHAR
ios->MajorFunction, // LOGUCHAR
sendBytes, // LOGULONG
recvBytes); // LOGULONG => AV's
DONE DONE DONE DONE DONE DONE DONE DONE DONE DONE
later handle KdPrintEx
later add "if" to templates (at least if not empty)
requirement md5
later detect WPP_INIT_TRACING
main Time ourselves and print how fast we are
useful ignore anything but .c .cxx .cpp .c++
useful -gen{a,b,c}h.tpl
-ext:.c.cxx.cpp.c++
-v verbose
-q extra quiet
footprint merge multiple formats together
difficult? Think how to unify DoTraceMacro0 TraceLazy, etc
figureout DbgPrint
-odir:path
trivial add "\" to -odir if needed
nice scan
-gen:{a.tpl}auto.c
-gen:{a.tpl}*.c
outstanding take plain DebugPrint format as well
difficult? Add generation of by value types
--*/
#define STRICT
#include <stdio.h>
#include <windows.h>
#include <rpc.h>
#include <rpcdce.h>
#pragma warning(disable: 4702) // unreachable code
#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: 4267) // 'return' : conversion from 'size_t' to 'int'
#include <xmemory>
#include <xstring>
#include <set>
#include <map>
#pragma warning(disable: 4663)
#pragma warning(disable: 4018) // signed/unsigned mismatch
#include <vector>
//#pragma warning(default: 4018 4663) // signed/unsigned mismatch
#pragma warning(default: 4100)
#include <algorithm>
#include "ezparse.h"
#include "crc32.h"
#include "md5.h"
#include "fieldtable.h"
#include "tpl.h"
#define override
//typedef ULONG CRC32;
using namespace std;
BOOL md5 = TRUE;
BOOL reorder = FALSE;
BOOL userMode = TRUE;
BOOL noshrieks = FALSE;
UINT SeparateTraceGuidPerFile = 1;
INT MSGTYPBASE = 10;
UINT MessageCount = 0;
UINT ArgBase = 1;
BOOL CheckTimestamp = TRUE;
BOOL CheckHash = TRUE;
BOOL IgnoreDupTypes = FALSE;
int arglimit = 32;
//string OutputInc ("_tracewpp.h");
//string OutputMof ("_tracewpp.mof");
string OutputMac;
string OutputDir (".\\");
//string ArrayPrefix("WPP_");
string MacroPrefix("LOG");
string AllowedExtensions(".c.cxx.cpp.c++.C.CPP");
string PreserveExtensions("");
string CurrentDir;
#define DEFAULT_UM_GEN_OPTION "{um-default.tpl}*.tmh"
#define DEFAULT_KM_GEN_OPTION "{km-default.tpl}*.tmh"
#define DEFAULT_GEN_OPTION (userMode?DEFAULT_UM_GEN_OPTION:DEFAULT_KM_GEN_OPTION)
#define DEFAULT_CONFIG_NAME "defaultwpp.ini"
string LOCAL_CONFIG_NAME("localwpp.ini");
string WppDefault(DEFAULT_CONFIG_NAME);
vector<string> SearchDirs;
//string Revision;
string LocalConfig;
string ComponentName;
BOOL CheckExtension(const string& str, string::size_type pos, const string& Extensions)
{
string::size_type q, n;
n = str.size() - pos; // extension length
q = Extensions.find(&str[pos], 0, n);
if ( (q == string::npos)
|| (q + n < Extensions.size() && Extensions[q + n] != '.') )
{
return FALSE;
}
return TRUE;
}
// Checks whether string passed has an allowed extension
// Lame: uses global variable AllowedExtensions
BOOL AllowedExtension(const string str)
{
string::size_type p = str.rfind('.');
if (p == string::npos) {
Unusual("File %s has no extension\n", str.c_str() );
return FALSE;
}
if ( !CheckExtension(str, p, AllowedExtensions) )
{
Unusual("File %s has unrecognized extension\n", str.c_str() );
return FALSE;
}
return TRUE;
}
inline size_t fprint_str(FILE* f, const string& str, LPCSTR replace, LPCSTR replace_with) {
int skipped = 0;
for (int i = 0; i < str.length(); ++i) {
char ch = str[i];
char* pos = strchr(replace, ch);
if (pos) {
ch = replace_with[ pos - replace ];
if (ch == '@') {
++skipped;
continue;
}
}
fputc(ch, f);
}
return str.length() - skipped;
}
inline size_t fprint_str(FILE* f, const string& str) {
return fwrite(str.c_str(), str.length(), 1, f );
}
inline size_t fprint_str(FILE* f, LPCSTR beg, LPCSTR end) {
return fwrite(beg, end - beg, 1, f );
}
inline size_t fprint_str(FILE* f, const STR_PAIR& Pair) {
return fwrite(Pair.beg, Pair.end - Pair.beg, 1, f );
}
inline int stoi(const STR_PAIR& str, LPCSTR name)
{
LPCSTR p = str.beg;
int sum = 0;
int sgn = 1;
if (p < str.end && *p == '-') {
++p; sgn = -1;
}
while (p < str.end) {
if (!isdigit(*p)) {
ReportError("%s should be a number (%s supplied)\n",
name, string(str.beg, str.end).c_str() );
break;
}
sum = 10 * sum + (*p - '0');
++p;
}
return sgn * sum;
}
inline unsigned char HexVal(int ch) {
return (unsigned char)(isdigit(ch) ? ch - '0' : ch - 'a' + 10); }
UINT Hex(LPCSTR s, int n)
{
UINT res = 0;
while(n--) {
res = res * 16 + HexVal(*s++);
}
return res;
}
CHAR HexDigit(int val) {
if (val < 10) return (CHAR)(val + '0');
else return (CHAR)(val + 'a' - 10);
}
ULONGLONG GetFileModificationTime(const string& str)
{
WIN32_FIND_DATA FindData;
ULONGLONG Time;
HANDLE FindHandle = FindFirstFile(str.c_str(), &FindData);
if (FindHandle == INVALID_HANDLE_VALUE) {
return 0;
}
FindClose(FindHandle);
CopyMemory(&Time, &FindData.ftLastWriteTime, sizeof(Time));
return Time;
}
// Prototypes //
void DealWithCmdLineOptions(LPCSTR s);
void DealWithCmdLineOptions(LPCSTR beg, LPCSTR end)
{
DealWithCmdLineOptions(string(beg,end).c_str() );
}
struct Hasher {
virtual void Init() = 0;
virtual void Finalize() = 0;
virtual void Hash(const void* buf, int nBytes) = 0;
virtual int Size() const = 0;
virtual const unsigned char * Buf() const = 0;
void Hash(const std::string& str) {
Hash(str.begin(), (int)str.size());
}
void HashStr(const std::string& str) { Hash(str); }
void fromString(LPCSTR beg, LPCSTR end)
{
int n = min((ULONG)(end - beg)/2, Size());
unsigned char * buf = BufRW();
ZeroMemory(buf, Size());
for(int i = 0; i < n; ++i) {
buf[i] = HexVal(beg[2*i]) * 16 + HexVal(beg[2*i + 1]);
}
}
void print(FILE* out) const {
int n = Size();
const UCHAR* buf = Buf();
for(int i = 0; i < n; ++i) {
fprintf(out, "%x%x", buf[i] >> 4, buf[i] & 0xF);
}
}
bool operator ==(const Hasher& b) const {
return Size() == b.Size() && memcmp(Buf(), b.Buf(), Size()) == 0;
}
private:
unsigned char* BufRW() { return (unsigned char*)Buf(); }
};
class Crc32Hasher : public Hasher
{
ULONG Crc;
public:
override void Init() { Crc = (ULONG)~0; }
override void Finalize() {}
override void Hash(const void* buf, int nBytes) {
Crc = FstubCrc32(Crc, (PVOID)buf, nBytes);
}
override int Size() const { return sizeof(Crc); }
override const unsigned char * Buf() const { return (UCHAR*)&Crc; }
};
class Md5Hasher : public Hasher
{
MD5_CTX Ctx;
public:
override void Init()
{
MD5Init(&Ctx);
}
override void Finalize()
{
MD5Final(&Ctx);
}
override void Hash(const void* buf, int nBytes)
{
MD5Update(&Ctx, (const unsigned char*)buf, nBytes);
}
virtual int Size() const
{
// assert(MD5DIGESTLEN == sizeof(GUID));
return MD5DIGESTLEN; // == 16
}
virtual const unsigned char * Buf() const
{
return Ctx.digest;
}
};
//typedef Crc32Hasher DefaultHasher;
typedef Md5Hasher DefaultHasher;
void ReplaceCrOrLfWith(string& s, char ch)
{
for(int i = 0; i < s.size(); ++i) {
if (s[i] == '\n' || s[i] == '\r') s[i] = ch;
if (s[i] == '"') s[i] = '\'';
}
}
string currentFileName();
bool
Files_AddFile(const string& Name, string Path, const WIN32_FIND_DATA& FindData) ;
enum {
WT_MACRONAME = 0x00000001, // TypeName is actually a MacroName. Don't have to be prepended with Log Macro Prefix
};
struct WppType : FieldHolder {
string TypeName;
string EquivType; // c-type
string MacroStart; // WPP_LOGVALARG(%s, if not specified
string MofType; // yeah
string Extension; // something to be merged with MofType
string FormatSpec; // sprintf style
string Sig;
int priority;
int ArgConsumed;
mutable BOOL Used;
DWORD Flags;
void hash(Hasher& hash) const {hash.Hash(TypeName);} // BUGBUG ?? Maybe we need to hash it
string sig() const { return Sig; }
bool isConstant() const { return EquivType.size() == 0; }
WppType() {} // To make STL happy
WppType(const string& a, const string&b, const string& c,
const string& d, const string& e, const string& f,
const string& g, int prio, int argConsumed):
TypeName(a), EquivType(b), MacroStart(c), MofType(d),
Extension(e), FormatSpec(f), Sig(g),
priority(prio),Used(0),ArgConsumed(argConsumed) {}
// WppType(LPCSTR beg, LPCSTR end):TypeName(beg, end),Used(0){}
// explicit WppType(char name):TypeName(&name, &name+1),Used(0){}
bool operator < (const WppType& b) const {
int diff = b.priority - priority; // higher prio first
if (diff < 0) return TRUE;
if (diff > 0) return FALSE;
return TypeName.compare(b.TypeName) < 0;
}
virtual BOOL Hidden(std::string) const { return !Used; }
BEGIN_FIELD_TABLE(WppType, f)
TEXT_FIELD(Name) fprint_str(f, TypeName);
TEXT_FIELD(EquivType) fprint_str(f, EquivType);
TEXT_FIELD(MacroName)
{
if (!(Flags & WT_MACRONAME)) fprint_str(f, MacroPrefix);
fprint_str(f, TypeName);
}
TEXT_FIELD(MacroStart) fprintf(f, MacroStart.c_str(), EquivType.c_str());
TEXT_FIELD(MacroEnd) fprintf(f, ")");
TEXT_FIELD(MofType) fprint_str(f, MofType);
TEXT_FIELD(Extension) fprint_str(f, Extension);
TEXT_FIELD(FormatSpec) fprint_str(f, FormatSpec);
END_FIELD_TABLE
};
typedef map<string,WppType,strless> TYPE_SET;
string SimpleValueMacroStart("WPP_LOGTYPEVAL(%s,");
string SimplePtrMacroStart("WPP_LOGTYPEPTR(");
TYPE_SET TypeSet;
struct Argument : FieldHolder {
const WppType* Type;
string Name;
string OverrideName;
int No;
bool operator < (const Argument& b) const { return *Type < *b.Type; }
Argument(){} // To make STL happy //
Argument(string name, const WppType* type):Type(type),Name(name)
{ ReplaceCrOrLfWith(Name, ' '); }
void hash(Hasher& hash) const {
hash.Hash(Name);
if(Type) Type->hash(hash);
}
BEGIN_FIELD_TABLE(Argument, f)
TEXT_FIELD(No) {fprintf(f, "%d", No);}
TEXT_FIELD(Name) {fprint_str(f, Name, ",\"\n", ".' ");}
TEXT_FIELD(RawName) {fprint_str(f, Name);}
TEXT_FIELD(MofType) {fprint_str(f, Type->MofType); fprint_str(f,Type->Extension);}
END_FIELD_TABLE
};
struct Reorder : FieldHolder {
string Name;
vector<int> Args;
Reorder(){} // to make stl happy
explicit Reorder(string name, const vector<Argument>& args): Name(name)
{ Args.resize( args.size() );
for(int i = 0; i < args.size(); ++i) {
Args[i] = args[i].No - MSGTYPBASE;
}
}
bool operator < (const Reorder& b) const
{ return Name.compare(b.Name) < 0; }
BEGIN_FIELD_TABLE(Reorder, f)
TEXT_FIELD(Name) fprint_str(f, Name);
TEXT_FIELD(Permutation)
{
for(int i = 0; i < Args.size(); ++i) {
fprintf(f, ", a%d", Args[i]);
}
}
TEXT_FIELD(Arguments)
{
fprintf(f,"MSG");
for(int i = 0; i < Args.size(); ++i) {
fprintf(f, ", a%d", i);
}
}
END_FIELD_TABLE
};
set<Reorder> ReorderSet;
string GetReorderSig(const vector<Argument>& args)
{
string sig;
if (args.size() > 256) {
ReportError("Only upto 256 arguments are supported\n");
return sig;
}
if (args.size() <= 16) {
sig.resize(args.size());
for(int i = 0; i < args.size(); ++i) {
sig[i] = HexDigit(args[i].No - MSGTYPBASE);
}
} else {
sig.resize(2 * args.size());
for(int i = 0; i < args.size(); ++i) {
int val = args[i].No;
sig[2 * i] = HexDigit(val >> 16);
sig[2 * i + 1] = HexDigit(val & 15);
}
}
ReorderSet.insert( Reorder(sig, args) );
return sig;
}
#include "parsed-string.hxx"
struct TypeSig : FieldHolder {
string Name;
vector<const WppType*> Types;
BOOL Unsafe;
TypeSig() {} // To make STL happy
TypeSig(const vector<Argument>& args, const string& sig, BOOL unsafe):
Unsafe(unsafe)
{
Name.assign(sig);
for(int i = 0; i < args.size(); ++i) {
Types.push_back(args[i].Type);
}
}
bool operator < (const TypeSig& b) const {
return Name.compare(b.Name) < 0;
}
virtual BOOL Hidden(std::string str) const {
if (str.size() == 0) { return FALSE; }
else if (str.compare("UnsafeArgs") == 0) { return !Unsafe; }
else if (str.compare("!UnsafeArgs") == 0) { return Unsafe; }
else { ReportError("Unknown filter '%s'\n", str.c_str()); exit(1); }
return NULL;
}
BEGIN_FIELD_TABLE(TypeSig, f)
TEXT_FIELD(Name) fprint_str(f, Name);
TEXT_FIELD(Count) fprintf(f, "%d", Name.size());
TEXT_FIELD(Arguments)
{
for(int i = 0; i < Types.size(); ++i) {
fprintf(f,", ");
fprint_str(f, Types[i]->EquivType);
fprintf(f, " _a%d", i+1);
}
}
TEXT_FIELD(LogArgs)
{
for(int i = 0; i < Types.size(); ++i) {
fprintf(f, Types[i]->MacroStart.c_str(),
Types[i]->EquivType.c_str());
fprintf(f, "_a%d) ", i+1);
}
}
TEXT_FIELD(DeclVars)
{
for(int i = 0; i < Types.size(); ++i) {
fprintf(f, "%s _a%d = va_arg(ap, %s); ",
Types[i]->EquivType.c_str(), i+1,
Types[i]->EquivType.c_str());
}
}
END_FIELD_TABLE
};
typedef map<string,TypeSig,strless> TYPESIG_MAP;
TYPESIG_MAP TypeSigMap;
TypeSig* GetTypeSig(const vector<Argument>& args, BOOL unsafe)
{
string sig;
if (unsafe) {
sig.assign("v");
}
for(int i = 0; i < args.size(); ++i) {
sig.append(args[i].Type->sig() );
}
TYPESIG_MAP::iterator it = TypeSigMap.find( sig );
if ( it == TypeSigMap.end() ) {
// we need to add one //
return &(TypeSigMap[ sig ] = TypeSig(args, sig, unsafe));
}
return &it->second;
}
void Fill(
string pattern)
{
WIN32_FIND_DATA findData;
HANDLE handle;
handle = FindFirstFile(pattern.c_str(), &findData);
if (handle == INVALID_HANDLE_VALUE) {
DWORD status = GetLastError();
if (status != ERROR_FILE_NOT_FOUND) {
Noise("FindFirstFile(%s): error %d\n", pattern.c_str(), GetLastError() );
}
return;
}
string::size_type p = pattern.find_last_of(":\\");
if (p != string::npos) {
pattern.resize(p+1); // to include the symbol
} else {
pattern.resize(0);
}
do {
Files_AddFile(findData.cFileName, pattern, findData);
} while( FindNextFile(handle, &findData) );
FindClose(handle);
}
struct Group : FieldHolder {
UINT GrpId;
vector<string> MsgIds;
string _Name;
Group(){}
Group(UINT id, string Name, string Msg):GrpId(id),_Name(Name) { MsgIds.push_back(Msg); }
BEGIN_FIELD_TABLE(Group, f)
TEXT_FIELD(GuidNo) fprintf(f, "%d", GrpId / 32);
TEXT_FIELD(BitNo) fprintf(f, "%d", GrpId & 31);
TEXT_FIELD(Name) fprint_str(f, _Name);
TEXT_FIELD(References)
{
vector<string>::const_iterator i;
for(i = MsgIds.begin(); i != MsgIds.end(); ++i) {
putc(' ', f); fprint_str(f, *i);
}
}
END_FIELD_TABLE
};
//void RegenerateMacroMap();
struct Prefix{
string FuncName;
// string MsgPrefix;
// vector<Argument> Args;
ParsedFormatString FmtStr;
Prefix(){}
Prefix(PSTR_PAIR str, UINT count);
};
int inline sign(UCHAR val) { return val?1:0; }
enum FuncOptions {
FO_VAR_ARGS = 0x01,
FO_UNSAFE = 0x02,
FO_DOUBLEP = 0x04,
FO_LINE_BEG = 0x08,
FO_NOMACRO = 0x10,
FO_NOMSG = 0x20,
};
struct Func : FieldHolder {
const Prefix *prefix, *suffix;
string _name;
vector<string> Args; // all supplied args - var args
vector<string> Goo; // values for GooId, usually = GooId
vector<string> GooId; // all unrecognized args + LEV
STR_PAIR assumedMsg;
UCHAR Grp, MsgArg, Msg, Arg;
UCHAR Num, Indent, MsgVal, NullArg; // BUGBUG -- what if multiple NULL args?
ULONG Options;
size_t nAssumedArgs;
void SetPS(const Prefix* val, const Prefix*& var, LPCSTR msg)
{
if (val && val->FmtStr.ArgCount > 0 && MsgArg) {
ReportError("Function %s has (something,(MSG,...)) type\n"
"It cannot have non-const %s\n",
_name.c_str(), msg );
}
var = val;
}
void set(ULONG flag) { Options |= flag; }
ULONG is(ULONG flag) const { return Options & flag; }
void SetLineBeg() { Options |= FO_LINE_BEG; }
void SetVarArgs() { Options |= FO_VAR_ARGS | FO_LINE_BEG; }
void SetUnsafe() { Options |= FO_UNSAFE; }
void SetDoubleP() { Options |= FO_DOUBLEP; }
void SetNoMsg() { Options |= FO_NOMSG; }
BOOL LineBeg() const { return Options & FO_LINE_BEG; }
BOOL VarArgs() const { return Options & FO_VAR_ARGS; }
BOOL Unsafe() const { return Options & FO_UNSAFE; }
BOOL DoubleP() const { return Options & FO_DOUBLEP; }
BOOL NoMsg() const { return Options & FO_NOMSG; }
void SetPrefix(const Prefix* pr) { SetPS(pr,prefix,"prefix"); }
void SetSuffix(const Prefix* sf) { SetPS(sf,suffix,"suffix"); }
virtual BOOL Hidden(std::string str) const {
if (str.size() == 0) { return FALSE; }
if (is(FO_NOMACRO)) return str.compare("NoMacro") != 0;
else if (str.compare("MsgArgs") == 0) { return !MsgArg; }
else if (str.compare("!MsgArgs") == 0) { return MsgArg; }
else if (str.compare("!DoubleP && !MsgArgs") == 0) { return !(!DoubleP() && !MsgArg); }
else if (str.compare("DoubleP && !MsgArgs") == 0) { return !(DoubleP() && !MsgArg); }
else if (str.compare("NoMsg") == 0) { return !NoMsg(); }
else if (str.compare("!NoMsg") == 0) { return NoMsg(); }
else if (str.compare("!NoMsg && !MsgArgs") == 0) { return !(!NoMsg() && !MsgArg); }
else if (str.compare("NoMsg && !MsgArgs") == 0) { return !(NoMsg() && !MsgArg); }
else if (str.compare("!DoubleP && !MsgArgs && !NoMsg") == 0) { return !(!DoubleP() && !MsgArg && !NoMsg()); }
else if (str.compare("DoubleP && !MsgArgs && !NoMsg") == 0) { return !(DoubleP() && !MsgArg && !NoMsg()); }
else { ReportError("Unknown filter '%s'\n", str.c_str()); exit(1); }
return NULL;
}
#if 0
// returns recognized argument count
int count() const {
return sign(Grp) + sign(Id) + sign(Msg) + sign(Arg);
}
#endif
void printArgs(FILE* f) const {
for(int i = 1; i <= Num; ++i) {
if (i > 1) fprintf(f, ", ");
else fprintf(f, "(");
fprint_str(f, Args[i]);
/*
if (i == Grp) fprintf(f,"GRP");
else if (i == Id) fprintf(f,"ID");
else if (i == Msg) fprintf(f,"MSG");
else if (i == Arg) fprintf(f,"ARG");
else fprintf(f,"_unknown%d", i);
*/
}
fprintf(f, ")");
}
BEGIN_FIELD_TABLE(Func, out)
TEXT_FIELD(Name) fprintf(out, "%s", _name.c_str() );
TEXT_FIELD(Arguments) {printArgs(out);}
TEXT_FIELD(MSG) {fputs(Msg?"MSG":"\"\"",out);}
TEXT_FIELD(ARG) {fputs(Arg?"ARG":"",out);}
TEXT_FIELD(GRP) {fputs(Grp?"GRP":"WPP_DEFAULT_GROUP_ID",out);}
// TEXT_FIELD(ID) {fprintf(out, Id?"ID":"WPP_AUTO_ID");}
TEXT_FIELD(FixedArgs)
{
for(int i = 0; i < Args.size(); ++i) {
fprint_str(out, Args[i]);
fprintf(out,", ");
}
}
TEXT_FIELD(GooArgs)
{
for(int i = 0; i < GooId.size(); ++i) {
if (i > 0) fprintf(out,",");
fprint_str(out, GooId[i]);
}
}
TEXT_FIELD(GooVals)
{
for(int i = 0; i < Goo.size(); ++i) {
if (i > 0) fprintf(out,", ");
fprint_str(out, Goo[i]);
}
}
TEXT_FIELD(GooId)
{
for(int i = 0; i < GooId.size(); ++i) {
fprintf(out,"_");
fprint_str(out, GooId[i]);
}
}
END_FIELD_TABLE
Func(){}
Func(std::string Name):_name(Name),prefix(0),suffix(0),
Grp(0), MsgArg(0), Msg(0), Arg(0),
Num(0), Indent(0), NullArg(0), MsgVal(0), Options(0),nAssumedArgs(0) {}
};
#define GRP(x) ((x).Grp)
#define MSG(x) ((x).Msg)
#define ARG(x) ((x).Arg)
#define NUM(x) ((x).Num)
//BOOL
//UpgradeFormatSpecifiers(string& str, int startCount, string* TypeSig = 0);
void printTraceGuid(FILE* f, int guidno);
struct Message : FieldHolder {
// string Msg;
string IdName;
string msgval;
string Indent;
string reorderSig;
int LineNo;
ParsedFormatString FormatStr;
TypeSig* typeSig;
const Func* func;
vector<Argument> Args;
vector<string> GooActualValues;
UINT id;
string TypeListHost;
virtual BOOL Hidden(std::string str) const { return func->Hidden(str); }
void hash(Hasher& hash) const;
int ArgConsumed() const ;
// void ChkArgs() const;
BEGIN_FIELD_TABLE(Message, f)
TEXT_FIELD(Text) { FormatStr.printMofTxt(f,LineNo); }
TEXT_FIELD(RawText) { FormatStr.print(f); }
TEXT_FIELD(Indent)
{
if(Indent.size()) {
fprintf(f, "INDENT=");
fprint_str(f, Indent);
}
}
TEXT_FIELD(MsgNo) { fprintf(f, "%d", id & 0xFFFF); }
TEXT_FIELD(GuidNo) { fprintf(f, "%d", id >> 16); }
TEXT_FIELD(Guid) { printTraceGuid(f, id >> 16); }
TEXT_FIELD(Name) { fprint_str(f, IdName ); }
TEXT_FIELD(Line) { fprintf(f, "%d", LineNo ); }
TEXT_FIELD(MsgVal) { fprint_str(f, msgval ); }
TEXT_FIELD(ReorderSig) { fprint_str(f, reorderSig ); }
TEXT_FIELD(TypeSig) { fprint_str(f, typeSig->Name); }
TEXT_FIELD(Func) { fprint_str(f, func->_name); }
TEXT_FIELD(Count) { fprintf(f, "%d", FormatStr.ArgCount); }
// TEXT_FIELD(CtlMsg) { ??? }
// TEXT_FIELD(Enabled)
ENUM_FIELD(Arguments, Args, VectorTag)
TEXT_FIELD(GooActualValues)
{
size_t k = func->nAssumedArgs;
size_t n = min(func->GooId.size()-k, GooActualValues.size());
for(size_t i = 0; i < n; ++i) {
fprintf(f, "_");
fprint_str(f, GooActualValues[i], "\"\n\r", "'@@");
}
}
TEXT_FIELD(GooPairs)
{
size_t k = func->nAssumedArgs;
// print assumed arguments
for(size_t i = 0; i < k; ++i) {
fprintf(f, " ");
fprint_str(f, func->GooId[i]);
fprintf(f, "=");
fprint_str(f, func->Goo[i], "\"\n", "' ");
}
// print the goo we pulled out of the trace statement itself
size_t n = min(func->GooId.size()-k, GooActualValues.size());
for(size_t i = 0; i < n; ++i) {
fprintf(f, " ");
fprint_str(f, func->GooId[i+k]);
fprintf(f, "=");
fprint_str(f, GooActualValues[i], "\"\n\r", "'@@");
}
}
TEXT_FIELD(FixedArgs)
{
for(int i = 0; i < func->Args.size(); ++i) {
fprint_str(f, func->Args[i]);
if ((unsigned)i == func->Args.size()-1 && func->NoMsg())
{;} // no comma needed
else
fprintf(f, ", ");
}
}
TEXT_FIELD(GooVals)
{
for(int i = 0; i < func->Goo.size(); ++i) {
if (i > 0) fprintf(f,", ");
fprint_str(f, func->Goo[i]);
}
}
TEXT_FIELD(GooArgs)
{
for(int i = 0; i < func->GooId.size(); ++i) {
if (i > 0) fprintf(f,",");
fprint_str(f, func->GooId[i]);
}
}
TEXT_FIELD(GooId)
{
for(int i = 0; i < func->GooId.size(); ++i) {
fprintf(f,"_");
fprint_str(f, func->GooId[i]);
}
}
TEXT_FIELD(MacroExprs)
{
for(int i = 0; i < Args.size(); ++i) {
if (Args[i].OverrideName.size() > 0) {
fprintf(f,",");
fprint_str(f, Args[i].OverrideName);
} else {
fprintf(f, ",a%d", Args[i].No);
}
}
}
TEXT_FIELD(MacroArgs)
{
if (!func->NoMsg()) {
fprintf(f,"MSG");
}
for(int i = 0; i < Args.size(); ++i) {
if (Args[i].OverrideName.size() > 0) {
// fprint_str(f, Args[i].OverrideName);
} else {
if (i != 0 || !func->NoMsg()) fprintf(f,",");
fprintf(f, "a%d", i + MSGTYPBASE);
}
}
}
END_FIELD_TABLE
Message(){}
Message(
const string& idTxt,
const string&,
const vector<Argument>&,
UINT msgid,
int lineno,
const ParsedFormatString& fmtStr,
const Func& aFunc
)
:IdName(idTxt), id(msgid), LineNo(lineno),
FormatStr(fmtStr),func(&aFunc)
{
Args.resize(FormatStr.ArgCount, Argument("",0) );
for(int i = 0; i < FormatStr.Items.size(); ++i) {
int no = FormatStr.Items[i].no;
if (no > 0) {
Args[no-1] = Argument
(FormatStr.Items[i].expr, FormatStr.Items[i].type);
Args[no-1].OverrideName = FormatStr.Items[i].argName;
Args[no-1].No = no-1 + MSGTYPBASE;
}
}
if (reorder) {
sort(Args.begin(), Args.end());
}
for (int j = 0; j < FormatStr.Items.size(); ++j) {
int& no = FormatStr.Items[j].no;
if (no > 0) {
// find new msg no for that
for(int k = 0; k < Args.size(); ++k) {
if (Args[k].No == no-1 + MSGTYPBASE) {
no = MSGTYPBASE + k;
break;
}
}
// Args[no-1].OverrideName = FormatStr.Items[i].argName;
}
}
typeSig = GetTypeSig(Args, func->Unsafe());
if (func->MsgArg) {
reorderSig = GetReorderSig(Args);
}
// UpgradeFormatSpecifiers(Msg, 10, (typeSig.size())?0:&TypeSig);
// ChkArgs();
}
};
struct File : FieldHolder {
string _CanonicalName; // lower case, bad chars => _
string _Path;
string _UppercaseName; // uppercase canonical
string _Name;
string _BaseName; // no extension
ULONGLONG ModificationTime;
//
// Since the file is a member of the set type
// fields that don't affect set membership should be mutables
//
mutable vector<Message*> Msgs;
mutable set<string, strless> IdsFound;
BEGIN_FIELD_TABLE(File, out)
TEXT_FIELD(Name) fprint_str(out, _Name );
TEXT_FIELD(Path) fprint_str(out, _Path );
TEXT_FIELD(CanonicalName) fprint_str(out, _CanonicalName );
TEXT_FIELD(UppercaseName) fprint_str(out, _UppercaseName );
ENUM_FIELD(Messages, Msgs, VectorPtrTag)
END_FIELD_TABLE
File(){} // to make STL and compiler happy
string FullFileName() const {
string Tmp(_Path);
Tmp.append(_Name);
return Tmp;
}
File(const string& FileName, const string& Path, const WIN32_FIND_DATA& FindData)
:_Name(FileName), _Path(Path)
{
CopyMemory(&ModificationTime, &FindData.ftLastWriteTime, sizeof(ModificationTime));
// Canonicalize FileName
// i.e. make it suitable to be a DEFINE name
//
// Cut the path out. Replace all non-alphanumeric symbols
// with an underscore.
string::size_type pos = _Name.rfind('\\');
if (pos == string::npos) {
// No back slash
_CanonicalName.assign(_Name);
} else {
// Get only the name part
_CanonicalName.assign(_Name.begin() + pos, _Name.end());
}
// strip extension for a base name
string::size_type ext = _CanonicalName.rfind('.');
if (ext != string::npos && !CheckExtension(_CanonicalName, ext, PreserveExtensions)) {
// there was a "."
_BaseName.assign(_CanonicalName.begin(),
_CanonicalName.begin() + ext);
} else {
_BaseName.assign(_CanonicalName);
}
// Can't have a digit in the first position
// Let's prepend it with an underscore
{
char ch = _CanonicalName[0];
if ( ch >= '0' && ch <= '9' ) {
_CanonicalName.insert(0, '_');
}
}
_UppercaseName.assign( _CanonicalName );
for(UINT i = 0; i < _CanonicalName.size(); ++i) {
char ch = _CanonicalName[i];
if ( (ch == '_')
|| ((ch >= 'A') && (ch <= 'Z'))
|| ((ch >= '0') && (ch <= '9')) )
{
// Good Character. Do nothing
} else if ( (ch >= 'a') && (ch <= 'z') ) {
_UppercaseName[i] = static_cast<char>(ch - 'a' + 'A');
} else {
_CanonicalName[i] = '_';
_UppercaseName[i] = '_';
}
}
}
bool operator < (const File& b) const {
return _CanonicalName.compare(b._CanonicalName) < 0;
}
};
int Message::ArgConsumed() const
{
vector<Argument>::const_iterator i = Args.begin();
int sum = 0;
for(;i != Args.end(); ++i) {
sum += i->Type->ArgConsumed;
}
return sum;
}
void Message::hash(Hasher& hash) const
{
vector<Argument>::const_iterator i = Args.begin();
for(;i != Args.end(); ++i) {
i->hash(hash);
}
hash.Hash(&id, sizeof(id));
hash.Hash(IdName);
hash.Hash(FormatStr.HostString);
}
typedef map<string, string, strless > TYPE_MAP;
typedef map<string, Message, strless > MSG_MAP;
typedef map<string, Func, strless > FUNC_MAP;
typedef map<string, Group, strless> GROUP_MAP;
typedef vector< Prefix > PREFIX_VECTOR;
typedef set<File> FILES;
FILES::iterator CurrentFile;
string CurrentTpl;
string currentFileName()
{ return CurrentFile->_Name; }
void MsgMapHash(
const MSG_MAP& msgmap,
Hasher& hash
)
{
MSG_MAP::iterator i = msgmap.begin();
for(;i != msgmap.end(); ++i) {
i->second.hash(hash);
}
}
void expand(IN OUT string& str, const string& id);
BOOL
parseMsg(
IN LPCSTR beg,
IN LPCSTR end,
IN OUT string& str,
OUT LPCSTR* strend = 0
)
{
LPCSTR p = beg;
for(;;) {
if (*p == '_' || isalpha(*p)) {
const char * id = p;
for(;;) {
if (++p == end) { expand(str, string(id,p)); goto done; }
if (*p == '_' || isalpha(*p) || isdigit(*p)) continue;
break;
}
expand(str, string(id,p));
}
if (*p == ',') goto done;
if (*p == '"') {
const char * run = ++p;
for(;;) {
if (p == end) goto unterminated;
if (*p == '"' && p[-1] != '\\') break;
++p;
}
str.append(run,p);
}
if (++p == end) break;
}
done:
if (str.size() == 0) {
ReportError("parsing %s. Cannot find format string\n",
string(beg,end).c_str() );
return FALSE;
}
// eat whitespace at the end
while (str.size() > 0 && isspace( str.end()[-1]) )
{ str.resize(str.size()-1); }
// get rid of pesky trailing \n
if (str.size() >= 2 && str.end()[-1] == 'n' && str.end()[-2] == '\\') {
str.resize(str.size()-2);
}
Flood("msg: \"%s\".\n", str.c_str() );
if (strend) {
*strend = p;
}
return TRUE;
unterminated:
ReportError("Unterminated string constant\n");
return FALSE;
}
BOOL
parseArgs(
IN LPCSTR beg,
IN LPCSTR end,
IN OUT vector<Argument>& args
);
#define isvar(x) ( isalnum(x) || (x == '_') )
Prefix::Prefix(
IN PSTR_PAIR str,
IN UINT count):FuncName(str[1].beg, str[1].end)
{
vector<string> ArgNames;
if (count > 2) {
string msg;
if (str[2].beg < str[2].end &&
str[2].beg[0] == '"') parseMsg(str[2].beg, str[2].end, msg);
else msg.assign(str[2].beg, str[2].end);
FmtStr.init( msg );
}
if (count > 3) {
count -= 3; str += 2;
while (count-- > 0) {
++str; ArgNames.push_back( string(str->beg,str->end) );
}
}
// check that sizes match //
if (ArgNames.size() > FmtStr.ArgCount ) {
ReportError("Prefix: Extra argument. Only %d are specified in the string\n", FmtStr.ArgCount);
exit(1);
}
if (ArgNames.size() < FmtStr.ArgCount ) {
ReportError("Prefix: Not enough args. %d are specified in the string\n", FmtStr.ArgCount);
exit(1);
}
for(int i = 0; i < FmtStr.Items.size(); ++i) {
int no = FmtStr.Items[i].no;
if (no > 0) {
FmtStr.Items[i].argName = ArgNames[no-1];
FmtStr.Items[i].expr = ArgNames[no-1];
}
}
}
struct GuidEntry : FieldHolder {
GUID guid;
string comment;
int count;
vector<Message*> Msgs;
void printTxt(FILE* f) const {
fprintf(f,"%08x-%04x-%04x-%02x%02x-%02x%02x%02x%02x%02x%02x",
guid.Data1,guid.Data2,guid.Data3,
guid.Data4[0],guid.Data4[1],guid.Data4[2],guid.Data4[3],
guid.Data4[4],guid.Data4[5],guid.Data4[6],guid.Data4[7] );
}
void printDat(FILE* f) const {
fprintf(f,"{0x%08x,0x%04x,0x%04x,{0x%02x,0x%02x,0x%02x,0x%02x,0x%02x,0x%02x,0x%02x,0x%02x}}",
guid.Data1,guid.Data2,guid.Data3,
guid.Data4[0],guid.Data4[1],guid.Data4[2],guid.Data4[3],
guid.Data4[4],guid.Data4[5],guid.Data4[6],guid.Data4[7] );
}
BEGIN_FIELD_TABLE(GuidEntry, f)
TEXT_FIELD(Text) { printTxt(f); }
TEXT_FIELD(Struct) { printDat(f);}
TEXT_FIELD(Comment) fprintf(f, "%s", comment.c_str() );
ENUM_FIELD(Messages, Msgs, VectorPtrTag);
END_FIELD_TABLE
GuidEntry(){}
GuidEntry(int cnt, const string& cmnt):count(cnt),comment(cmnt) {
RPC_STATUS status = UuidCreate( &guid );
if ( status != RPC_S_OK ) {
ReportError("UuidCreate failed with error 0x%0X\n", status);
exit(1);
}
}
GuidEntry(PSTR_PAIR p, int cnt, const string& cmnt):count(cnt),comment(cmnt)
{
guid.Data1 = Hex(p->beg + 0, 8);
guid.Data2 = (USHORT)Hex(p->beg + 9, 4);
guid.Data3 = (USHORT)Hex(p->beg + 14, 4);
guid.Data4[0] = (UCHAR) Hex(p->beg + 19, 2);
guid.Data4[1] = (UCHAR) Hex(p->beg + 21, 2);
guid.Data4[2] = (UCHAR) Hex(p->beg + 24, 2);
guid.Data4[3] = (UCHAR) Hex(p->beg + 26, 2);
guid.Data4[4] = (UCHAR) Hex(p->beg + 28, 2);
guid.Data4[5] = (UCHAR) Hex(p->beg + 30, 2);
guid.Data4[6] = (UCHAR) Hex(p->beg + 32, 2);
guid.Data4[7] = (UCHAR) Hex(p->beg + 34, 2);
}
};
struct Guids : FieldHolder {
UINT min_val;
UINT max_val;
vector<GuidEntry> guids;
int current;
string currentFileName;
Guids(int lo, int hi) : min_val(lo),max_val(hi),current(0) {}
void erase() {current = 0; guids.erase(guids.begin(), guids.end());}
BEGIN_FIELD_TABLE(Guids, f)
TEXT_FIELD(Count) fprintf(f, "%d", min(current+1, (signed)guids.size()) );
DEFAULT_ENUM_FIELD(guids, VectorTag)
END_FIELD_TABLE
ULONG getid() {
for(;;) {
if ( current == (signed)guids.size() ) {
guids.push_back(GuidEntry(max_val - min_val + 1, currentFileName) );
}
if ( guids[current].count ) {
ULONG ret = (max_val + 1 - guids[current].count--) | ( current << 16 );
return ret;
}
++current;
}
}
void add_guids(PSTR_PAIR p, int count) {
--count; ++p; // skip func name //
string comment(p->beg,p->end);
while (--count) {
guids.push_back(GuidEntry(++p, max_val - min_val + 1, comment));
}
}
void new_file(const char* fname) {
currentFileName.assign(fname);
// this will force a new guid to be allocated
// when a new id is required
current = static_cast<int>(guids.size());
}
};
struct GenPair {
string tpl;
string out;
GenPair(){} // STL pleaser
GenPair(const string& a, const string& b)
:tpl(a),out(b) {}
};
//TYPE_MAP AutoGen;
TYPE_MAP TypeMap;
MSG_MAP MsgMap;
//MACRO_MAP MacroMap;
FUNC_MAP Funcs;
GROUP_MAP Groups;
PREFIX_VECTOR Prefixes;
PREFIX_VECTOR Suffixes;
//Guids CtrlGuids(0,31);
Guids TraceGuids(10,65535);
FILES Files;
vector<string> ScanForMacros;
vector<string> Touch;
vector<GenPair>GenMulti;
vector<GenPair>GenSingle;
set<string, strless> LookFor;
void printTraceGuid(FILE* f, int guidno)
{
TraceGuids.guids[guidno].printTxt(f);
}
struct Keyword {
int nParams; // if < 0 then at least -nParams, otherwise = nParams //
EZPARSE_CALLBACK handler;
int id;
PVOID context;
Keyword(){}
Keyword(EZPARSE_CALLBACK func,
int n, UINT i, PVOID ctx) : nParams(n), handler(func), id(i), context(ctx) {}
};
typedef map<string, Keyword, strless> KEYWORD_MAP;
KEYWORD_MAP Keywords;
#if DEPRECATED
//BOOL GenerateTypeTable;
string TypeTableBegin;
string TypeTableEntry;
string TypeTableEnd;
string TypeTablePrefix;
string GuidStore;
#endif
const string CtlStr("LEVEL"); // the same as GRP
const string MakeStr("MAKESTR");
const string MsgArgStrUnsafe("(MSG,..unsafe..)");
const string MsgArgStr("(MSG,...)");
const string MsgValStr("MsgVal");
const string MsgStr("MSG");
const string ArgStr("...");
const string ArgStrUnsafe("..unsafe..");
const string IndentStr("INDENT");
const string NullStr("NULL");
bool
Files_AddFile(const string& Name, string Path, const WIN32_FIND_DATA& FindData)
{
if (!AllowedExtension(Name)) {
return FALSE;
}
pair<FILES::iterator, bool> result = Files.insert(File(Name,Path,FindData));
if (!result.second) {
if (Name.compare(result.first->_Name) == 0) {
ReportError("File %s was already in the processing list\n",Name.c_str());
} else {
ReportError("Files %s and %s have the same canonical name %s\n",
Name.c_str(),
result.first->_Name.c_str(),
result.first->_CanonicalName.c_str());
}
}
return result.second;
}
int
compare(const string& a, const STR_PAIR& b)
{
return a.compare(0, a.length(), b.beg, b.end - b.beg);
}
BOOL
parseLegacy(
IN PSTR_PAIR strs,
IN INT count,
IN INT hole, // args - msg, usually one
IN OUT string& Msg,
IN OUT vector<Argument>&,
OUT ParsedFormatString& FmtStr
)
{
vector<string> ArgNames;
STR_PAIR buf = *strs;
LPCSTR p = buf.beg, q;
int i;
if ( p < buf.end && buf.end[-1] == ')' ) --buf.end;
if(!parseMsg(buf.beg, buf.end, Msg, &p)) {
return FALSE;
}
if( !FmtStr.init(Msg) ) {
return FALSE;
}
ArgNames.reserve(FmtStr.ArgCount);
if (count == 1) {
// need to grok args from the string
// Let's get to the first ','
if (p == buf.end) goto success;
while(*p != ',') if (++p == buf.end) goto success;
q = ++p;
for(i = 0;;++i) {
int parlevel = 0;
// currently we don't deal with the case of commas or parents within '"'
while (p < buf.end) {
if (parlevel == 0 && *p == ',') break;
if (*p == '(') ++parlevel;
if (*p == ')') {
if (--parlevel < 0)
{ReportError("Too many ')'\n"); return FALSE;}
}
++p;
}
if (parlevel > 0) {
ReportError("No closing ')'");
}
LPCSTR tmp = p; // remember where we were
while (q < p && isspace(*q)) ++ q;
while (q < p && isspace(p[-1])) --p;
ArgNames.push_back( string(q,p) );
p = tmp;
if (p == buf.end) break;
q = ++p;
}
} else {
// arguments were supplied in strs
strs += hole; count -= hole;
while (count-- > 0) {
ArgNames.push_back( string(strs->beg,strs->end) );
++strs;
}
}
success:
// check that sizes match //
if ((unsigned)ArgNames.size() != (unsigned)FmtStr.ArgCount ) {
ReportError("%d argument(s) expected, argument(s) supplied: %d\n", FmtStr.ArgCount, ArgNames.size());
return FALSE;
}
for(i = 0; i < FmtStr.Items.size(); ++i) {
int no = FmtStr.Items[i].no;
if (no > 0) {
FmtStr.Items[i].expr = ArgNames[no-1];
}
}
return TRUE;
}
BOOL
parseArgs(
IN LPCSTR beg,
IN LPCSTR end,
IN OUT vector<Argument>& args
)
{
LPCSTR p = beg;
// MACRO_MAP::iterator macro;
TYPE_SET::iterator macro;
for(;;) {
while ( isspace(*p) ) {
if (++p == end) return TRUE;
}
LPCSTR q = p; // id start
while ( isvar(*p) ) {
if (++p == end) return TRUE;
}
macro = TypeSet.find( string(q, p) );
if ( macro == TypeSet.end() ) {
if (p == q) {
q = beg;
p = end;
}
ReportError("Unknown type: %s\n", string(q, p).c_str() );
return FALSE;
}
macro->second.Used = TRUE;
while ( *p != '(') {
if (++p == end) return TRUE;
}
q = p; // now q points to '('
int level = 0;
for (;;) {
if (*p == '(') {
++level;
} else if (*p == ')') {
if (--level == 0) {
break;
}
}
if (++p == end) return TRUE;
}
args.push_back( Argument(string(q+1,p), &macro->second) );
if (++p == end) return TRUE;
}
// return TRUE; // unreachable code
}
enum {
ID_TraceRange,
ID_ProcessFiles,
ID_TemplateFile,
ID_TypeMacro,
ID_TypevMacro,
ID_Include,
ID_UsePrefix,
ID_UseSuffix,
ID_NoPrefix,
ID_DefineSimpleTypePtr,
ID_DefineSimpleType,
ID_DefineCplxType,
ID_CustomType,
ID_DefineFlavor,
ID_SeparateTraceGuidPerFile,
ID_Touch,
ID_ScanForMacros,
ID_GenerateTypeTable,
ID_Exceptions,
ID_WppFlags,
};
void
UpdateIntVar(
UINT* var,
PSTR_PAIR Str
)
{
LPCSTR p = Str->beg, q = Str->end;
UINT x = 0;
while (p != q && isdigit(*p) ) {
x = (x * 10) + (*p - '0');
++p;
}
*var = x;
}
void parseAssumedArgs(Func& f,LPCSTR beg, LPCSTR end)
{
// we have a string of A=value1,B=value2,C=value3,...,F=valuen}
// we need to put A,B,C into f.GooId,
// and value1, value2, value3 into f.Goo
LPCSTR p = beg, q, stop;
if (p >= end) return;
Flood("Got %p %p\n", beg, end );
for(;p < end;) {
while (isspace(*p)) ++p; // cannot have spaces all the way. there is '}'
if (p == end) return;
q = p++;
while (p < end && *p != '=') ++p;
if (p == end) {
ReportError("Missing '=' in %s\n", string(beg,end).c_str() ); exit(1); }
stop = p;
while (--p > q && isspace(*p));
if (p == q) {
ReportError("Id required before '=' in %s\n", string(beg,end).c_str() ); exit(1); }
int isMsg = memcmp(q,"MSG=",4) == 0;
if (!isMsg)
f.GooId.push_back( string(q,p+1) );
p = stop + 1;
while (isspace(*p)) ++p; // cannot have spaces all the way. there is '}'
q = p++;
while (*p != '}' && *p != ',') ++p;
stop = p;
while (--p > q && isspace(*p));
if (isspace(*p)) { // BUGBUG Verify that this condition is correct
ReportError("value required after '=' in %s\n", string(beg,end).c_str() ); exit(1); }
if (isMsg)
f.assumedMsg = STR_PAIR(q, p+1);
else
f.Goo.push_back( string(q,p+1) );
if (*stop == '}') break;
p = stop + 1;
}
f.nAssumedArgs = f.Goo.size();
}
DWORD
ParseConfigCallback (
IN PSTR_PAIR Str,
IN INT Count,
IN PVOID Context,
IN PEZPARSE_CONTEXT ParseContext
)
{
string Name1(Str[0].beg, Str[0].end);
KEYWORD_MAP::iterator keyword = Keywords.find(Name1);
if (Context) {
// We need to ignore all the keywords besides ID_TypeMacro //
if (keyword != Keywords.end() && keyword->second.id == ID_TypeMacro) {
// proceed and do the job //
} else {
// Ignore everything else
return ERROR_SUCCESS;
}
}
if (keyword != Keywords.end() ) {
if (keyword->second.nParams < 0 && Count - 1 < -keyword->second.nParams ) {
ReportError("%s requires at least %d parameter(s)\n",
Name1.c_str(), -keyword->second.nParams);
return ERROR_INVALID_PARAMETER;
}
if (keyword->second.nParams > 0 && Count - 1 != keyword->second.nParams ) {
ReportError("%s requires %d parameter(s) (we have %s)\n",
Name1.c_str(), keyword->second.nParams, std::string(Str[1].beg, Str[Count-1].end).c_str() );
return ERROR_INVALID_PARAMETER;
}
switch (keyword->second.id) {
#ifdef DEPRECATED
case ID_UseTraceGuid:
TraceGuids.add_guids(Str+1, Count-1);
break;
case ID_UseCtrlGuid:
CtrlGuids.add_guids(Str+1, Count-1);
break;
case ID_SeparateTraceGuidPerFile:
UpdateIntVar((UINT*)(keyword->second.context), Str+1);
break;
case ID_GrpidRange:
UpdateIntVar(&CtrlGuids.min_val, Str+1); UpdateIntVar(&CtrlGuids.max_val, Str+2);
break;
#endif
case ID_TraceRange:
UpdateIntVar(&TraceGuids.min_val, Str+1); UpdateIntVar(&TraceGuids.max_val, Str+2);
break;
case ID_ProcessFiles:
while(--Count) { ++Str; Fill( string(Str->beg, Str->end) ); }
break;
case ID_WppFlags:
while(--Count) { ++Str; if(*Str->beg == '-') ++Str->beg; DealWithCmdLineOptions( Str->beg, Str->end ); }
Unusual("\n");
break;
case ID_ScanForMacros:
while(--Count) { ++Str; ScanForMacros.push_back( string(Str->beg, Str->end) ); }
break;
case ID_Touch:
while(--Count) { ++Str; Touch.push_back( string(Str->beg, Str->end) ); }
break;
case ID_DefineSimpleType:
{
string Name(Str[1].beg, Str[1].end);
if (Str[4].beg < Str[4].end && *Str[4].beg == '"') Str[4].beg++;
if (Str[4].beg < Str[4].end && Str[4].end[-1] == '"') Str[4].end--;
if (TypeSet.find(Name) != TypeSet.end()) {
if (IgnoreDupTypes) {
return ERROR_SUCCESS;
}
ReportError("Type %s is already defined\n", Name.c_str() );
break;
}
Flood(" type %s\n", Name.c_str());
TypeSet[Name] = WppType( Name , // name
string(Str[2].beg, Str[2].end), // c-type
SimpleValueMacroStart,
string(Str[3].beg, Str[3].end), // mof type
"", // MofExtension
string(Str[4].beg, Str[4].end),
string(Str[5].beg, Str[5].end),
stoi(Str[6], "priority"),
Count > 7?stoi(Str[7], "argused"):1
);
break;
}
case ID_DefineSimpleTypePtr:
{
string Name(Str[1].beg, Str[1].end);
if (Str[4].beg < Str[4].end && *Str[4].beg == '"') Str[4].beg++;
if (Str[4].beg < Str[4].end && Str[4].end[-1] == '"') Str[4].end--;
if (TypeSet.find(Name) != TypeSet.end()) {
if (IgnoreDupTypes) {
return ERROR_SUCCESS;
}
ReportError("Type %s is already defined\n", Name.c_str() );
break;
}
TypeSet[Name] = WppType( Name , // name
string(Str[2].beg, Str[2].end), // c-type
SimplePtrMacroStart,
string(Str[3].beg, Str[3].end), // mof type
"", // MofExtension
string(Str[4].beg, Str[4].end),
string(Str[5].beg, Str[5].end),
stoi(Str[6], "priority"),
Count > 7?stoi(Str[7], "argused"):1
);
break;
}
case ID_DefineCplxType:
{
string Name(Str[1].beg, Str[1].end);
string MacroStart(Str[2].beg, Str[2].end);
MacroStart.append("(");
if (Str[5].beg < Str[5].end && *Str[5].beg == '"') Str[5].beg++;
if (Str[5].beg < Str[5].end && Str[5].end[-1] == '"') Str[5].end--;
if (TypeSet.find(Name) != TypeSet.end()) {
if (IgnoreDupTypes) {
return ERROR_SUCCESS;
}
ReportError("Type %s is already defined\n", Name.c_str() );
break;
}
TypeSet[Name] = WppType( Name , // name
string(Str[3].beg, Str[3].end), // equiv type
MacroStart,
string(Str[4].beg, Str[4].end), // mof type
"", // MofExtension
string(Str[5].beg, Str[5].end),
string(Str[6].beg, Str[6].end),
stoi(Str[7], "priority"),
Count > 8?stoi(Str[8], "argused"):1
);
}
break;
case ID_DefineFlavor:
{
string Name(Str[1].beg, Str[1].end);
string BaseType(Str[2].beg, Str[2].end);
TYPE_SET::const_iterator it = TypeSet.find(BaseType);
if (Str[4].beg < Str[4].end && *Str[4].beg == '"') Str[4].beg++;
if (Str[4].beg < Str[4].end && Str[4].end[-1] == '"') Str[4].end--;
if (it == TypeSet.end()) {
ReportError("Type Not Found %s\n", BaseType.c_str());
break;
}
if (TypeSet.find(Name) != TypeSet.end()) {
if (IgnoreDupTypes) {
return ERROR_SUCCESS;
}
ReportError("Type %s is already defined\n", Name.c_str() );
break;
}
WppType Flavor(it->second);
Flavor.TypeName = Name;
if (!Str[3].empty())
{ Flavor.MofType.assign(Str[3].beg, Str[3].end); }
if (!Str[4].empty())
{ Flavor.FormatSpec.assign(Str[4].beg, Str[4].end); }
TypeSet[Name] = Flavor;
break;
}
case ID_TypeMacro:
case ID_CustomType:
{
string Name(Str[1].beg, Str[1].end);
LPCSTR p = Str[2].beg, q = Str[2].end;
while (p < Str[2].end && *p != '(') ++p;
if (p < Str[2].end) {
q = p-1;
while (q >= Str[2].beg && isspace(*q)) --q;
++q;
}
string BaseType(Str[2].beg, q);
TYPE_SET::const_iterator it = TypeSet.find(BaseType);
if (it == TypeSet.end()) {
ReportError("Type Not Found %s\n", BaseType.c_str());
break;
}
if (TypeSet.find(Name) != TypeSet.end()) {
if (IgnoreDupTypes) {
return ERROR_SUCCESS;
}
ReportError("Type %s is already defined\n", Name.c_str() );
break;
}
if (Count == 4) {
if (Str[3].beg < Str[3].end && *Str[3].beg == '"') Str[3].beg++;
if (Str[3].beg < Str[3].end && Str[3].end[-1] == '"') Str[3].end--;
}
{
string ExtendedType;
LPCSTR end = Str[2].end;
if ( p != end ) {
ExtendedType.reserve(end - p);
do {
while(p != end && isspace(*p)) ++p;
LPCSTR qq = p;
while(qq != end && !isspace(*qq)) ++qq;
ExtendedType.append(p, qq);
p = qq;
} while ( p != end );
}
WppType Flavor(it->second);
Flavor.TypeName = Name;
Flavor.Extension = ExtendedType;
if (Count == 4)
{ Flavor.FormatSpec.assign(Str[3].beg,Str[3].end); }
if (keyword->second.id == ID_TypeMacro) {
Flavor.Flags = WT_MACRONAME;
} else {
Flavor.Flags = 0;
}
TypeSet[Name] = Flavor;
}
break;
}
case ID_Include:
return EzParse(std::string(Str[1].beg, Str[1].end).c_str(),ParseConfigCallback,0);
case ID_UseSuffix:
Suffixes.push_back( Prefix(Str, Count) );
break;
case ID_UsePrefix:
case ID_NoPrefix:
Prefixes.push_back( Prefix(Str, Count) );
break;
default:;
//return Keywords->handler(Str, Count, Context);
}
} else {
//
// a macro can have a list of assumed arguments in
// curly braces right after the name
//
// Ex: TraceNoise{LEVEL=Noise}(MSG,...)
//
Flood("Got %s\n", Name1.c_str() );
LPCSTR assumed_beg = find(Str[0].beg,Str[0].end, '{');
LPCSTR assumed_end = find(Str[0].beg,Str[0].end, '}');
if (assumed_beg != Str[0].end) {
if (assumed_end == Str[0].end) {
ReportError("No closing brace in '%s'\n",
Name1.c_str() );
exit(1);
}
Name1.assign(Str[0].beg, assumed_beg);
Noise("Real name is %s (%s)\n", Name1.c_str(),
string(assumed_beg+1, assumed_end).c_str() );
} else if (assumed_end != Str[0].end) {
ReportError("No openning brace in '%s'\n",
Name1.c_str() );
exit(1);
}
FUNC_MAP::iterator it = Funcs.find(Name1);
if (it != Funcs.end()) {
ReportError("Function %s is already defined\n",
Name1.c_str() );
exit(1);
}
Func f(Name1);
BOOL SeenMsg = FALSE;
if (ParseContext->doubleParent) {
f.SetDoubleP();
}
if (assumed_end > assumed_beg) {
parseAssumedArgs(f, assumed_beg+1, assumed_end+1);
}
for (int i = 1; i < Count; ++i) {
// check for '='
LPCSTR div = find(Str[i].beg, Str[i].end, '=' );
if ( div != Str[i].end ) {
//f.Args.push_back( string(div+1, Str[i].end) );
f.Goo.push_back( string(Str[i].beg, div ) );
f.GooId.push_back( string(Str[i].beg, div ) );
f.MsgVal = static_cast<UCHAR>(i); // can I have more of those?
continue;
}
if ( compare(CtlStr, Str[i]) == 0 ) {
f.Grp = static_cast<UCHAR>(i);
f.Goo.push_back( string(Str[i].beg, Str[i].end) );
f.GooId.push_back( string(Str[i].beg, Str[i].end) );
} else if ( compare(MsgArgStr, Str[i]) == 0 ) {
f.MsgArg = static_cast<UCHAR>(i); // can I have more of those?
SeenMsg = TRUE;
// f.SetLineBeg();
continue;
} else if ( compare(MsgArgStrUnsafe, Str[i]) == 0 ) {
f.SetUnsafe();
f.MsgArg = static_cast<UCHAR>(i); // can I have more of those?
continue;
} else if ( compare(MsgValStr, Str[i]) == 0 ) {
f.MsgVal = static_cast<UCHAR>(i); // can I have more of those?
} else if ( compare(MakeStr, Str[i]) == 0 ) {
f.MsgVal = static_cast<UCHAR>(i); // can I have more of those?
f.set(FO_NOMACRO);
// f.SetLineBeg();
// should i continue?
} else if ( compare(MsgStr, Str[i]) == 0 ) {
f.Msg = static_cast<UCHAR>(i);
SeenMsg = TRUE;
continue;
} else if ( compare(NullStr, Str[i]) == 0 ) {
f.NullArg = static_cast<UCHAR>(i);
} else if ( compare(IndentStr, Str[i]) == 0 ) {
f.Indent = static_cast<UCHAR>(i);
} else if ( compare(ArgStrUnsafe, Str[i]) == 0 ) {
f.Arg = static_cast<UCHAR>(i);
f.SetUnsafe();
f.SetVarArgs();
if (i != Count - 1) {
ReportError("func %s: '...' has to be the very last argument\n",
Name1.c_str() );
}
continue;
} else if ( compare(ArgStr, Str[i]) == 0 ) {
f.Arg = static_cast<UCHAR>(i);
f.SetVarArgs();
if (i != Count - 1) {
ReportError("func %s: '...' has to be the very last argument\n",
Name1.c_str() );
}
continue;
} else {
f.Goo.push_back( string(Str[i].beg, Str[i].end) );
f.GooId.push_back( string(Str[i].beg, Str[i].end) );
}
f.Args.push_back( string(Str[i].beg, Str[i].end) );
}
if (!SeenMsg) { f.SetNoMsg(); }
f.Num = static_cast<UCHAR>(Count - 1);
Noise(" func %s\n", Name1.c_str() );
Funcs[ Name1 ] = f;
}
return ERROR_SUCCESS;
}
#if 0
DWORD
PrintCallback (
IN PSTR_PAIR Str,
IN UINT Count,
IN PVOID
)
{
UINT i;
for (i = 0; i < Count; ++i) {
putchar('<'); fwrite(Str[i].beg, Str[i].end - Str[i].beg, 1, stdout); putchar('>');
}
putchar('\n');
return ERROR_SUCCESS;
}
#endif
void CleanupString(LPCSTR beg, LPCSTR end, OUT string& msg)
{
LPCSTR p = beg, q;
msg.resize(0);
while (p < end) {
// skip spaces
while (p < end && isspace(*p)) ++p;
q = p;
if (p < end && *p == '"') {
++p;
while (p < end && (*p != '"' || p[-1] != '\\')) ++p;
}
if (p < end) ++p;
msg.append(q,p);
}
}
DWORD
ParseSrcCallback (
IN PSTR_PAIR Str,
IN INT Count,
IN PVOID,
IN PEZPARSE_CONTEXT ParseContext
)
{
string FuncName(Str[0].beg, Str[0].end);
string msg, msgval, id, indent;
vector<Argument> args;
ParsedFormatString FmtStr;
int LineNo;
// UINT argno = 10;
if (FuncName.compare("WPP_COMPONENT_NAME") == 0) {
if (Count != 2) {
ReportError("WPP_COMPONENT_NAME requires 1 argument.\n");
exit(1);
}
ComponentName.assign(Str[1].beg, Str[1].end);
return ERROR_SUCCESS;
}
// Look for special directive
if ( LookFor.find( FuncName ) != LookFor.end() ) {
Noise("SpecialString found %s\n", FuncName.c_str());
CurrentFile->IdsFound.insert(FuncName);
}
FUNC_MAP::iterator func = Funcs.find( FuncName );
if (func == Funcs.end() ) {
return ERROR_SUCCESS;
}
if (func->second.DoubleP() && !ParseContext->doubleParent) {
ReportError("%s requires ((args))\n", func->first.c_str() );
exit(1);
}
if (!func->second.DoubleP() && ParseContext->doubleParent) {
ReportError("%s doesn't take ((args))\n", func->first.c_str() );
exit(1);
}
msg.reserve(255);
msgval.reserve(255);
if (func->second.VarArgs()) {
if (Count < func->second.Num ) {
ReportError("%s requires at least %d arguments. (Found only %d)\n",
func->first.c_str(), func->second.Num, Count-1);
return ERROR_INVALID_PARAMETER;
}
} else {
if (Count-1 != func->second.Num ) {
ReportError("%s requires %d arguments. (Found only %d)\n",
func->first.c_str(), func->second.Num, Count-1);
return ERROR_INVALID_PARAMETER;
}
}
if (func->second.LineBeg() && !func->second.DoubleP() ) {
LineNo = EzGetLineNo(Str[0].beg, ParseContext);
} else {
LineNo = EzGetLineNo(ParseContext->macroEnd, ParseContext);
}
{
char Num[64];
sprintf(Num,"%d", LineNo);
id = CurrentFile->_CanonicalName;
id += Num;
}
if (GRP(func->second)) {
PSTR_PAIR p = Str + GRP(func->second);
string group(p->beg, p->end);
GROUP_MAP::iterator i = Groups.find(group);
if (i == Groups.end()) {
Groups[ string(p->beg, p->end) ] = Group(0, string(p->beg, p->end), id);
} else {
i->second.MsgIds.push_back(id);
}
}
if (func->second.MsgArg) {
PSTR_PAIR p = Str + func->second.MsgArg;
if (p->beg == p->end || *p->beg != '(') {
ReportError("MsgArg argument should have form (MSG,...)\n");
exit(1);
}
if (!parseLegacy(p, 1, 1, msg, args, FmtStr)) {
return ERROR_INVALID_PARAMETER;
}
}
if (!func->second.assumedMsg.empty() || MSG(func->second)) {
PSTR_PAIR p = Str + MSG(func->second); // MSG(.) == 0 if assumed
if (!func->second.assumedMsg.empty()) {
*p = func->second.assumedMsg; // Str[0] now holds the assumed message
}
if (func->second.VarArgs()) {
if (!parseLegacy(p, Count-MSG(func->second),
ARG(func->second) - MSG(func->second),
msg, args, FmtStr)) {
return ERROR_INVALID_PARAMETER;
}
} else if ( !parseMsg(p->beg, p->end, msg) ) {
return ERROR_INVALID_PARAMETER;
}
}
if (func->second.Indent) {
PSTR_PAIR p = Str + func->second.Indent;
indent.assign(p->beg, p->end);
}
if (func->second.MsgVal) {
PSTR_PAIR p = Str + func->second.MsgVal;
CleanupString(p->beg, p->end, msgval);
//msgval.assign(p->beg, p->end);
string::size_type div = msgval.find('=');
if (div != string::npos) {
FmtStr.HostString += msgval.substr(div+1);
} else {
FmtStr.HostString += msgval;
}
msgval.append(","); // Why comma?
}
if (func->second.prefix) {
FmtStr.insert_prefix(func->second.prefix->FmtStr);
}
if (func->second.suffix) {
FmtStr.append(func->second.suffix->FmtStr);
}
{
MSG_MAP::iterator i = MsgMap.find(id);
if ( i != MsgMap.end() ) {
// Id already exists. Can't happen //
ReportError("Can't handle multiple trace statements on the same line\n");
exit(1);
}
Message* Msg =
&(MsgMap[id] =
Message(id, msg, args, TraceGuids.getid(), LineNo, FmtStr,
func->second) ) ;
Msg->msgval = msgval;
Msg->Indent = indent;
for(int ic = 1; ic < Count; ++ic) {
if (ic == func->second.Indent)
continue;
if (ic == func->second.NullArg)
continue;
if (ic == func->second.MsgVal ||
ic == func->second.MsgArg ||
ic == MSG(func->second) )
{
break;
}
Msg->GooActualValues.push_back(string(Str[ic].beg, Str[ic].end));
}
CurrentFile->Msgs.push_back( Msg );
TraceGuids.guids[TraceGuids.current].Msgs.push_back( Msg );
++MessageCount;
if (Msg->ArgConsumed() > arglimit) {
ReportError("Too many arguments supplied: %d > %d\n",
Msg->ArgConsumed(), arglimit);
}
}
return ERROR_SUCCESS;
}
void
InitKeywords()
{
#define HANDLER(A,B,C,D) Keywords[B] = Keyword(0,C,ID_ ## A, D)
HANDLER( WppFlags, "WPP_FLAGS", -1, NULL);
HANDLER( TraceRange, "TRACERANGE", 2, NULL);
HANDLER( ProcessFiles, "PROCESSFILES", -1, NULL);
HANDLER( Touch, "TOUCH", -1, NULL);
HANDLER( ScanForMacros,"SCANFORMACROS", -1, NULL);
HANDLER( Exceptions, "EXCEPTIONS", -2, NULL);
HANDLER( TypeMacro, "TYPEMACRO", 2, NULL);
HANDLER( TypevMacro, "TYPEVMACRO", 3, NULL);
HANDLER( Include, "INCLUDE", 1, NULL);
HANDLER( UsePrefix, "USEPREFIX", -2, NULL);
HANDLER( UseSuffix, "USESUFFIX", -2, NULL); // 3
HANDLER( NoPrefix, "NOPREFIX", 1, NULL);
HANDLER( SeparateTraceGuidPerFile, "SEPARATE_TRACE_GUID_PERFILE", 1, &SeparateTraceGuidPerFile);
HANDLER( DefineSimpleTypePtr, "DEFINE_SIMPLE_TYPE_PTR",-6, NULL);
HANDLER( DefineSimpleType, "DEFINE_SIMPLE_TYPE", -6, NULL);
HANDLER( DefineCplxType, "DEFINE_CPLX_TYPE", -7, NULL);
HANDLER( DefineFlavor, "DEFINE_FLAVOR", 4, NULL);
HANDLER( CustomType, "CUSTOM_TYPE", -2, NULL);
}
struct iterless {
typedef MSG_MAP::iterator ty;
bool operator() (const ty& a, const ty&b) const { return a->second.id < b->second.id; }
};
/*
bool
FileExists(
LPSTR fileName
)
{
HANDLE hFile = CreateFile(fileName, 0, 0, 0, OPEN_EXISTING, 0, 0);
if (hFile == INVALID_HANDLE_VALUE) {
return FALSE;
} else {
CloseHandle( hFile );
return TRUE;
}
}
*/
void
MyGetCurrentDirectory(string& str, int level = 0)
{
char buf[4096];
int n;
n = GetCurrentDirectory(sizeof(buf), buf);
if (n == 0 || n > sizeof(buf) ) {
ReportError("GetCurrentDirectory failed: %d\n", GetLastError());
exit(1);
}
{
char *p, *q;
p = buf;
while (level) {
q = strrchr(buf, '\\');
if (q == 0) {
break;
}
*q = '_'; p = q+1;
--level;
}
str.assign(p);
}
}
ContainerAdapter<MSG_MAP, MapTag> MsgMap_tpl("Messages", MsgMap);
ContainerAdapter<GROUP_MAP, MapTag> GroupMap_tpl("Groups", Groups);
ContainerAdapter<FUNC_MAP, MapTag> FuncMap_tpl("Funcs", Funcs);
ContainerAdapter<set<Reorder>, VectorTag> Reorder_tpl("Reorder", ReorderSet);
ContainerAdapter<FILES, VectorTag> Files_tpl("Files", Files);
ContainerAdapter<TYPE_SET, MapTag> TypeSet_tpl("TypeSet", TypeSet);
ContainerAdapter<TYPESIG_MAP, MapTag> TypeSigSet_tpl("TypeSigSet", TypeSigMap);
StringAdapter CurrentDir_tpl("CurrentDir", CurrentDir);
StringAdapter MacroPrefix_tpl("MacroPrefix", MacroPrefix);
//StringAdapter GuidStore_tpl("GuidStore", GuidStore);
StringAdapter TemplateFile_tpl("TemplateFile", CurrentTpl);
IteratorAdapter<FILES::iterator> CurrentFile_tpl(&CurrentFile);
struct NameAlias : FieldHolder {
string _Name, _Alias;
NameAlias(){} // To make STL happy
NameAlias(const string& Name, const string& Alias): _Name(Name), _Alias(Alias){}
bool operator < (const NameAlias& b) const { return _Name.compare(b._Name) < 0; }
BEGIN_FIELD_TABLE(NameAlias, f)
TEXT_FIELD(Name) fprint_str(f, _Name);
TEXT_FIELD(Alias) fprint_str(f, _Alias);
TEXT_FIELD(MacroName) {
string::size_type br = _Name.find('(');
if (br == string::npos) {
fprint_str(f, _Name);
} else {
fprint_str(f, _Name.begin(), _Name.begin() + br);
}
}
END_FIELD_TABLE
};
struct Compiler : FieldHolder {
const Hasher* Checksum;
BEGIN_FIELD_TABLE(Compiler, f)
TEXT_FIELD(Name) fprintf(f, "WPP");
TEXT_FIELD(Version) fprintf(f, "0.01");
TEXT_FIELD(Checksum) if (Checksum) Checksum->print(f);
TEXT_FIELD(Timestamp) fprintf(f, __TIMESTAMP__);
END_FIELD_TABLE
};
struct SystemObj : FieldHolder {
BEGIN_FIELD_TABLE(SystemObject, out)
TEXT_FIELD(Date) {
SYSTEMTIME UtcTime; GetSystemTime(&UtcTime);
fprintf(out, "%02d/%02d/%04d", UtcTime.wMonth, UtcTime.wDay, UtcTime.wYear);
}
TEXT_FIELD(Time) {
SYSTEMTIME UtcTime; GetSystemTime(&UtcTime);
fprintf(out, "%02d:%02d:%02d", UtcTime.wHour, UtcTime.wMinute, UtcTime.wSecond);
}
END_FIELD_TABLE
};
struct FoundTpl: FieldHolder {
virtual BOOL Hidden(std::string filter="") const {
return CurrentFile->IdsFound.find(filter) == CurrentFile->IdsFound.end();
}
BEGIN_FIELD_TABLE(FoundTpl, f)
__f__;
END_FIELD_TABLE
};
FoundTpl Found_tpl;
SystemObj System_tpl;
Compiler Compiler_tpl;
//vector<NameAlias> AutoGenMacros;
//ContainerAdapter< vector<NameAlias>, VectorTag > AutoGenMacros_tpl("AutoGenMacros", AutoGenMacros);
set<NameAlias> MacroDefinitions;
ContainerAdapter< set<NameAlias>, VectorTag > MacroDefintions_tpl("MacroDefinitions", MacroDefinitions);
void AddMacroDefinition(LPCSTR s)
{
LPCSTR end = s + strlen(s);
LPCSTR q = strchr(s, '=');
if (q == NULL) {
MacroDefinitions.insert( NameAlias( s , "" ) );
} else {
MacroDefinitions.insert( NameAlias( string(s,q) , string(q+1,end) ) );
}
}
void expand(IN OUT string& str, const string& id)
{
NameAlias dummy(id, "");
set<NameAlias>::const_iterator it = MacroDefinitions.find(dummy);
if (it != MacroDefinitions.end()) {
str.append(it->_Alias);
}
}
void processGenOption(LPCSTR s)
{
LPCSTR end = s + strlen(s), div = strchr(s,'}'), star = strchr(s,'*');
vector<GenPair>& v = star ? GenMulti : GenSingle;
if (*s != '{') { goto usage; }
if (div == 0) { goto usage; }
if (s+1 == div || div+1 == end) { goto usage; }
if (star) {
if (div+1 != star) {
ReportError(" '*' can appear only as a first charachter of output-filename part\n");
goto usage;
}
++star;
} else {
star = div+1;
}
if (star == end) goto usage;
v.push_back( GenPair(string(s+1, div), string(star, end)) );
return;
usage:
ReportError(" use -gen:{template-fpathname}output-fpathname\n"
" you supplied -gen:%s\n", s);
ExitProcess(1);
}
void
PrepareSearchPaths(LPCSTR s)
{
//
// split /Isdf;sdf;y into a vector of strings
//
while(s)
{
LPCSTR semi = strchr(s, ';');
if (semi) {
SearchDirs.push_back(string(s,semi));
++semi;
} else {
SearchDirs.push_back(string(s));
}
s = semi;
if (SearchDirs.back().size() > 0
&& SearchDirs.back().end()[-1] != '\\')
{
SearchDirs.back().append("\\");
}
}
}
// 2fb37eda-004b-4b64-a1c4-84c53cb55df5
// 0 1 2 3
// 01234567typeName1234567typeName1234567typeName12345
void processCtlOption(string s)
{
// first let's check that the guid is OK
if ( (s.size() == 16 * 2 + 4)
&& (s[8] == '-') && (s[13] == '-')
&& (s[18] == '-') && s[23] == '-')
{
s[8] = s[13] = s[18] = s[23] = ',';
MacroDefinitions.insert( NameAlias(
string("WPP_DEFAULT_CONTROL_GUID"), s) );
}
}
void processCfgItem(const string& s)
{
LPCSTR beg = s.begin(), end = s.end();
ULONG Status;
EZPARSE_CONTEXT ParseContext;
ZeroMemory(&ParseContext, sizeof(ParseContext) );
ParseContext.start = beg;
ParseContext.filename = "cmdline";
ParseContext.scannedLineCount = 1;
ParseContext.lastScanned = beg;
// EzParseCurrentContext = &ParseContext;
Status = ScanForFunctionCallsEx
(beg, end, ParseConfigCallback, 0, &ParseContext, NO_SEMICOLON);
if (Status != ERROR_SUCCESS) {
exit(1);
}
}
vector<string> cmdinit; // config commands specified on the command line.
void ParseConfigFromCmdline()
{
for(int i = 0; i < cmdinit.size(); ++i) {
processCfgItem(cmdinit[i]);
}
}
void DealWithCmdLineOptions(LPCSTR s)
{
Unusual(" -%s", s);
if (lstrcmp(s,"km") == 0) {
AddMacroDefinition("WPP_KERNEL_MODE");
userMode = FALSE;
} else if (lstrcmp(s,"um") == 0) {
AddMacroDefinition("WPP_USER_MODE");
userMode = TRUE;
} else if (strcmp(s, "dll")==0) {
AddMacroDefinition("WPP_DLL");
} else if (strcmp(s, "ignoreduptypes")==0) {
IgnoreDupTypes = TRUE;
} else if (strncmp(s, "gen:", 4)==0) {
processGenOption(s+4);
} else if (strncmp(s, "gen{", 4)==0) { // I was always forgetting to put :
processGenOption(s+3);
} else if (strncmp(s, "ctl:", 4)==0) {
processCtlOption(string(s+4));
} else if (strncmp(s, "scan:", 5)==0) {
ScanForMacros.push_back(string(s+5));
} else if (strncmp(s, "defwpp:", 7)==0) {
WppDefault.assign(s+7);
} else if (strncmp(s, "v", 1)==0) {
if ( isdigit(s[1]) ) {
DbgLevel = s[1] - '0';
} else {
DbgLevel = 1;
}
} else if (strncmp(s, "ini:", 4)==0) {
LocalConfig.assign(s+4);
} else if (strncmp(s, "lookfor:", 8)==0) {
LookFor.insert(string(s+8));
} else if (strncmp(s, "ext:", 4)==0) {
AllowedExtensions.assign(s+4);
} else if (strncmp(s, "preserveext:", 4)==0) {
PreserveExtensions.assign(s+4);
} else if (strncmp(s, "cfgdir:", 7)==0) { // OBSOLETE
PrepareSearchPaths(s+7);
} else if (strncmp(s, "arglimit:", 9)==0) {
arglimit = atoi(s+9);
} else if (strncmp(s, "I", 1)==0) {
PrepareSearchPaths(s+1);
} else if (strcmp(s, "reorder")==0) {
reorder = TRUE;
} else if (strcmp(s, "noreorder")==0) {
reorder = FALSE;
} else if (strcmp(s, "donothing")==0) {
exit(0);
} else if (strcmp(s, "notimechk")==0) {
CheckTimestamp = FALSE;
} else if (strcmp(s, "noshrieks")==0) {
noshrieks = TRUE;
} else if (strcmp(s, "nohashchk")==0) {
CheckHash = FALSE;
} else if (strncmp(s, "func:",5)==0) {
cmdinit.push_back( string(s+5) );
} else if (strcmp(s, "md5")==0) {
md5 = TRUE;
} else if (strcmp(s, "nomd5")==0) {
md5 = FALSE;
} else if (strncmp(s, "omac:", 5)==0) {
OutputMac.assign(s+5);
} else if (strncmp(s, "argbase:", 8)==0) {
ArgBase = atoi(s+8);
} else if (strncmp(s, "odir:", 5)==0) {
OutputDir.assign(s+5);
if (OutputDir.size() > 0 && OutputDir.end()[-1] != '\\') {
OutputDir.append("\\");
}
} else if (strncmp(s, "D", 1)==0) {
AddMacroDefinition(s+1);
} else {
ReportError("Unknown cmdline option: -%s\n", s);
exit(1);
}
}
void parseStringAsCmdLine(
LPCSTR beg, LPCSTR end
)
{
LPCSTR p = beg;
if( end == NULL ) {
end = beg + strlen(beg);
}
// need to skip spaces. " is processed specially
for(;;) {
while (p < end && isspace(*p)) ++p;
if (p == end) return;
LPCSTR q = p; // beginning of the the string
if (*p == '"') {
++q; // skip openning quote
do {
++p; while (p < end && *p != '"') ++p;
if (p == end) {
ReportError("Unterminated string in %s\n", string(beg, end).c_str() );
exit(1);
}
}
while (*p == '\\');
// now p points to '"' which is not prefixed by '\'
} else {
while (p < end && !isspace(*p)) ++p;
}
if (*q != '-') {
ReportError("Option doesn't start with '-' %s in \"%s\"\n",
string(q,p).c_str(), string(beg, end).c_str() );
}
DealWithCmdLineOptions(q+1, p);
if (*p == '"') ++p;
}
}
enum {
FAVOR_LOCAL_DIR = 0x01,
UPDATE_NAME = 0x02,
COMPLAIN_BITTERLY = 0x04,
};
BOOL
FileExists(
string& fname,
string path = "",
ULONG options = 0)
{
string file(path);
file.append(fname);
WIN32_FIND_DATA Dummy;
HANDLE ffh = FindFirstFile(file.c_str(), &Dummy);
if (ffh != INVALID_HANDLE_VALUE) {
FindClose(ffh);
Noise("found %s in %s\n", fname.c_str(), path.c_str());
if (options & UPDATE_NAME) {
fname = file;
}
return TRUE;
}
return FALSE;
}
void
FindFileOnPath(string& Name, ULONG Options = 0)
{
if (Options & FAVOR_LOCAL_DIR) {
if(FileExists(Name)) { goto found; }
}
int i;
for(i = 0; i < SearchDirs.size(); ++i) {
if(FileExists(Name, SearchDirs[i], Options))
{ goto found; }
}
if ( !(Options & FAVOR_LOCAL_DIR) ) {
if(FileExists(Name)) { goto found; }
}
if ( Options & COMPLAIN_BITTERLY ) {
ReportError("File %s not found\n", Name.c_str());
exit(1);
}
found:;
}
void InitGlobals()
{
InitKeywords();
MyGetCurrentDirectory(CurrentDir, 1);
// ArrayPrefix.assign(currentDir);
// ArrayPrefix.append("_wpp_");
PopulateFieldMap();
if (OutputMac.size() > 0) {
DeleteFile( OutputMac.c_str() );
}
ObjectMap["Reorder"] = &Reorder_tpl;
ObjectMap["TraceGuids"] = &TraceGuids;
// ObjectMap["CtrlGuids"] = &CtrlGuids;
ObjectMap["Messages"] = &MsgMap_tpl;
ObjectMap["Groups"] = &GroupMap_tpl;
ObjectMap["Funcs"] = &FuncMap_tpl;
// ObjectMap["AutoMacros"] = &AutoGenMacros_tpl;
ObjectMap["CurrentDir"] = &CurrentDir_tpl;
ObjectMap["MacroPrefix"] = &MacroPrefix_tpl;
ObjectMap["Compiler"] = &Compiler_tpl;
ObjectMap["Files"] = &Files_tpl;
ObjectMap["TypeSet"] = &TypeSet_tpl;
ObjectMap["System"] = &System_tpl;
ObjectMap["MacroDefinitions"] = &MacroDefintions_tpl;
ObjectMap["TemplateFile"] = &TemplateFile_tpl;
ObjectMap["SourceFile"] = &CurrentFile_tpl;
ObjectMap["TypeSigSet"] = &TypeSigSet_tpl;
ObjectMap["FOUND"] = &Found_tpl;
}
VOID ReadCommandLineArgs(int argc, char** argv)
{
Unusual(" tracewpp");
for (int i = 1; i < argc; ++i) {
char* s = argv[i];
int len = (int)strlen(s);
if (len > 1 && s[0] == '-') {
DealWithCmdLineOptions(s+1);
} else {
Unusual(" %s", argv[i]);
Fill( string(argv[i]));
}
}
Unusual("\n");
{
LPCSTR p = getenv("WPP_FLAGS");
if (p) {
Noise("WPP_FLAGS defined: %s\n", p);
parseStringAsCmdLine(p, NULL);
}
}
/*
// THINK do we need revision at all?
if (Revision.size() > 0) {
if(SearchDirs.size() == 0) {
ReportError("Revision can be specified only when -cfgdir directive is specified");
ExitProcess(3);
} else {
if (Revision.end()[-1] != '\\') {
Revision.append("\\");
}
int i, n = SearchDirs.size();
for (i = 0; i < n; ++i) {
SearchDirs.push_back( SearchDirs[i] );
SearchDirs.back().append(Revision);
}
}
}
*/
}
void ReadConfig()
{
DWORD status;
BOOL WppParsed = FALSE;
// If default config file was specified,
// process it
if (WppDefault.size() > 0) {
WppParsed = TRUE;
Noise("parsing config file %s\n", WppDefault.c_str());
status = EzParse(WppDefault.c_str(), ParseConfigCallback,0);
if (status != ERROR_SUCCESS) {
ExitProcess( status );
}
}
if (LocalConfig.size() == 0) {
if (FileExists(LOCAL_CONFIG_NAME) ) {
WppParsed = TRUE;
Noise("parsing config file %s\n", LOCAL_CONFIG_NAME.c_str());
status = EzParse(LOCAL_CONFIG_NAME.c_str(),ParseConfigCallback,0);
if (status != ERROR_SUCCESS) {
ExitProcess( status );
}
}
}
else
{
WppParsed = TRUE;
Noise("parsing config file %s\n", LocalConfig.c_str());
status = EzParse(LocalConfig.c_str(),ParseConfigCallback,0);
if (status != ERROR_SUCCESS) {
ExitProcess( status );
}
}
if (!WppParsed) {
ReportError("configuration file not found\n");
ExitProcess( ERROR_FILE_NOT_FOUND );
}
if (Files.begin() == Files.end()) {
Unusual("Nothing to compile\n");
ExitProcess( ERROR_FILE_NOT_FOUND );
}
{
// Scan files for macro //
vector<string>::iterator i;
for (i = ScanForMacros.begin(); i != ScanForMacros.end(); ++i) {
Noise("scanning %s... \n", i->c_str());
EzParseEx(i->c_str(), SmartScan, ParseConfigCallback,0, 0);
}
}
ParseConfigFromCmdline();
//
// We need to add prefix and suffix information to FuncMap
//
{
PREFIX_VECTOR::iterator i;
Prefix * defaultPrefix = 0;
Prefix * defaultSuffix = 0;
string star("*");
for (i = Prefixes.begin(); i != Prefixes.end(); ++i) {
if ( 0 == i->FuncName.compare(star) ) {
defaultPrefix = i;
} else {
Funcs[i->FuncName].SetPrefix(i);
}
}
for (i = Suffixes.begin(); i != Suffixes.end(); ++i) {
if ( 0 == i->FuncName.compare(star) ) {
defaultSuffix = i;
} else {
Funcs[i->FuncName].SetSuffix(i);
}
}
//
// Assign the appendicies to the all other functions
//
FUNC_MAP::iterator j = Funcs.begin();
for(; j != Funcs.end(); ++j) {
if (!j->second.prefix) {
j->second.SetPrefix( defaultPrefix );
}
if (!j->second.suffix) {
j->second.SetSuffix( defaultSuffix );
}
}
}
//
// Generates names of the log macros based on type names
// Thus users of the tool can have LOGULONG or log_ulong
// or w_ulong(). Whatever they like best
//
// RegenerateMacroMap();
}
// used by tpl.cpp to do template include [BUGBUG] currently broken
void ProcessTemplate(LPCSTR b, LPCSTR e, void* Context)
{
string prev(CurrentTpl);
CurrentTpl.assign(b,e);
FindFileOnPath(CurrentTpl, COMPLAIN_BITTERLY | UPDATE_NAME | FAVOR_LOCAL_DIR);
EzParseEx(CurrentTpl.c_str(), processTemplate,0, Context, 0);
CurrentTpl.assign(prev);
}
void FormOutputFileName(
IN const string& to,
IN const string& suffix,
OUT string& OutputFile,
OUT BOOL &StdOut
)
{
string::size_type backSlash = to.find('\\');
string::size_type colon = to.find(':');
if (to.compare("-") == 0) {
StdOut = TRUE;
OutputFile.assign("-");
} else {
StdOut = FALSE;
if (backSlash == string::npos && colon == string::npos) {
// can prepend odir
OutputFile.assign(OutputDir);
OutputFile.append(to);
} else {
OutputFile.assign(to);
}
OutputFile.append(suffix);
}
}
void GenerateOutput(string tpl, string to, string suffix = "")
{
string OutputFile;
FILE * f = 0;
BOOL useStdOut = FALSE;
FormOutputFileName(to, suffix, OutputFile, useStdOut);
DefaultHasher computedHash;
computedHash.Init();
MsgMapHash(MsgMap, computedHash);
computedHash.HashStr( CurrentFile->FullFileName() );
computedHash.Finalize();
Compiler_tpl.Checksum = &computedHash;
if (md5) {
size_t len = min(sizeof(GUID), computedHash.Size() );
Flood("Going MD5... %d guid(s) ", TraceGuids.guids.size());
for (int i = 0; i < TraceGuids.guids.size(); ++i) {
// The following two lines are just needed in case
// we ever decide DefaultHasher to be CRC32
// If it will always be MD5 hasher, we can remove them
ZeroMemory(&TraceGuids.guids[i].guid, sizeof(TraceGuids.guids[i].guid));
TraceGuids.guids[i].guid.Data2 = (USHORT)i;
CopyMemory(&TraceGuids.guids[i].guid, computedHash.Buf(), len);
if (DbgLevel >= DBG_FLOOD) {
TraceGuids.guids[i].printTxt(stdout); putchar(' ');
}
}
Flood("\n");
}
// check whether we need to regenerate the file
if (ErrorCount == 0 && CheckHash) {
// Scan the beginning of the
FILE *inc = fopen(OutputFile.c_str(), "r");
if (inc) {
DefaultHasher readHash;
char buf[128], *p = buf;
int n = (int)fread(buf, 1, sizeof(buf)-1, inc);
fclose(inc);
buf[n] = 0;
while (!isalnum(*p) && *p ) ++p;
readHash.fromString(p, buf + n);
if (DbgLevel >= DBG_FLOOD) {
computedHash.print(stdout); putchar(' ');
readHash.print(stdout); putchar(' ');
}
if (computedHash == readHash) {
Noise("No changes in %s.\n",
CurrentFile->_Name.c_str() );
return;
}
}
}
if (ErrorCount) {
DeleteFile(OutputFile.c_str());
Noise("errors detected. deleting %s...\n", OutputFile.c_str());
return;
}
Noise("generating %s...\n", OutputFile.c_str());
if (useStdOut) {
f = stdout;
} else {
f = fopen(OutputFile.c_str(), "w");
if (!f) {
ReportError("Cannot open '%s' for writing, error %d\n", OutputFile.c_str(), GetLastError() );
ExitProcess( GetLastError() );
}
}
ProcessTemplate(tpl.begin(), tpl.end(), f);
if (!useStdOut) {
fclose(f);
}
}
void
VerifyAndUpdateTemplatesNames(
vector<GenPair>& Gen
)
{
// verify that the template
vector<GenPair>::iterator i;
for(i = Gen.begin();i != Gen.end();++i){
FindFileOnPath(i->tpl, COMPLAIN_BITTERLY | UPDATE_NAME | FAVOR_LOCAL_DIR);
}
}
int RealMain(int argc, char** argv)
{
InitGlobals();
ReadCommandLineArgs(argc, argv);
if (WppDefault.size()) {
FindFileOnPath(WppDefault, COMPLAIN_BITTERLY | UPDATE_NAME);
}
ReadConfig();
if (GenSingle.size() == 0 && GenMulti.size() == 0)
{
// add default template
processGenOption(DEFAULT_GEN_OPTION);
}
BOOL OneAtATime = GenMulti.size() > 0;
VerifyAndUpdateTemplatesNames(GenMulti);
VerifyAndUpdateTemplatesNames(GenSingle);
{
FILES::iterator file;
if (CheckTimestamp)
{
//
// Before we do any heavyweight processing
// let's check timestamps of source
// and output files
//
ULONGLONG Now, MaxSrcTime, MinOutTime;
string OutputFile;
GetSystemTimeAsFileTime((LPFILETIME)&Now);
MaxSrcTime = 0;
MinOutTime = Now;
for (file = Files.begin(); file != Files.end(); ++file) {
if (file->ModificationTime == 0) {
Always("%s file has invalid modification time\n",
file->_BaseName.c_str());
goto out_of_date;
}
if (file->ModificationTime > Now) {
Always("%s file modification time is in the future\n",
file->_BaseName.c_str());
goto out_of_date;
}
if (file->ModificationTime > MaxSrcTime) {
MaxSrcTime = file->ModificationTime;
}
vector<GenPair>::iterator i;
for(i = GenMulti.begin();i != GenMulti.end();++i){
BOOL StdOut;
ULONGLONG Time;
FormOutputFileName(file->_BaseName, i->out.c_str(),
OutputFile, StdOut);
if (StdOut) {
// StdOut is always out of date //
goto out_of_date;
}
Time = GetFileModificationTime(OutputFile);
if (Time == 0) {
goto out_of_date;
}
if (Time > Now) {
Always("%s file modification time is in the future\n",
OutputFile.c_str());
goto out_of_date;
}
if (Time < MinOutTime) {
MinOutTime = Time;
}
}
}
vector<GenPair>::iterator i;
for(i = GenSingle.begin();i != GenSingle.end();++i){
BOOL StdOut;
ULONGLONG Time;
FormOutputFileName(file->_BaseName, i->out.c_str(),
OutputFile, StdOut);
if (StdOut) {
// StdOut is always out of date //
goto out_of_date;
}
Time = GetFileModificationTime(OutputFile);
if (Time == 0) {
goto out_of_date;
}
if (Time > Now) {
Always("%s file modification time is in the future\n",
OutputFile.c_str());
goto out_of_date;
}
if (Time < MinOutTime) {
MinOutTime = Time;
}
}
if (MaxSrcTime <= MinOutTime) {
return 0;
}
} // if (CheckTimestamp) //
out_of_date:
//
// Now we are ready to start scanning the source files
//
for (file = Files.begin(); file != Files.end(); ++file) {
Noise("processing %s... \n", file->FullFileName().c_str());
CurrentFile = file;
if (SeparateTraceGuidPerFile) {
TraceGuids.new_file( file->_Name.c_str() );
}
EzParseWithOptions(file->FullFileName().c_str(), ParseSrcCallback, 0, IGNORE_CPP_COMMENT);
{
vector<GenPair>::iterator i;
for(i = GenMulti.begin();i != GenMulti.end();++i){
GenerateOutput(i->tpl, file->_BaseName, i->out.c_str());
}
}
if (OneAtATime) {
// Clear the tables for the next iteration
MsgMap.erase(MsgMap.begin(), MsgMap.end());
TypeSigMap.erase(TypeSigMap.begin(), TypeSigMap.end());
TraceGuids.erase();
for (TYPE_SET::const_iterator i = TypeSet.begin();
i != TypeSet.end(); ++i)
{
i->second.Used = FALSE;
}
}
}
Noise("done. Errors: %d.\n", ErrorCount);
}
#if 0
// update Used field for every type in TypeSigSet
{
for (TYPESIG_MAP::const_iterator i = TypeSigMap.begin();
i != TypeSigMap.end(); ++i)
{
for(int j = 0; j < i->Types.size(); ++j) {
i->second.Types[j]->Used = TRUE;
}
}
}
#endif
// generate global files
{
vector<GenPair>::iterator i;
for(i = GenSingle.begin();i != GenSingle.end();++i){
GenerateOutput(i->tpl, i->out.c_str());
}
}
return ErrorCount;
}
int __cdecl main(int argc, char** argv)
{
int status = 0;
__int64 a, b;
GetSystemTimeAsFileTime((LPFILETIME)&a);
__try
{
status = RealMain(argc, argv);
}
__except(EXCEPTION_EXECUTE_HANDLER)
{
status = 1;
ReportError("Internal Error : Contact GorN\n");
}
{
SYSTEMTIME st;
GetSystemTimeAsFileTime((LPFILETIME)&b);
b -= a;
FileTimeToSystemTime((LPFILETIME)&b, &st);
Always("tracewpp: %d file(s) with %d message(s), processed in %d.%03d seconds\n",
Files.size(), MessageCount,
st.wSecond, st.wMilliseconds);
}
return status;
}