1
0
Fork 0
forked from fte/fteqw
fteqw/engine/qclib/qccmain.c

6178 lines
199 KiB
C

#if !defined(MINIMAL) && !defined(OMIT_QCC)
#define PROGSUSED
#include "qcc.h"
#include <sys/stat.h>
#ifdef _WIN32
#include <direct.h>
#endif
#include <time.h>
#include "errno.h"
#define countof(array) (sizeof(array)/sizeof(array[0]))
//#define TODO_READWRITETRACK
//#define DEBUG_DUMP
//#define DISASM "M_Preset"
extern QCC_def_t tempsdef;
char QCC_copyright[1024];
int QCC_packid;
char QCC_Packname[5][128];
extern int optres_test1;
extern int optres_test2;
pbool writeasm;
int verbose;
#define VERBOSE_WARNINGSONLY -1
#define VERBOSE_PROGRESS 0
#define VERBOSE_STANDARD 1
#define VERBOSE_DEBUG 2
#define VERBOSE_DEBUGSTATEMENTS 3 //figuring out the files can be expensive.
pbool qcc_nopragmaoptimise;
pbool opt_stripunusedfields;
extern unsigned int locals_marshalled;
extern int qccpersisthunk;
pbool QCC_PR_SimpleGetToken (void);
void QCC_PR_LexWhitespace (pbool inhibitpreprocessor);
static char *QCC_PR_String (char *string);
void *FS_ReadToMem(char *fname, size_t *len);
void FS_CloseFromMem(void *mem);
unsigned int MAX_REGS;
unsigned int MAX_LOCALS = 0x10000;
unsigned int MAX_TEMPS = 0x10000;
int MAX_STRINGS;
int MAX_GLOBALS;
int MAX_FIELDS;
int MAX_STATEMENTS;
int MAX_FUNCTIONS;
int MAX_CONSTANTS;
int max_temps;
int *qcc_tempofs;
int tempsstart;
#define MAXSOURCEFILESLIST 8
char sourcefileslist[MAXSOURCEFILESLIST][1024];
QCC_def_t *sourcefilesdefs[MAXSOURCEFILESLIST]; //for the gui to peek at.
int sourcefilesnumdefs; //maximum used...
int currentsourcefile; //currently compiling file.
int numsourcefiles; //count pending.
extern char *compilingfile; //file currently being compiled
char compilingrootfile[1024]; //the .src file we started from (the current one, not original)
char qccmsourcedir[1024]; //the -src path, for #includes
void QCC_PR_ResetErrorScope(void);
static void StartNewStyleCompile(void);
pbool compressoutput;
pbool newstylesource;
char destfile[1024]; //the file we're going to output to
pbool destfile_explicit; //destfile was override on the commandline, don't let qc change it.
QCC_eval_basic_t *qcc_pr_globals;
unsigned int numpr_globals;
char *strings;
int strofs;
QCC_statement_t *statements;
int numstatements;
QCC_function_t *functions;
//dfunction_t *dfunctions;
int numfunctions;
QCC_ddef_t *qcc_globals;
int numglobaldefs;
QCC_ddef_t *fields;
int numfielddefs;
//typedef char PATHSTRING[MAX_DATA_PATH];
precache_t *precache_sound;
int numsounds;
precache_t *precache_texture;
int numtextures;
precache_t *precache_model;
int nummodels;
precache_t *precache_file;
int numfiles;
extern int numCompilerConstants;
hashtable_t compconstantstable;
hashtable_t globalstable;
hashtable_t localstable;
hashtable_t typedeftable;
#ifdef WRITEASM
FILE *asmfile;
pbool asmfilebegun;
#endif
hashtable_t floatconstdefstable;
hashtable_t stringconstdefstable;
hashtable_t stringconstdefstable_trans;
extern int dotranslate_count;
unsigned char qccwarningaction[WARN_MAX]; //0 = disabled, 1 = warn, 2 = error.
unsigned int qcc_targetversion;
qcc_targetformat_t qcc_targetformat;
pbool bodylessfuncs;
QCC_type_t *qcc_typeinfo;
int numtypeinfos;
int maxtypeinfos;
pbool preprocessonly;
static pbool flag_dumpfilenames;
static pbool flag_dumpfields;
static pbool flag_dumpsymbols;
static pbool flag_dumpautocvars;
static pbool flag_dumplocalisation;
static pbool flag_dumptags;
struct {
char *name;
int index;
} warningnames[] =
{
// {"", WARN_NOTREFERENCEDCONST},
// {"", WARN_CONFLICTINGRETURNS},
{" Q100", WARN_PRECOMPILERMESSAGE},
{" Q101", WARN_TOOMANYPARAMS},
//102: Indirect function: too many parameters
//103: vararg func cannot have more than
//104: type mismatch on parm %i
{" Q105", WARN_TOOFEWPARAMS},
// {"", WARN_UNEXPECTEDPUNCT},
{" Q106", WARN_ASSIGNMENTTOCONSTANT},
//107: Array index should be type int
//108: Mixed float and int types
//109: Expecting int, float a parameter found
//110: Expecting int, float b parameter found
//112: Null 'if' statement
//113: Null 'else' statement
//114: Type mismatch on redeclaration
//115: redeclared with different number of parms
//116: Local %s redeclared
//117: too many initializers
//118: Too many closing braces
//119: Too many #endifs
{" Q120", WARN_BADPRAGMA},
//121: unknown directive
//122: strofs exceeds limit
//123: numstatements exceeds limit
//124: numfunctions exceeds limit
//125: numglobaldefs exceeds limit
//126: numfielddefs exceeds limit
//127: numpr_globals exceeds limit
//128: rededeclared with different parms
{" F129", WARN_COMPATIBILITYHACK}, //multiple errors are replaced by this, for compat purposes.
{" Q203", WARN_MISSINGRETURNVALUE},
{" Q204", WARN_WRONGRETURNTYPE},
{" Q205", WARN_POINTLESSSTATEMENT},
{" Q206", WARN_MISSINGRETURN},
{" Q207", WARN_DUPLICATEDEFINITION}, //redeclared different scope
{" Q208", WARN_SYSTEMCRC},
// {"", WARN_STRINGTOOLONG},
// {"", WARN_BADTARGET},
//301: %s defined as local in %s
{" Q302", WARN_NOTREFERENCED}, //302: Unreferenced local variable %s from line %i
//401: In function %s parameter %s is unused
// {"", WARN_HANGINGSLASHR},
// {"", WARN_NOTDEFINED},
// {"", WARN_SWITCHTYPEMISMATCH},
// {"", WARN_CONFLICTINGUNIONMEMBER},
// {"", WARN_KEYWORDDISABLED},
// {"", WARN_ENUMFLAGS_NOTINTEGER},
// {"", WARN_ENUMFLAGS_NOTBINARY},
{" Q111", WARN_DUPLICATELABEL},
{" Q201", WARN_ASSIGNMENTINCONDITIONAL},
{" F300", WARN_DEADCODE},
{" F301", WARN_NOTUTF8},
{" F302", WARN_UNINITIALIZED},
{" F303", WARN_EVILPREPROCESSOR},
{" F304", WARN_UNARYNOTSCOPE},
{" F305", WARN_CASEINSENSITIVEFRAMEMACRO},
{" F306", WARN_SAMENAMEASGLOBAL},
{" F307", WARN_STRICTTYPEMISMATCH},
{" F308", WARN_TYPEMISMATCHREDECOPTIONAL},
{" F309", WARN_IGNORECOMMANDLINE},
{" F310", WARN_MISUSEDAUTOCVAR},
{" F311", WARN_FTE_SPECIFIC},
{" F312", WARN_OVERFLOW},
{" F313", WARN_DENORMAL},
{" F314", WARN_LAXCAST},
{" F315", WARN_DUPLICATEPRECOMPILER},
{" F316", WARN_IDENTICALPRECOMPILER},
{" F317", WARN_STALEMACRO},
{" F318", WARN_DUPLICATEMACRO},
{" F319", WARN_CONSTANTCOMPARISON},
{" F320", WARN_PARAMWITHNONAME},
{" F321", WARN_GMQCC_SPECIFIC},
{" F322", WARN_IFSTRING_USED},
{" F323", WARN_UNREACHABLECODE},
{" F324", WARN_FORMATSTRING},
{" F325", WARN_NESTEDCOMMENT},
{" F326", WARN_DEPRECATEDVARIABLE},
{" F327", WARN_ENUMFLAGS_NOTINTEGER},
{" F328", WARN_DEPRECACTEDSYNTAX},
{" F329", WARN_REDECLARATIONMISMATCH},
{" F330", WARN_MUTEDEPRECATEDVARIABLE},
{" F331", WARN_SELFNOTTHIS},
{" F332", WARN_DIVISIONBY0},
{" F333", WARN_ARGUMENTCHECK},
{" F334", WARN_MISSINGMEMBERQUALIFIER},
{" F335", WARN_MEMBERNOTDEFINED}, //defining a new member function inside a class that didn't list it.
{" F207", WARN_NOTREFERENCEDFIELD},
{" F208", WARN_NOTREFERENCEDCONST},
{" F209", WARN_EXTRAPRECACHE},
{" F210", WARN_NOTPRECACHED},
{" F211", WARN_SYSTEMCRC2},
//frikqcc errors
//Q608: PrecacheSound: numsounds
//Q609: PrecacheModels: nummodels
//Q610: PrecacheFile: numfiles
//Q611: Bad parm order
//Q612: PR_CompileFile: Didn't clear (internal error)
//Q614: PR_DefForFieldOfs: couldn't find
//Q613: Error writing error.log
//Q615: Error writing
//Q616: No function named
//Q617: Malloc failure
//Q618: Ran out of mem pointer space (malloc failure again)
//we can put longer alternative names here...
{" field-redeclared", WARN_REMOVEDWARNING},
{" deprecated", WARN_DEPRECATEDVARIABLE},
{NULL}
};
char *QCC_NameForWarning(int idx)
{
int i;
for (i = 0; warningnames[i].name; i++)
{
if (warningnames[i].index == idx)
return warningnames[i].name;
}
return NULL;
}
int QCC_WarningForName(const char *name)
{
int i;
for (i = 0; warningnames[i].name; i++)
{
if (!stricmp(name, warningnames[i].name+1))
return warningnames[i].index;
}
return -1;
}
optimisations_t optimisations[] =
{
//level 0 = no optimisations
//level 1 = size optimisations
//level 2 = speed optimisations
//level 3 = unsafe optimisations (they break multiprogs).
//level 4 = experimental or extreeme features... must be used explicitly
{&opt_assignments, "t", 1, FLAG_ASDEFAULT, "assignments", "c = a*b is performed in one operation rather than two, and can cause older decompilers to fail."},
{&opt_shortenifnots, "i", 1, FLAG_ASDEFAULT, "shortenifs", "if (!a) was traditionally compiled in two statements. This optimisation does it in one, but can cause some decompilers to get confused."},
{&opt_nonvec_parms, "p", 1, FLAG_ASDEFAULT, "nonvec_parms", "In the original qcc, function parameters were specified as a vector store even for floats. This fixes that."},
{&opt_constant_names, "c", 2, FLAG_KILLSDEBUGGERS, "constant_names", "This optimisation strips out the names of constants (but not strings) from your progs, resulting in smaller files. It makes decompilers leave out names or fabricate numerical ones."},
{&opt_constant_names_strings, "cs", 3, FLAG_KILLSDEBUGGERS, "constant_names_strings", "This optimisation strips out the names of string constants from your progs. However, this can break addons, so don't use it in those cases."},
{&opt_dupconstdefs, "d", 1, FLAG_ASDEFAULT, "dupconstdefs", "This will merge definitions of constants which are the same value. Pay extra attention to assignment to constant warnings."},
{&opt_noduplicatestrings, "s", 1, FLAG_ASDEFAULT, "noduplicatestrings", "This will compact the string table that is stored in the progs. It will be considerably smaller with this."},
{&opt_locals, "l", 1, FLAG_KILLSDEBUGGERS, "locals", "Strips out local names and definitions. Most decompiles will break on this."},
{&opt_function_names, "n", 1, FLAG_KILLSDEBUGGERS, "function_names", "This strips out the names of functions which are never called. Doesn't make much of an impact though."},
{&opt_filenames, "f", 1, FLAG_KILLSDEBUGGERS, "filenames", "This strips out the filenames of the progs. This can confuse the really old decompilers, but is nothing to the more recent ones."},
// {&opt_unreferenced, "u", 1, FLAG_ASDEFAULT, "unreferenced", "Removes the entries of unreferenced variables. Doesn't make a difference in well maintained code."},
{&opt_overlaptemps, "r", 1, FLAG_ASDEFAULT, "overlaptemps", "Optimises the pr_globals count by overlapping temporaries. In QC, every multiplication, division or operation in general produces a temporary variable. This optimisation prevents excess, and in the case of Hexen2's gamecode, reduces the count by 50k. This is the most important optimisation, ever."},
{&opt_constantarithmatic, "a", 1, FLAG_ASDEFAULT, "constantarithmatic", "5*6 actually emits an operation into the progs. This prevents that happening, effectivly making the compiler see 30"},
{&opt_precache_file, "pf", 2, 0, "precache_file", "Strip out stuff wasted used in function calls and strings to the precache_file builtin (which is actually a stub in quake)."},
{&opt_return_only, "ro", 2, FLAG_KILLSDEBUGGERS, "return_only", "Functions ending in a return statement do not need a done statement at the end of the function. This can confuse some decompilers, making functions appear larger than they were."},
{&opt_compound_jumps, "cj", 2, FLAG_KILLSDEBUGGERS, "compound_jumps", "This optimisation plays an effect mostly with nested if/else statements, instead of jumping to an unconditional jump statement, it'll jump to the final destination instead. This will bewilder decompilers."},
// {&opt_comexprremoval, "cer", 4, 0, "expression_removal", "Eliminate common sub-expressions"}, //this would be too hard...
{&opt_stripfunctions, "sf", 3, FLAG_KILLSDEBUGGERS, "strip_functions", "Strips out the 'defs' of functions that were only ever called directly. This does not affect saved games, but will prevent FTE_MULTIPROGS/mutators from being able to hook functions."},
{&opt_locals_overlapping, "lo", 2, FLAG_KILLSDEBUGGERS, "locals_overlapping", "Store all locals in a single section of the pr_globals. Vastly reducing it. This effectivly does the job of overlaptemps.\nHowever, locals are no longer automatically initialised to 0 (and never were in the case of recursion, but at least then its the same type).\nIf locals appear uninitialised, fteqcc will disable this optimisation for the affected functions, you can optionally get a warning about these locals using: #pragma warning enable F302"},
{&opt_vectorcalls, "vc", 4, FLAG_KILLSDEBUGGERS, "vectorcalls", "Where a function is called with just a vector, this causes the function call to store three floats instead of one vector. This can save a good number of pr_globals where those vectors contain many duplicate coordinates but do not match entirly."},
{&opt_classfields, "cf", 2, FLAG_KILLSDEBUGGERS, "class_fields", "Strip class field names. This will harm debugging and can result in 'gibberish' names appearing in saved games. Has no effect on engines other than FTEQW, which will not recognise these anyway."},
{&opt_stripunusedfields, "uf", 4, FLAG_KILLSDEBUGGERS, "strip_unused_fields","Strips any fields that have no references. This may result in extra warnings at load time, or disabling support for mapper-specified alpha in engines that do not provide that. FIXME: this is a little pointless until relocs are properly implemented."},
{NULL}
};
#define defaultkeyword FLAG_HIDDENINGUI|FLAG_ASDEFAULT|FLAG_MIDCOMPILE
#define typekeyword FLAG_HIDDENINGUI|FLAG_ASDEFAULT
#define nondefaultkeyword FLAG_HIDDENINGUI|0|FLAG_MIDCOMPILE
#define hideflag FLAG_HIDDENINGUI|FLAG_MIDCOMPILE
#define defaultflag FLAG_ASDEFAULT|FLAG_MIDCOMPILE
#define hidedefaultflag FLAG_HIDDENINGUI|FLAG_ASDEFAULT|FLAG_MIDCOMPILE
//global to store useage to, flags, codename, human-readable name, help text
compiler_flag_t compiler_flag[] = {
//keywords
{&keyword_asm, defaultkeyword, "asm", "Keyword: asm", "Disables the 'asm' keyword. Use the writeasm flag to see an example of the asm."},
{&keyword_break, defaultkeyword, "break", "Keyword: break", "Disables the 'break' keyword."},
{&keyword_case, defaultkeyword, "case", "Keyword: case", "Disables the 'case' keyword."},
{&keyword_class, defaultkeyword, "class", "Keyword: class", "Disables the 'class' keyword."},
{&keyword_accessor, defaultkeyword, "accessor", "Keyword: accessor", "Disables the 'accessor' keyword."},
{&keyword_const, defaultkeyword, "const", "Keyword: const", "Disables the 'const' keyword."},
{&keyword_continue, defaultkeyword, "continue", "Keyword: continue", "Disables the 'continue' keyword."},
{&keyword_default, defaultkeyword, "default", "Keyword: default", "Disables the 'default' keyword."},
{&keyword_entity, defaultkeyword, "entity", "Keyword: entity", "Disables the 'entity' keyword."},
{&keyword_enum, defaultkeyword, "enum", "Keyword: enum", "Disables the 'enum' keyword."}, //kinda like in c, but typedef not supported.
{&keyword_enumflags, defaultkeyword, "enumflags", "Keyword: enumflags", "Disables the 'enumflags' keyword."}, //like enum, but doubles instead of adds 1.
{&keyword_extern, defaultkeyword, "extern", "Keyword: extern", "Disables the 'extern' keyword. Use only on functions inside addons."}, //function is external, don't error or warn if the body was not found
{&keyword_float, defaultkeyword, "float", "Keyword: float", "Disables the 'float' keyword. (Disables the float keyword without 'local' preceeding it)"},
{&keyword_for, defaultkeyword, "for", "Keyword: for", "Disables the 'for' keyword. Syntax: for(assignment; while; increment) {codeblock;}"},
{&keyword_goto, defaultkeyword, "goto", "Keyword: goto", "Disables the 'goto' keyword."},
{&keyword_int, typekeyword, "int", "Keyword: int", "Disables the 'int' keyword."},
{&keyword_integer, typekeyword, "integer", "Keyword: integer", "Disables the 'integer' keyword."},
{&keyword_double, nondefaultkeyword, "double", "Keyword: double", "Disables the 'double' keyword."},
{&keyword_long, nondefaultkeyword, "long", "Keyword: long", "Disables the 'long' keyword."},
{&keyword_short, nondefaultkeyword, "short", "Keyword: short", "Disables the 'short' keyword."},
{&keyword_char, nondefaultkeyword, "char", "Keyword: char", "Disables the 'char' keyword."},
{&keyword_signed, nondefaultkeyword, "signed", "Keyword: signed", "Disables the 'signed' keyword."},
{&keyword_unsigned, defaultkeyword, "unsigned", "Keyword: unsigned", "Disables the 'unsigned' keyword."},
{&keyword_noref, defaultkeyword, "noref", "Keyword: noref", "Disables the 'noref' keyword."}, //nowhere else references this, don't warn about it.
{&keyword_unused, nondefaultkeyword, "unused", "Keyword: unused", "Disables the 'unused' keyword. 'unused' means that the variable is unused, you're aware that its unused, and you'd rather not know about all the warnings this results in."},
{&keyword_used, nondefaultkeyword, "used", "Keyword: used", "Disables the 'used' keyword. 'used' means that the variable is used even if the qcc can't see how - thus preventing it from ever being stripped."},
{&keyword_static, defaultkeyword, "static", "Keyword: static", "Disables the 'static' keyword. 'static' means that a variable has altered scope. On globals, the variable is visible only to the current .qc file. On locals, the variable's value does not change between calls to the function. On class variables, specifies that the field is a scoped global instead of a local. On class functions, specifies that 'this' is expected to be invalid and that the function will access any memembers via it."},
{&keyword_nonstatic, defaultkeyword, "nonstatic", "Keyword: nonstatic", "Disables the 'nonstatic' keyword. 'nonstatic' acts upon globals+functions, reverting the defaultstatic pragma on a per-variable basis. For use by people who prefer to keep their APIs explicit."},
{&keyword_ignore, nondefaultkeyword, "ignore", "Keyword: ignore", "Disables the 'ignore' keyword. 'ignore' is expected to typically be hidden behind a 'csqconly' define, and in such a context can be used to conditionally compile functions a little more gracefully. The opposite of the 'used' keyword. These variables/functions/members are ALWAYS stripped, and effectively ignored."},
{&keyword_nosave, defaultkeyword, "nosave", "Keyword: nosave", "Disables the 'nosave' keyword."}, //don't write the def to the output.
{&keyword_inline, defaultkeyword, "inline", "Keyword: inline", "Disables the 'inline' keyword."}, //don't write the def to the output.
{&keyword_strip, defaultkeyword, "strip", "Keyword: strip", "Disables the 'strip' keyword."}, //don't write the def to the output.
{&keyword_shared, defaultkeyword, "shared", "Keyword: shared", "Disables the 'shared' keyword."}, //mark global to be copied over when progs changes (part of FTE_MULTIPROGS)
{&keyword_state, nondefaultkeyword,"state", "Keyword: state", "Disables the 'state' keyword."},
{&keyword_optional, defaultkeyword,"optional", "Keyword: optional", "Disables the 'optional' keyword."},
{&keyword_inout, nondefaultkeyword,"inout", "Keyword: inout", "Disables the 'inout' keyword."},
{&keyword_string, defaultkeyword, "string", "Keyword: string", "Disables the 'string' keyword."},
{&keyword_struct, defaultkeyword, "struct", "Keyword: struct", "Disables the 'struct' keyword."},
{&keyword_switch, defaultkeyword, "switch", "Keyword: switch", "Disables the 'switch' keyword."},
{&keyword_thinktime, nondefaultkeyword,"thinktime", "Keyword: thinktime", "Disables the 'thinktime' keyword which is used in HexenC"},
{&keyword_until, nondefaultkeyword,"until", "Keyword: until", "Disables the 'until' keyword which is used in HexenC"},
{&keyword_loop, nondefaultkeyword,"loop", "Keyword: loop", "Disables the 'loop' keyword which is used in HexenC"},
{&keyword_typedef, defaultkeyword, "typedef", "Keyword: typedef", "Disables the 'typedef' keyword."}, //fixme
{&keyword_union, defaultkeyword, "union", "Keyword: union", "Disables the 'union' keyword."}, //you surly know what a union is!
{&keyword_var, defaultkeyword, "var", "Keyword: var", "Disables the 'var' keyword."},
{&keyword_vector, defaultkeyword, "vector", "Keyword: vector", "Disables the 'vector' keyword."},
{&keyword_wrap, defaultkeyword, "wrap", "Keyword: wrap", "Disables the 'wrap' keyword."},
{&keyword_weak, defaultkeyword, "weak", "Keyword: weak", "Disables the 'weak' keyword."},
{&keyword_accumulate, nondefaultkeyword,"accumulate", "Keyword: accumulate", "Disables the 'accumulate' keyword."},
{&keyword_using, nondefaultkeyword,"using", "Keyword: using", "Disables the 'using' keyword."},
//options
{&flag_acc, 0, "acc", "Reacc support", "Reacc is a pascall like compiler. It was released before the Quake source was released. This flag has a few effects. It sorts all qc files in the current directory into alphabetical order to compile them. It also allows Reacc global/field distinctions, as well as allows | for linebreaks. Whilst case insensitivity and lax type checking are supported by reacc, they are seperate compiler flags in fteqcc."}, //reacc like behaviour of src files.
{&flag_qccx, FLAG_MIDCOMPILE,"qccx", "QCCX syntax", "WARNING: This syntax makes mods inherantly engine specific.\nDo NOT use unless you know what you're doing.This is provided for compatibility only\nAny entity hacks will be unsupported in FTEQW, DP, and others, resulting in engine crashes if the code in question is executed."},
{&keywords_coexist, defaultflag, "kce", "Keywords Coexist", "If you want keywords to NOT be disabled when they a variable by the same name is defined, check here."},
// {&flag_lno, defaultflag, "lno", "Write Line Numbers", "Writes line number information. This is required for any real kind of debugging. Will be ignored if filenames were stripped."},
{&output_parms, 0, "parms", "Define offset parms", "if PARM0 PARM1 etc should be defined by the compiler. These are useful if you make use of the asm keyword for function calls, or you wish to create your own variable arguments. This is an easy way to break decompilers."}, //controls weather to define PARMx for the parms (note - this can screw over some decompilers)
{&autoprototype, 0, "autoproto", "Automatic Prototyping","Causes compilation to take two passes instead of one. The first pass, only the definitions are read. The second pass actually compiles your code. This means you never have to remember to prototype functions again."}, //so you no longer need to prototype functions and things in advance.
{&writeasm, 0, "wasm", "Dump Assembler", "Writes out a qc.asm which contains all your functions but in assembler. This is a great way to look for bugs in fteqcc, but can also be used to see exactly what your functions turn into, and thus how to optimise statements better."}, //spit out a qc.asm file, containing an assembler dump of the ENTIRE progs. (Doesn't include initialisation of constants)
{&flag_guiannotate, FLAG_MIDCOMPILE,"annotate", "Annotate Sourcecode", "Annotate source code with assembler statements on compile (requires gui)."},
{&flag_nullemptystr, FLAG_MIDCOMPILE,"nullemptystr", "Null String Immediates", "Empty string immediates will have the raw value 0 instead of 1."},
{&flag_ifstring, FLAG_MIDCOMPILE,"ifstring", "if(string) fix", "Causes if(string) to behave identically to if(string!="") This is most useful with addons of course, but also has adverse effects with FRIK_FILE's fgets, where it becomes impossible to determin the end of the file. In such a case, you can still use asm {IF string 2;RETURN} to detect eof and leave the function."}, //correction for if(string) no-ifstring to get the standard behaviour.
{&flag_iffloat, FLAG_MIDCOMPILE,"iffloat", "if(-0.0) fix","Fixes certain floating point logic."},
{&flag_ifvector, defaultflag, "ifvector", "if('0 1 0') fix","Fixes conditional vector logic."},
{&flag_vectorlogic, defaultflag, "vectorlogic", "v&&v||v fix", "Fixes conditional vector logic."},
{&flag_brokenarrays, FLAG_MIDCOMPILE,"brokenarray", "array[0] omission", "Treat references to arrays as references to the first index of said array, to replicate an old fteqcc bug."},
{&flag_rootconstructor, FLAG_MIDCOMPILE,"rootconstructor","root constructor first", "When enabled, the root constructor should be called first like in c++."},
{&flag_caseinsensitive, 0, "caseinsens", "Case insensitivity", "Causes fteqcc to become case insensitive whilst compiling names. It's generally not advised to use this as it compiles a little more slowly and provides little benefit. However, it is required for full reacc support."}, //symbols will be matched to an insensitive case if the specified case doesn't exist. This should b usable for any mod
{&flag_laxcasts, FLAG_MIDCOMPILE,"lax", "Lax type checks", "Disables many errors (generating warnings instead) when function calls or operations refer to two normally incompatible types. This is required for reacc support, and can also allow certain (evil) mods to compile that were originally written for frikqcc."}, //Allow lax casting. This'll produce loadsa warnings of course. But allows compilation of certain dodgy code.
{&flag_hashonly, FLAG_MIDCOMPILE,"hashonly", "Hash-only constants", "Allows use of only #constant for precompiler constants, allows certain preqcc using mods to compile"},
{&flag_macroinstrings, FLAG_MIDCOMPILE,"macroinstrings","Expand macros in strings", "Allows the use #constant inside string immediates. This is a feature of preqcc, but should otherwise probably be left disabled."},
{&opt_logicops, FLAG_MIDCOMPILE,"lo", "Logic ops", "This changes the behaviour of your code. It generates additional if operations to early-out in if statements. With this flag, the line if (0 && somefunction()) will never call the function. It can thus be considered an optimisation. However, due to the change of behaviour, it is not considered so by fteqcc. Note that due to inprecisions with floats, this flag can cause runaway loop errors within the player walk and run functions (without iffloat also enabled). This code is advised:\nplayer_stand1:\n if (self.velocity_x || self.velocity_y)\nplayer_run\n if (!(self.velocity_x || self.velocity_y))"},
{&flag_msvcstyle, FLAG_MIDCOMPILE,"msvcstyle", "MSVC-style errors", "Generates warning and error messages in a format that msvc understands, to facilitate ide integration."},
{&flag_debugmacros, FLAG_MIDCOMPILE,"debugmacros", "Verbose Macro Expansion", "Print out the contents of macros that are expanded. This can help look inside macros that are expanded and is especially handy if people are using preprocessor hacks."},
{&flag_filetimes, 0, "filetimes", "Check Filetimes", "Recompiles the progs only if the file times are modified."},
{&flag_fasttrackarrays, FLAG_MIDCOMPILE,"fastarrays", "fast arrays where possible", "Generates extra instructions inside array handling functions to detect engine and use extension opcodes only in supporting engines.\nAdds a global which is set by the engine if the engine supports the extra opcodes. Note that this applies to all arrays or none."},
{&flag_assume_integer, FLAG_MIDCOMPILE,"assumeint", "Assume Integers", "Numerical constants are assumed to be integers, instead of floats."},
{&pr_subscopedlocals, FLAG_MIDCOMPILE,"subscope", "Subscoped Locals", "Restrict the scope of locals to the block they are actually defined within, as in C."},
{&verbose, FLAG_MIDCOMPILE,"verbose", "Verbose", "Lots of extra compiler messages."},
{&flag_typeexplicit, FLAG_MIDCOMPILE,"typeexplicit", "Explicit types", "All type conversions must be explicit or directly supported by instruction set."},
{&flag_boundchecks, defaultflag, "boundchecks", "Enforce Bound Checks", "Enforce array index checks to avoid accessing arrays out of bounds. This can be disabled for a speedup (the qcvm will still verify that the access is within the qcvm's memory, but it can't verify that its within the intended array)."},
{&flag_attributes, hideflag, "attributes", "[[attributes]]", "WARNING: This syntax conflicts with vector constructors."},
{&flag_assumevar, hideflag, "assumevar", "explicit consts", "Initialised globals will be considered non-const by default."},
{&flag_dblstarexp, hideflag, "ssp", "** exponent", "Treat ** as an operator for exponents, instead of multiplying by a dereferenced pointer."},
{&flag_cpriority, hideflag, "cpriority", "C Operator Priority", "QC treats !a&&b as equivelent to !(a&&b). When this is set, behaviour will be (!a)&&b as in C. Other operators are also affected in similar ways."},
{&flag_assume_double, hideflag, "assumedouble", "Assume Doubles", "Floating point immediates will be treated as doubles, for C compat."},
{&flag_qcfuncs, hidedefaultflag,"qcfuncs", "Parse QC-style funcs", "Recognise void() as a function type. Required for QC compat."},
{&flag_allowuninit, hideflag, "allowuninit", "Uninitialised Locals", "Permit optimisations that may result in locals being uninitialised. This may allow for greater reductions in temps."},
{&flag_nopragmafileline,FLAG_MIDCOMPILE,"nofileline", "Ignore #pragma file", "Ignores #pragma file(foo) and #pragma line(foo), so that errors and symbols reflect the actual lines, instead of the original source."},
// {&flag_lno, hidedefaultflag,"lno", "Gen Debugging Info", "Writes debugging info."},
{&flag_utf8strings, FLAG_MIDCOMPILE,"utf8", "Unicode", "String immediates will use utf-8 encoding, instead of quake's encoding."},
{&flag_embedsrc, FLAG_MIDCOMPILE,"embedsrc", "Embed Sources", "Write the sourcecode into the output file. The resulting .dat can be opened as a standard zip archive (or by fteqccgui).\nGood for GPL compliance!"},
// {&flag_noreflection, FLAG_MIDCOMPILE,"omitinternals","Omit Reflection Info", "Keeps internal symbols private (equivelent to unix's hidden visibility). This has the effect of reducing filesize, thwarting debuggers, and breaking saved games. This allows you to use arrays without massively bloating the size of your progs.\nWARNING: The bit about breaking saved games was NOT a joke, but does not apply to menuqc or csqc. It also interferes with FTE_MULTIPROGS."},
{&flag_dumpfilenames, FLAG_MIDCOMPILE,"dumpfilenames","Write a .lst file", "Writes a .lst file which contains a list of all file names that we can detect from the qc. This file list can then be passed into external compression tools."},
{&flag_dumpfields, FLAG_MIDCOMPILE,"dumpfields", "Write a .fld file", "Writes a .fld file that shows which fields are defined, along with their offsets etc, for weird debugging."},
{&flag_dumpsymbols, FLAG_MIDCOMPILE,"dumpsymbols", "Write a .sym file", "Writes a .sym file alongside the dat which contains a list of all global symbols defined in the code (before stripping)"},
{&flag_dumpautocvars, FLAG_MIDCOMPILE,"dumpautocvars","Write a .cfg file", "Writes a .cfg file that contains a default value for each autocvar listed in the code"},
{&flag_dumplocalisation,FLAG_MIDCOMPILE,"dumplocalisation","Write a .pot file", "Writes a .po template file from your _("") strings that can be edited (with eg gettext's tools) for translations, resulting in eg csprogs.en_US.po vs csprogs.en.po and other various other dialects vs languages."},
{&flag_dumptags, FLAG_MIDCOMPILE,"dumptags", "Write vi's tags file", "Writes a .tags file for text editors compatible with vi to locate symbols by name. This file is unsorted and per-module, so you will need to 'cat ../*.tags|LC_COLLATE=C sort>tags' for it to be used."},
{&flag_dumptags, FLAG_MIDCOMPILE|FLAG_HIDDENINGUI,"tags","aka dumptags", "See dumptags"},
{NULL}
};
static const char *QCC_VersionString(void)
{
#if defined(SVNREVISION) && defined(SVNDATE)
return "FTEQCC: " STRINGIFY(SVNREVISION) " (" STRINGIFY(SVNDATE) ")";
#elif defined(SVNREVISION)
return "FTEQCC: " STRINGIFY(SVNREVISION) " (" __DATE__")";
#else
return "FTEQCC: " __DATE__;
#endif
}
/*
=================
BspModels
Runs qbsp and light on all of the models with a .bsp extension
=================
*/
static void QCC_BspModels (void)
{
/*
int p;
char *gamedir;
int i;
char *m;
char cmd[1024];
char name[256];
size_t result;
p = QCC_CheckParm ("-bspmodels");
if (!p)
return;
if (p == myargc-1)
QCC_Error (ERR_BADPARMS, "-bspmodels must preceed a game directory");
gamedir = myargv[p+1];
for (i=0 ; i<nummodels ; i++)
{
m = precache_models[i];
if (strcmp(m+strlen(m)-4, ".bsp"))
continue;
strcpy (name, m);
name[strlen(m)-4] = 0;
sprintf (cmd, "qbsp %s/%s ; light -extra %s/%s", gamedir, name, gamedir, name);
result = system (cmd); // do something with the result
if (result != 0)
QCC_Error(ERR_INTERNAL, "QCC_BspModels() system returned non zero (failure) with: qbsp %s/%s ; light -extra %s/%s (%i)\n", gamedir, name, gamedir, name, errno);
}
*/
}
typedef struct stringtab_s
{
struct stringtab_s *prev;
char *ofs;
int len;
} stringtab_t;
stringtab_t *stringtablist[256];
// CopyString returns an offset from the string heap
int QCC_CopyString (const char *str)
{
int old;
size_t len;
if (!str)
return 0;
if (!*str)
return !flag_nullemptystr;
if (opt_noduplicatestrings || !strcmp(str, "IMMEDIATE"))
{
#if 1
//more scalable (faster) version.
stringtab_t *t;
int len = strlen(str);
unsigned char key = str[len-1];
for (t = stringtablist[key]; t; t = t->prev)
{
if (t->len >= len)
if (!strcmp(t->ofs + t->len - len, str))
{
optres_noduplicatestrings += len;
return (t->ofs + t->len - len)-strings;
}
}
t = qccHunkAlloc(sizeof(*t));
t->prev = stringtablist[key];
stringtablist[key] = t;
t->ofs = strings+strofs;
t->len = len;
#elif 1
//bruteforce
char *s;
for (s = strings; s < strings+strofs; s++)
if (!strcmp(s, str))
{
optres_noduplicatestrings += strlen(str);
return s-strings;
}
#endif
}
old = strofs;
len = strlen(str)+1;
if ( (strofs + len) > MAX_STRINGS)
QCC_Error(ERR_INTERNAL, "QCC_CopyString: stringtable size limit exceeded\n");
memcpy (strings+strofs, str, len);
strofs += len;
return old;
}
int QCC_CopyStringLength (const char *str, size_t length)
{
int old;
if (!str)
return 0;
if (!*str && length == 1)
return !flag_nullemptystr;
old = strofs;
if ( (strofs + length) > MAX_STRINGS)
QCC_Error(ERR_INTERNAL, "QCC_CopyString: stringtable size limit exceeded\n");
memcpy (strings+strofs, str, length);
strings[strofs+length] = 0;
strofs += length+1;
return old;
}
/*static int QCC_CopyDupBackString (const char *str)
{
size_t length;
int old;
char *s;
for (s = strings+strofs-1; s>strings ; s--)
if (!strcmp(s, str))
return s-strings;
old = strofs;
length = strlen(str)+1;
if ( (strofs + length) > MAX_STRINGS)
QCC_Error(ERR_INTERNAL, "QCC_CopyString: stringtable size limit exceeded\n");
strcpy (strings+strofs, str);
strofs += length;
return old;
}
static void QCC_PrintStrings (void)
{
int i, l, j;
for (i=0 ; i<strofs ; i += l)
{
l = strlen(strings+i) + 1;
externs->Printf ("%5i : ",i);
for (j=0 ; j<l ; j++)
{
if (strings[i+j] == '\n')
{
putchar ('\\');
putchar ('n');
}
else
putchar (strings[i+j]);
}
externs->Printf ("\n");
}
}
void QCC_PrintFunctions (void)
{
int i,j;
QCC_dfunction_t *d;
for (i=0 ; i<numfunctions ; i++)
{
d = &functions[i];
externs->Printf ("%s : %s : %i %i (", strings + d->s_file, strings + d->s_name, d->first_statement, d->parm_start);
for (j=0 ; j<d->numparms ; j++)
externs->Printf ("%i ",d->parm_size[j]);
externs->Printf (")\n");
}
}*/
static void QCC_SortFields (void)
{
int i, j;
QCC_ddef32_t t;
//insertion sort, of sorts.
//(qsort doesn't guarentee ordering)
for (i = 1; i < numfielddefs; i++)
{
if (fields[i].ofs < fields[i-1].ofs)
{ //this entry is out of order
for (j = i-1; j > 0; j--)
{
if (fields[j].ofs <= fields[i].ofs)
{ //okay, so we're putting it after this element.
j++;
break;
}
}
t = fields[i];
memmove(&fields[j+1], &fields[j], (i-j)*sizeof(t));
fields[j] = t;
}
}
}
static void QCC_DumpFields (const char *outputname)
{
char line[1024];
extern char *basictypenames[];
int i;
QCC_ddef_t *d;
int h;
snprintf(line, sizeof(line), "%s.fld", outputname);
h = SafeOpenWrite (line, 2*1024*1024);
if (h >= 0)
{
for (i=0 ; i<numfielddefs ; i++)
{
d = &fields[i];
snprintf(line, sizeof(line), "%5i : (%s) %s\n", d->ofs, basictypenames[d->type], strings + d->s_name);
SafeWrite(h, line, strlen(line));
}
SafeClose(h);
}
}
static void QCC_DumpSymbolNames (const char *outputname)
{
char line[1024];
QCC_def_t *def;
int h;
snprintf(line, sizeof(line), "%s.sym", outputname);
h = SafeOpenWrite (line, 2*1024*1024);
if (h >= 0)
{
for (def = pr.def_head.next ; def ; def = def->next)
{
if ((def->scope && !def->isstatic) || !strcmp(def->name, "IMMEDIATE"))
continue;
if (def->symbolheader != def && def->symbolheader->type != def->type)
continue; //try to exclude vector components.
snprintf(line, sizeof(line), "%s\n", def->name);
SafeWrite(h, line, strlen(line));
}
SafeClose(h);
}
}
/*static void QCC_DumpSymbolInfo (const char *outputname)
{
char line[1024];
char tname[512];
QCC_def_t *def;
int h;
snprintf(line, sizeof(line), "%s.sym", outputname);
h = SafeOpenWrite (line, 2*1024*1024);
if (h >= 0)
{
for (def = pr.def_head.next ; def ; def = def->next)
{
// if ((def->scope && !def->isstatic) || !strcmp(def->name, "IMMEDIATE"))
// continue;
// if (def->symbolheader != def && def->symbolheader->type != def->type)
// continue; //try to exclude vector components.
if (def->arraysize)
snprintf(line, sizeof(line), "%s%i: %s[%i] %s = ", def->used?"":"(unused)", def->ofs, TypeName(def->type, tname, sizeof(tname)), def->arraysize, def->name);
else
snprintf(line, sizeof(line), "%s%i: %s %s = ", def->used?"":"(unused)", def->ofs, TypeName(def->type, tname, sizeof(tname)), def->name);
SafeWrite(h, line, strlen(line));
switch(def->type->type)
{
case ev_vector:
snprintf(line, sizeof(line), "%g %g %g\n", def->symboldata[0]._float, def->symboldata[1]._float, def->symboldata[2]._float);
break;
case ev_float:
snprintf(line, sizeof(line), "%g\n", def->symboldata[0]._float);
break;
case ev_double:
snprintf(line, sizeof(line), "%g\n", def->symboldata[0]._float);
break;
case ev_string:
snprintf(line, sizeof(line), "%+i, %s\n", def->symboldata[0].string, QCC_PR_String(strings + def->symboldata[0].string));
break;
case ev_function:
snprintf(line, sizeof(line), "%+i, %s\n", def->symboldata[0]._int, functions[def->symboldata[0]._int].name);
break;
case ev_int64:
case ev_uint64:
snprintf(line, sizeof(line), "%i\n", def->symboldata[0]._int);
break;
default:
snprintf(line, sizeof(line), "%i\n", def->symboldata[0]._int);
break;
}
SafeWrite(h, line, strlen(line));
}
SafeClose(h);
}
}*/
/*
static void QCC_PrintGlobals (void)
{
int i;
QCC_ddef_t *d;
externs->Printf("Globals Listing:\n");
for (i=0 ; i<numglobaldefs ; i++)
{
d = &qcc_globals[i];
externs->Printf ("%5i : (%i) %s\n", d->ofs, d->type, strings + d->s_name);
}
}*/
static void QCC_DumpAutoCvars (const char *outputname)
{
char line[1024];
int i, h;
QCC_ddef_t *d;
char *n;
snprintf(line, sizeof(line), "%s.cfg", outputname);
h = SafeOpenWrite (line, 2*1024*1024);
if (h >= 0)
{
for (i=0 ; i<numglobaldefs ; i++)
{
d = &qcc_globals[i];
n = strings + d->s_name;
if (!strncmp(n, "autocvar_", 9))
{
char *desc;
const QCC_eval_t *val = (const QCC_eval_t*)&qcc_pr_globals[d->ofs];
QCC_def_t *def = QCC_PR_GetDef(NULL, n, NULL, false, 0, 0);
n += 9;
if (!def)
continue; //erk?
if (def->comment)
desc = def->comment;
else
desc = NULL;
switch(d->type & ~(DEF_SAVEGLOBAL|DEF_SHARED))
{
case ev_float:
snprintf(line, sizeof(line), "set %s\t%g%s%s\n", n, val->_float, desc?"\t//":"", desc?desc:"");
break;
case ev_double:
snprintf(line, sizeof(line), "set %s\t%g%s%s\n", n, val->_double, desc?"\t//":"", desc?desc:"");
break;
case ev_vector:
snprintf(line, sizeof(line), "set %s\t\"%g %g %g\"%s%s\n", n, val->vector[0], val->vector[1], val->vector[2], desc?"\t//":"", desc?desc:"");
break;
case ev_integer:
snprintf(line, sizeof(line), "set %s\t%"pPRIi"%s%s\n", n, val->_int, desc?"\t//":"", desc?desc:"");
break;
case ev_uint:
snprintf(line, sizeof(line), "set %s\t%"pPRIu"%s%s\n", n, val->_uint, desc?"\t//":"", desc?desc:"");
break;
case ev_int64:
snprintf(line, sizeof(line), "set %s\t%"pPRIi64"%s%s\n", n, val->i64, desc?"\t//":"", desc?desc:"");
break;
case ev_uint64:
snprintf(line, sizeof(line), "set %s\t%"pPRIu64"%s%s\n", n, val->u64, desc?"\t//":"", desc?desc:"");
break;
case ev_string:
snprintf(line, sizeof(line), "set %s\t\"%s\"%s%s\n", n, strings + val->_int, desc?"\t//":"", desc?desc:"");
break;
default:
snprintf(line, sizeof(line), "//set %s\t ?%s%s\n", n, desc?"\t//":"", desc?desc:"");
break;
}
SafeWrite(h, line, strlen(line));
}
}
}
}
static void QCC_DumpLocalisation (const char *outputname)
{
char line[65536];
int h, o;
QCC_def_t *def;
char *n;
snprintf(line, sizeof(line), "%s.pot", outputname);
h = SafeOpenWrite (line, 2*1024*1024);
if (h >= 0)
{
for (def = pr.def_head.next ; def ; def = def->next)
{
if (!strncmp(def->name, "dotranslate_", 12))
{
const QCC_eval_t *val = (const QCC_eval_t*)def->symboldata;
if (def->type->type != ev_string)
continue;
if (def->comment)
{
n = def->comment;
snprintf(line, sizeof(line), "#. ");
for (o = strlen(line); *n && o < countof(line)-10; n++)
{
if (*n == '\n')
{
if (n[1])
{
line[o++] = '\n';
line[o++] = '#';
line[o++] = '.';
line[o++] = ' ';
continue;
}
else line[++o] = 'n';
}
else if (*n == '\\') line[++o] = '\\';
else if (*n == '\"') line[++o] = '\"';
else if (*n == '\n') line[++o] = 'n';
else if (*n == '\r') line[++o] = 'r';
else if (*n == '\t') line[++o] = 't';
else
{ //hopefully the programmer used utf-8...
line[o++] = *n;
continue;
}
line[o++-1] = '\\';
}
line[o++] = '\n';
line[o++] = 0;
SafeWrite(h, line, strlen(line));
}
if (def->filen)
{ //strip any extra macro info there...
char *c = strchr(def->filen, ':');
if (c && (c[1] < '0' || c[1] > '9')) //don't get fooled by windows paths...
c = strchr(c+1, ':');
if (c)
{
*c = 0;
snprintf(line, sizeof(line), "#: %s:%i\n", def->filen, def->s_line);
*c = ':';
}
else
snprintf(line, sizeof(line), "#: %s:%i\n", def->filen, def->s_line);
SafeWrite(h, line, strlen(line));
}
n = strings + val->_int;
snprintf(line, sizeof(line), "msgid \"");
for (o = strlen(line); *n && o < countof(line)-5; n++)
{
if (*n == '\n' && n[1] && o < countof(line)-10)
{ //split multi-line stuff onto multiple lines, becase we can.
line[o++] = '\\';
line[o++] = 'n';
line[o++] = '\"';
line[o++] = '\n';
line[o++] = '\"';
continue;
}
else if (*n == '\\') line[++o] = '\\';
else if (*n == '\"') line[++o] = '\"';
else if (*n == '\n') line[++o] = 'n';
else if (*n == '\r') line[++o] = 'r';
else if (*n == '\t') line[++o] = 't';
else
{ //hopefully the programmer used utf-8...
line[o++] = *n;
continue;
}
line[o++-1] = '\\';
}
line[o++] = '\"';
line[o++] = '\n';
line[o++] = 0;
SafeWrite(h, line, strlen(line));
snprintf(line, sizeof(line), "msgstr \"\"\n\n");
SafeWrite(h, line, strlen(line));
}
}
}
SafeClose(h);
}
static void QCC_DumpFiles (const char *outputname)
{
struct
{
precache_t *list;
int count;
} precaches[] =
{
{ precache_sound, numsounds},
{ precache_texture, numtextures},
{ precache_model, nummodels},
{ precache_file, numfiles}
};
int g, i, b;
pbool header;
externs->Printf("\nFile lists:\n");
for (b = 0; b < 64; b++)
{
for (header = false, g = 0; g < sizeof(precaches)/sizeof(precaches[0]); g++)
{
for (i = 0; i < precaches[g].count; i++)
{
if (precaches[g].list[i].block == b)
{
if (*precaches[g].list[i].name == '*')
continue; //*-prefixed models are not real, and shouldn't be included in file lists.
if (!header)
{
externs->Printf("pak%i:\n", b-1);
header=true;
}
externs->Printf("%s\n", precaches[g].list[i].name);
}
}
}
if (header)
externs->Printf("\n");
}
}
void QCC_DumpPreProcTags(void *ctx, void *data)
{
char line[2048];
int h = *(int*)ctx;
CompilerConstant_t *def = data;
if (def->fromfile)
{
snprintf(line, sizeof(line), "%s\t%s\t%i;\"\td\n", def->name, def->fromfile, def->fromline);
SafeWrite(h, line, strlen(line));
}
}
static void QCC_DumpTags(const char *outputname)
{
char line[65536];
char scope[2048];
int h;
QCC_def_t *def;
QCC_function_t *fnc;
int i;
QCC_type_t *t;
snprintf(line, sizeof(line), "%s.tags", outputname);
h = SafeOpenWrite (line, 2*1024*1024);
if (h >= 0)
{
Hash_Enumerate(&compconstantstable, QCC_DumpPreProcTags, &h);
for (i = 0; i < numtypeinfos; i++)
{
t = &qcc_typeinfo[i];
if (!t->line || !t->filen)
continue; //predefined type, not from any file.
if (!*t->name || strchr(t->name, '<'))
continue; //anonymous stuff happens.
if (t->typedefed)
snprintf(line, sizeof(line), "%s\t%s\t%i;\"\tt\n", t->name, t->filen, t->line);
else if (t->type == ev_struct)
snprintf(line, sizeof(line), "%s\t%s\t%i;\"\ts\n", t->name, t->filen, t->line);
else if (t->type == ev_union)
snprintf(line, sizeof(line), "%s\t%s\t%i;\"\tu\n", t->name, t->filen, t->line);
else if (t->type == ev_enum)
snprintf(line, sizeof(line), "%s\t%s\t%i;\"\tg\n", t->name, t->filen, t->line);
else if ((t->type == ev_entity||t->type == ev_accessor) && t->parentclass)
snprintf(line, sizeof(line), "%s\t%s\t%i;\"\tc\n", t->name, t->filen, t->line);
else
continue;
SafeWrite(h, line, strlen(line));
}
for (def = pr.def_head.next; def; def = def->next)
{
//immediates are uninteresting...
if (!strcmp(def->name, "IMMEDIATE"))
continue;
if (strchr(def->name, '.') || strchr(def->name, '[') || strchr(def->name, '*'))
continue; //weird symbols, listed for savedgames only...
if (def->scope && !strchr(def->scope->name, ':'))
snprintf(scope, sizeof(scope), "\tfunction:%s\n", def->scope->name);
else if (def->isstatic)
snprintf(scope, sizeof(scope), "\file:\n"); //local scope
else
*scope = 0;
if (def->type->type == ev_function && def->constant && !def->arraysize)
{
int fnum = def->symboldata[0].function;
if (fnum > 0 && fnum < numfunctions)
{
fnc = &functions[fnum];
if (fnc->code>=0 && fnc->filen)
{
snprintf(line, sizeof(line), "%s\t%s\t%i;\"\tf%s\n", def->name, fnc->filen, fnc->line, scope);
SafeWrite(h, line, strlen(line));
}
}
if (def->filen)
{
snprintf(line, sizeof(line), "%s\t%s\t%i;\"\tp%s\n", def->name, def->filen, def->s_line, scope);
SafeWrite(h, line, strlen(line));
}
}
else if (def->filen)
{
snprintf(line, sizeof(line), "%s\t%s\t%i;\"\tv%s\n", def->name, def->filen, def->s_line, scope);
SafeWrite(h, line, strlen(line));
}
}
}
SafeClose(h);
}
int WriteSourceFiles(qcc_cachedsourcefile_t *filelist, int h, pbool sourceaswell, pbool legacyembed)
{
//helpers to deal with misaligned data. writes little-endian.
#define misbyte(ptr,ofs,data) ((unsigned char*)(ptr))[ofs] = (data)&0xff;
#define misshort(ptr,ofs,data) misbyte((ptr),(ofs),(data));misbyte((ptr),(ofs)+1,(data)>>8);
#define misint(ptr,ofs,data) misshort((ptr),(ofs),(data));misshort((ptr),(ofs)+2,(data)>>16);
includeddatafile_t *idf;
qcc_cachedsourcefile_t *f;
int num=0;
int ofs;
pbool zipembed = true;
int startofs;
sourceaswell |= flag_embedsrc;
for (f = filelist,num=0; f ; f=f->next)
{
if (f->type == FT_CODE && !sourceaswell)
continue;
num++;
}
if (!num)
{
if (zipembed)
{ //zips are found by scanning. so make sure something can be found so noone will erroneously find something
char centralheader[22];
int centraldirofs = SafeSeek(h, 0, SEEK_CUR);
misint (centralheader, 0, 0x06054b50);
misshort(centralheader, 4, 0); //this disk number
misshort(centralheader, 6, 0); //centraldir first disk
misshort(centralheader, 8, 0); //centraldir entries
misshort(centralheader, 10, 0); //total centraldir entries
misint (centralheader, 12, 0); //centraldir size
misint (centralheader, 16, centraldirofs); //centraldir offset
misshort(centralheader, 20, 0); //comment length
SafeWrite(h, centralheader, 22);
}
return 0;
}
startofs = SafeSeek(h, 0, SEEK_CUR);
idf = qccHunkAlloc(sizeof(includeddatafile_t)*num);
for (f = filelist,num=0; f ; f=f->next)
{
if (f->type == FT_CODE && !sourceaswell)
continue;
if (zipembed)
{
size_t end;
char header[32];
size_t fnamelen = strlen(f->filename);
f->zcrc = QC_encodecrc(f->size, f->file);
misint (header, 0, 0x04034b50);
misshort(header, 4, 0);//minver
misshort(header, 6, 0);//general purpose flags
misshort(header, 8, 0);//compression method, 0=store, 8=deflate
misshort(header, 10, 0);//lastmodfiletime
misshort(header, 12, 0);//lastmodfiledate
misint (header, 14, f->zcrc);//crc32
misint (header, 18, f->size);//compressed size
misint (header, 22, f->size);//uncompressed size
misshort(header, 26, fnamelen);//filename length
misshort(header, 28, 0);//extradata length
f->zhdrofs = SafeSeek(h, 0, SEEK_CUR);
SafeWrite(h, header, 30);
SafeWrite(h, f->filename, fnamelen);
strcpy(idf[num].filename, f->filename);
idf[num].size = f->size;
idf[num].compmethod = 8; //must be 0(raw) or 8(raw deflate) for zips to work
idf[num].ofs = SafeSeek(h, 0, SEEK_CUR);
if (idf[num].compmethod==0)
SafeWrite(h, f->file, f->size);
else
{
idf[num].compsize = QC_encode(progfuncs, f->size, idf[num].compmethod, f->file, h);
misshort(header, 8, idf[num].compmethod);//compression method, 0=store, 8=deflate
misint (header, 18, idf[num].compsize);
end = SafeSeek(h, 0, SEEK_CUR);
SafeSeek(h, f->zhdrofs, SEEK_SET);
SafeWrite(h, header, 30);
SafeSeek(h, end, SEEK_SET);
}
}
else
{
if (strlen(f->filename) >= sizeof(idf[num].filename))
continue;
strcpy(idf[num].filename, f->filename);
idf[num].size = f->size;
#ifdef AVAIL_ZLIB
idf[num].compmethod = 2;
#else
idf[num].compmethod = 0;
#endif
idf[num].ofs = SafeSeek(h, 0, SEEK_CUR);
idf[num].compsize = QC_encode(progfuncs, f->size, idf[num].compmethod, f->file, h);
}
num++;
}
if (zipembed)
{
char centralheader[46];
int centraldirsize;
ofs = SafeSeek(h, 0, SEEK_CUR);
for (f = filelist,num=0; f ; f=f->next)
{
size_t fnamelen;
if (f->type == FT_CODE && !sourceaswell)
continue;
fnamelen = strlen(f->filename);
misint (centralheader, 0, 0x02014b50);
misshort(centralheader, 4, 0);//ourver
misshort(centralheader, 6, 0);//minver
misshort(centralheader, 8, 0);//general purpose flags
misshort(centralheader, 10, idf[num].compmethod);//compression method, 0=store, 8=deflate
misshort(centralheader, 12, 0);//lastmodfiletime
misshort(centralheader, 14, 0);//lastmodfiledate
misint (centralheader, 16, f->zcrc);//crc32
misint (centralheader, 20, idf[num].compsize);//compressed size
misint (centralheader, 24, f->size);//uncompressed size
misshort(centralheader, 28, fnamelen);//filename length
misshort(centralheader, 30, 0);//extradata length
misshort(centralheader, 32, 0);//comment length
misshort(centralheader, 34, 0);//first disk number
misshort(centralheader, 36, 0);//internal file attribs
misint (centralheader, 38, 0);//external file attribs
misint (centralheader, 42, f->zhdrofs);//local header offset
SafeWrite(h, centralheader, 46);
SafeWrite(h, f->filename, fnamelen);
num++;
}
centraldirsize = SafeSeek(h, 0, SEEK_CUR)-ofs;
misint (centralheader, 0, 0x06054b50);
misshort(centralheader, 4, 0); //this disk number
misshort(centralheader, 6, 0); //centraldir first disk
misshort(centralheader, 8, num); //centraldir entries
misshort(centralheader, 10, num); //total centraldir entries
misint (centralheader, 12, centraldirsize); //centraldir size
misint (centralheader, 16, ofs); //centraldir offset
misshort(centralheader, 20, 0); //comment length
SafeWrite(h, centralheader, 22);
ofs = 0;
}
else if (legacyembed)
{
ofs = SafeSeek(h, 0, SEEK_CUR);
SafeWrite(h, &num, sizeof(int));
SafeWrite(h, idf, sizeof(includeddatafile_t)*num);
}
else
ofs = 0;
externs->Printf("Embedded files take %u bytes\n", SafeSeek(h, 0, SEEK_CUR) - startofs);
return ofs;
}
static void QCC_InitData (void)
{
static char parmname[12][MAX_PARMS];
int i;
qcc_sourcefile = NULL;
memset(stringtablist, 0, sizeof(stringtablist));
numstatements = 1; //first statement should be an OP_DONE, matching the null function(ish), in case it somehow doesn't get caught by the vm.
strofs = 2; //null, empty, *other stuff*
numfunctions = 1; //first function is a null function.
numglobaldefs = 1; //first globaldef is a null def. just because. doesn't include parms+ret. is there any point to this?
numfielddefs = 0;
fields[numfielddefs].type = ev_void;
fields[numfielddefs].s_name = 0; //should map to null
fields[numfielddefs].ofs = 0;
numfielddefs++; //FIXME: do we actually need a null field? is there any point to this at all?
memset(&def_ret, 0, sizeof(def_ret));
def_ret.ofs = OFS_RETURN;
def_ret.name = "return";
def_ret.constant = false;
def_ret.type = NULL;
def_ret.symbolheader = &def_ret;
for (i=0 ; i<MAX_PARMS ; i++)
{
def_parms[i].symbolheader = &def_parms[i];
def_parms[i].temp = NULL;
def_parms[i].type = NULL;
def_parms[i].ofs = OFS_PARM0 + 3*i;
def_parms[i].name = parmname[i];
sprintf(parmname[i], "parm%i", i);
}
}
static int WriteBodylessFuncs (int handle)
{
QCC_def_t *d;
int ret=0;
for (d=pr.def_head.next ; d ; d=d->next)
{
if (!d->used || !d->constant || d->symbolheader != d)
continue;
if (d->type->type == ev_function && !d->scope)// function parms are ok
{
if (d->isextern)
{
SafeWrite(handle, d->name, strlen(d->name)+1);
ret++;
}
if (d->initialized == 0)
{
QCC_PR_Warning(ERR_NOFUNC, d->filen, d->s_line, "function %s has no body", d->name);
QCC_PR_ParsePrintDef(ERR_NOFUNC, d);
}
}
}
return ret;
}
static void QCC_DetermineNeededSymbols(QCC_def_t *endsyssym)
{
QCC_def_t *sym = pr.def_head.next;
size_t i;
//make sure system defs are not hurt by this.
if (endsyssym)
{
for (; sym; sym = sym->next)
{
if (sym->unused && !sym->initialized)
sym->initialized = 1; //even if its not.
sym->used = true;
sym->referenced = true; //silence warnings about unreferenced things that can't be stripped
if (sym == endsyssym)
break;
}
}
//non-system fields should maybe be present too.
if (!opt_stripunusedfields)
{
for (; sym; sym = sym->next)
{
if (sym->constant && sym->type->type == ev_field)
{
sym->used = true;
sym->referenced = true;
}
}
}
for (i=0 ; i<numstatements ; i++)
{
if ((sym = statements[i].a.sym))
if (sym->symbolheader)
sym->used = true;
if ((sym = statements[i].b.sym))
if (sym->symbolheader)
sym->used = true;
if ((sym = statements[i].c.sym))
if (sym->symbolheader)
sym->used = true;
}
}
//allocates final space for the def, making it a true def
static void QCC_FinaliseDef(QCC_def_t *def)
{
//#define DEBUG_DUMP_GLOBALMAP
#if defined(DEBUG_DUMP) || defined(DEBUG_DUMP_GLOBALMAP)
int ssize = def->symbolsize;
const QCC_eval_t *v;
QCC_sref_t sr;
#endif
if (def->symboldata == qcc_pr_globals + def->ofs)
{
#ifdef DEBUG_DUMP_GLOBALMAP
externs->Printf("Prefinalised %s @ %i+%i\n", def->name, def->ofs, ssize);
#endif
return; //was already finalised.
}
if (def->symbolheader != def)
{
def->symbolheader->used |= def->used;
QCC_FinaliseDef(def->symbolheader);
def->referenced = true;
}
if (def->symbolheader == def && def->deftail)
{
//for head symbols, we go through and touch all of their children
QCC_def_t *prev, *sub;
if (def->used && !def->referenced)
{
pbool ignoreone = true;
//touch all but one child
for (prev = def, sub = prev->next; prev != def->deftail; sub = (prev=sub)->next)
{
if (sub->referenced)
def->referenced=true; //if one child is referenced, the composite is referenced
else if (!sub->referenced && ignoreone)
ignoreone = false; //this is the one we're going to warn about
}
// if (def->referenced) //no child defs were referenced at all. if we're going to be warning about this then at least mute warnings for any other fields
// for (prev = def, sub = prev->next; prev != def->deftail; sub = (prev=sub)->next)
// sub->referenced = true;
}
else
{
//touch children to silence annoying warnings.
for (prev = def, sub = prev->next; prev != def->deftail; sub = (prev=sub)->next)
sub->referenced |= true;
}
#ifdef TODO_READWRITETRACK
if (!def->read)
{
pbool ignoreone = true;
//touch all but one child
for (prev = def, sub = prev->next; prev != def->deftail; sub = (prev=sub)->next)
{
if (sub->read)
def->read=true; //if one child is referenced, the composite is referenced
else if (!sub->read && ignoreone)
ignoreone = false; //this is the one we're going to warn about
else
sub->read |= def->read;
}
if (!def->read) //no child defs were referenced at all. if we're going to be warning about this then at least mute warnings for any other fields
for (prev = def, sub = prev->next; prev != def->deftail; sub = (prev=sub)->next)
sub->read = true;
}
else
{
//touch children
for (prev = def, sub = prev->next; prev != def->deftail; sub = (prev=sub)->next)
sub->read |= def->read;
}
if (!def->written)
{
pbool ignoreone = true;
//touch all but one child
for (prev = def, sub = prev->next; prev != def->deftail; sub = (prev=sub)->next)
{
if (sub->written)
def->written=true; //if one child is referenced, the composite is referenced
else if (!sub->written && ignoreone)
ignoreone = false; //this is the one we're going to warn about
else
sub->written |= def->written;
}
if (!def->written) //no child defs were referenced at all. if we're going to be warning about this then at least mute warnings for any other fields
for (prev = def, sub = prev->next; prev != def->deftail; sub = (prev=sub)->next)
sub->written = true;
}
else
{
//touch children
for (prev = def, sub = prev->next; prev != def->deftail; sub = (prev=sub)->next)
sub->written |= def->written;
}
#endif
}
// else if (def->symbolheader) //if a child symbol is referenced, mark the entire parent as referenced too. this avoids vec_x+vec_y with no vec or vec_z from generating warnings about vec being unreferenced
// def->symbolheader->referenced |= def->referenced;
if (!def->symbolheader->used)
{
if (def->symboldata != qcc_pr_globals+def->ofs && def->symbolheader != def && def->symbolheader->symboldata == qcc_pr_globals + def->symbolheader->ofs)
{
def->ofs += def->symbolheader->ofs;
def->symboldata = qcc_pr_globals + def->ofs;
}
if (verbose >= VERBOSE_DEBUG)
externs->Printf("not needed: %s\n", def->name);
return;
}
else if (def->symbolheader != def && def->symbolheader->symboldata == qcc_pr_globals + def->symbolheader->ofs)
{
def->ofs += def->symbolheader->ofs;
}
else
{
/* globalspertype[def->type->type].size += def->symbolsize;
globalspertype[def->type->type].entries += 1;
globalspertype[def->type->type].vars += !def->constant;
globalspertype[def->type->type].consts += def->constant;
*/
if (def->ofs)
{
if (def->symbolheader == def)
QCC_Error(ERR_INTERNAL, "root symbol %s has an offset", def->name);
else
def->ofs = 0;
}
if (def->arraylengthprefix)
{
//for hexen2 arrays, we need to emit a length first
if (numpr_globals+1+def->symbolsize >= MAX_REGS)
{
if (!opt_overlaptemps || !opt_locals_overlapping)
QCC_Error(ERR_TOOMANYGLOBALS, "numpr_globals exceeded MAX_REGS - you'll need to use more optimisations");
else
QCC_Error(ERR_TOOMANYGLOBALS, "numpr_globals exceeded MAX_REGS of %u. Increase with eg: -max_regs %u", MAX_REGS, MAX_REGS*2);
}
if (def->type->type == ev_vector)
((int *)qcc_pr_globals)[numpr_globals] = def->arraysize-1;
else
((int *)qcc_pr_globals)[numpr_globals] = (def->arraysize*def->type->size)-1; //using float arrays for structs.
numpr_globals+=1;
}
else
{
if (numpr_globals+def->symbolsize >= MAX_REGS)
{
if (!opt_overlaptemps || !opt_locals_overlapping)
QCC_Error(ERR_TOOMANYGLOBALS, "numpr_globals exceeded MAX_REGS - you'll need to use more optimisations");
else
QCC_Error(ERR_TOOMANYGLOBALS, "numpr_globals exceeded MAX_REGS of %u. Increase with eg: -max_regs %u", MAX_REGS, MAX_REGS*2);
}
}
def->ofs += numpr_globals;
numpr_globals += def->symbolsize;
if (def->symboldata)
memcpy(qcc_pr_globals+def->ofs, def->symboldata, def->symbolsize*sizeof(float));
else
memset(qcc_pr_globals+def->ofs, 0, def->symbolsize*sizeof(float));
}
def->symboldata = qcc_pr_globals + def->ofs;
def->symbolsize = numpr_globals - def->ofs;
if (def->reloc)
{
def->reloc->used = true;
QCC_FinaliseDef(def->reloc);
qcc_pr_globals[def->ofs]._int += def->reloc->ofs;
}
#ifdef DEBUG_DUMP_GLOBALMAP
if (!def->referenced)
externs->Printf("Unreferenced ");
sr.sym = def;
sr.ofs = 0;
sr.cast = def->type;
v = (QCC_eval_t*)&sr.sym->symboldata[sr.ofs];
if (v && def->type->type == ev_float)
externs->Printf("Finalise %s(%f) @ %i+%i\n", def->name, v->_float, def->ofs, ssize);
else if (v && def->type->type == ev_vector)
externs->Printf("Finalise %s(%f %f %f) @ %i+%i\n", def->name, v->vector[0], v->vector[1], v->vector[2], def->ofs, ssize);
else if (v && def->type->type == ev_integer)
externs->Printf("Finalise %s(%i) @ %i+%i\n", def->name, v->_int, def->ofs, ssize);
else if (v && def->type->type == ev_function)
externs->Printf("Finalise %s(@%i) @ %i+%i\n", def->name, v->_int, def->ofs, ssize);
else if (v && def->type->type == ev_field)
externs->Printf("Finalise %s(.%i) @ %i+%i\n", def->name, v->_int, def->ofs, ssize);
else if (v && def->type->type == ev_string)
externs->Printf("Finalise %s(\"%s\") @ %i+%i\n", def->name, strings+v->_int, def->ofs, ssize);
else
externs->Printf("Finalise %s(?) @ %i+%i\n", def->name, def->ofs, ssize);
#endif
}
//marshalled locals still point to the FIRST_LOCAL range.
//this function remaps all the locals back into actual usable defs.
static void QCC_UnmarshalLocals(void)
{
QCC_def_t *d;
unsigned int onum, biggest, eog;
size_t i;
//finalise all the globals that we've seen so far
for (d = pr.def_head.next ; d ; d = d->next)
{
if (!d->localscope || d->isstatic)
QCC_FinaliseDef(d);
}
//first, finalize all static locals that shouldn't form part of the local defs.
for (i=0 ; i<numfunctions ; i++)
{
// if (functions[i].privatelocals)
{
for (d = functions[i].firstlocal; d; d = d->nextlocal)
if (d->isstatic || (d->constant && d->initialized))
QCC_FinaliseDef(d);
}
}
eog = numpr_globals;
//next, finalize non-static non-shared locals.
for (i=0 ; i<numfunctions ; i++)
{
if (functions[i].privatelocals)
{
onum = numpr_globals;
#ifdef DEBUG_DUMP
externs->Printf("function %s locals:\n", functions[i].name);
#endif
for (d = functions[i].firstlocal; d; d = d->nextlocal)
if (!d->isstatic && !(d->constant && d->initialized))
QCC_FinaliseDef(d);
if (verbose >= VERBOSE_DEBUG)
{
if (onum == numpr_globals)
externs->Printf("code: %s:%i: function %s no private locals\n", functions[i].filen, functions[i].line, functions[i].name);
else
externs->Printf("code: %s:%i: function %s private locals %i-%i\n", functions[i].filen, functions[i].line, functions[i].name, onum, numpr_globals);
}
}
}
//these functions don't have any initialisation issues, allowing us to merge them
onum = biggest = numpr_globals;
for (i=0 ; i<numfunctions ; i++)
{
if (!functions[i].privatelocals)
{
numpr_globals = onum;
for (d = functions[i].firstlocal; d; d = d->nextlocal)
if (!d->isstatic && !(d->constant && d->initialized))
QCC_FinaliseDef(d);
if (biggest < numpr_globals)
biggest = numpr_globals;
if (verbose >= VERBOSE_DEBUG)
{
if (onum == numpr_globals)
externs->Printf("code: %s:%i: function %s no locals\n", functions[i].filen, functions[i].line, functions[i].name);
else
{
externs->Printf("code: %s:%i: function %s overlapped locals %i-%i\n", functions[i].filen, functions[i].line, functions[i].name, onum, numpr_globals);
for (d = functions[i].firstlocal; d; d = d->nextlocal)
{
externs->Printf("code: %s:%i: %s @%i\n", functions[i].filen, functions[i].line, d->name, d->ofs);
}
}
}
}
}
numpr_globals = biggest;
if (verbose >= VERBOSE_STANDARD)
externs->Printf("%i shared locals, %i private, %i total\n", biggest - onum, onum - eog, numpr_globals-eog);
}
static void QCC_GenerateFieldDefs(QCC_def_t *def, char *fieldname, int ofs, QCC_type_t *type)
{
string_t sname;
QCC_ddef32_t *dd;
if (type->type == ev_struct || type->type == ev_union)
{ //the qcvm cannot cope with struct fields. so we need to generate lots of fake ones.
char sub[256];
unsigned int p, a;
unsigned int parms = type->num_parms;
if (type->type == ev_union)
parms = 1; //unions only generate the first element. it simplifies things (should really just be the biggest).
for (p = 0; p < parms; p++)
{
if (type->params[p].arraysize)
{
for (a = 0; a < type->params[p].arraysize; a++)
{
QC_snprintfz(sub, sizeof(sub), "%s.%s[%u]", fieldname, type->params[p].paramname, a);
QCC_GenerateFieldDefs(def, sub, ofs + type->params[p].ofs + a * type->params[p].type->size, type->params[p].type);
}
}
else
{
QC_snprintfz(sub, sizeof(sub), "%s.%s", fieldname, type->params[p].paramname);
QCC_GenerateFieldDefs(def, sub, ofs + type->params[p].ofs, type->params[p].type);
}
}
return;
}
if (numfielddefs >= MAX_FIELDS)
QCC_PR_ParseError(0, "Too many fields. Limit is %u\n", MAX_FIELDS);
dd = &fields[numfielddefs];
numfielddefs++;
dd->type = type->type;
dd->s_name = sname = QCC_CopyString (fieldname);
dd->ofs = def->symboldata[ofs]._int;
if (numglobaldefs >= MAX_GLOBALS)
QCC_PR_ParseError(0, "Too many globals. Limit is %u\n", MAX_GLOBALS);
//and make sure that there's a global defined too, so field remapping isn't screwed.
dd = &qcc_globals[numglobaldefs];
numglobaldefs++;
dd->type = ev_field;
dd->ofs = def->ofs+ofs;
dd->s_name = sname;
}
static const char *QCC_FileForStatement(int st)
{
const char *ret = "???";
int i;
for (i = 0; i < numfunctions; i++)
{
if (functions[i].code > 0)
{
if (st < functions[i].code)
break;
ret = functions[i].filen;
}
}
return ret;
}
static const char *QCC_FunctionForStatement(int st)
{
const char *ret = "???";
int i;
for (i = 0; i < numfunctions; i++)
{
if (functions[i].code > 0)
{
if (st < functions[i].code)
break;
ret = functions[i].name;
}
}
return ret;
}
static void QCC_PR_CRCMessages(unsigned short crc);
CompilerConstant_t *QCC_PR_CheckCompConstDefined(char *def);
static pbool QCC_WriteData (int crc)
{
QCC_def_t *def;
QCC_ddef_t *dd;
dprograms_t progs;
int h;
int i, len;
pbool debugtarget = false;
pbool types = false;
int outputsttype = PST_DEFAULT;
int dupewarncount = 0, extwarncount = 0;
int *statement_linenums;
void *funcdata;
size_t funcdatasize;
const char *bigjumps = NULL;
extern char *basictypenames[];
memset(&progs, 0, sizeof(progs));
if (numstatements==1 && numfunctions==1 && numglobaldefs==1 && numfielddefs==1)
{
externs->Printf("nothing to write\n");
return false;
}
progs.blockscompressed=0;
if (numstatements > MAX_STATEMENTS)
QCC_Error(ERR_TOOMANYSTATEMENTS, "Too many statements - %i\nAdd '-max_statements %i' to the commandline", numstatements, (numstatements+32768)&~32767);
if (strofs > MAX_STRINGS)
QCC_Error(ERR_TOOMANYSTRINGS, "Too many strings - %i\nAdd '-max_strings %i' to the commandline", strofs, (strofs+32768)&~32767);
//part of how compilation works. This def is always present, and never used.
def = QCC_PR_GetDef(NULL, "end_sys_globals", NULL, false, 0, false);
if (def)
def->referenced = true;
def = QCC_PR_GetDef(NULL, "end_sys_fields", NULL, false, 0, false);
if (def)
def->referenced = true;
QCC_PR_FinaliseFunctions();
QCC_DetermineNeededSymbols(def);
QCC_UnmarshalLocals();
QCC_FinaliseTemps();
for (i=0 ; i<numstatements ; i++)
{
if (!statements[i].a.sym && ((int)statements[i].a.ofs > 0x7fff || (int)statements[i].a.ofs < -0x7fff))
break;
if (!statements[i].a.sym && ((int)statements[i].a.ofs > 0x7fff || (int)statements[i].a.ofs < -0x7fff))
break;
if (!statements[i].a.sym && ((int)statements[i].a.ofs > 0x7fff || (int)statements[i].a.ofs < -0x7fff))
break;
}
if (i < numstatements)
bigjumps = QCC_FunctionForStatement(i);
QCC_PR_CRCMessages(crc);
switch (qcc_targetformat)
{
case QCF_HEXEN2:
case QCF_STANDARD:
case QCF_DARKPLACES: //grr.
if (bodylessfuncs)
externs->Printf("Warning: There are some functions without bodies.\n");
if (bigjumps)
{
externs->Printf("Forcing target to FTE32 due to large function %s\n", bigjumps);
outputsttype = PST_FTE32;
}
else if (numpr_globals > 65530)
{
if (qcc_targetformat == QCF_HEXEN2)
{
externs->Printf("Forcing target to uHexen2 due to numpr_globals\n");
outputsttype = PST_UHEXEN2;
}
else
{
externs->Printf("Forcing target to FTE32 due to numpr_globals\n");
outputsttype = PST_FTE32;
}
}
else if (qcc_targetformat == QCF_FTEH2)
{
externs->Printf("Progs execution will require FTE\n");
break;
}
else if (qcc_targetformat == QCF_HEXEN2)
{
externs->Printf("Progs execution requires a Hexen2 compatible HCVM\n");
break;
}
else if (qcc_targetformat == QCF_DARKPLACES)
{
externs->Printf("Progs execution uses extended opcodes.\n");
break;
}
else
{
if (numpr_globals >= 32768) //not much of a different format. Rewrite output to get it working on original executors?
externs->Printf("Globals exceeds 32k - an enhanced QCVM will be required\n");
else if (verbose >= VERBOSE_STANDARD)
externs->Printf("Progs should run on any QuakeC VM\n");
break;
}
QCC_OPCodeSetTarget((qcc_targetformat==QCF_HEXEN2)?QCF_FTEH2:QCF_FTE, 0);
//intentional fallthrough
case QCF_FTEDEBUG:
case QCF_FTE:
case QCF_FTEH2:
case QCF_QSS:
if (qcc_targetformat == QCF_FTEDEBUG)
debugtarget = true;
if (outputsttype != PST_FTE32 && outputsttype != PST_UHEXEN2)
{
if (bigjumps)
{
externs->Printf("Using 32 bit target due to large function %s\n", bigjumps);
outputsttype = PST_FTE32;
}
else if (numpr_globals > 65530)
{
externs->Printf("Using 32 bit target due to numpr_globals\n");
outputsttype = PST_FTE32;
}
}
if (qcc_targetformat == QCF_QSS || qcc_targetformat == QCF_DARKPLACES)
compressoutput = 0;
//compression of blocks?
if (compressoutput) progs.blockscompressed |=1; //statements
if (compressoutput) progs.blockscompressed |=2; //defs
if (compressoutput) progs.blockscompressed |=4; //fields
if (compressoutput) progs.blockscompressed |=8; //functions
if (compressoutput) progs.blockscompressed |=16; //strings
if (compressoutput) progs.blockscompressed |=32; //globals
if (compressoutput) progs.blockscompressed |=64; //line numbers
if (compressoutput) progs.blockscompressed |=128; //types
//include a type block?
//types = debugtarget;
if (types && sizeof(char *) != sizeof(string_t))
{
//qcc_typeinfo_t has a char* inside it, which changes size
externs->Printf("AMD64 builds cannot write typeinfo structures\n");
types = false;
}
if (verbose >= VERBOSE_STANDARD)
{
if (qcc_targetformat == QCF_QSS)
externs->Printf("QSS or FTE will be required\n");
else if (qcc_targetformat == QCF_DARKPLACES)
externs->Printf("DarkPlaces or FTE will be required\n");
else if (outputsttype == PST_UHEXEN2)
externs->Printf("FTE or uHexen2 will be required\n");
else
externs->Printf("FTE's QCLib will be required\n");
}
break;
case QCF_UHEXEN2:
debugtarget = false;
outputsttype = PST_UHEXEN2;
if (verbose >= VERBOSE_STANDARD)
externs->Printf("uHexen2 will be required\n");
if (numpr_globals < 65535)
externs->Printf("Warning: outputting 32 uHexen2 format when 16bit would suffice\n");
break;
case QCF_KK7:
if (bodylessfuncs)
externs->Printf("Warning: There are some functions without bodies.\n");
if (numpr_globals > 65530)
externs->Printf("Warning: Saving is not fully supported. Ensure all engine read fields and globals are defined early on.\n");
externs->Printf("A KK compatible executor will be required (FTE/KK)\n");
outputsttype = PST_KKQWSV;
break;
case QCF_QTEST:
externs->Printf("Compiled QTest progs will most likely not work at all. YOU'VE BEEN WARNED!\n");
outputsttype = PST_QTEST;
break;
default:
externs->Sys_Error("invalid progs type chosen!");
}
switch (outputsttype)
{
case PST_QTEST:
{
// this sucks but the structures are just too different
qtest_function_t *funcs = (qtest_function_t *)qccHunkAlloc(sizeof(qtest_function_t)*numfunctions);
for (i=0 ; i<numfunctions ; i++)
{
int j;
funcs[i].unused1 = 0;
funcs[i].profile = 0;
if (functions[i].code == -1)
funcs[i].first_statement = PRLittleLong (-functions[i].builtin);
else
funcs[i].first_statement = PRLittleLong (functions[i].code);
funcs[i].first_statement = PRLittleLong (functions[i].code);
funcs[i].parm_start = 0;//PRLittleLong (functions[i].parm_start);
funcs[i].s_name = PRLittleLong (QCC_CopyString(functions[i].name));
funcs[i].s_file = PRLittleLong (functions[i].s_filed);
funcs[i].numparms = 0;//PRLittleLong ((functions[i].numparms>MAX_PARMS)?MAX_PARMS:functions[i].numparms);
funcs[i].locals = 0;//PRLittleLong (functions[i].locals);
for (j = 0; j < MAX_PARMS; j++)
funcs[i].parm_size[j] = 0;//PRLittleLong((int)functions[i].parm_size[j]);
}
funcdata = funcs;
funcdatasize = numfunctions*sizeof(*funcs);
}
break;
case PST_UHEXEN2:
case PST_DEFAULT:
case PST_KKQWSV:
case PST_FTE32:
{
dfunction_t *funcs = (dfunction_t *)qccHunkAlloc(sizeof(dfunction_t)*numfunctions);
for (i=0 ; i<numfunctions ; i++)
{
QCC_def_t *local;
unsigned int p, a;
if (functions[i].code == -1)
funcs[i].first_statement = PRLittleLong (-functions[i].builtin);
else
funcs[i].first_statement = PRLittleLong (functions[i].code);
if (opt_function_names && functions[i].builtin)
{
//builtins don't need a name entry. the def is constant, so no string need be emitted at all.
optres_function_names += strlen(functions[i].name)+1;
funcs[i].s_name = 0;
}
else
funcs[i].s_name = PRLittleLong (QCC_CopyString(functions[i].name));
funcs[i].s_file = PRLittleLong (functions[i].s_filed);
if (functions[i].merged)
{
funcs[i].parm_start = functions[i].merged->parm_start;
funcs[i].locals = functions[i].merged->locals;
funcs[i].numparms = functions[i].merged->numparms;
for(p = 0; p < (unsigned)funcs[i].numparms; p++)
funcs[i].parm_size[p] = functions[i].merged->parm_size[p];
}
else if (functions[i].code == -1)
{
funcs[i].parm_start = 0;
funcs[i].locals = 0;
funcs[i].numparms = functions[i].type->num_parms;
if (funcs[i].numparms > MAX_PARMS)
funcs[i].numparms = MAX_PARMS; //shouldn't happen for builtins.
p = 0;
}
else if (functions[i].firstlocal)
{
unsigned int size;
funcs[i].parm_start = 0;
for (local = functions[i].firstlocal, a = 0, p = 0; local && a < MAX_PARMS && a < functions[i].type->num_parms; local = local->deftail->nextlocal)
{
if (!local->used)
{ //all params should have been assigned space. logically we could have safely omitted the last ones, but blurgh.
QCC_PR_Warning(ERR_INTERNAL, local->filen, local->s_line, "Argument %s was not marked used.\n", local->name);
continue;
}
if (!p)
funcs[i].parm_start = local->ofs;
size = local->type->size;
if (local->arraysize) //arrays are annoying
size = local->arraylengthprefix+size*local->arraysize;
funcs[i].locals += size;
for (; size > 3 && p < MAX_PARMS; size -= 3)
funcs[i].parm_size[p++] = 3; //the engine will copy PARMi over, it can't cope with larger types, the following args would be wrong.
if (p < MAX_PARMS)
funcs[i].parm_size[p++] = size;
a++;
}
for (; local && (!local->used || local->isstatic || (local->constant && local->initialized)); local = local->nextlocal)
;
if (!p && local)
funcs[i].parm_start = local->ofs;
for (; local; local = local->nextlocal)
{
if (!local->used || local->isstatic || (local->constant && local->initialized))
continue;
size = local->type->size;
if (local->arraysize) //arrays are annoying
size = local->arraylengthprefix+size*local->arraysize;
funcs[i].locals += size;
}
funcs[i].numparms = p;
}
else
{
//fucked up functions with no params.
funcs[i].parm_start = 0;
funcs[i].locals = 0;
funcs[i].numparms = 0;
p = 0;
}
while(p < MAX_PARMS)
funcs[i].parm_size[p++] = 0;
funcs[i].parm_start = PRLittleLong(funcs[i].parm_start);
funcs[i].locals = PRLittleLong(funcs[i].locals);
funcs[i].numparms = PRLittleLong(funcs[i].numparms);
if (funcs[i].locals && !funcs[i].parm_start)
QCC_PR_Warning(0, strings + funcs[i].s_file, functions[i].line, "%s:%i: func %s @%i locals@%i+%i, %i parms\n", functions[i].filen, functions[i].line, strings+funcs[i].s_name, funcs[i].first_statement, funcs[i].parm_start, funcs[i].locals, funcs[i].numparms);
if (verbose >= VERBOSE_DEBUG)
{
externs->Printf("code: %s:%i: func %s @%i locals@%i+%i, %i parms\n", functions[i].filen, functions[i].line, strings+funcs[i].s_name, funcs[i].first_statement, funcs[i].parm_start, funcs[i].locals, funcs[i].numparms);
externs->Printf("code: %s:%i: (%i,%i,%i,%i,%i,%i,%i,%i)\n", functions[i].filen, functions[i].line, funcs[i].parm_size[0], funcs[i].parm_size[1], funcs[i].parm_size[2], funcs[i].parm_size[3], funcs[i].parm_size[4], funcs[i].parm_size[5], funcs[i].parm_size[6], funcs[i].parm_size[7]);
}
}
funcdata = funcs;
funcdatasize = numfunctions*sizeof(*funcs);
}
break;
default:
externs->Sys_Error("structtype error");
funcdata = NULL;
funcdatasize = 0;
}
for (dupewarncount = 0, def = pr.def_head.next ; def ; def = def->next)
{
if (def->scope && !def->isstatic && !def->scope->privatelocals)
{ //if we're merging locals, then we shouldn't ever bother writing those globals. it just confuses debuggers etc. they're utterly pointless.
//which may be a problem if they're things that the engine is going to be swapping around... which they shouldn't be.
continue;
}
/* if (def->type->type == ev_vector || (def->type->type == ev_field && def->type->aux_type->type == ev_vector))
{ //do the references, so we don't get loadsa not referenced VEC_HULL_MINS_x
s_file = def->s_file;
QC_snprintfz (element, sizeof(element), "%s_x", def->name);
comp_x = QCC_PR_GetDef(NULL, element, def->scope, false, 0, false);
QC_snprintfz (element, sizeof(element), "%s_y", def->name);
comp_y = QCC_PR_GetDef(NULL, element, def->scope, false, 0, false);
QC_snprintfz (element, sizeof(element), "%s_z", def->name);
comp_z = QCC_PR_GetDef(NULL, element, def->scope, false, 0, false);
h = def->references;
if (comp_x && comp_y && comp_z)
{
h += comp_x->references;
h += comp_y->references;
h += comp_z->references;
if (!def->references)
if (!comp_x->references || !comp_y->references || !comp_z->references) //one of these vars is useless...
h=0;
def->references = h;
if (!h)
h = 1;
if (comp_x)
comp_x->references = h;
if (comp_y)
comp_y->references = h;
if (comp_z)
comp_z->references = h;
}
}
*/
#ifdef TODO_READWRITETRACK
if (def->symbolheader->read && !def->symbolheader->written && !def->symbolheader->referenced)
{
char typestr[256];
QCC_sref_t sr = {def, 0, def->type};
QCC_PR_Warning(WARN_READNOTWRITTEN, def->filen, def->s_line, "%s %s = %s read, but not writte.", TypeName(def->type, typestr, sizeof(typestr)), def->name, QCC_VarAtOffset(sr));
}
if (def->symbolheader->written && !def->symbolheader->read && !def->symbolheader->referenced)
{
char typestr[256];
QCC_sref_t sr = {def, 0, def->type};
QCC_PR_Warning(WARN_WRITTENNOTREAD, def->filen, def->s_line, "%s %s = %s written, but not read.", TypeName(def->type, typestr, sizeof(typestr)), def->name, QCC_VarAtOffset(sr));
}
#endif
if (!def->symbolheader->read && !def->symbolheader->written && !def->referenced)
{
int wt = def->constant?WARN_NOTREFERENCEDCONST:WARN_NOTREFERENCED;
if (def->type->type == ev_field && def->constant)
wt = WARN_NOTREFERENCEDFIELD;
pr_scope = def->scope;
if (!strncmp(def->name, "spawnfunc_", 10))
; //no warnings from unreferenced entry points.
else if (strcmp(def->name, "IMMEDIATE") && qccwarningaction[wt] && !(def->type->type == ev_function && def->symbolheader->timescalled) && !def->symbolheader->used)
{
char typestr[256];
if (QC_strcasestr(def->filen, "extensions") && verbose < VERBOSE_STANDARD)
{ //try to avoid annoying warnings from dpextensions.qc
extwarncount++;
QCC_PR_Warning(wt, def->filen, def->s_line, NULL);
}
else if (def->arraysize)
QCC_PR_Warning(wt, def->filen, def->s_line, (dupewarncount++ >= 10 && verbose < VERBOSE_STANDARD)?NULL:"%s %s%s%s[%i] no references.", TypeName(def->type, typestr, sizeof(typestr)), col_symbol, def->name, col_none, def->arraysize);
else
QCC_PR_Warning(wt, def->filen, def->s_line, (dupewarncount++ >= 10 && verbose < VERBOSE_STANDARD)?NULL:"%s %s%s%s no references.", TypeName(def->type, typestr, sizeof(typestr)), col_symbol, def->name, col_none);
}
pr_scope = NULL;
if (def->symbolheader->used)
{
char typestr[256];
QCC_sref_t sr = {def, 0, def->type};
QCC_PR_Warning(WARN_NOTREFERENCED, def->filen, def->s_line, "%s %s%s%s = %s used, but not referenced.", TypeName(def->type, typestr, sizeof(typestr)), col_symbol, def->name, col_none, QCC_VarAtOffset(sr));
}
/*if (opt_unreferenced && def->type->type != ev_field)
{
optres_unreferenced++;
#ifdef DEBUG_DUMP
externs->Printf("code: %s:%i: strip noref %s %s@%i;\n", def->filen, def->s_line, def->type->name, def->name, def->ofs);
#endif
continue;
}*/
}
if ((def->type->type == ev_struct || def->type->type == ev_union || def->arraysize) && def->deftail)
{
#ifdef DEBUG_DUMP
externs->Printf("code: %s:%i: strip struct %s %s@%i;\n", def->filen, def->s_line, def->type->name, def->name, def->ofs);
#endif
//the head of an array/struct is never written. only, its member fields are.
continue;
}
if (def->strip || !def->symbolheader->used)
{
optres_unreferenced++;
#ifdef DEBUG_DUMP
externs->Printf("code: %s:%i: strip %s %s@%i;\n", def->filen, def->s_line, def->type->name, def->name, def->ofs);
#endif
continue;
}
if (def->type->type == ev_function)
{
if (opt_function_names && def->initialized && functions[def->symboldata[0].function].code<0)
{
optres_function_names++;
def->name = "";
}
#if IAMNOTLAZY
if (!def->timescalled)
{
if (def->references<=1 && strncmp(def->name, "spawnfunc_", 10))
QCC_PR_Warning(WARN_DEADCODE, strings + def->s_file, def->s_line, "%s is never directly called or referenced (spawn function or dead code)", def->name);
// else
// QCC_PR_Warning(WARN_DEADCODE, strings + def->s_file, def->s_line, "%s is never directly called", def->name);
}
if (opt_stripfunctions && def->constant && def->timescalled >= def->references-1) //make sure it's not copied into a different var.
{ //if it ever does self.think then it could be needed for saves.
optres_stripfunctions++; //if it's only ever called explicitly, the engine doesn't need to know.
#ifdef DEBUG_DUMP
externs->Printf("code: %s:%i: strip const %s %s@%i;\n", strings+def->s_file, def->s_line, def->type->name, def->name, def->ofs);
#endif
continue;
}
#endif
}
else if (def->type->type == ev_field && def->constant && (!def->scope || def->isstatic || def->initialized))
{
QCC_GenerateFieldDefs(def, def->name, 0, def->type->aux_type);
continue;
}
else if ((def->scope||def->constant) && (def->type->type != ev_string || (strncmp(def->name, "dotranslate_", 12) && opt_constant_names_strings)))
{
if (opt_constant_names)
{
if (def->type->type == ev_string)
optres_constant_names_strings += strlen(def->name);
else
optres_constant_names += strlen(def->name);
#ifdef DEBUG_DUMP
if (def->scope)
externs->Printf("code: %s:%i: strip local %s %s@%i;\n", def->filen, def->s_line, def->type->name, def->name, def->ofs);
else if (def->constant)
externs->Printf("code: %s:%i: strip const %s %s@%i;\n", def->filen, def->s_line, def->type->name, def->name, def->ofs);
#endif
continue;
}
}
// if (!def->saved && def->type->type != ev_string)
// continue;
dd = &qcc_globals[numglobaldefs];
numglobaldefs++;
if (types)
dd->type = def->type-qcc_typeinfo;
else
dd->type = def->type->type;
#ifdef DEF_SAVEGLOBAL
if ( def->saved && !def->constant// || def->type->type == ev_function)
// && def->type->type != ev_function
&& def->type->type != ev_field
&& def->scope == NULL)
{
dd->type |= DEF_SAVEGLOBAL;
}
#endif
if (def->shared)
dd->type |= DEF_SHARED;
if (opt_locals && ((def->scope&&!def->isstatic) || !strcmp(def->name, "IMMEDIATE")))
{
dd->s_name = 0;
optres_locals += strlen(def->name);
}
else
dd->s_name = QCC_CopyString (def->name);
dd->ofs = def->ofs;
#ifdef DEBUG_DUMP
if ((dd->type&~(DEF_SHARED|DEF_SAVEGLOBAL)) == ev_string)
externs->Printf("code: %s:%i: %s%s%s %s@%i = \"%s\"\n", def->filen, def->s_line, dd->type&DEF_SAVEGLOBAL?"save ":"nosave ", dd->type&DEF_SHARED?"shared ":"", basictypenames[dd->type&~(DEF_SHARED|DEF_SAVEGLOBAL)], strings+dd->s_name, dd->ofs, ((unsigned)def->symboldata[0].string>=(unsigned)strofs)?"???":(strings + def->symboldata[0].string));
else if ((dd->type&~(DEF_SHARED|DEF_SAVEGLOBAL)) == ev_float)
externs->Printf("code: %s:%i: %s%s%s %s@%i = %g\n", def->filen, def->s_line, dd->type&DEF_SAVEGLOBAL?"save ":"nosave ", dd->type&DEF_SHARED?"shared ":"", basictypenames[dd->type&~(DEF_SHARED|DEF_SAVEGLOBAL)], strings+dd->s_name, dd->ofs, def->symboldata[0]._float);
else if ((dd->type&~(DEF_SHARED|DEF_SAVEGLOBAL)) == ev_integer)
externs->Printf("code: %s:%i: %s%s%s %s@%i = %i\n", def->filen, def->s_line, dd->type&DEF_SAVEGLOBAL?"save ":"nosave ", dd->type&DEF_SHARED?"shared ":"", basictypenames[dd->type&~(DEF_SHARED|DEF_SAVEGLOBAL)], strings+dd->s_name, dd->ofs, def->symboldata[0]._int);
else if ((dd->type&~(DEF_SHARED|DEF_SAVEGLOBAL)) == ev_vector)
externs->Printf("code: %s:%i: %s%s%s %s@%i = '%g %g %g'\n", def->filen, def->s_line, dd->type&DEF_SAVEGLOBAL?"save ":"nosave ", dd->type&DEF_SHARED?"shared ":"", basictypenames[dd->type&~(DEF_SHARED|DEF_SAVEGLOBAL)], strings+dd->s_name, dd->ofs, def->symboldata[0].vector[0], def->symboldata[0].vector[1], def->symboldata[0].vector[2]);
else if ((dd->type&~(DEF_SHARED|DEF_SAVEGLOBAL)) == ev_function)
externs->Printf("code: %s:%i: %s%s%s %s@%i = %i(%s)\n", def->filen, def->s_line, dd->type&DEF_SAVEGLOBAL?"save ":"nosave ", dd->type&DEF_SHARED?"shared ":"", basictypenames[dd->type&~(DEF_SHARED|DEF_SAVEGLOBAL)], strings+dd->s_name, dd->ofs, def->symboldata[0].function, def->symboldata[0].function >= numfunctions?"???":functions[def->symboldata[0].function].name);
else if ((dd->type&~(DEF_SHARED|DEF_SAVEGLOBAL)) == ev_field)
externs->Printf("code: %s:%i: %s%s%s %s@%i = @%i\n", def->filen, def->s_line, dd->type&DEF_SAVEGLOBAL?"save ":"nosave ", dd->type&DEF_SHARED?"shared ":"", basictypenames[dd->type&~(DEF_SHARED|DEF_SAVEGLOBAL)], strings+dd->s_name, dd->ofs, def->symboldata[0]._int);
else
externs->Printf("code: %s:%i: %s%s%s %s@%i\n", def->filen, def->s_line, dd->type&DEF_SAVEGLOBAL?"save ":"nosave ", dd->type&DEF_SHARED?"shared ":"", basictypenames[dd->type&~(DEF_SHARED|DEF_SAVEGLOBAL)], strings+dd->s_name, dd->ofs);
#endif
}
QCC_SortFields();
if (dupewarncount > 10 && verbose < VERBOSE_STANDARD)
QCC_PR_Note(WARN_NOTREFERENCED, NULL, 0, "suppressed %i more warnings about unreferenced variables, as you clearly don't care about the first 10.", dupewarncount-10);
if (extwarncount)
QCC_PR_Note(WARN_NOTREFERENCED, NULL, 0, "suppressed %i warnings about unused extensions.", extwarncount);
for (i = 0; i < numglobaldefs; i++)
{
dd = &qcc_globals[i];
if (!(dd->type & DEF_SAVEGLOBAL)) //only warn about saved ones.
continue;
for (h = 0; h < numglobaldefs; h++)
{
if (i == h || !(qcc_globals[h].type & DEF_SAVEGLOBAL))
continue;
if (dd->ofs == qcc_globals[h].ofs)
{
if ((dd->type&~DEF_SAVEGLOBAL) != (qcc_globals[h].type&~DEF_SAVEGLOBAL))
{
if (!(((dd->type&~DEF_SAVEGLOBAL) == ev_vector && (qcc_globals[h].type&~DEF_SAVEGLOBAL) == ev_float) ||
((dd->type&~DEF_SAVEGLOBAL) == ev_struct || (dd->type&~DEF_SAVEGLOBAL) == ev_union)))
QCC_PR_Warning(0, NULL, 0, "Mismatched union global types (%s and %s)", strings+dd->s_name, strings+qcc_globals[h].s_name);
}
//remove the saveglobal flag on the duplicate globals.
qcc_globals[h].type &= ~DEF_SAVEGLOBAL;
}
}
}
for (i = 1; i < numfielddefs; i++)
{
dd = &fields[i];
if (dd->type == ev_vector || dd->type == ev_struct || dd->type == ev_union) //just ignore vectors, structs, and unions.
continue;
for (h = 1; h < numfielddefs; h++)
{
if (i == h)
continue;
if (dd->ofs == fields[h].ofs)
{
if (dd->type != fields[h].type)
{
if (fields[h].type != ev_vector && fields[h].type != ev_struct && fields[h].type != ev_union)
{
QCC_PR_Warning(0, NULL, 0, "Mismatched union field types (%s and %s @%i)", strings+dd->s_name, strings+fields[h].s_name, dd->ofs);
}
}
}
}
}
if (numglobaldefs > MAX_GLOBALS)
QCC_Error(ERR_TOOMANYGLOBALS, "Too many globals - %i\nAdd '-max_globals %i' to the commandline", numglobaldefs, (numglobaldefs+32768)&~32767);
dupewarncount = 0;
for (i = 0; i < nummodels; i++)
{
if (!precache_model[i].used)
dupewarncount+=QCC_PR_Warning(WARN_EXTRAPRECACHE, precache_model[i].filename, precache_model[i].fileline, (dupewarncount>10&&verbose < VERBOSE_STANDARD)?NULL:"Model \"%s\" was precached but not directly used%s", precache_model[i].name, dupewarncount?"":" (annotate the usage with the used_model intrinsic to silence this warning)");
else if (!precache_model[i].block)
dupewarncount+=QCC_PR_Warning(WARN_NOTPRECACHED, precache_model[i].filename, precache_model[i].fileline, (dupewarncount>10&&verbose < VERBOSE_STANDARD)?NULL:"Model \"%s\" was used but not directly precached", precache_model[i].name);
}
for (i = 0; i < numsounds; i++)
{
if (!precache_sound[i].used)
dupewarncount+=QCC_PR_Warning(WARN_EXTRAPRECACHE, precache_sound[i].filename, precache_sound[i].fileline, (dupewarncount>10&&verbose < VERBOSE_STANDARD)?NULL:"Sound \"%s\" was precached but not directly used", precache_sound[i].name, dupewarncount?"":" (annotate the usage with the used_sound intrinsic to silence this warning)");
else if (!precache_sound[i].block)
dupewarncount+=QCC_PR_Warning(WARN_NOTPRECACHED, precache_sound[i].filename, precache_sound[i].fileline, (dupewarncount>10&&verbose < VERBOSE_STANDARD)?NULL:"Sound \"%s\" was used but not directly precached", precache_sound[i].name);
}
if (dupewarncount > 10 && verbose < VERBOSE_STANDARD)
QCC_PR_Note(WARN_NOTREFERENCED, NULL, 0, "suppressed %i more %swarnings%s about precaches.", dupewarncount-10, col_warning, col_none);
//PrintStrings ();
//PrintFunctions ();
//PrintFields ();
//PrintGlobals ();
strofs = (strofs+3)&~3;
if (verbose >= VERBOSE_STANDARD)
{
externs->Printf ("%6i strofs (of %i)\n", strofs, MAX_STRINGS);
externs->Printf ("%6i numstatements (of %i)\n", numstatements, MAX_STATEMENTS);
externs->Printf ("%6i numfunctions (of %i)\n", numfunctions, MAX_FUNCTIONS);
externs->Printf ("%6i numglobaldefs (of %i)\n", numglobaldefs, MAX_GLOBALS);
externs->Printf ("%6i numfielddefs (%i unique) (of %i)\n", numfielddefs, pr.size_fields, MAX_FIELDS);
externs->Printf ("%6i numpr_globals (of %i)\n", numpr_globals, MAX_REGS);
externs->Printf ("%6i nummodels\n", nummodels);
externs->Printf ("%6i numsounds\n", numsounds);
}
if (!*destfile)
strcpy(destfile, "progs.dat");
if (verbose >= VERBOSE_PROGRESS)
externs->Printf("Writing %s\n", destfile);
h = SafeOpenWrite (destfile, 2*1024*1024);
SafeWrite (h, &progs, sizeof(progs));
SafeWrite (h, "\r\n\r\n", 4);
SafeWrite (h, QCC_copyright, strlen(QCC_copyright)+1);
SafeWrite (h, "\r\n\r\n", 4);
while(SafeSeek (h, 0, SEEK_CUR) & 3)//this is a lame way to do it
{
SafeWrite (h, "\0", 1);
}
progs.ofs_strings = SafeSeek (h, 0, SEEK_CUR);
progs.numstrings = strofs;
if (progs.blockscompressed&16)
{
SafeWrite (h, &len, sizeof(int)); //save for later
len = QC_encode(progfuncs, strofs*sizeof(char), 2, (char *)strings, h); //write
i = SafeSeek (h, 0, SEEK_CUR);
SafeSeek(h, progs.ofs_strings, SEEK_SET);//seek back
len = PRLittleLong(len);
SafeWrite (h, &len, sizeof(int)); //write size.
SafeSeek(h, i, SEEK_SET);
}
else
SafeWrite (h, strings, strofs);
progs.ofs_statements = SafeSeek (h, 0, SEEK_CUR);
progs.numstatements = numstatements;
if (qcc_targetformat == QCF_HEXEN2 || qcc_targetformat == QCF_UHEXEN2 || qcc_targetformat == QCF_FTEH2)
{
for (i=0 ; i<numstatements ; i++)
{
if (statements[i].op >= OP_CALL1 && statements[i].op <= OP_CALL8)
QCC_Error(ERR_BADTARGETSWITCH, "Target switching produced incompatible instructions");
else if (statements[i].op >= OP_CALL1H && statements[i].op <= OP_CALL8H)
statements[i].op = statements[i].op - OP_CALL1H + OP_CALL1;
}
}
if (!opt_filenames)
{
statement_linenums = qccHunkAlloc(sizeof(statement_linenums) * numstatements);
for (i = 0; i < numstatements; i++)
statement_linenums[i] = statements[i].linenum;
}
else
statement_linenums = NULL;
switch(outputsttype)
{
case PST_UHEXEN2:
{
QCC_dstatement32_t *statements32 = qccHunkAlloc(sizeof(*statements32) * numstatements);
for (i=0 ; i<numstatements ; i++)
{
statements32[i].op = PRLittleLong(statements[i].op<<16);
statements32[i].a = PRLittleLong((statements[i].a.sym?statements[i].a.sym->ofs:0) + statements[i].a.ofs);
statements32[i].b = PRLittleLong((statements[i].b.sym?statements[i].b.sym->ofs:0) + statements[i].b.ofs);
statements32[i].c = PRLittleLong((statements[i].c.sym?statements[i].c.sym->ofs:0) + statements[i].c.ofs);
if (verbose >= VERBOSE_DEBUGSTATEMENTS)
externs->Printf("code: %s:%i: @%i %s %i %i %i\n", QCC_FileForStatement(i), statements[i].linenum, i, pr_opcodes[statements[i].op].name, statements32[i].a, statements32[i].b, statements32[i].c);
}
SafeWrite (h, statements32, numstatements*sizeof(QCC_dstatement32_t));
}
break;
case PST_KKQWSV:
case PST_FTE32:
{
QCC_dstatement32_t *statements32 = qccHunkAlloc(sizeof(*statements32) * numstatements);
for (i=0 ; i<numstatements ; i++)
{
statements32[i].op = PRLittleLong(statements[i].op);
statements32[i].a = PRLittleLong((statements[i].a.sym?statements[i].a.sym->ofs:0) + statements[i].a.ofs);
statements32[i].b = PRLittleLong((statements[i].b.sym?statements[i].b.sym->ofs:0) + statements[i].b.ofs);
statements32[i].c = PRLittleLong((statements[i].c.sym?statements[i].c.sym->ofs:0) + statements[i].c.ofs);
if (verbose >= VERBOSE_DEBUGSTATEMENTS)
externs->Printf("code: %s:%i: @%i %s %i %i %i\n", QCC_FileForStatement(i), statements[i].linenum, i, pr_opcodes[statements[i].op].name, statements32[i].a, statements32[i].b, statements32[i].c);
}
if (progs.blockscompressed&1)
{
SafeWrite (h, &len, sizeof(int)); //save for later
len = QC_encode(progfuncs, numstatements*sizeof(QCC_dstatement32_t), 2, (char *)statements, h); //write
i = SafeSeek (h, 0, SEEK_CUR);
SafeSeek(h, progs.ofs_statements, SEEK_SET);//seek back
len = PRLittleLong(len);
SafeWrite (h, &len, sizeof(int)); //write size.
SafeSeek(h, i, SEEK_SET);
}
else
SafeWrite (h, statements32, numstatements*sizeof(QCC_dstatement32_t));
}
break;
case PST_QTEST:
{
qtest_statement_t *qtst = qccHunkAlloc(sizeof(*qtst) * numstatements);
for (i=0 ; i<numstatements ; i++) // scale down from 16-byte internal to 12-byte qtest
{
unsigned int a = (statements[i].a.sym?statements[i].a.sym->ofs:0) + statements[i].a.ofs;
unsigned int b = (statements[i].b.sym?statements[i].b.sym->ofs:0) + statements[i].b.ofs;
unsigned int c = (statements[i].c.sym?statements[i].c.sym->ofs:0) + statements[i].c.ofs;
qtst[i].line = statements[i].linenum;
qtst[i].op = PRLittleShort((unsigned short)statements[i].op);
qtst[i].a = (unsigned short)PRLittleShort((unsigned short)a);
qtst[i].b = (unsigned short)PRLittleShort((unsigned short)b);
qtst[i].c = (unsigned short)PRLittleShort((unsigned short)c);
}
// no compression
SafeWrite (h, qtst, numstatements*sizeof(qtest_statement_t));
}
break;
case PST_DEFAULT:
{
QCC_dstatement16_t *statements16 = qccHunkAlloc(sizeof(*statements16) * numstatements);
#ifdef DISASM
int start, end;
for (i = 1; i < numfunctions; i++)
{
if (!strcmp(functions[i].name, DISASM))
{
start = functions[i].code;
end = functions[i+1].code;
}
}
#endif
for (i=0 ; i<numstatements ; i++) //resize as we go - scaling down
{
unsigned int a = (statements[i].a.sym?statements[i].a.sym->ofs:0) + statements[i].a.ofs;
unsigned int b = (statements[i].b.sym?statements[i].b.sym->ofs:0) + statements[i].b.ofs;
unsigned int c = (statements[i].c.sym?statements[i].c.sym->ofs:0) + statements[i].c.ofs;
statements16[i].op = PRLittleShort((unsigned short)statements[i].op);
if (
#if defined(DISASM)
(i >= start && i < end) ||
#endif
verbose >= VERBOSE_DEBUGSTATEMENTS)
{
char line[2048];
QC_snprintfz(line, sizeof(line), "code: %s:%i: @%i %s %i %i %i (%s", QCC_FileForStatement(i), statements[i].linenum, i, pr_opcodes[statements[i].op].opname, a, b, c, QCC_VarAtOffset(statements[i].a));
QC_snprintfz(line+strlen(line), sizeof(line)-strlen(line), " %s", QCC_VarAtOffset(statements[i].b));
QC_snprintfz(line+strlen(line), sizeof(line)-strlen(line), " %s)\n", QCC_VarAtOffset(statements[i].c));
externs->Printf("%s", line);
}
#ifdef _DEBUG
if (((signed)a >= (signed)numpr_globals && statements[i].a.sym) || ((signed)b >= (signed)numpr_globals && statements[i].b.sym) || ((signed)c >= (signed)numpr_globals && statements[i].c.sym))
externs->Printf("invalid offset on %s instruction (from line %i)\n", pr_opcodes[statements[i].op].opname, statements[i].linenum);
#endif
//truncate to 16bit. should probably warn if the high bits are not 0x0000 or 0xffff
statements16[i].a = (unsigned short)PRLittleShort((unsigned short)a);
statements16[i].b = (unsigned short)PRLittleShort((unsigned short)b);
statements16[i].c = (unsigned short)PRLittleShort((unsigned short)c);
}
if (progs.blockscompressed&1)
{
SafeWrite (h, &len, sizeof(int)); //save for later
len = QC_encode(progfuncs, numstatements*sizeof(QCC_dstatement16_t), 2, (char *)statements16, h); //write
i = SafeSeek (h, 0, SEEK_CUR);
SafeSeek(h, progs.ofs_statements, SEEK_SET);//seek back
len = PRLittleLong(len);
SafeWrite (h, &len, sizeof(int)); //write size.
SafeSeek(h, i, SEEK_SET);
}
else
SafeWrite (h, statements16, numstatements*sizeof(QCC_dstatement16_t));
}
break;
default:
externs->Sys_Error("structtype error");
}
progs.ofs_functions = SafeSeek (h, 0, SEEK_CUR);
progs.numfunctions = numfunctions;
if (progs.blockscompressed&8)
{
SafeWrite (h, &len, sizeof(int)); //save for later
len = QC_encode(progfuncs, funcdatasize, 2, (char *)funcdata, h); //write
i = SafeSeek (h, 0, SEEK_CUR);
SafeSeek(h, progs.ofs_functions, SEEK_SET);//seek back
len = PRLittleLong(len);
SafeWrite (h, &len, sizeof(int)); //write size.
SafeSeek(h, i, SEEK_SET);
}
else
SafeWrite (h, funcdata, funcdatasize);
if (flag_dumpfilenames)
QCC_DumpFiles(destfile);
if (flag_dumpfields)
QCC_DumpFields(destfile);
if (flag_dumpsymbols)
QCC_DumpSymbolNames(destfile);
// QCC_DumpSymbolInfo(destfile);
if (flag_dumpautocvars)
QCC_DumpAutoCvars(destfile);
if (flag_dumplocalisation)
QCC_DumpLocalisation(destfile);
if (flag_dumptags)
QCC_DumpTags(destfile);
switch(outputsttype)
{
case PST_QTEST:
// qtest needs a struct remap but should be able to get away with a simple swap here
for (i=0 ; i<numglobaldefs ; i++)
{
qtest_def_t qtdef = ((qtest_def_t *)qcc_globals)[i];
qcc_globals[i].type = qtdef.type;
qcc_globals[i].ofs = qtdef.ofs;
qcc_globals[i].s_name = qtdef.s_name;
}
for (i=0 ; i<numfielddefs ; i++)
{
qtest_def_t qtdef = ((qtest_def_t *)fields)[i];
fields[i].type = qtdef.type;
fields[i].ofs = qtdef.ofs;
fields[i].s_name = qtdef.s_name;
}
// passthrough.. reuse FTE32 code
case PST_FTE32:
progs.ofs_globaldefs = SafeSeek (h, 0, SEEK_CUR);
progs.numglobaldefs = numglobaldefs;
for (i=0 ; i<numglobaldefs ; i++)
{
qcc_globals[i].type = PRLittleLong/*PRLittleShort*/ (qcc_globals[i].type);
qcc_globals[i].ofs = PRLittleLong/*PRLittleShort*/ (qcc_globals[i].ofs);
qcc_globals[i].s_name = PRLittleLong (qcc_globals[i].s_name);
}
if (progs.blockscompressed&2)
{
SafeWrite (h, &len, sizeof(int)); //save for later
len = QC_encode(progfuncs, numglobaldefs*sizeof(QCC_ddef_t), 2, (char *)qcc_globals, h); //write
i = SafeSeek (h, 0, SEEK_CUR);
SafeSeek(h, progs.ofs_globaldefs, SEEK_SET);//seek back
len = PRLittleLong(len);
SafeWrite (h, &len, sizeof(int)); //write size.
SafeSeek(h, i, SEEK_SET);
}
else
SafeWrite (h, qcc_globals, numglobaldefs*sizeof(QCC_ddef_t));
progs.ofs_fielddefs = SafeSeek (h, 0, SEEK_CUR);
progs.numfielddefs = numfielddefs;
for (i=0 ; i<numfielddefs ; i++)
{
fields[i].type = PRLittleLong/*PRLittleShort*/ (fields[i].type);
fields[i].ofs = PRLittleLong/*PRLittleShort*/ (fields[i].ofs);
fields[i].s_name = PRLittleLong (fields[i].s_name);
}
if (progs.blockscompressed&4)
{
SafeWrite (h, &len, sizeof(int)); //save for later
len = QC_encode(progfuncs, numfielddefs*sizeof(QCC_ddef_t), 2, (char *)fields, h); //write
i = SafeSeek (h, 0, SEEK_CUR);
SafeSeek(h, progs.ofs_fielddefs, SEEK_SET);//seek back
len = PRLittleLong(len);
SafeWrite (h, &len, sizeof(int)); //write size.
SafeSeek(h, i, SEEK_SET);
}
else
SafeWrite (h, fields, numfielddefs*sizeof(QCC_ddef_t));
break;
case PST_UHEXEN2:
progs.ofs_globaldefs = SafeSeek (h, 0, SEEK_CUR);
progs.numglobaldefs = numglobaldefs;
for (i=0 ; i<numglobaldefs ; i++)
{
qcc_globals[i].type = PRLittleLong (qcc_globals[i].type<<16);
qcc_globals[i].ofs = PRLittleLong (qcc_globals[i].ofs);
qcc_globals[i].s_name = PRLittleLong (qcc_globals[i].s_name);
}
if (progs.blockscompressed&2)
{
SafeWrite (h, &len, sizeof(int)); //save for later
len = QC_encode(progfuncs, numglobaldefs*sizeof(QCC_ddef_t), 2, (char *)qcc_globals, h); //write
i = SafeSeek (h, 0, SEEK_CUR);
SafeSeek(h, progs.ofs_globaldefs, SEEK_SET);//seek back
len = PRLittleLong(len);
SafeWrite (h, &len, sizeof(int)); //write size.
SafeSeek(h, i, SEEK_SET);
}
else
SafeWrite (h, qcc_globals, numglobaldefs*sizeof(QCC_ddef_t));
progs.ofs_fielddefs = SafeSeek (h, 0, SEEK_CUR);
progs.numfielddefs = numfielddefs;
for (i=0 ; i<numfielddefs ; i++)
{
fields[i].type = PRLittleLong/*PRLittleShort*/ (fields[i].type<<16);
fields[i].ofs = PRLittleLong/*PRLittleShort*/ (fields[i].ofs);
fields[i].s_name = PRLittleLong (fields[i].s_name);
}
if (progs.blockscompressed&4)
{
SafeWrite (h, &len, sizeof(int)); //save for later
len = QC_encode(progfuncs, numfielddefs*sizeof(QCC_ddef_t), 2, (char *)fields, h); //write
i = SafeSeek (h, 0, SEEK_CUR);
SafeSeek(h, progs.ofs_fielddefs, SEEK_SET);//seek back
len = PRLittleLong(len);
SafeWrite (h, &len, sizeof(int)); //write size.
SafeSeek(h, i, SEEK_SET);
}
else
SafeWrite (h, fields, numfielddefs*sizeof(QCC_ddef_t));
break;
case PST_KKQWSV:
case PST_DEFAULT:
#define qcc_globals16 ((QCC_ddef16_t*)qcc_globals)
#define fields16 ((QCC_ddef16_t*)fields)
progs.ofs_globaldefs = SafeSeek (h, 0, SEEK_CUR);
progs.numglobaldefs = numglobaldefs;
for (i=0 ; i<numglobaldefs ; i++)
{
qcc_globals16[i].type = (unsigned short)PRLittleShort ((unsigned short)qcc_globals[i].type);
qcc_globals16[i].ofs = (unsigned short)PRLittleShort ((unsigned short)qcc_globals[i].ofs);
qcc_globals16[i].s_name = PRLittleLong (qcc_globals[i].s_name);
}
if (progs.blockscompressed&2)
{
SafeWrite (h, &len, sizeof(int)); //save for later
len = QC_encode(progfuncs, numglobaldefs*sizeof(QCC_ddef16_t), 2, (char *)qcc_globals16, h); //write
i = SafeSeek (h, 0, SEEK_CUR);
SafeSeek(h, progs.ofs_globaldefs, SEEK_SET);//seek back
len = PRLittleLong(len);
SafeWrite (h, &len, sizeof(int)); //write size.
SafeSeek(h, i, SEEK_SET);
}
else
SafeWrite (h, qcc_globals16, numglobaldefs*sizeof(QCC_ddef16_t));
progs.ofs_fielddefs = SafeSeek (h, 0, SEEK_CUR);
progs.numfielddefs = numfielddefs;
for (i=0 ; i<numfielddefs ; i++)
{
fields16[i].type = (unsigned short)PRLittleShort ((unsigned short)fields[i].type);
if (fields[i].ofs > 0xffff)
{
externs->Printf("Offset for field %s overflowed 16 bits.\n", strings+fields[i].s_name);
fields16[i].ofs = 0;
}
else
fields16[i].ofs = (unsigned short)PRLittleShort ((unsigned short)fields[i].ofs);
fields16[i].s_name = PRLittleLong (fields[i].s_name);
}
if (progs.blockscompressed&4)
{
SafeWrite (h, &len, sizeof(int)); //save for later
len = QC_encode(progfuncs, numfielddefs*sizeof(QCC_ddef16_t), 2, (char *)fields16, h); //write
i = SafeSeek (h, 0, SEEK_CUR);
SafeSeek(h, progs.ofs_fielddefs, SEEK_SET);//seek back
len = PRLittleLong(len);
SafeWrite (h, &len, sizeof(int)); //write size.
SafeSeek(h, i, SEEK_SET);
}
else
SafeWrite (h, fields16, numfielddefs*sizeof(QCC_ddef16_t));
break;
default:
externs->Sys_Error("structtype error");
}
progs.ofs_globals = SafeSeek (h, 0, SEEK_CUR);
progs.numglobals = numpr_globals;
for (i=0 ; (unsigned)i<numpr_globals ; i++)
((int *)qcc_pr_globals)[i] = PRLittleLong (((int *)qcc_pr_globals)[i]);
if (progs.blockscompressed&32)
{
SafeWrite (h, &len, sizeof(int)); //save for later
len = QC_encode(progfuncs, numpr_globals*4, 2, (char *)qcc_pr_globals, h); //write
i = SafeSeek (h, 0, SEEK_CUR);
SafeSeek(h, progs.ofs_globals, SEEK_SET);//seek back
len = PRLittleLong(len);
SafeWrite (h, &len, sizeof(int)); //write size.
SafeSeek(h, i, SEEK_SET);
}
else
SafeWrite (h, qcc_pr_globals, numpr_globals*4);
/*
if (types)
for (i=0 ; i<numtypeinfos ; i++)
{
if (qcc_typeinfo[i].aux_type)
qcc_typeinfo[i].aux_type = (QCC_type_t*)(qcc_typeinfo[i].aux_type - qcc_typeinfo);
if (qcc_typeinfo[i].next)
qcc_typeinfo[i].next = (QCC_type_t*)(qcc_typeinfo[i].next - qcc_typeinfo);
qcc_typeinfo[i].name = (char*)QCC_CopyDupBackString(qcc_typeinfo[i].name);
}
*/
progs.ofsfiles = 0;
progs.ofslinenums = 0;
progs.secondaryversion = 0;
progs.ofsbodylessfuncs = 0;
progs.numbodylessfuncs = 0;
progs.ofs_types = 0;
progs.numtypes = 0;
progs.ofsbodylessfuncs = SafeSeek (h, 0, SEEK_CUR);
progs.numbodylessfuncs = WriteBodylessFuncs(h);
switch(qcc_targetformat)
{
case QCF_QTEST:
progs.version = PROG_QTESTVERSION;
progs.ofsfiles = WriteSourceFiles(qcc_sourcefile, h, debugtarget, false);
break;
default:
case QCF_STANDARD:
case QCF_HEXEN2: //urgh
progs.version = PROG_VERSION;
progs.ofsfiles = WriteSourceFiles(qcc_sourcefile, h, debugtarget, false);
break;
case QCF_DARKPLACES:
case QCF_FTE:
case QCF_FTEH2:
case QCF_FTEDEBUG:
case QCF_QSS:
case QCF_UHEXEN2:
case QCF_KK7:
progs.version = PROG_EXTENDEDVERSION;
if (outputsttype == PST_UHEXEN2)
progs.secondaryversion = PROG_SECONDARYUHEXEN2; //prepadded...
else if (outputsttype == PST_KKQWSV)
progs.secondaryversion = PROG_SECONDARYKKQWSV; //messed up
else if (outputsttype == PST_FTE32)
progs.secondaryversion = PROG_SECONDARYVERSION32; //post-extended.
else
{
progs.secondaryversion = PROG_SECONDARYVERSION16;
if (qcc_targetformat == QCF_DARKPLACES)
progs.version = PROG_VERSION;
}
if (debugtarget && statement_linenums)
{
progs.ofslinenums = SafeSeek (h, 0, SEEK_CUR);
if (progs.blockscompressed&64)
{
SafeWrite (h, &len, sizeof(int)); //save for later
len = QC_encode(progfuncs, numstatements*sizeof(int), 2, (char *)statement_linenums, h); //write
i = SafeSeek (h, 0, SEEK_CUR);
SafeSeek(h, progs.ofslinenums, SEEK_SET);//seek back
len = PRLittleLong(len);
SafeWrite (h, &len, sizeof(int)); //write size.
SafeSeek(h, i, SEEK_SET);
}
else
SafeWrite (h, statement_linenums, numstatements*sizeof(int));
statement_linenums = NULL;
}
else
progs.ofslinenums = 0;
if (types)
{
progs.ofs_types = SafeSeek (h, 0, SEEK_CUR);
if (progs.blockscompressed&128)
{
SafeWrite (h, &len, sizeof(int)); //save for later
len = QC_encode(progfuncs, sizeof(QCC_type_t)*numtypeinfos, 2, (char *)qcc_typeinfo, h); //write
i = SafeSeek (h, 0, SEEK_CUR);
SafeSeek(h, progs.ofs_types, SEEK_SET);//seek back#
len = PRLittleLong(len);
SafeWrite (h, &len, sizeof(int)); //write size.
SafeSeek(h, i, SEEK_SET);
}
else
SafeWrite (h, qcc_typeinfo, sizeof(QCC_type_t)*numtypeinfos);
progs.numtypes = numtypeinfos;
}
else
{
progs.ofs_types = 0;
progs.numtypes = 0;
}
progs.ofsfiles = WriteSourceFiles(qcc_sourcefile, h, debugtarget, true);
break;
}
qcc_sourcefile = NULL;
if (progs.version != PROG_EXTENDEDVERSION && progs.numbodylessfuncs)
externs->Printf ("WARNING: progs format cannot handle extern functions\n");
if (verbose >= VERBOSE_STANDARD)
externs->Printf ("%6i TOTAL SIZE\n", (int)SafeSeek (h, 0, SEEK_CUR));
progs.entityfields = pr.size_fields;
progs.crc = crc;
if (flag_qccx)
{
def = QCC_PR_GetDef(NULL, "progs", NULL, false, 0, 0); //this is for qccx support
if (def && (def->type->type == ev_entity || (def->type->type == ev_accessor && def->type->parentclass->type == ev_entity)))
{
int size = SafeSeek (h, 0, SEEK_CUR);
size += 1; //the engine will add a null terminator
size = (size+15)&(~15); //and will allocate it on the hunk with 16-byte alignment
//this global receives the offset from world to the start of the progs def _IN VANILLA QUAKE_.
//this is a negative index due to allocation ordering with the assumption that the progs.dat was loaded on the heap directly followed by the entities.
//this will NOT work in FTE, DP, QuakeForge due to entity indexes. Various other engines will likely mess up too, if they change the allocation order or sizes etc. 64bit is screwed.
if (progs.blockscompressed&32)
externs->Printf("unable to write value for 'entity progs'\n"); //would not work anyway
else
{
QCC_PR_Warning(WARN_DENORMAL, def->filen, def->s_line, "'entity progs' is non-portable and will not work across engines nor cpus.");
if (def->initialized)
i = PRLittleLong(qcc_pr_globals[def->ofs]._int);
else
{ //entsize(=96)+hunk header size(=32)
if (verbose >= VERBOSE_STANDARD)
externs->Printf("qccx hack - 'entity progs' uninitialised. Assuming 112.\n");
i = 112; //match qccx.
}
i = -(size + i);
i = PRLittleLong(i);
SafeSeek (h, progs.ofs_globals + 4 * def->ofs, SEEK_SET);
SafeWrite (h, &i, 4);
}
}
}
// qbyte swap the header and write it out
for (i=0 ; i<sizeof(progs)/4 ; i++)
((int *)&progs)[i] = PRLittleLong ( ((int *)&progs)[i] );
SafeSeek (h, 0, SEEK_SET);
SafeWrite (h, &progs, sizeof(progs));
if (!SafeClose (h))
{
externs->Printf("%sError%s while writing output %s\n", col_error, col_none, destfile);
return false;
}
if (verbose >= VERBOSE_PROGRESS)
switch(qcc_targetformat)
{
case QCF_QTEST:
externs->Printf("Compile finished: %s (qtest format)\n", destfile);
break;
case QCF_KK7:
externs->Printf("Compile finished: %s (kk7 format)\n", destfile);
break;
case QCF_STANDARD:
externs->Printf("Compile finished: %s (id format)\n", destfile);
break;
case QCF_HEXEN2:
case QCF_UHEXEN2:
if (progs.version == PROG_VERSION)
externs->Printf("Compile finished: %s (hexen2 format)\n", destfile);
else
externs->Printf("Compile finished: %s (uhexen2 format)\n", destfile);
break;
case QCF_DARKPLACES:
externs->Printf("Compile finished: %s (fte+dp format)\n", destfile);
break;
case QCF_QSS:
externs->Printf("Compile finished: %s (fte+qss format)\n", destfile);
break;
case QCF_FTE:
externs->Printf("Compile finished: %s (fte format)\n", destfile);
break;
case QCF_FTEH2:
externs->Printf("Compile finished: %s (fteh2 format)\n", destfile);
break;
case QCF_FTEDEBUG:
externs->Printf("Compile finished: %s (ftedbg format)\n", destfile);
break;
default:
externs->Printf("Compile finished: %s\n", destfile);
break;
}
if (statement_linenums)
{
unsigned int lnotype = *(unsigned int*)"LNOF";
unsigned int version = 1;
pbool gz = false;
while(1)
{
char *ext;
ext = strrchr(destfile, '.');
if (!ext || strchr(ext, '/') || strchr(ext, '\\'))
break;
if (!stricmp(ext, ".gz"))
{
*ext = 0;
gz = true;
continue;
}
*ext = 0;
break;
}
if (strlen(destfile) < sizeof(destfile)-(4+3))
{
strcat(destfile, ".lno");
if (gz)
strcat(destfile, ".gz");
if (verbose >= VERBOSE_STANDARD)
externs->Printf("Writing %s for debugging\n", destfile);
h = SafeOpenWrite (destfile, 2*1024*1024);
SafeWrite (h, &lnotype, sizeof(int));
SafeWrite (h, &version, sizeof(int));
SafeWrite (h, &numglobaldefs, sizeof(int));
SafeWrite (h, &numpr_globals, sizeof(int));
SafeWrite (h, &numfielddefs, sizeof(int));
SafeWrite (h, &numstatements, sizeof(int));
SafeWrite (h, statement_linenums, numstatements*sizeof(int));
SafeClose (h);
}
}
return true;
}
/*
#merge "oldprogs"
wrap void() worldspawn =
{
print("hello world\n");
prior();
};
Progs merging is done by loading in an existing progs.dat and essentially appending new stuff on the end.
The resulting output should be the same, other than wraps (which replaces the previous function global with the new one).
*/
static void QCC_MergeStrings(char *in, unsigned int num)
{
memcpy(strings, in, num);
strofs = num;
}
static int QCC_MergeValidateString(int str)
{
if (str < 0 || str >= strofs)
str = 0;
return str;
}
static void QCC_MergeFunctions(dfunction_t *in, unsigned int num)
{
numfunctions = 0;
while(num --> 0)
{
if (in->first_statement <= 0)
{
functions[numfunctions].builtin = -in->first_statement;
functions[numfunctions].code = -1;
}
else
{
functions[numfunctions].builtin = 0;
functions[numfunctions].code = in->first_statement;
}
functions[numfunctions].s_filed = QCC_MergeValidateString(in->s_file);
functions[numfunctions].filen = strings+functions[numfunctions].s_filed;
functions[numfunctions].line = 0;
functions[numfunctions].name = strings+QCC_MergeValidateString(in->s_name);
functions[numfunctions].parentscope = NULL;
functions[numfunctions].type = NULL;
functions[numfunctions].def = NULL;
functions[numfunctions].firstlocal = NULL;
functions[numfunctions].privatelocals = true;
functions[numfunctions].merged = in;
numfunctions++;
in++;
}
}
static void QCC_MergeStatements16(dstatement16_t *in, unsigned int num)
{
QCC_statement_t *out = statements;
numstatements = num;
for (; num --> 0; out++, in++)
{
out->op = in->op;
out->a.sym = NULL;
out->a.cast = NULL;
out->a.ofs = in->a;
out->b.sym = NULL;
out->b.cast = NULL;
out->b.ofs = in->b;
out->c.sym = NULL;
out->c.cast = NULL;
out->c.ofs = in->c;
out->linenum = 0;
if (in->op < OP_NUMREALOPS)
{
if (!pr_opcodes[in->op].type_a)
out->a.ofs = (short)in->a;
if (!pr_opcodes[in->op].type_b)
out->b.ofs = (short)in->b;
if (!pr_opcodes[in->op].type_c)
out->c.ofs = (short)in->c;
}
}
out->op = OP_DONE;
out->a.ofs = 0;
out->b.ofs = 0;
out->c.ofs = 0;
out->linenum = 0;
numstatements++;
}
static etype_t QCC_MergeFindFieldType(unsigned int ofs, const char *fldname, ddef16_t *fields, size_t numfields)
{
size_t i;
etype_t best = ev_void;
for (i = 0; i < numfields; i++)
{
if (fields[i].ofs == ofs)
{ //sometimes we have field unions. go for the exact name match if we can so we don't get confused over vectors/floats. otherwise just go with the first (and hope they're correctly ordered)
char *name = strings+QCC_MergeValidateString(fields[i].s_name);
if (!strcmp(name, fldname))
return fields[i].type;
if (best == ev_void)
best = fields[i].type;
}
}
return best;
}
static void QCC_MergeUnstrip(dfunction_t *in, unsigned int num)
{
size_t i;
char *name;
QCC_def_t *def;
//functions may have been stripped. this results in an annoying lack of errors, and will likely confuse function wrapping...
//generate a new def for each function, if it doesn't already exist.
//these are probably going to be wasteful dupes, but they'll just get stripped again if they're still not used.
for (i = 0; i < num; i++)
{
if (!in[i].s_name)
continue;
name = strings+QCC_MergeValidateString(in[i].s_name);
def = QCC_PR_GetDef(NULL, name, NULL, false, 0, GDF_BASICTYPE);
if (!def)
{
def = QCC_PR_GetDef(type_function, name, NULL, true, 0, GDF_BASICTYPE);
def->symboldata[0].function = i;
def->initialized = true;
def->referenced = true;
def->assumedtype = true;
}
QCC_FreeDef(def);
}
}
QCC_type_t *QCC_PR_FieldType (QCC_type_t *pointsto);
static void QCC_MergeGlobalDefs16(ddef16_t *in, size_t num, void *values, size_t defscount, ddef16_t *fields, size_t numfields)
{
QCC_def_t *root, *def;
QCC_type_t *type;
etype_t evt;
char *name;
unsigned int flags;
pbool referrable;
numpr_globals = 0; //that root object will replace the normal reserved globals.
root = QCC_PR_GetDef(type_void, "", NULL, true, 0, GDF_USED);
root->symboldata = values;
root->symbolsize = defscount;
for (; num --> 0; in++)
{
name = strings+QCC_MergeValidateString(in->s_name);
flags = GDF_USED;
if (in->type & DEF_SAVEGLOBAL)
flags |= GDF_SAVED;
evt = in->type&~DEF_SAVEGLOBAL;
if (evt == ev_field)
evt = QCC_MergeFindFieldType(root->symboldata[in->ofs]._int, name, fields, numfields);
switch(evt)
{
case ev_void:
type = type_void;
break;
case ev_vector:
type = type_vector;
break;
case ev_float:
type = type_float;
break;
case ev_string:
type = type_string;
break;
case ev_entity:
type = type_entity;
break;
case ev_integer:
type = type_integer;
break;
case ev_function:
type = type_function;
break;
default:
type = type_variant;
break;
}
if ((in->type&~DEF_SAVEGLOBAL) == ev_field)
{
type = QCC_PR_FieldType(type);
flags |= GDF_CONST;
}
referrable = true; //fixme: disable if this appears to be within a function's local storage
def = QCC_PR_DummyDef(type, name, NULL, 0, root, in->ofs, referrable, flags);
def->initialized = 1;
def->referenced = true;
def->assumedtype = true;
if (evt == ev_vector)
{
int j = 3;
if ((in->type&~DEF_SAVEGLOBAL) == ev_field)
{
for (j = 0; j < 3; j++)
{
if (in[j+1].ofs == in->ofs+j && (in[j+1].type&~DEF_SAVEGLOBAL) == ev_field && QCC_MergeFindFieldType(root->symboldata[in[j+1].ofs]._int, strings+QCC_MergeValidateString(in[j+1].s_name), fields, numfields) == ev_float)
continue;
break;
}
}
else
{
for (j = 0; j < 3; j++)
{
if (in[j+1].ofs == in->ofs+j && (in[j+1].type&~DEF_SAVEGLOBAL) == ev_float)
continue;
break;
}
}
in += j;
num -= j;
}
}
QCC_FreeDef(root);
}
static unsigned char *PDECL QCC_LoadFileHunkAlloc(void *ctx, size_t size)
{
return (unsigned char*)qccHunkAlloc(size+1);
}
/*load a progs into the current compile state.*/
void QCC_ImportProgs(const char *filename)
{
size_t flen;
dprograms_t *prog;
//these keywords are implicitly enabled by #merge
keyword_weak = true;
keyword_wrap = true;
// if (strofs != 0) //could be fixed with relocs
// QCC_Error(ERR_BADEXTENSION, "#merge used too late. It must be used before any other definitions.");
if (numstatements != 1) //should be easy to deal with.
QCC_Error(ERR_BADEXTENSION, "#merge used too late. It must be used before any other definitions.");
if (numfunctions != 1) //could be fixed with relocs
QCC_Error(ERR_BADEXTENSION, "#merge used too late. It must be used before any other definitions.");
if (numglobaldefs != 1) //could be fixed by inserting it properly. any already-defined defs must have their parentdef changed to union them with imported ones.
QCC_Error(ERR_BADEXTENSION, "#merge used too late. It must be used before any other definitions (globals).");
if (numfielddefs != 1) //could be fixed with relocs
QCC_Error(ERR_BADEXTENSION, "#merge used too late. It must be used before any other definitions (fields).");
if (numpr_globals != RESERVED_OFS) //not normally changed until after compiling
QCC_Error(ERR_BADEXTENSION, "#merge used too late. It must be used before any other definitions (regs).");
externs->Printf ("\nnote: The #merge feature is still experimental\n\n");
//FIXME: find overlapped locals. strip them. merge with new ones.
//FIXME: find temps. strip them. you get the idea.
//FIXME: find immediates. set up hash tables for them for reuse. HAH!
prog = externs->ReadFile(filename, QCC_LoadFileHunkAlloc, NULL, &flen, false);
if (!prog)
{
QCC_Error(ERR_COULDNTOPENFILE, "Couldn't open file %s", filename);
return;
}
if (prog->version == 7 && prog->secondaryversion == PROG_SECONDARYVERSION16 && !prog->blockscompressed && !prog->numtypes)
;
else if (prog->version == 7 && prog->secondaryversion == PROG_SECONDARYVERSION32 && !prog->blockscompressed && !prog->numtypes)
;
else if (prog->version != 6)
{
QCC_Error(ERR_COULDNTOPENFILE, "Unsupported version: %s", filename);
return;
}
QCC_MergeStrings(((char*)prog+prog->ofs_strings), prog->numstrings);
QCC_MergeFunctions((dfunction_t*)((char*)prog+prog->ofs_functions), prog->numfunctions);
pr.size_fields = prog->entityfields;
if (prog->version == 7 && prog->secondaryversion == PROG_SECONDARYVERSION32)
{
// QCC_MergeStatements32((dstatement32_t*)((char*)prog+prog->ofs_statements), prog->numstatements);
// QCC_MergeGlobalDefs32((ddef32_t*)((char*)prog+prog->ofs_globaldefs), prog->numglobaldefs, ((char*)prog)+prog->ofs_globals, prog->numglobals, (ddef16_t*)((char*)prog+prog->ofs_fielddefs), prog->numfielddefs);
QCC_Error(ERR_COULDNTOPENFILE, "32bit versions not supported: %s", filename);
}
else
{
QCC_MergeStatements16((dstatement16_t*)((char*)prog+prog->ofs_statements), prog->numstatements);
QCC_MergeGlobalDefs16((ddef16_t*)((char*)prog+prog->ofs_globaldefs), prog->numglobaldefs, ((char*)prog)+prog->ofs_globals, prog->numglobals, (ddef16_t*)((char*)prog+prog->ofs_fielddefs), prog->numfielddefs);
}
QCC_MergeUnstrip((dfunction_t*)((char*)prog+prog->ofs_functions), prog->numfunctions);
}
/*
===============
PR_String
Returns a string suitable for printing (no newlines, max 60 chars length)
===============
*/
static char *QCC_PR_String (char *string)
{
static char buf[80];
char *s;
s = buf;
*s++ = '"';
while (string && *string)
{
if (s == buf + sizeof(buf) - 2)
break;
if (*string == '\n')
{
*s++ = '\\';
*s++ = 'n';
}
else if (*string == '"')
{
*s++ = '\\';
*s++ = '"';
}
else
*s++ = *string;
string++;
if (s - buf > 60)
{
*s++ = '.';
*s++ = '.';
*s++ = '.';
break;
}
}
*s++ = '"';
*s++ = 0;
return buf;
}
static QCC_def_t *QCC_PR_DefForFieldOfs (gofs_t ofs)
{
QCC_def_t *d;
for (d=pr.def_head.next ; d ; d=d->next)
{
if (d->type->type != ev_field)
continue;
if (*((unsigned int *)&qcc_pr_globals[d->ofs]) == ofs)
return d;
}
QCC_Error (ERR_NOTDEFINED, "PR_DefForFieldOfs: couldn't find %i",ofs);
return NULL;
}
/*
============
PR_ValueString
Returns a string describing *data in a type specific manner
=============
*/
char *QCC_PR_ValueString (etype_t type, void *val)
{
static char line[256];
QCC_def_t *def;
QCC_function_t *f;
switch (type)
{
case ev_string:
QC_snprintfz (line, sizeof(line), "%s", QCC_PR_String(strings + *(int *)val));
break;
case ev_entity:
QC_snprintfz (line, sizeof(line), "entity %i", *(int *)val);
break;
case ev_function:
f = functions + *(int *)val;
if (!f)
QC_snprintfz (line, sizeof(line), "undefined function");
else
QC_snprintfz (line, sizeof(line), "%s()", f->name);
break;
case ev_field:
def = QCC_PR_DefForFieldOfs ( *(int *)val );
QC_snprintfz (line, sizeof(line), ".%s", def->name);
break;
case ev_void:
QC_snprintfz (line, sizeof(line), "void");
break;
case ev_float:
QC_snprintfz (line, sizeof(line), "%5.1f", *(float *)val);
break;
case ev_integer:
QC_snprintfz (line, sizeof(line), "%i", *(int *)val);
break;
case ev_vector:
QC_snprintfz (line, sizeof(line), "'%5.1f %5.1f %5.1f'", ((float *)val)[0], ((float *)val)[1], ((float *)val)[2]);
break;
case ev_pointer:
QC_snprintfz (line, sizeof(line), "pointer");
break;
default:
QC_snprintfz (line, sizeof(line), "bad type %i", type);
break;
}
return line;
}
/*
============
PR_GlobalString
Returns a string with a description and the contents of a global,
padded to 20 field width
============
*/
/*char *QCC_PR_GlobalStringNoContents (gofs_t ofs)
{
int i;
QCC_def_t *def;
void *val;
static char line[128];
val = (void *)&qcc_pr_globals[ofs];
def = pr_global_defs[ofs];
if (!def)
// Error ("PR_GlobalString: no def for %i", ofs);
QC_snprintfz (line, sizeof(line), "%i(?""?""?)", ofs);
else
QC_snprintfz (line, sizeof(line), "%i(%s)", ofs, def->name);
i = strlen(line);
for ( ; i<16 ; i++)
Q_strlcat (line," ", sizeof(line));
Q_strlcat (line," ", sizeof(line));
return line;
}
char *QCC_PR_GlobalString (gofs_t ofs)
{
char *s;
int i;
QCC_def_t *def;
void *val;
static char line[128];
val = (void *)&qcc_pr_globals[ofs];
def = pr_global_defs[ofs];
if (!def)
return QCC_PR_GlobalStringNoContents(ofs);
if (def->initialized && def->type->type != ev_function)
{
s = QCC_PR_ValueString (def->type->type, &qcc_pr_globals[ofs]);
QC_snprintfz (line, sizeof(line), "%i(%s)", ofs, s);
}
else
QC_snprintfz (line, sizeof(line), "%i(%s)", ofs, def->name);
i = strlen(line);
for ( ; i<16 ; i++)
strcat (line," ");
strcat (line," ");
return line;
}*/
/*
============
PR_PrintOfs
============
*/
/*void QCC_PR_PrintOfs (gofs_t ofs)
{
externs->Printf ("%s\n",QCC_PR_GlobalString(ofs));
}*/
/*
=================
PR_PrintStatement
=================
*/
/*void QCC_PR_PrintStatement (QCC_dstatement_t *s)
{
int i;
externs->Printf ("%4i : %4i : %s ", (int)(s - statements), statement_linenums[s-statements], pr_opcodes[s->op].opname);
i = strlen(pr_opcodes[s->op].opname);
for ( ; i<10 ; i++)
externs->Printf (" ");
if (s->op == OP_IF || s->op == OP_IFNOT)
externs->Printf ("%sbranch %i",QCC_PR_GlobalString(s->a),s->b);
else if (s->op == OP_GOTO)
{
externs->Printf ("branch %i",s->a);
}
else if ( (unsigned)(s->op - OP_STORE_F) < 6)
{
externs->Printf ("%s",QCC_PR_GlobalString(s->a));
externs->Printf ("%s", QCC_PR_GlobalStringNoContents(s->b));
}
else
{
if (s->a)
externs->Printf ("%s",QCC_PR_GlobalString(s->a));
if (s->b)
externs->Printf ("%s",QCC_PR_GlobalString(s->b));
if (s->c)
externs->Printf ("%s", QCC_PR_GlobalStringNoContents(s->c));
}
externs->Printf ("\n");
}*/
/*
============
PR_PrintDefs
============
*/
/*void QCC_PR_PrintDefs (void)
{
QCC_def_t *d;
for (d=pr.def_head.next ; d ; d=d->next)
QCC_PR_PrintOfs (d->ofs);
}*/
QCC_type_t *QCC_PR_NewType (const char *name, int basictype, pbool typedefed)
{
if (numtypeinfos>= maxtypeinfos)
QCC_Error(ERR_TOOMANYTYPES, "Too many types");
memset(&qcc_typeinfo[numtypeinfos], 0, sizeof(QCC_type_t));
qcc_typeinfo[numtypeinfos].type = basictype;
qcc_typeinfo[numtypeinfos].name = name;
qcc_typeinfo[numtypeinfos].num_parms = 0;
qcc_typeinfo[numtypeinfos].params = NULL;
qcc_typeinfo[numtypeinfos].size = type_size[basictype];
qcc_typeinfo[numtypeinfos].typedefed = typedefed;
qcc_typeinfo[numtypeinfos].filen = s_filen;
qcc_typeinfo[numtypeinfos].line = pr_source_line;
if (typedefed)
pHash_Add(&typedeftable, name, &qcc_typeinfo[numtypeinfos], qccHunkAlloc(sizeof(bucket_t)));
numtypeinfos++;
return &qcc_typeinfo[numtypeinfos-1];
}
/*
==============
PR_BeginCompilation
called before compiling a batch of files, clears the pr struct
==============
*/
static void QCC_PR_BeginCompilation (void *memory, int memsize)
{
extern int recursivefunctiontype;
int i;
char name[16];
pr.memory = memory;
pr.max_memory = memsize;
pr.def_tail = &pr.def_head;
pr.local_tail = &pr.local_head;
QCC_PR_ResetErrorScope();
pr_scope = NULL;
/* numpr_globals = RESERVED_OFS;
for (i=0 ; i<RESERVED_OFS ; i++)
pr_global_defs[i] = &def_void;
*/
type_void = QCC_PR_NewType("void", ev_void, true);
type_string = QCC_PR_NewType("string", ev_string, true);
type_float = QCC_PR_NewType("float", ev_float, true);
type_double = QCC_PR_NewType("__double", ev_double, true);
type_vector = QCC_PR_NewType("vector", ev_vector, true);
type_entity = QCC_PR_NewType("entity", ev_entity, true);
type_field = QCC_PR_NewType("__field", ev_field, false);
type_field->aux_type = type_void;
type_function = QCC_PR_NewType("__function", ev_function, false);
type_function->aux_type = type_void;
type_pointer = QCC_PR_NewType("__pointer", ev_pointer, false);
type_integer = QCC_PR_NewType("__int", ev_integer, true);
type_uint = QCC_PR_NewType("__uint", ev_uint, true);
type_int64 = QCC_PR_NewType("__int64", ev_int64, true);
type_uint64 = QCC_PR_NewType("__uint64", ev_uint64, true);
type_variant = QCC_PR_NewType("__variant", ev_variant, true);
type_invalid = QCC_PR_NewType("invalid", ev_void, false);
type_floatfield = QCC_PR_NewType("__fieldfloat", ev_field, false);
type_floatfield->aux_type = type_float;
type_pointer->aux_type = QCC_PR_NewType("__pointeraux", ev_float, false);
type_intpointer = QCC_PR_NewType("__intpointer", ev_pointer, false);
type_intpointer->aux_type = type_integer;
type_floatpointer = QCC_PR_NewType("__floatpointer", ev_pointer, false);
type_floatpointer->aux_type = type_float;
type_floatfunction = QCC_PR_NewType("__floatfunction", ev_function, false);
type_floatfunction->aux_type = type_float;
type_bfloat = QCC_PR_NewType("__bfloat", ev_boolean, true);
type_bfloat->parentclass = type_float;
type_bint = QCC_PR_NewType("__bint", ev_boolean, true);
type_bint->parentclass = type_uint;
//type_field->aux_type = type_float;
// QCC_PR_NewType("_Bool", ev_boolean, true);
// QCC_PR_NewType("bool", ev_boolean, true);
// QCC_PR_NewType("__int", ev_integer, keyword_integer?true:false);
QCC_PR_NewType("variant", ev_variant, true);
if (output_parms)
{ //this tends to confuse the brains out of decompilers. :)
numpr_globals = 1;
QCC_PR_GetDef(type_vector, "RETURN", NULL, true, 0, false)->referenced=true;
for (i = 0; i < MAX_PARMS; i++)
{
QC_snprintfz (name, sizeof(name), "PARM%i", i);
QCC_PR_GetDef(type_vector, name, NULL, true, 0, false)->referenced=true;
}
}
else
{
numpr_globals = RESERVED_OFS;
// for (i=0 ; i<RESERVED_OFS ; i++)
// pr_global_defs[i] = NULL;
}
// link the function type in so state forward declarations match proper type
pr.types = NULL;
// type_function->next = NULL;
pr_error_count = 0;
pr_warning_count = 0;
recursivefunctiontype = 0;
QCC_PrioritiseOpcodes();
}
static void QCC_PR_FinishFieldDef(QCC_def_t *d)
{
int i;
if (d->symboldata)
return; //nothing to finish
d->symbolsize = (d->arraysize?d->arraysize:1) * d->type->size;
if (d->symbolheader != d)
{
QCC_PR_FinishFieldDef(d->symbolheader);
d->symboldata = d->symbolheader->symboldata + d->ofs;
}
else
{
d->symboldata = qccHunkAlloc (d->symbolsize * sizeof(float));
for (i = 0; i < d->symbolsize; i++)
d->symboldata[i]._int = pr.size_fields++;
}
}
/*
==============
PR_FinishCompilation
called after all files are compiled to check for errors
Returns false if errors were detected.
==============
*/
static int QCC_PR_FinishCompilation (void)
{
QCC_def_t *d;
QCC_type_t *t;
int errors;
pbool externokay = false;
errors = false;
if (pr_error_count)
return false;
if (qcc_targetformat == QCF_FTE || qcc_targetformat == QCF_FTEDEBUG || qcc_targetformat == QCF_FTEH2)
externokay = true;
// check to make sure all functions prototyped have code
for (d=pr.def_head.next ; d ; d=d->next)
{
if (d->type->type == ev_field && !d->symboldata)
QCC_PR_FinishFieldDef(d);
if (d->type->type == ev_function && d->constant && d->symbolheader == d)// function parms are ok
{
if (d->isextern)
{
if (!externokay)
{
QCC_PR_Warning(ERR_NOFUNC, d->filen, d->s_line, "extern is not supported with this target format",d->name);
QCC_PR_ParsePrintDef(ERR_NOFUNC, d);
errors = true;
}
bodylessfuncs = true;
}
if (!d->initialized)
{
if (!strncmp(d->name, "ArrayGet*", 9))
{
QCC_PR_EmitArrayGetFunction(d, d->generatedfor, d->name+9);
pr_scope = NULL;
continue;
}
if (!strncmp(d->name, "ArraySet*", 9))
{
QCC_PR_EmitArraySetFunction(d, d->generatedfor, d->name+9);
pr_scope = NULL;
continue;
}
if (!strncmp(d->name, "spawnfunc_", 10))
{
//not all of these will have a class defined, as some will be regular spawn functions, so don't error on that
t = QCC_TypeForName(d->name+10);
if (t && t->type == ev_entity)
{
QCC_PR_EmitClassFromFunction(d, t);
pr_scope = NULL;
continue;
}
}
if (d->unused)
{
d->initialized = 1;
continue;
}
QCC_PR_Warning(ERR_NOFUNC, d->filen, d->s_line, "function %s has no body",d->name);
QCC_PR_ParsePrintDef(ERR_NOFUNC, d);
bodylessfuncs = true;
errors = true;
}
}
}
pr_scope = NULL;
return !errors;
}
//=============================================================================
// FIXME: byte swap?
// this is a 16 bit, non-reflected CRC using the polynomial 0x1021
// and the initial and final xor values shown below... in other words, the
// CCITT standard CRC used by XMODEM
#define CRC_INIT_VALUE 0xffff
#define CRC_XOR_VALUE 0x0000
static unsigned short QCC_crctable[256] =
{
0x0000, 0x1021, 0x2042, 0x3063, 0x4084, 0x50a5, 0x60c6, 0x70e7,
0x8108, 0x9129, 0xa14a, 0xb16b, 0xc18c, 0xd1ad, 0xe1ce, 0xf1ef,
0x1231, 0x0210, 0x3273, 0x2252, 0x52b5, 0x4294, 0x72f7, 0x62d6,
0x9339, 0x8318, 0xb37b, 0xa35a, 0xd3bd, 0xc39c, 0xf3ff, 0xe3de,
0x2462, 0x3443, 0x0420, 0x1401, 0x64e6, 0x74c7, 0x44a4, 0x5485,
0xa56a, 0xb54b, 0x8528, 0x9509, 0xe5ee, 0xf5cf, 0xc5ac, 0xd58d,
0x3653, 0x2672, 0x1611, 0x0630, 0x76d7, 0x66f6, 0x5695, 0x46b4,
0xb75b, 0xa77a, 0x9719, 0x8738, 0xf7df, 0xe7fe, 0xd79d, 0xc7bc,
0x48c4, 0x58e5, 0x6886, 0x78a7, 0x0840, 0x1861, 0x2802, 0x3823,
0xc9cc, 0xd9ed, 0xe98e, 0xf9af, 0x8948, 0x9969, 0xa90a, 0xb92b,
0x5af5, 0x4ad4, 0x7ab7, 0x6a96, 0x1a71, 0x0a50, 0x3a33, 0x2a12,
0xdbfd, 0xcbdc, 0xfbbf, 0xeb9e, 0x9b79, 0x8b58, 0xbb3b, 0xab1a,
0x6ca6, 0x7c87, 0x4ce4, 0x5cc5, 0x2c22, 0x3c03, 0x0c60, 0x1c41,
0xedae, 0xfd8f, 0xcdec, 0xddcd, 0xad2a, 0xbd0b, 0x8d68, 0x9d49,
0x7e97, 0x6eb6, 0x5ed5, 0x4ef4, 0x3e13, 0x2e32, 0x1e51, 0x0e70,
0xff9f, 0xefbe, 0xdfdd, 0xcffc, 0xbf1b, 0xaf3a, 0x9f59, 0x8f78,
0x9188, 0x81a9, 0xb1ca, 0xa1eb, 0xd10c, 0xc12d, 0xf14e, 0xe16f,
0x1080, 0x00a1, 0x30c2, 0x20e3, 0x5004, 0x4025, 0x7046, 0x6067,
0x83b9, 0x9398, 0xa3fb, 0xb3da, 0xc33d, 0xd31c, 0xe37f, 0xf35e,
0x02b1, 0x1290, 0x22f3, 0x32d2, 0x4235, 0x5214, 0x6277, 0x7256,
0xb5ea, 0xa5cb, 0x95a8, 0x8589, 0xf56e, 0xe54f, 0xd52c, 0xc50d,
0x34e2, 0x24c3, 0x14a0, 0x0481, 0x7466, 0x6447, 0x5424, 0x4405,
0xa7db, 0xb7fa, 0x8799, 0x97b8, 0xe75f, 0xf77e, 0xc71d, 0xd73c,
0x26d3, 0x36f2, 0x0691, 0x16b0, 0x6657, 0x7676, 0x4615, 0x5634,
0xd94c, 0xc96d, 0xf90e, 0xe92f, 0x99c8, 0x89e9, 0xb98a, 0xa9ab,
0x5844, 0x4865, 0x7806, 0x6827, 0x18c0, 0x08e1, 0x3882, 0x28a3,
0xcb7d, 0xdb5c, 0xeb3f, 0xfb1e, 0x8bf9, 0x9bd8, 0xabbb, 0xbb9a,
0x4a75, 0x5a54, 0x6a37, 0x7a16, 0x0af1, 0x1ad0, 0x2ab3, 0x3a92,
0xfd2e, 0xed0f, 0xdd6c, 0xcd4d, 0xbdaa, 0xad8b, 0x9de8, 0x8dc9,
0x7c26, 0x6c07, 0x5c64, 0x4c45, 0x3ca2, 0x2c83, 0x1ce0, 0x0cc1,
0xef1f, 0xff3e, 0xcf5d, 0xdf7c, 0xaf9b, 0xbfba, 0x8fd9, 0x9ff8,
0x6e17, 0x7e36, 0x4e55, 0x5e74, 0x2e93, 0x3eb2, 0x0ed1, 0x1ef0
};
static void QCC_CRC_Init(unsigned short *crcvalue)
{
*crcvalue = CRC_INIT_VALUE;
}
static void QCC_CRC_ProcessByte(unsigned short *crcvalue, pbyte data)
{
*crcvalue = ((*crcvalue << 8) ^ QCC_crctable[(*crcvalue >> 8) ^ data]) & 0xffff;
}
/*static unsigned short QCC_CRC_Value(unsigned short crcvalue)
{
return crcvalue ^ CRC_XOR_VALUE;
}*/
//=============================================================================
/*
============
PR_WriteProgdefs
Writes the global and entity structures out
Returns a crc of the header, to be stored in the progs file for comparison
at load time.
============
*/
/*
char *Sva(char *msg, ...)
{
va_list l;
static char buf[1024];
va_start(l, msg);
QC_vsnprintf (buf,sizeof(buf)-1, msg, l);
va_end(l);
return buf;
}
*/
#define PROGDEFS_MAX_SIZE 16384
//write (to file buf) and add to the crc
static void Add_WithCRC(char *p, unsigned short *crc, char *file)
{
char *s;
int i = strlen(file);
if (i + strlen(p)+1 >= PROGDEFS_MAX_SIZE)
return;
for(s=p;*s;s++,i++)
{
QCC_CRC_ProcessByte(crc, *s);
file[i] = *s;
}
file[i]='\0';
}
#define ADD_CRC(p) Add_WithCRC(p, &crc, file)
//#define ADD(p) {char *s;int i = strlen(p);for(s=p;*s;s++,i++){QCC_CRC_ProcessByte(&crc, *s);file[i] = *s;}file[i]='\0';}
static void Add_CrcOnly(char *p, unsigned short *crc, char *file)
{
char *s;
for(s=p;*s;s++)
QCC_CRC_ProcessByte(crc, *s);
}
#define EAT_CRC(p) Add_CrcOnly(p, &crc, file)
static void QCC_PR_CRCMessages(unsigned short crc)
{
switch (crc)
{
case 12923: //#pragma sourcefile usage
break;
case 54730:
if (verbose >= VERBOSE_STANDARD)
externs->Printf("Recognised progs as QuakeWorld\n");
break;
case 5927:
if (verbose >= VERBOSE_STANDARD)
externs->Printf("Recognised progs as NetQuake server gamecode\n");
break;
case 26940:
if (verbose >= VERBOSE_STANDARD)
externs->Printf("Recognised progs as Quake pre-release...\n");
break;
case 38488:
if (verbose >= VERBOSE_STANDARD)
externs->Printf("Recognised progs as original Hexen2\n");
break;
case 26905:
if (verbose >= VERBOSE_STANDARD)
externs->Printf("Recognised progs as Hexen2 Mission Pack\n");
break;
case 14046:
if (verbose >= VERBOSE_STANDARD)
externs->Printf("Recognised progs as Hexen2 (demo)\n");
break;
case 22390: //EXT_CSQC_1
if (verbose >= VERBOSE_STANDARD)
externs->Printf("Recognised progs as an EXT_CSQC_1 module\n");
break;
case 17105:
case 32199: //outdated ext_csqc
QCC_PR_Warning(WARN_SYSTEMCRC2, NULL, 0, "Recognised progs as outdated CSQC module");
break;
case 52195: //this is what DP requires. don't print it as the warning that it is as that would royally piss off xonotic and their use of -Werror.
if (verbose >= VERBOSE_PROGRESS)
externs->Printf("Recognised progs as DP-specific CSQC module\n");
break;
case 10020:
if (verbose >= VERBOSE_STANDARD)
externs->Printf("Recognised progs as a MenuQC module\n");
break;
case 32401:
QCC_PR_Warning(WARN_SYSTEMCRC, NULL, 0, "please update your tenebrae system defs.");
break;
default:
QCC_PR_Warning(WARN_SYSTEMCRC, NULL, 0, "system defs not recognised from quake nor clones, probably buggy (sys)defs.qc");
break;
}
}
static unsigned short QCC_PR_WriteProgdefs (char *filename)
{
#define ADD_ONLY(p) QC_strlcat(file, p, sizeof(file)) //no crc (later changes)
char file[PROGDEFS_MAX_SIZE];
QCC_def_t *d;
int f;
unsigned short crc;
// int c;
file[0] = '\0';
QCC_CRC_Init (&crc);
// print global vars until the first field is defined
ADD_CRC("\n/* ");
if (qcc_targetformat == QCF_HEXEN2 || qcc_targetformat == QCF_UHEXEN2 || qcc_targetformat == QCF_FTEH2)
EAT_CRC("generated by hcc, do not modify");
else
EAT_CRC("file generated by qcc, do not modify");
ADD_ONLY("File generated by FTEQCC, relevent for engine modding only, the generated crc must be the same as your engine expects.");
ADD_CRC(" */\n\ntypedef struct");
ADD_ONLY(" globalvars_s");
ADD_CRC(qcva("\n{"));
ADD_ONLY("\tint pad;\n"
"\tint ofs_return[3];\n" //makes it easier with the get globals func
"\tint ofs_parm0[3];\n"
"\tint ofs_parm1[3];\n"
"\tint ofs_parm2[3];\n"
"\tint ofs_parm3[3];\n"
"\tint ofs_parm4[3];\n"
"\tint ofs_parm5[3];\n"
"\tint ofs_parm6[3];\n"
"\tint ofs_parm7[3];\n");
EAT_CRC(qcva("\tint\tpad[%i];\n", RESERVED_OFS));
for (d=pr.def_head.next ; d ; d=d->next)
{
if (!strcmp (d->name, "end_sys_globals"))
break;
if (!*d->name)
continue;
// if (d->symbolheader->ofs<RESERVED_OFS)
// continue;
switch (d->type->type)
{
case ev_float:
ADD_CRC(qcva("\tfloat\t%s;\n",d->name));
break;
case ev_vector:
ADD_CRC(qcva("\tvec3_t\t%s;\n",d->name));
if (d->deftail)
d=d->deftail; // skip the elements
break;
case ev_string:
ADD_CRC(qcva("\tstring_t\t%s;\n",d->name));
break;
case ev_function:
ADD_CRC(qcva("\tfunc_t\t%s;\n",d->name));
break;
case ev_entity:
ADD_CRC(qcva("\tint\t%s;\n",d->name));
break;
case ev_integer:
ADD_CRC(qcva("\tint\t%s;\n",d->name));
break;
default:
ADD_CRC(qcva("\tint\t%s;\n",d->name));
break;
}
}
ADD_CRC("} globalvars_t;\n\n");
// print all fields
ADD_CRC("typedef struct");
ADD_ONLY(" entvars_s");
ADD_CRC("\n{\n");
for (d=pr.def_head.next ; d ; d=d->next)
{
if (!strcmp (d->name, "end_sys_fields"))
break;
if (d->type->type != ev_field)
continue;
switch (d->type->aux_type->type)
{
case ev_float:
ADD_CRC(qcva("\tfloat\t%s;\n",d->name));
break;
case ev_vector:
ADD_CRC(qcva("\tvec3_t\t%s;\n",d->name));
if (d->deftail)
d=d->deftail; // skip the elements
break;
case ev_string:
ADD_CRC(qcva("\tstring_t\t%s;\n",d->name));
break;
case ev_function:
ADD_CRC(qcva("\tfunc_t\t%s;\n",d->name));
break;
case ev_entity:
ADD_CRC(qcva("\tint\t%s;\n",d->name));
break;
case ev_integer:
ADD_CRC(qcva("\tint\t%s;\n",d->name));
break;
default:
ADD_CRC(qcva("\tint\t%s;\n",d->name));
break;
}
}
ADD_CRC("} entvars_t;\n\n");
/*
///temp
ADD_ONLY("//with this the crc isn't needed for fields.\n#ifdef FIELDSSTRUCT\nstruct fieldvars_s {\n\tint ofs;\n\tint type;\n\tchar *name;\n} fieldvars[] = {\n");
f=0;
for (d=pr.def_head.next ; d ; d=d->next)
{
if (!strcmp (d->name, "end_sys_fields"))
break;
if (d->type->type != ev_field)
continue;
if (f)
ADD_ONLY(",\n");
ADD_ONLY(qcva("\t{%i,\t%i,\t\"%s\"}",G_INT(d->ofs), d->type->aux_type->type, d->name));
f = 1;
}
ADD_ONLY("\n};\n#endif\n\n");
//end temp
*/
ADD_ONLY(qcva("#define PROGHEADER_CRC %i\n", crc));
if (QCC_CheckParm("-progdefs"))
{
externs->Printf ("writing %s\n", filename);
f = SafeOpenWrite(filename, 16384);
SafeWrite(f, file, strlen(file));
SafeClose(f);
}
if (ForcedCRC)
crc = ForcedCRC;
return crc;
}
/*void QCC_PrintFunction (char *name)
{
int i;
QCC_dstatement_t *ds;
QCC_dfunction_t *df;
for (i=0 ; i<numfunctions ; i++)
if (!strcmp (name, strings + functions[i].s_name))
break;
if (i==numfunctions)
QCC_Error (ERR_NOFUNC, "No function named \"%s\"", name);
df = functions + i;
externs->Printf ("Statements for function %s:\n", name);
ds = statements + df->first_statement;
while (1)
{
QCC_PR_PrintStatement (ds);
if (!ds->op)
break;
ds++;
}
}*/
/*
void QCC_PrintOfs(unsigned int ofs)
{
int i;
bool printfunc;
QCC_dstatement_t *ds;
QCC_dfunction_t *df;
for (i=0 ; i<numfunctions ; i++)
{
df = functions + i;
ds = statements + df->first_statement;
printfunc = false;
while (1)
{
if (!ds->op)
break;
if (ds->a == ofs || ds->b == ofs || ds->c == ofs)
{
QCC_PR_PrintStatement (ds);
printfunc = true;
}
ds++;
}
if (printfunc)
{
QCC_PrintFunction(strings + functions[i].s_name);
externs->Printf(" \n \n");
}
}
}
*/
/*
==============================================================================
DIRECTORY COPYING / PACKFILE CREATION
==============================================================================
*/
static packfile_t pfiles[4096], *pf;
static int packhandle;
static int packbytes;
/*
===========
PackFile
Copy a file into the pak file
===========
*/
static void QCC_PackFile (char *src, char *name)
{
size_t remaining;
#if 1
char *f;
#else
int in;
int count;
char buf[4096];
#endif
if ( (pbyte *)pf - (pbyte *)pfiles > sizeof(pfiles) )
QCC_Error (ERR_TOOMANYPAKFILES, "Too many files in pak file");
#if 1
f = FS_ReadToMem(src, &remaining);
if (!f)
{
externs->Printf ("%64s : %7s\n", name, "");
// QCC_Error("Failed to open file %s", src);
return;
}
pf->filepos = PRLittleLong (SafeSeek (packhandle, 0, SEEK_CUR));
pf->filelen = PRLittleLong (remaining);
strcpy (pf->name, name);
externs->Printf ("%64s : %7u\n", pf->name, (unsigned int)remaining);
packbytes += remaining;
SafeWrite (packhandle, f, remaining);
FS_CloseFromMem(f);
#else
in = SafeOpenRead (src);
remaining = filelength (in);
pf->filepos = PRLittleLong (lseek (packhandle, 0, SEEK_CUR));
pf->filelen = PRLittleLong (remaining);
strcpy (pf->name, name);
externs->Printf ("%64s : %7u\n", pf->name, (unsigned int)remaining);
packbytes += remaining;
while (remaining)
{
if (remaining < sizeof(buf))
count = remaining;
else
count = sizeof(buf);
SafeRead (in, buf, count);
SafeWrite (packhandle, buf, count);
remaining -= count;
}
close (in);
#endif
pf++;
}
/*
===========
CopyFile
Copies a file, creating any directories needed
===========
*/
static void QCC_CopyFile (char *src, char *dest)
{
/*
int in, out;
int remaining, count;
char buf[4096];
print ("%s to %s\n", src, dest);
in = SafeOpenRead (src);
remaining = filelength (in);
QCC_CreatePath (dest);
out = SafeOpenWrite (dest, remaining+10);
while (remaining)
{
if (remaining < sizeof(buf))
count = remaining;
else
count = sizeof(buf);
SafeRead (in, buf, count);
SafeWrite (out, buf, count);
remaining -= count;
}
close (in);
SafeClose (out);
*/
}
/*
===========
CopyFiles
===========
*/
static void _QCC_CopyFiles (int blocknum, int copytype, char *srcdir, char *destdir)
{
int i;
int dirlen;
unsigned short crc;
packheader_t header;
char name[1024];
char srcfile[1024], destfile[1024];
packbytes = 0;
if (copytype == 2)
{
pf = pfiles;
packhandle = SafeOpenWrite (destdir, 1024*1024);
SafeWrite (packhandle, &header, sizeof(header));
}
for (i=0 ; i<numsounds ; i++)
{
if (precache_sound[i].block != blocknum)
continue;
sprintf (srcfile,"%s%s",srcdir, precache_sound[i].name);
sprintf (destfile,"%s%s",destdir, precache_sound[i].name);
if (copytype == 1)
QCC_CopyFile (srcfile, destfile);
else
QCC_PackFile (srcfile, precache_sound[i].name);
}
for (i=0 ; i<nummodels ; i++)
{
if (precache_model[i].block != blocknum)
continue;
sprintf (srcfile,"%s%s",srcdir, precache_model[i].name);
sprintf (destfile,"%s%s",destdir, precache_model[i].name);
if (copytype == 1)
QCC_CopyFile (srcfile, destfile);
else
QCC_PackFile (srcfile, precache_model[i].name);
}
for (i=0 ; i<numtextures ; i++)
{
if (precache_texture[i].block != blocknum)
continue;
{
sprintf (name, "%s", precache_texture[i].name);
sprintf (srcfile,"%s%s",srcdir, name);
sprintf (destfile,"%s%s",destdir, name);
if (copytype == 1)
QCC_CopyFile (srcfile, destfile);
else
QCC_PackFile (srcfile, name);
}
{
sprintf (name, "%s.bmp", precache_texture[i].name);
sprintf (srcfile,"%s%s",srcdir, name);
sprintf (destfile,"%s%s",destdir, name);
if (copytype == 1)
QCC_CopyFile (srcfile, destfile);
else
QCC_PackFile (srcfile, name);
}
{
sprintf (name, "%s.tga", precache_texture[i].name);
sprintf (srcfile,"%s%s",srcdir, name);
sprintf (destfile,"%s%s",destdir, name);
if (copytype == 1)
QCC_CopyFile (srcfile, destfile);
else
QCC_PackFile (srcfile, name);
}
}
for (i=0 ; i<numfiles ; i++)
{
if (precache_file[i].block != blocknum)
continue;
sprintf (srcfile,"%s%s",srcdir, precache_file[i].name);
sprintf (destfile,"%s%s",destdir, precache_file[i].name);
if (copytype == 1)
QCC_CopyFile (srcfile, destfile);
else
QCC_PackFile (srcfile, precache_file[i].name);
}
if (copytype == 2)
{
header.id[0] = 'P';
header.id[1] = 'A';
header.id[2] = 'C';
header.id[3] = 'K';
dirlen = (pbyte *)pf - (pbyte *)pfiles;
header.dirofs = PRLittleLong(SafeSeek (packhandle, 0, SEEK_CUR));
header.dirlen = PRLittleLong(dirlen);
SafeWrite (packhandle, pfiles, dirlen);
SafeSeek (packhandle, 0, SEEK_SET);
SafeWrite (packhandle, &header, sizeof(header));
SafeClose (packhandle);
// do a crc of the file
QCC_CRC_Init (&crc);
for (i=0 ; i<dirlen ; i++)
QCC_CRC_ProcessByte (&crc, ((pbyte *)pfiles)[i]);
i = pf - pfiles;
externs->Printf ("%i files packed in %i bytes (%i crc)\n",i, packbytes, crc);
}
}
static void QCC_CopyFiles (void)
{
char *s;
char srcdir[1024], destdir[1024];
int p;
if (verbose)
{
if (numsounds > 0)
externs->Printf ("%3i unique precache_sounds\n", numsounds);
if (nummodels > 0)
externs->Printf ("%3i unique precache_models\n", nummodels);
if (numtextures > 0)
externs->Printf ("%3i unique precache_textures\n", numtextures);
if (numfiles > 0)
externs->Printf ("%3i unique precache_files\n", numfiles);
}
p = QCC_CheckParm ("-copy");
if (p && p < myargc-2)
{ // create a new directory tree
strcpy (srcdir, myargv[p+1]);
strcpy (destdir, myargv[p+2]);
if (srcdir[strlen(srcdir)-1] != '/')
strcat (srcdir, "/");
if (destdir[strlen(destdir)-1] != '/')
strcat (destdir, "/");
_QCC_CopyFiles(0, 1, srcdir, destdir);
return;
}
for ( p = 0; p < countof(QCC_Packname); p++)
{
s = QCC_Packname[p];
if (!*s)
continue;
strcpy(destdir, s);
strcpy(srcdir, "");
_QCC_CopyFiles(p+1, 2, srcdir, destdir);
}
return;
/*
blocknum = 1;
p = QCC_CheckParm ("-pak2");
if (p && p <myargc-2)
blocknum = 2;
else
p = QCC_CheckParm ("-pak");
if (p && p < myargc-2)
{ // create a pak file
strcpy (srcdir, myargv[p+1]);
strcpy (destdir, myargv[p+2]);
if (srcdir[strlen(srcdir)-1] != '/')
strcat (srcdir, "/");
DefaultExtension (destdir, ".pak");
copytype = 2;
_QCC_CopyFiles(blocknum, copytype, srcdir, destdir);
}
*/
}
//============================================================================
#ifdef _WIN32
#define WINDOWSARG(x) x
#else
#define WINDOWSARG(x) false
#endif
pbool QCC_RegisterSourceFile(const char *filename)
{
int i;
for (i = 0; i < numsourcefiles; i++)
{
if (!strcmp(sourcefileslist[i], filename))
return true;
}
if (numsourcefiles < MAXSOURCEFILESLIST)
{
strcpy(sourcefileslist[numsourcefiles++], filename);
return true;
}
return false;
}
static void QCC_PR_CommandLinePrecompilerOptions (void)
{
CompilerConstant_t *cnst;
int i, j, p;
const char *name, *val;
pbool werror = false;
qcc_nopragmaoptimise = false;
for (i = 1;i<myargc;i++)
{
if ( !strcmp(myargv[i], "-v") )
verbose++; //verbose
else if ( !strcmp(myargv[i], "-srcfile") )
{
if (++i == myargc)
break;
if (!QCC_RegisterSourceFile(myargv[i]))
QCC_PR_Warning(WARN_BADPARAMS, "cmdline", 0, "too many -srcfile arguments");
}
else if ( !strcmp(myargv[i], "-src") )
{
i++;
strcpy (qccmsourcedir, myargv[i]);
strcat (qccmsourcedir, "/");
}
else if ( !strcmp(myargv[i], "-o") )
{ //explicit output file
i++;
strcpy(destfile, myargv[i]);
if (!destfile_explicit)
verbose--;
destfile_explicit = true;
}
else if ( !strncmp(myargv[i], "-o", 2) )
{ //explicit output file
strcpy(destfile, myargv[i]+2);
if (!destfile_explicit)
verbose--;
destfile_explicit = true;
}
else if ( !strcmp(myargv[i], "-qc") )
QCC_PR_Warning(WARN_BADPARAMS, "cmdline", 0, "Argument %s is experimental", myargv[i]); //compile without linking. output cannot be read by engines.
else if ( !strcmp(myargv[i], "-E") )
QCC_PR_Warning(WARN_BADPARAMS, "cmdline", 0, "Argument %s is experimental", myargv[i]); //preprocess only
else if ( !strcmp(myargv[i], "-progdefs") )
; //write progdefs.h
else if ( !strcmp(myargv[i], "-copy") )
; //copy files / write pak files
else if ( !strcmp(myargv[i], "-bspmodels") )
QCC_PR_Warning(WARN_BADPARAMS, "cmdline", 0, "Argument %s is not supported", myargv[i]);
else if ( !strcmp(myargv[i], "-h2") || !strcmp(myargv[i], "-fteh2") || !strcmp(myargv[i], "-fte") || !strcmp(myargv[i], "-dp") )
; //various targets
else if ( !strcmp(myargv[i], "-pak") || !strcmp(myargv[i], "-pak2") )
QCC_PR_Warning(WARN_BADPARAMS, "cmdline", 0, "Argument %s is not supported", myargv[i]);
else
//compiler constant
if ( !strncmp(myargv[i], "-D", 2) )
{
name = myargv[i] + 2;
val = strchr(name, '=');
if (val)
{
char *t = malloc(val-name+1);
memcpy(t, name, val-name);
t[val-name] = 0;
cnst = QCC_PR_DefineName(t);
free(t);
val++;
}
else
cnst = QCC_PR_DefineName(name);
if (val)
{
cnst->value = qccHunkAlloc(strlen(val)+1);
memcpy(cnst->value, val, strlen(val)+1);
}
}
else if ( !strncmp(myargv[i], "-I", 2) )
{
name = myargv[i] + 2;
if (!*name && i+1<myargc)
name = myargv[++i];
QCC_PR_AddIncludePath(name);
}
//optimisations.
else if ( !strnicmp(myargv[i], "-O", 2) || WINDOWSARG(!strnicmp(myargv[i], "/O", 2)) )
{
qcc_nopragmaoptimise = true;
p = 0;
if (myargv[i][2] >= '0' && myargv[i][2] <= '3')
{
}
else
{
const char *a = myargv[i]+2;
pbool state = true;
if (!strnicmp(a, "no-", 3))
{
a+=3;
state = false;
}
for (p = 0; optimisations[p].enabled; p++)
{
if ((*optimisations[p].abbrev && !stricmp(a, optimisations[p].abbrev)) || !stricmp(a, optimisations[p].fullname))
{
*optimisations[p].enabled = state;
break;
}
}
if (!optimisations[p].enabled)
{
if (!stricmp(a, "overlap-locals"))
opt_locals_overlapping = state;
else
QCC_PR_Warning(WARN_BADPARAMS, "cmdline", 0, "Unrecognised optimisation parameter (%s)", myargv[i]);
}
}
}
else if ( !strnicmp(myargv[i], "-K", 2) || WINDOWSARG(!strnicmp(myargv[i], "/K", 2)) )
{
p = 0;
if (!strnicmp(myargv[i]+2, "no-", 3))
{
for (p = 0; compiler_flag[p].enabled; p++)
if (!stricmp(myargv[i]+5, compiler_flag[p].abbrev))
{
*compiler_flag[p].enabled = false;
break;
}
}
else
{
for (p = 0; compiler_flag[p].enabled; p++)
if (!stricmp(myargv[i]+2, compiler_flag[p].abbrev))
{
*compiler_flag[p].enabled = true;
break;
}
}
if (!compiler_flag[p].enabled)
QCC_PR_Warning(WARN_BADPARAMS, "cmdline", 0, "Unrecognised keyword parameter (%s)", myargv[i]);
}
else if ( !strnicmp(myargv[i], "-std=", 5))
{
for (p = 0; compiler_flag[p].enabled; p++)
{
if (compiler_flag[p].flags & FLAG_ASDEFAULT)
*compiler_flag[p].enabled = true;
else
*compiler_flag[p].enabled = false;
}
if (!stricmp(myargv[i]+5, "C") || !stricmp(myargv[i]+5, "c++") ||!stricmp(myargv[i]+5, "c89") || !stricmp(myargv[i]+5, "c90") || !stricmp(myargv[i]+5, "c99") || !stricmp(myargv[i]+5, "c11") || !stricmp(myargv[i]+5, "c17"))
{ //set up for greatest C compatibility... variations from C are bugs, not features.
keyword_asm = false;
keyword_break = keyword_continue = keyword_for = keyword_goto = keyword_const = keyword_extern = keyword_static = true;
keyword_switch = keyword_case = keyword_default = true;
keyword_accessor = keyword_class = keyword_var = keyword_inout = keyword_optional = keyword_state = keyword_inline = keyword_nosave = keyword_shared = keyword_noref = keyword_unused = keyword_used = keyword_nonstatic = keyword_ignore = keyword_strip = false;
keyword_vector = keyword_entity = keyword_float = keyword_string = false; //not to be confused with actual types, but rather the absence of the keyword local.
keyword_integer = keyword_enumflags = false;
keyword_float = keyword_int = keyword_typedef = keyword_struct = keyword_union = keyword_enum = true;
keyword_double = keyword_long = keyword_short = keyword_char = keyword_signed = keyword_unsigned = true;
keyword_thinktime = keyword_until = keyword_loop = false;
keyword_integer = true;
opt_logicops = true; //early out like C.
flag_assumevar = true; //const only if explicitly const.
pr_subscopedlocals = true; //locals shadow other locals rather than being the same one.
flag_cpriority = true; //fiddle with operator precedence.
flag_assume_integer = true; //unqualified numeric constants are assumed to be ints, consistent with C.
flag_assume_double = true; //and any immediates with a decimal points are assumed to be doubles, consistent with C.
flag_qcfuncs = false;
flag_macroinstrings = false;
qccwarningaction[WARN_UNINITIALIZED] = WA_WARN; //C doesn't like that, might as well warn here too.
qccwarningaction[WARN_TOOMANYPARAMS] = WA_ERROR; //too many args to function is weeeeird.
qccwarningaction[WARN_TOOFEWPARAMS] = WA_ERROR; //missing args should be fatal.
qccwarningaction[WARN_ASSIGNMENTTOCONSTANT] = WA_ERROR; //const is const. at least its not const by default.
qccwarningaction[WARN_SAMENAMEASGLOBAL] = WA_IGNORE; //shadowing of globals.
if (!stricmp(myargv[i]+5, "c++"))
{
keyword_class = /*keyword_new =*/ keyword_inline = true;
cnst = QCC_PR_DefineName("__cplusplus");
val = NULL;
}
else if (!stricmp(myargv[i]+5, "c89") || !stricmp(myargv[i]+5, "c90"))
val = "199409L"; //it was ammended, apparently.
else if (!stricmp(myargv[i]+5, "c99"))
val = "199901L";
else if (!stricmp(myargv[i]+5, "c11"))
val = "201112L";
else if (!stricmp(myargv[i]+5, "c17"))
val = "201710L";
else
val = NULL;
cnst = QCC_PR_DefineName("__STDC_VERSION__");
if (val)
{
cnst->value = qccHunkAlloc(strlen(val)+1);
memcpy(cnst->value, val, strlen(val)+1);
}
}
else if (!strcmp(myargv[i]+5, "qccx"))
{
flag_qccx = true; //fixme: extra stuff
qccwarningaction[WARN_DENORMAL] = WA_IGNORE; //this is just too spammy
qccwarningaction[WARN_LAXCAST] = WA_IGNORE; //more plausable, but still too spammy. easier to fix at least.
}
else if (!strcmp(myargv[i]+5, "preqcc"))
{
flag_hashonly = true;
flag_macroinstrings = true;
}
else if (!strcmp(myargv[i]+5, "reacc"))
{
flag_acc = true;
flag_macroinstrings = false;
}
else if (!strcmp(myargv[i]+5, "frikqcc"))
{
keyword_state = true;
flag_macroinstrings = false;
}
else if (!strcmp(myargv[i]+5, "fteqcc"))
; //as above, its the default.
else if (!strcmp(myargv[i]+5, "qcc") || !strcmp(myargv[i]+5, "id"))
{
flag_ifvector = flag_vectorlogic = false;
keyword_asm = keyword_break = keyword_continue = keyword_for = keyword_goto = false;
keyword_const = keyword_var = keyword_inout = keyword_optional = keyword_state = keyword_inline = keyword_nosave = keyword_extern = keyword_shared = keyword_noref = keyword_unused = keyword_used = keyword_static = keyword_nonstatic = keyword_ignore = keyword_strip = false;
keyword_switch = keyword_case = keyword_default = keyword_accessor = keyword_class = keyword_const = false;
keyword_vector = keyword_entity = keyword_float = keyword_string = false; //not to be confused with actual types, but rather the absence of the keyword local.
keyword_int = keyword_integer = keyword_typedef = keyword_struct = keyword_union = keyword_enum = keyword_enumflags = false;
keyword_thinktime = keyword_until = keyword_loop = false;
keyword_wrap = keyword_weak = false;
qccwarningaction[WARN_PARAMWITHNONAME] = WA_ERROR;
}
else if (!strcmp(myargv[i]+5, "hcc") || !strcmp(myargv[i]+5, "hexenc"))
{
flag_ifvector = flag_vectorlogic = false;
flag_macroinstrings = false;
keyword_asm = keyword_continue = keyword_for = keyword_goto = false;
keyword_const = keyword_var = keyword_inout = keyword_optional = keyword_state = keyword_inline = keyword_nosave = keyword_extern = keyword_shared = keyword_noref = keyword_unused = keyword_used = keyword_static = keyword_nonstatic = keyword_ignore = keyword_strip = false;
keyword_accessor = keyword_class = keyword_const = false;
keyword_vector = keyword_entity = keyword_float = keyword_string = false; //not to be confused with actual types, but rather the absence of the keyword local.
keyword_int = keyword_integer = keyword_typedef = keyword_struct = keyword_union = keyword_enum = keyword_enumflags = false;
keyword_wrap = keyword_weak = false;
keyword_thinktime = keyword_until = keyword_loop = true;
keyword_switch = keyword_case = keyword_default = keyword_break = true;
}
else if (!strcmp(myargv[i]+5, "gmqcc"))
{
flag_ifvector = flag_vectorlogic = true;
flag_dblstarexp = flag_attributes = flag_assumevar = pr_subscopedlocals = flag_cpriority = flag_allowuninit = true;
flag_boundchecks = false; //gmqcc doesn't support dynamic bound checks, so xonotic is buggy shite. we don't want to generate code that will crash.
flag_macroinstrings = false;
opt_logicops = true;
//we have to disable some of these warnings, because xonotic insists on using -Werror. use -Wextra to override.
qccwarningaction[WARN_NOTREFERENCEDCONST] = WA_IGNORE; //gmqcc doesn't warn about function prototypes without any names
qccwarningaction[WARN_CONSTANTCOMPARISON] = WA_IGNORE; //xonotic abuses this, and gmqcc doesn't warn.
qccwarningaction[WARN_POINTLESSSTATEMENT] = WA_IGNORE; //so many macro expansions that we can't mute because of xonotic using an external preprocessor.
qccwarningaction[WARN_OVERFLOW] = WA_IGNORE; //xonotic has data loss from implicit conversions too.
qccwarningaction[WARN_STRICTTYPEMISMATCH] = WA_IGNORE; //gmqcc doesn't enforce checks on auxilliary types.
qccwarningaction[WARN_PARAMWITHNONAME] = WA_IGNORE; //nor does it care if a parameter isn't named.
qccwarningaction[WARN_IFSTRING_USED] = WA_IGNORE; //and many people would argue that this was a feature rather than a bug
qccwarningaction[WARN_UNINITIALIZED] = WA_IGNORE; //all locals get 0-initialised anyway, and our checks are not quite up to scratch.
qccwarningaction[WARN_GMQCC_SPECIFIC] = WA_IGNORE; //we shouldn't warn about gmqcc syntax when we're trying to be compatible with it. there's always -Wextra.
qccwarningaction[WARN_SYSTEMCRC] = WA_IGNORE; //lameness
qccwarningaction[WARN_SYSTEMCRC2] = WA_IGNORE; //extra lameness
qccwarningaction[WARN_ARGUMENTCHECK] = WA_IGNORE; //gmqcc is often used on DP mods, and DP is just too horrible to fix its problems. Also there's a good chance its using some undocumented/new thing.
qccwarningaction[WARN_ASSIGNMENTTOCONSTANT] = WA_ERROR; //some sanity.
keyword_asm = false;
keyword_inout = keyword_optional = keyword_state = keyword_inline = keyword_nosave = keyword_extern = keyword_shared = keyword_unused = keyword_used = keyword_nonstatic = keyword_ignore = keyword_strip = false;
keyword_accessor = keyword_class = false;
keyword_vector = keyword_entity = keyword_float = keyword_string = false; //not to be confused with actual types, but rather the absence of the keyword local.
keyword_int = keyword_integer = keyword_struct = keyword_union = keyword_enum = keyword_enumflags = false;
keyword_thinktime = keyword_until = keyword_loop = false;
keyword_wrap = keyword_weak = false;
keyword_enum = true;
keyword_break = keyword_continue = keyword_for = keyword_goto = true;
keyword_typedef = true;
keyword_switch = keyword_case = keyword_default = true;
keyword_const = keyword_var = keyword_static = keyword_noref = true;
}
else
QCC_PR_Warning(WARN_BADPARAMS, "cmdline", 0, "Unrecognised std parameter (%s)", myargv[i]);
}
else if ( !strnicmp(myargv[i], "-F", 2) || WINDOWSARG(!strnicmp(myargv[i], "/F", 2)) )
{
pbool state;
const char *arg;
p = 0;
if (!strnicmp(myargv[i]+2, "no-", 3))
{
arg = myargv[i]+5;
state = false;
}
else
{
arg = myargv[i]+2;
state = true;
}
for (p = 0; compiler_flag[p].enabled; p++)
if (!stricmp(arg, compiler_flag[p].abbrev))
{
*compiler_flag[p].enabled = state;
break;
}
if (!compiler_flag[p].enabled)
{
//compat flags.
if (!stricmp(arg, "short-logic"))
opt_logicops = state;
else if (!stricmp(arg, "correct-logic"))
flag_ifvector = flag_vectorlogic = state;
else if (!stricmp(arg, "false-empty-strings"))
flag_ifstring = state;
else if (!stricmp(arg, "true-empty-strings"))
flag_brokenifstring = state;
else if (!stricmp(arg, "arithmetic-exceptions"))
qccwarningaction[WARN_DIVISIONBY0] = state?WA_ERROR:WA_IGNORE;
else if (!stricmp(arg, "lno"))
{
//currently we always try to write lno files, when filename info isn't stripped
if (opt_filenames)
{
QCC_PR_Warning(WARN_BADPARAMS, "cmdline", 0, "Disabling -Ofilenames to satisfy -flno request");
opt_filenames = false;
}
}
else if (!stricmp(arg, "return-assignments"))
; //should really be a warning instead
else if (!stricmp(arg, "relaxed-switch"))
; //again should be a warning/werror
else if (!stricmp(arg, "bail-on-werror"))
;
else
QCC_PR_Warning(WARN_BADPARAMS, "cmdline", 0, "Unrecognised flag parameter (%s)", myargv[i]);
}
}
else if ( !strncmp(myargv[i], "-T", 2) || WINDOWSARG(!strncmp(myargv[i], "/T", 2)) )
{
p = 0;
if (!strcmp("parse", myargv[i]+2))
parseonly = true;
else
{
if (!QCC_OPCodeSetTargetName(myargv[i]+2))
QCC_PR_Warning(WARN_BADPARAMS, "cmdline", 0, "Unrecognised target parameter (%s)", myargv[i]);
}
}
else if ( !strnicmp(myargv[i], "-W", 2) || WINDOWSARG(!strnicmp(myargv[i], "/W", 2)) )
{
const char *a = myargv[i]+2;
if (!stricmp(a, "all"))
{
for (j = 0; j < ERR_PARSEERRORS; j++)
if (qccwarningaction[j] == WA_IGNORE)
{
switch(j)
{
//these warnings do not get switched on with -Wall when using -std=gmqcc, because mods that use -Werror would screw up too much
case WARN_CONSTANTCOMPARISON:
case WARN_POINTLESSSTATEMENT:
case WARN_OVERFLOW:
case WARN_STRICTTYPEMISMATCH:
case WARN_PARAMWITHNONAME:
case WARN_IFSTRING_USED:
// case WARN_UNINITIALIZED:
case WARN_GMQCC_SPECIFIC:
case WARN_SYSTEMCRC:
case WARN_SYSTEMCRC2:
if (qccwarningaction[WARN_GMQCC_SPECIFIC])
qccwarningaction[j] = WA_WARN;
break;
//these warnings require -Wextra to enable, as they're too annoying to have to fix
case WARN_NOTREFERENCEDCONST: //warning about every single constant is annoying as heck. note that this includes both stuff like MOVETYPE_ and builtins.
case WARN_EXTRAPRECACHE: //we can't guarentee that we can parse this correctly. this warning is thus a common false positive. its available with -Wextra, and there's intrinsics to reduce false positives.
case WARN_FTE_SPECIFIC: //kinda annoying when its actually valid code.
case WARN_MUTEDEPRECATEDVARIABLE: //these were explicitly muted by the user using checkbuiltin/etc to mute specific symbols.
case WARN_DIVISIONBY0: //breaks xonotic, which seems to want nans.
break;
default:
qccwarningaction[j] = WA_WARN;
break;
}
}
}
else if (!stricmp(a, "extra"))
{
for (j = 0; j < ERR_PARSEERRORS; j++)
if (qccwarningaction[j] == WA_IGNORE)
qccwarningaction[j] = WA_WARN;
}
else if (!stricmp(a, "none"))
{
for (j = 0; j < ERR_PARSEERRORS; j++)
qccwarningaction[j] = WA_IGNORE;
}
else if(!stricmp(a, "error"))
{
werror = true;
}
else if (!stricmp(a, "no-mundane"))
{ //disable mundane performance/efficiency/blah warnings that don't affect code.
qccwarningaction[WARN_SAMENAMEASGLOBAL] = WA_IGNORE;
qccwarningaction[WARN_DUPLICATEDEFINITION] = WA_IGNORE;
qccwarningaction[WARN_CONSTANTCOMPARISON] = WA_IGNORE;
qccwarningaction[WARN_ASSIGNMENTINCONDITIONAL] = WA_IGNORE;
qccwarningaction[WARN_DEADCODE] = WA_IGNORE;
qccwarningaction[WARN_NOTREFERENCEDCONST] = WA_IGNORE;
qccwarningaction[WARN_NOTREFERENCED] = WA_IGNORE;
qccwarningaction[WARN_POINTLESSSTATEMENT] = WA_IGNORE;
qccwarningaction[WARN_ASSIGNMENTTOCONSTANTFUNC] = WA_IGNORE;
qccwarningaction[WARN_BADPRAGMA] = WA_IGNORE; //C specs say that these should be ignored. We're close enough to C that I consider that a valid statement.
qccwarningaction[WARN_IDENTICALPRECOMPILER] = WA_IGNORE;
qccwarningaction[WARN_UNDEFNOTDEFINED] = WA_IGNORE;
qccwarningaction[WARN_EXTRAPRECACHE] = WA_IGNORE;
qccwarningaction[WARN_CORRECTEDRETURNTYPE] = WA_IGNORE;
qccwarningaction[WARN_NOTUTF8] = WA_IGNORE;
qccwarningaction[WARN_SELFNOTTHIS] = WA_IGNORE;
}
else
{
unsigned char action = WA_WARN;
p = -1;
if (!strnicmp(a, "error-", 6))
{
a+= 6;
action = WA_ERROR;
}
else if (!strnicmp(a, "no-", 3))
{
a+=3;
action = WA_IGNORE;
}
p = QCC_WarningForName(a);
if (p >= 0)
qccwarningaction[p] = action;
else
QCC_PR_Warning(WARN_BADPARAMS, "cmdline", 0, "Unrecognised warning parameter (%s)", myargv[i]);
}
}
else if ( !strcmp(myargv[i], "-stdout") )
{
}
else if ( !strcmp(myargv[i], "-log") || !strcmp(myargv[i], "-nolog") )
{
}
else if ( !strcmp(myargv[i], "-max_regs") || !strcmp(myargv[i], "-max_strings") || !strcmp(myargv[i], "-max_globals")
|| !strcmp(myargv[i], "-max_fields") || !strcmp(myargv[i], "-max_statements") || !strcmp(myargv[i], "-max_functions")
|| !strcmp(myargv[i], "-max_types") || !strcmp(myargv[i], "-max_temps") || !strcmp(myargv[i], "-max_macros") )
{
if (++i == myargc)
QCC_PR_Warning(WARN_BADPARAMS, "cmdline", 0, "Missing value for %s arg", myargv[--i]);
}
else if ( !strcmp(myargv[i], "--version") )
{
externs->Printf("%s\n", QCC_VersionString());
exit(EXIT_SUCCESS);
}
else if (*myargv[i] == '-' || WINDOWSARG(*myargv[i] == '/'))
QCC_PR_Warning(WARN_BADPARAMS, "cmdline", 0, "Unrecognised parameter (%s)", myargv[i]);
else
{
if (!QCC_RegisterSourceFile(myargv[i]))
QCC_PR_Warning(WARN_BADPARAMS, "cmdline", 0, "too many source filename arguments");
}
}
if (werror)
{
for (j = 0; j < ERR_PARSEERRORS; j++)
if (qccwarningaction[j])
qccwarningaction[j] = WA_ERROR;
}
}
/*
============
main
============
*/
int qccmline;
char *qccmsrc;
//char *qccmsrc2;
char qccmfilename[1024];
char qccmprogsdat[1024*2];
void QCC_FinishCompile(void);
void SetEndian(void);
static void QCC_SetDefaultProperties (void)
{
int level;
int i;
#ifdef _WIN32
#define FWDSLASHARGS 1
#else
#define FWDSLASHARGS 0
#endif
Hash_InitTable(&compconstantstable, MAX_CONSTANTS, qccHunkAlloc(Hash_BytesForBuckets(MAX_CONSTANTS)));
qcc_framerate = 0; //depends on target (engine's OP_STATE)
ForcedCRC = 0;
defaultstatic = 0;
verbose = VERBOSE_PROGRESS;
*qccmsourcedir = 0;
QCC_PR_CloseProcessor();
QCC_PR_DefineName("FTEQCC");
QCC_PR_DefineName("__FTEQCC__");
if ((FWDSLASHARGS && QCC_CheckParm("/O0")) || QCC_CheckParm("-O0"))
level = 0;
else if ((FWDSLASHARGS && QCC_CheckParm("/O1")) || QCC_CheckParm("-O1"))
level = 1;
else if ((FWDSLASHARGS && QCC_CheckParm("/O2")) || QCC_CheckParm("-O2"))
level = 2;
else if ((FWDSLASHARGS && QCC_CheckParm("/O3")) || QCC_CheckParm("-O3"))
level = 3;
else
level = -1;
if (level == -1)
{
for (i = 0; optimisations[i].enabled; i++)
{
if (optimisations[i].flags & FLAG_ASDEFAULT)
*optimisations[i].enabled = true;
else
*optimisations[i].enabled = false;
}
}
else
{
for (i = 0; optimisations[i].enabled; i++)
{
if (level >= optimisations[i].optimisationlevel)
*optimisations[i].enabled = true;
else
*optimisations[i].enabled = false;
}
}
{ //FIXME: outdated, should be using -Tfte
qcc_targetformat_t targ;
if (QCC_CheckParm ("-h2"))
targ = QCF_HEXEN2;
else if (QCC_CheckParm ("-fte"))
targ = QCF_FTE;
else if (QCC_CheckParm ("-fteh2"))
targ = QCF_FTEH2;
else if (QCC_CheckParm ("-dp"))
targ = QCF_DARKPLACES;
else
targ = QCF_STANDARD;
QCC_OPCodeSetTarget(targ, 0);
}
//enable all warnings
for (i = 0; i < ERR_PARSEERRORS; i++)
qccwarningaction[i] = WA_WARN;
for (; i < WARN_MAX; i++)
qccwarningaction[i] = WA_ERROR;
//play with default warnings.
qccwarningaction[WARN_NOTREFERENCEDCONST] = WA_IGNORE;
qccwarningaction[WARN_MACROINSTRING] = WA_IGNORE;
// qccwarningaction[WARN_ASSIGNMENTTOCONSTANT] = WA_IGNORE;
qccwarningaction[WARN_EXTRAPRECACHE] = WA_IGNORE;
qccwarningaction[WARN_DEADCODE] = WA_IGNORE;
qccwarningaction[WARN_FTE_SPECIFIC] = WA_IGNORE;
qccwarningaction[WARN_DIVISIONBY0] = WA_IGNORE;
qccwarningaction[WARN_MUTEDEPRECATEDVARIABLE] = WA_IGNORE;
qccwarningaction[WARN_EXTENSION_USED] = WA_IGNORE;
qccwarningaction[WARN_IFSTRING_USED] = WA_IGNORE;
qccwarningaction[WARN_CORRECTEDRETURNTYPE] = WA_IGNORE;
qccwarningaction[WARN_NOTUTF8] = WA_IGNORE;
qccwarningaction[WARN_UNINITIALIZED] = WA_IGNORE; //not sure about this being ignored by default.
qccwarningaction[WARN_SELFNOTTHIS] = WA_IGNORE;
qccwarningaction[WARN_EVILPREPROCESSOR] = WA_ERROR; //evil people do evil things. evil must be thwarted!
qccwarningaction[WARN_IDENTICALPRECOMPILER] = WA_IGNORE;
qccwarningaction[WARN_DENORMAL] = WA_ERROR; //DAZ provides a speedup on modern machines, so any engine compiled for sse2+ will have problems with denormals, so make their use look serious.
if (qcc_targetformat == QCF_HEXEN2 || qcc_targetformat == QCF_UHEXEN2 || qcc_targetformat == QCF_FTEH2)
qccwarningaction[WARN_CASEINSENSITIVEFRAMEMACRO] = WA_IGNORE; //hexenc consides these fair game.
if (QCC_CheckParm ("-Fqccx"))
{
qccwarningaction[WARN_DENORMAL] = WA_IGNORE; //this is just too spammy
qccwarningaction[WARN_LAXCAST] = WA_IGNORE; //more plausable, but still too spammy. easier to fix at least.
}
//Check the command line
QCC_PR_CommandLinePrecompilerOptions();
if (qcc_targetformat == QCF_HEXEN2 || qcc_targetformat == QCF_UHEXEN2 || qcc_targetformat == QCF_FTEH2) //force on the thinktime keyword if hexen2 progs.
{
keyword_thinktime = true; //thinktime self : 0.1;
keyword_until = true; //until(cond) {code}; or do{code}until(cond);
keyword_loop = true; //loop {code};
}
if ((FWDSLASHARGS && QCC_CheckParm("/Debug"))) //disable any debug optimisations
{
for (i = 0; optimisations[i].enabled; i++)
{
if (optimisations[i].flags & FLAG_KILLSDEBUGGERS)
*optimisations[i].enabled = false;
}
}
}
//builds a list of files, pretends that they came from a progs.src
//FIXME: use sourcedir!
static int QCC_FindQCFiles(const char *sourcedir)
{
#ifdef _WIN32
WIN32_FIND_DATA fd;
HANDLE h;
#endif
int numfiles = 0, i, j;
char *filelist[256], *temp;
qccmsrc = qccHunkAlloc(8192);
strcat(qccmsrc, "progs.dat\n");//"#pragma PROGS_DAT progs.dat\n");
#if defined(_WIN32) && !defined(WINRT)
h = FindFirstFile("*.qc", &fd);
if (h == INVALID_HANDLE_VALUE)
return 0;
do
{
filelist[numfiles] = qccHunkAlloc (strlen(fd.cFileName)+1);
strcpy(filelist[numfiles], fd.cFileName);
numfiles++;
} while(FindNextFile(h, &fd)!=0);
FindClose(h);
#else
externs->Printf("-Facc is not supported on this platform. Please make a progs.src file instead\n");
#endif
//Sort alphabetically.
//bubble. :(
for (i = 0; i < numfiles-1; i++)
{
for (j = i+1; j < numfiles; j++)
{
if (stricmp(filelist[i], filelist[j]) > 0)
{
temp = filelist[j];
filelist[j] = filelist[i];
filelist[i] = temp;
}
}
}
for (i = 0; i < numfiles; i++)
{
strcat(qccmsrc, filelist[i]);
strcat(qccmsrc, "\n");
// strcat(qccmsrc, "#include \"");
// strcat(qccmsrc, filelist[i]);
// strcat(qccmsrc, "\"\n");
}
return numfiles;
}
static pbool QCC_GenerateRelativePath(char *dest, size_t destsize, char *base, char *relative)
{
int p;
char *s1, *s2;
if (!QC_strlcpy (dest, base, destsize))
return false;
s1 = strchr(dest, '\\');
s2 = strchr(dest, '/');
if (s2 > s1)
s1 = s2;
if (s1)
*s1 = 0;
else
*dest = 0;
p=0;
s2 = relative;
for (;;)
{
if (!strncmp(s2, "./", 2))
s2+=2;
else if(!strncmp(s2, "../", 3))
{
s2+=3;
p++;
}
else
break;
}
for (s1=dest+strlen(dest)-1;p && s1>=dest; s1--)
{
if (*s1 == '/' || *s1 == '\\')
{
*s1 = '\0';
p--;
}
}
if (*dest)
{
if (p)
{ //we were still looking for a separator, but didn't find one, so kill the entire path.
(void)QC_strlcpy(dest, "", destsize);
p--;
}
else if (!QC_strlcat(dest, "/", destsize))
return false;
}
if (!QC_strlcat(dest, s2, destsize))
return false;
while (p>0)
{
if (strlen(dest)+3 >= destsize)
return false;
memmove(dest+3, dest, strlen(dest)+1);
dest[0] = '.';
dest[1] = '.';
dest[2] = '/';
p--;
}
return true;
}
const char *qcccol[COL_MAX];
int qcc_compileactive = false;
extern int accglobalsblock;
char *originalqccmsrc; //for autoprototype.
pbool QCC_main (int argc, const char **argv) //as part of the quake engine
{
extern int pr_bracelevel;
time_t long_time;
extern QCC_type_t *pr_classtype;
size_t p;
extern int qccpersisthunk;
const char *arg;
char *s;
//make sure any print colours are set up properly.
for (p = 0; p < COL_MAX; p++)
if (!qcccol[p])
qcccol[p] = "";
s_filen = "cmdline";
s_filed = 0;
pr_source_line = 0;
if (numsourcefiles && currentsourcefile == numsourcefiles)
{
numsourcefiles = 0;
return false;
}
else if (!numsourcefiles)
currentsourcefile = 0;
if (currentsourcefile && qccpersisthunk && numsourcefiles)
QCC_PR_ResetErrorScope(); //don't clear the ram if we're retaining def info
else
{
memset(sourcefilesdefs, 0, sizeof(sourcefilesdefs));
sourcefilesnumdefs = 0;
if (!PreCompile())
return false;
}
SetEndian();
myargc = argc;
myargv = argv;
pr_scope = NULL;
pr_classtype = NULL;
locals_marshalled = 0;
qcc_compileactive = true;
pHash_Get = &Hash_Get;
pHash_GetNext = &Hash_GetNext;
pHash_Add = &Hash_Add;
pHash_RemoveData = &Hash_RemoveData;
MAX_REGS = 1<<17;
MAX_STRINGS = 1<<21;
MAX_GLOBALS = 1<<17;
MAX_FIELDS = 1<<13;
MAX_STATEMENTS = 1<<20;
MAX_FUNCTIONS = 1<<15;
maxtypeinfos = 1<<16;
MAX_CONSTANTS = 1<<12;
strcpy(destfile, "");
compressoutput = 0;
if ((arg = QCC_ReadParm("-max_regs")))
MAX_REGS = max(100, atoi(arg));
if ((arg = QCC_ReadParm("-max_strings")))
MAX_STRINGS = max(100, atoi(arg));
if ((arg = QCC_ReadParm("-max_globals")))
MAX_GLOBALS = max(64, atoi(arg));
if ((arg = QCC_ReadParm("-max_fields")))
MAX_FIELDS = max(0, atoi(arg));
if ((arg = QCC_ReadParm("-max_statements")))
MAX_STATEMENTS = max(1, atoi(arg));
if ((arg = QCC_ReadParm("-max_functions")))
MAX_FUNCTIONS = max(1, atoi(arg));
if ((arg = QCC_ReadParm("-max_types")))
maxtypeinfos = max(100, atoi(arg));
if ((arg = QCC_ReadParm("-max_temps")))
max_temps = max(100, atoi(arg));
if ((arg = QCC_ReadParm("-max_macros")))
MAX_CONSTANTS = max(100, atoi(arg));
//FIXME: strip this.
s = externs->ReadFile("qcc.cfg", QCC_LoadFileHunkAlloc, NULL, &p, false);
if (s)
{
while(1)
{
s = QCC_COM_Parse(s);
if (!strcmp(qcc_token, "MAX_REGS"))
{
s = QCC_COM_Parse(s);
MAX_REGS = atoi(qcc_token);
} else if (!strcmp(qcc_token, "MAX_STRINGS")) {
s = QCC_COM_Parse(s);
MAX_STRINGS = atoi(qcc_token);
} else if (!strcmp(qcc_token, "MAX_GLOBALS")) {
s = QCC_COM_Parse(s);
MAX_GLOBALS = atoi(qcc_token);
} else if (!strcmp(qcc_token, "MAX_FIELDS")) {
s = QCC_COM_Parse(s);
MAX_FIELDS = atoi(qcc_token);
} else if (!strcmp(qcc_token, "MAX_STATEMENTS")) {
s = QCC_COM_Parse(s);
MAX_STATEMENTS = atoi(qcc_token);
} else if (!strcmp(qcc_token, "MAX_FUNCTIONS")) {
s = QCC_COM_Parse(s);
MAX_FUNCTIONS = atoi(qcc_token);
} else if (!strcmp(qcc_token, "MAX_TYPES")) {
s = QCC_COM_Parse(s);
maxtypeinfos = atoi(qcc_token);
} else if (!strcmp(qcc_token, "MAX_TEMPS")) {
s = QCC_COM_Parse(s);
max_temps = atoi(qcc_token);
} else if (!strcmp(qcc_token, "CONSTANTS")) {
s = QCC_COM_Parse(s);
MAX_CONSTANTS = atoi(qcc_token);
}
else if (!s)
break;
else
externs->Printf("Bad token in qcc.cfg file\n");
}
}
/* don't try to be clever
else if (p < 0)
{
s = qccHunkAlloc(8192);
sprintf(s, "MAX_REGS\t%i\r\nMAX_STRINGS\t%i\r\nMAX_GLOBALS\t%i\r\nMAX_FIELDS\t%i\r\nMAX_STATEMENTS\t%i\r\nMAX_FUNCTIONS\t%i\r\nMAX_TYPES\t%i\r\n",
MAX_REGS, MAX_STRINGS, MAX_GLOBALS, MAX_FIELDS, MAX_STATEMENTS, MAX_FUNCTIONS, maxtypeinfos);
externs->WriteFile("qcc.cfg", s, strlen(s));
}
*/
time(&long_time);
strftime(QCC_copyright, sizeof(QCC_copyright), "Compiled [%Y/%m/%d]"
#ifdef SVNREVISION
", by fteqcc "STRINGIFY(SVNREVISION)
#endif
". ", localtime( &long_time ));
(void)QC_strlcat(QCC_copyright, QCC_VersionString(), sizeof(QCC_copyright));
for (p = 0; p < 5; p++)
strcpy(QCC_Packname[p], "");
for (p = 0; compiler_flag[p].enabled; p++)
{
*compiler_flag[p].enabled = compiler_flag[p].flags & FLAG_ASDEFAULT;
}
parseonly = autoprototyped = autoprototype = false;
QCC_SetDefaultProperties();
autoprototype |= parseonly;
optres_shortenifnots = 0;
optres_overlaptemps = 0;
optres_noduplicatestrings = 0;
optres_constantarithmatic = 0;
optres_nonvec_parms = 0;
optres_constant_names = 0;
optres_constant_names_strings = 0;
optres_precache_file = 0;
optres_filenames = 0;
optres_assignments = 0;
optres_unreferenced = 0;
optres_function_names = 0;
optres_locals = 0;
optres_dupconstdefs = 0;
optres_return_only = 0;
optres_compound_jumps = 0;
// optres_comexprremoval = 0;
optres_stripfunctions = 0;
optres_locals_overlapping = 0;
optres_logicops = 0;
optres_inlines = 0;
optres_test1 = 0;
optres_test2 = 0;
accglobalsblock = 0;
tempsused = 0;
QCC_PurgeTemps();
strings = (void *)qccHunkAlloc(sizeof(char) * MAX_STRINGS);
strofs = 2;
statements = (void *)qccHunkAlloc(sizeof(QCC_statement_t) * MAX_STATEMENTS);
numstatements = 0;
functions = (void *)qccHunkAlloc(sizeof(QCC_function_t) * MAX_FUNCTIONS);
numfunctions=0;
pr_bracelevel = 0;
qcc_pr_globals = (void *)qccHunkAlloc(sizeof(float) * (MAX_REGS + MAX_LOCALS + MAX_TEMPS));
numpr_globals=0;
Hash_InitTable(&typedeftable, 1024, qccHunkAlloc(Hash_BytesForBuckets(1024)));
Hash_InitTable(&globalstable, MAX_REGS/2, qccHunkAlloc(Hash_BytesForBuckets(MAX_REGS/2)));
Hash_InitTable(&localstable, 128, qccHunkAlloc(Hash_BytesForBuckets(128)));
Hash_InitTable(&floatconstdefstable, MAX_REGS/2+1, qccHunkAlloc(Hash_BytesForBuckets(MAX_REGS/2+1)));
Hash_InitTable(&stringconstdefstable, MAX_REGS/2, qccHunkAlloc(Hash_BytesForBuckets(MAX_REGS/2)));
Hash_InitTable(&stringconstdefstable_trans, 1000, qccHunkAlloc(Hash_BytesForBuckets(1000)));
dotranslate_count = 0;
// pr_global_defs = (QCC_def_t **)qccHunkAlloc(sizeof(QCC_def_t *) * MAX_REGS);
qcc_globals = (void *)qccHunkAlloc(sizeof(QCC_ddef_t) * MAX_GLOBALS);
numglobaldefs=0;
fields = (void *)qccHunkAlloc(sizeof(QCC_ddef_t) * MAX_FIELDS);
numfielddefs=0;
memset(pr_immediate_string, 0, sizeof(pr_immediate_string));
precache_sound = (void *)qccHunkAlloc(sizeof(*precache_sound)*QCC_MAX_SOUNDS);
numsounds=0;
precache_texture = (void *)qccHunkAlloc(sizeof(*precache_texture)*QCC_MAX_TEXTURES);
numtextures=0;
precache_model = (void *)qccHunkAlloc(sizeof(*precache_model)*QCC_MAX_MODELS);
nummodels=0;
precache_file = (void *)qccHunkAlloc(sizeof(*precache_file)*QCC_MAX_FILES);
numfiles = 0;
qcc_typeinfo = (void *)qccHunkAlloc(sizeof(QCC_type_t)*maxtypeinfos);
numtypeinfos = 0;
qcc_tempofs = qccHunkAlloc(sizeof(int) * max_temps);
tempsstart = 0;
bodylessfuncs=0;
memset(&pr, 0, sizeof(pr));
#ifdef MAX_EXTRA_PARMS
memset(&extra_parms, 0, sizeof(extra_parms));
#endif
if ( QCC_CheckParm ("/?") || QCC_CheckParm ("?") || QCC_CheckParm ("-?") || QCC_CheckParm ("-help") || QCC_CheckParm ("--help"))
{
externs->Printf ("qcc looks for progs.src in the current directory.\n");
externs->Printf ("to look in a different directory: qcc -src <directory>\n");
// externs->Printf ("to build a clean data tree: qcc -copy <srcdir> <destdir>\n");
// externs->Printf ("to build a clean pak file: qcc -pak <srcdir> <packfile>\n");
// externs->Printf ("to bsp all bmodels: qcc -bspmodels <gamedir>\n");
externs->Printf ("-Fwasm causes FTEQCC to dump all asm to qc.asm\n");
externs->Printf ("-O0 to disable optimisations\n");
externs->Printf ("-O1 to optimise for size\n");
externs->Printf ("-O2 to optimise more - some behaviours may change\n");
externs->Printf ("-O3 to optimise lots - experimental or non-future-proof\n");
externs->Printf ("-Oname to enable an optimisation\n");
externs->Printf ("-Ono-name to disable optimisations\n");
externs->Printf ("-Kkeyword to activate keyword\n");
externs->Printf ("-Kno-keyword to disable keyword\n");
externs->Printf ("-Wall to give a stupid number of warnings\n");
externs->Printf ("-Ttarget to set a output format\n");
externs->Printf ("-Fautoproto to enable automatic prototyping\n");
externs->Printf ("-Fsubscope to make locals specific to their subscope\n");
qcc_compileactive = false;
return true;
}
if (flag_caseinsensitive)
{
externs->Printf("Compiling without case sensitivity\n");
pHash_Get = &Hash_GetInsensitive;
pHash_GetNext = &Hash_GetNextInsensitive;
pHash_Add = &Hash_AddInsensitive;
pHash_RemoveData = &Hash_RemoveDataInsensitive;
}
if (*qccmsourcedir)
externs->Printf ("Source directory: %s\n", qccmsourcedir);
QCC_InitData ();
QCC_PR_BeginCompilation ((void *)qccHunkAlloc (0x100000), 0x100000);
QCC_PR_ClearGrabMacros (false);
qccmsrc = NULL;
if (destfile_explicit && numsourcefiles && !currentsourcefile)
{ //generate an internal .src file from the argument list
int i;
qccmsrc = qccHunkAlloc(8192);
*qccmsrc = 0;
for (i = 0;i<numsourcefiles;i++)
QC_snprintfz(qccmsrc+strlen(qccmsrc), 8192-strlen(qccmsrc), "#include \"%s\"\n", sourcefileslist[i]);
currentsourcefile = i;
}
if (qccmsrc)
;
else if (flag_acc && !numsourcefiles)
{
if (!QCC_FindQCFiles(qccmsourcedir))
QCC_Error (ERR_COULDNTOPENFILE, "Couldn't find any qc files.");
}
else
{
if (!numsourcefiles)
{
p = QCC_CheckParm ("-qc");
if (p && p < argc-1 )
QC_strlcpy(qccmprogsdat, argv[p+1], sizeof(qccmprogsdat));
else
{ //look for a preprogs.src... :o)
char tmp[sizeof(qccmsourcedir)+16];
QC_snprintfz (tmp, sizeof(tmp), "%spreprogs.src", qccmsourcedir);
if (externs->FileSize(tmp) <= 0)
QC_snprintfz (qccmprogsdat, sizeof(qccmprogsdat), "progs.src");
else
QC_snprintfz (qccmprogsdat, sizeof(qccmprogsdat), "preprogs.src");
}
numsourcefiles = 0;
strcpy(sourcefileslist[numsourcefiles++], qccmprogsdat);
currentsourcefile = 0;
}
else if (currentsourcefile == numsourcefiles || (currentsourcefile && destfile_explicit))
{
//no more.
qcc_compileactive = false;
numsourcefiles = 0;
currentsourcefile = 0;
return true;
}
if (currentsourcefile)
externs->Printf("-------------------------------------\n");
else
externs->Printf("%s\n", QCC_VersionString());
QC_snprintfz (qccmprogsdat, sizeof(qccmprogsdat), "%s%s", qccmsourcedir, sourcefileslist[currentsourcefile++]);
externs->Printf ("Source file: %s\n", qccmprogsdat);
QC_strlcpy(compilingrootfile, qccmprogsdat, sizeof(compilingrootfile));
if (QCC_LoadFile (qccmprogsdat, (void *)&qccmsrc) == -1)
{
return true;
}
}
#ifdef WRITEASM
if (writeasm)
{
asmfile = fopen("qc.asm", "wb");
if (!asmfile)
QCC_Error (ERR_INTERNAL, "Couldn't open file for asm output.");
}
asmfilebegun = !!asmfile;
#endif
newstylesource = false;
if (qccmsrc[0] == '#' && qccmsrc[1] == '!')
qccmsrc = strchr(qccmsrc, '\n'); //ignore the first line if it starts with a #! for unix scripts. because we can.
compilingfile = qccmprogsdat;
preprocessonly = false;
if (QCC_CheckParm ("-E"))
{
pr_file_p = qccmsrc;
preprocessonly = true;
goto newstyle;
}
pr_file_p = QCC_COM_Parse(qccmsrc);
if (QCC_CheckParm ("-qc"))
{
strcpy(destfile, qccmprogsdat);
StripExtension(destfile);
strcat(destfile, ".qco");
p = QCC_CheckParm ("-o");
if (!p || p >= argc-1 || argv[p+1][0] == '-')
if (p && p < argc-1 )
sprintf (destfile, "%s%s", qccmsourcedir, argv[p+1]);
goto newstyle;
}
if (*qcc_token == '#')
{
newstyle:
if (flag_filetimes)
QCC_PR_Warning(0, qccmsrc, 0, "-ffiletimes unsupported with this input");
newstylesource = true;
originalqccmsrc = qccmsrc;
pr_source_line = qccmline = 1;
StartNewStyleCompile();
return true;
}
pr_source_line = qccmline = 1;
pr_file_p = qccmsrc;
QCC_PR_LexWhitespace(false);
qccmsrc = pr_file_p;
s = qccmsrc;
pr_file_p = qccmsrc;
QCC_PR_SimpleGetToken ();
strcpy(qcc_token, pr_token);
qccmsrc = pr_file_p;
qccmline = pr_source_line;
if (!qccmsrc)
QCC_Error (ERR_NOOUTPUT, "No destination filename. qcc -help for info.");
QCC_GenerateRelativePath(destfile, sizeof(destfile), qccmprogsdat, qcc_token);
p = QCC_CheckParm ("-o");
if (p > 0 && p < argc-1 && argv[p+1][0] != '-')
sprintf (destfile, "%s", argv[p+1]);
if (flag_filetimes)
{
struct stat s, os;
pbool modified = false;
if (stat(destfile, &os) != -1)
{
while ((pr_file_p=QCC_COM_Parse(pr_file_p)))
{
if (stat(qcc_token, &s) == -1 || s.st_mtime > os.st_mtime)
{
externs->Printf("%s changed\n", qcc_token);
modified = true;
break;
}
}
if (!modified)
{
externs->Printf("No changes\n");
qcc_compileactive = false;
return true;
}
else
{
pr_file_p = qccmsrc;
}
}
}
externs->Printf ("outputfile: %s\n", destfile);
pr_dumpasm = false;
currentchunk = NULL;
originalqccmsrc = qccmsrc;
return true;
}
void new_QCC_ContinueCompile(void);
//called between exe frames - won't loose net connection (is the theory)...
void QCC_ContinueCompile(void)
{
if (!qcc_compileactive)
//HEY!
return;
if (newstylesource)
{
char *ofp = pr_file_p;
do
{
new_QCC_ContinueCompile();
} while(currentchunk); //while parsing through preprocessor, make sure nothing gets hurt.
if (ofp == pr_file_p && qcc_compileactive && pr_token_type != tt_eof)
QCC_Error (ERR_INTERNAL, "Syntax error\n");
return;
}
pr_file_p = qccmsrc;
s_filen = compilingrootfile;
s_filed = 0;
pr_source_line = qccmline;
QCC_PR_LexWhitespace(false);
qccmsrc = pr_file_p;
qccmline = pr_source_line;
qccmsrc = QCC_COM_Parse(qccmsrc);
if (!qccmsrc)
{
if (parseonly)
{
qcc_compileactive = false;
if (sourcefilesnumdefs < countof(sourcefilesdefs) && qccpersisthunk)
sourcefilesdefs[currentsourcefile++] = pr.def_head.next;
}
else
{
if (autoprototype)
{
qccmsrc = originalqccmsrc;
autoprototyped = autoprototype;
QCC_SetDefaultProperties();
autoprototype = false;
return;
}
QCC_FinishCompile();
}
PostCompile();
if (currentsourcefile < numsourcefiles)
{
if (!QCC_main(myargc, myargv))
return;
}
else
{
qcc_compileactive = false;
numsourcefiles = 0;
currentsourcefile = 0;
}
return;
}
if(setjmp(pr_parse_abort))
{
if (++pr_error_count > MAX_ERRORS)
QCC_Error (ERR_PARSEERRORS, "Errors have occured\n");
return; //just try move onto the next file, gather errors.
}
else
QCC_FindBestInclude(qcc_token, compilingrootfile, 2);
/*
{
int includepath = 0;
while(1)
{
if (includepath)
{
if (includepath > MAXINCLUDEDIRS || !*qccincludedir[includepath-1])
{
QCC_GenerateRelativePath(qccmfilename, sizeof(qccmfilename), compilingrootfile, qcc_token);
break;
}
currentfile = qccincludedir[includepath-1];
}
QCC_Canonicalize(qccmfilename, sizeof(fullname), qcc_token, compilingrootfile);
{
extern progfuncs_t *qccprogfuncs;
if (qccprogfuncs->funcs.parms->FileSize(qccmfilename) == -1)
{
includepath++;
continue;
}
}
break;
}
}
QCC_GenerateRelativePath(qccmfilename, sizeof(qccmfilename), compilingrootfile, qcc_token);
if (autoprototype)
externs->Printf ("prototyping %s\n", qccmfilename);
else
{
externs->Printf ("compiling %s\n", qccmfilename);
}
QCC_LoadFile (qccmfilename, (void *)&qccmsrc2);
if (!QCC_PR_CompileFile (qccmsrc2, qccmfilename) )
QCC_Error (ERR_PARSEERRORS, "Errors have occured\n");
*/
}
void QCC_FinishCompile(void)
{
pbool donesomething;
int crc;
// int p;
currentchunk = NULL;
if (setjmp(pr_parse_abort))
QCC_Error(ERR_INTERNAL, "%s", "");
s_filen = "";
s_filed = 0;
pr_source_line = 0;
if (!QCC_PR_FinishCompilation ())
{
QCC_Error (ERR_PARSEERRORS, "compilation errors");
}
/* p = QCC_CheckParm ("-asm");
if (p)
{
for (p++ ; p<myargc ; p++)
{
if (myargv[p][0] == '-')
break;
QCC_PrintFunction (myargv[p]);
}
}*/
/*p = QCC_CheckParm ("-ofs");
if (p)
{
for (p++ ; p<myargc ; p++)
{
if (myargv[p][0] == '-')
break;
QCC_PrintOfs (atoi(myargv[p]));
}
}*/
// write progdefs.h
crc = QCC_PR_WriteProgdefs ("progdefs.h");
if (pr_error_count)
QCC_Error (ERR_PARSEERRORS, "compilation errors");
// write data file
donesomething = QCC_WriteData (crc);
// regenerate bmodels if -bspmodels
QCC_BspModels ();
// report / copy the data files
QCC_CopyFiles ();
if (sourcefilesnumdefs < countof(sourcefilesdefs) && qccpersisthunk)
sourcefilesdefs[sourcefilesnumdefs++] = pr.def_head.next;
if (donesomething)
{
if (verbose >= VERBOSE_STANDARD)
{
externs->Printf ("Compile Complete\n\n");
if (optres_shortenifnots)
externs->Printf("optres_shortenifnots %i\n", optres_shortenifnots);
if (optres_overlaptemps)
externs->Printf("optres_overlaptemps %i\n", optres_overlaptemps);
if (optres_noduplicatestrings)
externs->Printf("optres_noduplicatestrings %i\n", optres_noduplicatestrings);
if (optres_constantarithmatic)
externs->Printf("optres_constantarithmatic %i\n", optres_constantarithmatic);
if (optres_nonvec_parms)
externs->Printf("optres_nonvec_parms %i\n", optres_nonvec_parms);
if (optres_constant_names)
externs->Printf("optres_constant_names %i\n", optres_constant_names);
if (optres_constant_names_strings)
externs->Printf("optres_constant_names_strings %i\n", optres_constant_names_strings);
if (optres_precache_file)
externs->Printf("optres_precache_file %i\n", optres_precache_file);
if (optres_filenames)
externs->Printf("optres_filenames %i\n", optres_filenames);
if (optres_assignments)
externs->Printf("optres_assignments %i\n", optres_assignments);
if (optres_unreferenced)
externs->Printf("optres_unreferenced %i\n", optres_unreferenced);
if (optres_locals)
externs->Printf("optres_locals %i\n", optres_locals);
if (optres_function_names)
externs->Printf("optres_function_names %i\n", optres_function_names);
if (optres_dupconstdefs)
externs->Printf("optres_dupconstdefs %i\n", optres_dupconstdefs);
if (optres_return_only)
externs->Printf("optres_return_only %i\n", optres_return_only);
if (optres_compound_jumps)
externs->Printf("optres_compound_jumps %i\n", optres_compound_jumps);
// if (optres_comexprremoval)
// externs->Printf("optres_comexprremoval %i\n", optres_comexprremoval);
if (optres_stripfunctions)
externs->Printf("optres_stripfunctions %i\n", optres_stripfunctions);
if (optres_locals_overlapping)
externs->Printf("optres_locals_overlapping %i\n", optres_locals_overlapping);
if (optres_logicops)
externs->Printf("optres_logicops %i\n", optres_logicops);
if (optres_inlines)
externs->Printf("optres_inlines %i\n", optres_inlines);
if (optres_test1)
externs->Printf("optres_test1 %i\n", optres_test1);
if (optres_test2)
externs->Printf("optres_test2 %i\n", optres_test2);
externs->Printf("numtemps %u\n", (unsigned)tempsused);
}
if (!flag_msvcstyle && verbose >= VERBOSE_PROGRESS)
externs->Printf("Done. %i warnings\n", pr_warning_count);
}
qcc_compileactive = false;
}
extern char *pr_file_p;
extern int pr_source_line;
static void StartNewStyleCompile(void)
{
char *tmp;
if (setjmp(pr_parse_abort))
{
if (++pr_error_count > MAX_ERRORS)
return;
if (setjmp(pr_parse_abort))
return;
QCC_PR_SkipToSemicolon ();
if (pr_token_type == tt_eof)
return;
}
compilingfile = qccmprogsdat;
s_filen = tmp = qccHunkAlloc(strlen(compilingfile)+1);
strcpy(tmp, compilingfile);
if (opt_filenames)
{
optres_filenames += strlen(compilingfile)+1;
s_filed = 0;
}
else
s_filed = QCC_CopyString (compilingfile);
pr_file_p = qccmsrc;
pr_source_line = 0;
QCC_PR_NewLine (false);
QCC_PR_Lex (); // read first token
}
void new_QCC_ContinueCompile(void)
{
if (setjmp(pr_parse_abort))
{
// if (pr_error_count != 0)
{
QCC_Error (ERR_PARSEERRORS, "Errors have occured");
return;
}
QCC_PR_SkipToSemicolon ();
if (pr_token_type == tt_eof)
return;
}
if (pr_token_type == tt_eof)
{
if (pr_error_count)
QCC_Error (ERR_PARSEERRORS, "Errors have occured");
if (autoprototype && !parseonly)
{
char *tmp;
qccmsrc = originalqccmsrc;
s_filen = tmp = qccHunkAlloc(strlen(compilingfile)+1);
strcpy(tmp, compilingfile);
if (opt_filenames)
{
optres_filenames += strlen(compilingfile)+1;
s_filed = 0;
}
else
s_filed = QCC_CopyString (compilingfile);
pr_file_p = qccmsrc;
autoprototyped = autoprototype;
QCC_SetDefaultProperties();
autoprototype = false;
QCC_PR_NewLine(false);
QCC_PR_Lex();
return;
}
else
{
if (!parseonly)
QCC_FinishCompile();
else
{
if (sourcefilesnumdefs < countof(sourcefilesdefs) && qccpersisthunk)
sourcefilesdefs[currentsourcefile++] = pr.def_head.next;
}
PostCompile();
if (!QCC_main(myargc, myargv))
{
qcc_compileactive = false;
return;
}
return;
}
}
pr_scope = NULL; // outside all functions
if (preprocessonly)
{
pbool white = false;
static int line = 1;
while(pr_token_type != tt_eof)
{
//if there's whitespace next, make sure we represent that
if (line < pr_token_line)
{ //keep line numbers correct by splurging multiple newlines.
while(line++ < pr_source_line)
externs->Printf("\n");
}
else if (white)
externs->Printf(" ");
externs->Printf("%s", pr_token);
white = (qcc_iswhite(*pr_file_p) || (*pr_file_p == '/' && (pr_file_p[1] == '/' || pr_file_p[1] == '*')));
QCC_PR_Lex();
}
QCC_PR_Lex();
return;
}
QCC_PR_ParseDefs (NULL, false);
}
/*void new_QCC_ContinueCompile(void)
{
char *s, *s2;
if (!qcc_compileactive)
//HEY!
return;
// compile all the files
qccmsrc = QCC_COM_Parse(qccmsrc);
if (!qccmsrc)
{
QCC_FinishCompile();
return;
}
s = qcc_token;
strcpy (qccmfilename, qccmsourcedir);
while(1)
{
if (!strncmp(s, "..\\", 3))
{
s2 = qccmfilename + strlen(qccmfilename)-2;
while (s2>=qccmfilename)
{
if (*s2 == '/' || *s2 == '\\')
{
s2[1] = '\0';
break;
}
s2--;
}
s+=3;
continue;
}
if (!strncmp(s, ".\\", 2))
{
s+=2;
continue;
}
break;
}
// strcat (qccmfilename, s);
// externs->Printf ("compiling %s\n", qccmfilename);
// QCC_LoadFile (qccmfilename, (void *)&qccmsrc2);
// if (!new_QCC_PR_CompileFile (qccmsrc2, qccmfilename) )
// QCC_Error ("Errors have occured\n");
{
if (!pr.memory)
QCC_Error ("PR_CompileFile: Didn't clear");
QCC_PR_ClearGrabMacros (); // clear the frame macros
compilingfile = filename;
pr_file_p = qccmsrc2;
s_file = QCC_CopyString (filename);
pr_source_line = 0;
QCC_PR_NewLine ();
QCC_PR_Lex (); // read first token
while (pr_token_type != tt_eof)
{
if (setjmp(pr_parse_abort))
{
if (++pr_error_count > MAX_ERRORS)
return false;
QCC_PR_SkipToSemicolon ();
if (pr_token_type == tt_eof)
return false;
}
pr_scope = NULL; // outside all functions
QCC_PR_ParseDefs ();
}
}
return (pr_error_count == 0);
}*/
#endif