#if !defined(MINIMAL) && !defined(OMIT_QCC) #define PROGSUSED #include "qcc.h" #include #ifdef _WIN32 #include #endif #include #include "errno.h" char QCC_copyright[1024]; int QCC_packid; char QCC_Packname[5][128]; extern int optres_test1; extern int optres_test2; int writeasm; pbool verbose; pbool qcc_nopragmaoptimise; extern unsigned int locals_marshalled; pbool QCC_PR_SimpleGetToken (void); void QCC_PR_LexWhitespace (pbool inhibitpreprocessor); void *FS_ReadToMem(char *fname, void *membuf, int *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; int numtemps; #define MAXSOURCEFILESLIST 8 char sourcefileslist[MAXSOURCEFILESLIST][1024]; int currentsourcefile; int numsourcefiles; extern char *compilingfile; void QCC_PR_ResetErrorScope(void); pbool compressoutput; pbool newstylesource; char destfile[1024]; float *qcc_pr_globals; unsigned int numpr_globals; char *strings; int strofs; QCC_statement_t *statements; int numstatements; QCC_dfunction_t *functions; 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; #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. qcc_targetformat_t qcc_targetformat; pbool bodylessfuncs; QCC_type_t *qcc_typeinfo; int numtypeinfos; int maxtypeinfos; 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 {" 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}, //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) {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(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", 3, 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", 3, 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. This can affect FTE_MULTIPROGS."}, {&opt_locals_overlapping, "lo", 3, 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."}, {NULL} }; #define defaultkeyword FLAG_HIDDENINGUI|FLAG_ASDEFAULT|FLAG_MIDCOMPILE #define nondefaultkeyword FLAG_HIDDENINGUI|0|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_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, defaultkeyword, "int", "Keyword: int", "Disables the 'int' keyword."}, {&keyword_integer, defaultkeyword, "integer", "Keyword: integer", "Disables the 'integer' keyword."}, {&keyword_noref, defaultkeyword, "noref", "Keyword: noref", "Disables the 'noref' keyword."}, //nowhere else references this, don't strip it. {&keyword_nosave, defaultkeyword, "nosave", "Keyword: nosave", "Disables the 'nosave' 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_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."}, //options {&keywords_coexist, FLAG_ASDEFAULT, "kce", "Keywords Coexist", "If you want keywords to NOT be disabled when they a variable by the same name is defined, check here."}, {&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_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_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 ¦ as EOF. 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_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"}, {&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,"flag_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|FLAG_ASDEFAULT,"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_noboundchecks, FLAG_MIDCOMPILE,"noboundchecks","Disable Bound Checks", "Disable array index checks, speeding up array access but can result in your code misbehaving."}, {NULL} }; struct { qcc_targetformat_t target; char *name; } targets[] = { {QCF_STANDARD, "standard"}, {QCF_STANDARD, "q1"}, {QCF_STANDARD, "quakec"}, {QCF_HEXEN2, "hexen2"}, {QCF_HEXEN2, "h2"}, {QCF_KK7, "kkqwsv"}, {QCF_KK7, "kk7"}, {QCF_KK7, "bigprogs"}, {QCF_KK7, "version7"}, {QCF_KK7, "kkqwsv"}, {QCF_FTE, "fte"}, {QCF_FTEH2, "fteh2"}, {QCF_DARKPLACES,"darkplaces"}, {QCF_DARKPLACES,"dp"}, {QCF_QTEST, "qtest"}, {0, NULL} }; /* ================= BspModels Runs qbsp and light on all of the models with a .bsp extension ================= */ int QCC_CheckParm (char *check); 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 ; iprev) { 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; strcpy (strings+strofs, str); strofs += strlen(str)+1; return old; } int QCC_CopyDupBackString (char *str) { int old; char *s; for (s = strings+strofs-1; s>strings ; s--) if (!strcmp(s, str)) return s-strings; old = strofs; strcpy (strings+strofs, str); strofs += strlen(str)+1; return old; } void QCC_PrintStrings (void) { int i, l, j; for (i=0 ; is_file, strings + d->s_name, d->first_statement, d->parm_start); for (j=0 ; jnumparms ; j++) printf ("%i ",d->parm_size[j]); printf (")\n"); } }*/ void QCC_PrintFields (void) { extern char *basictypenames[]; int i; QCC_ddef_t *d; for (i=0 ; iofs, basictypenames[d->type], strings + d->s_name); } } void QCC_PrintGlobals (void) { int i; QCC_ddef_t *d; for (i=0 ; iofs, d->type, strings + d->s_name); } } int encode(int len, int method, char *in, int handle); int WriteSourceFiles(int h, dprograms_t *progs, pbool sourceaswell) { includeddatafile_t *idf; qcc_cachedsourcefile_t *f; int num=0; int ofs; /* for (f = qcc_sourcefile; f ; f=f->next) { if (f->type == FT_CODE && !sourceaswell) continue; SafeWrite(h, f->filename, strlen(f->filename)+1); i = PRLittleLong(f->size); SafeWrite(h, &i, sizeof(int)); i = PRLittleLong(encrpytmode); SafeWrite(h, &i, sizeof(int)); if (encrpytmode) for (i = 0; i < f->size; i++) f->file[i] ^= 0xA5; SafeWrite(h, f->file, f->size); }*/ for (f = qcc_sourcefile,num=0; f ; f=f->next) { if (f->type == FT_CODE && !sourceaswell) continue; num++; } if (!num) return 0; idf = qccHunkAlloc(sizeof(includeddatafile_t)*num); for (f = qcc_sourcefile,num=0; f ; f=f->next) { if (f->type == FT_CODE && !sourceaswell) continue; strcpy(idf[num].filename, f->filename); idf[num].size = f->size; #ifdef AVAIL_ZLIB idf[num].compmethod = 2; #else idf[num].compmethod = 1; #endif idf[num].ofs = SafeSeek(h, 0, SEEK_CUR); idf[num].compsize = QC_encode(progfuncs, f->size, idf[num].compmethod, f->file, h); num++; } ofs = SafeSeek(h, 0, SEEK_CUR); SafeWrite(h, &num, sizeof(int)); SafeWrite(h, idf, sizeof(includeddatafile_t)*num); qcc_sourcefile = NULL; return ofs; } void QCC_InitData (void) { static char parmname[12][MAX_PARMS]; static temp_t ret_temp; 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(&ret_temp, 0, sizeof(ret_temp)); def_ret.ofs = OFS_RETURN; def_ret.name = "return"; def_ret.temp = &ret_temp; def_ret.constant = false; def_ret.type = NULL; ret_temp.ofs = def_ret.ofs; ret_temp.scope = NULL; ret_temp.size = 3; ret_temp.next = NULL; for (i=0 ; inext) { if (d->type->type == ev_function && !d->scope)// function parms are ok { if (!(d->initialized & 1) && d->references>0) { SafeWrite(handle, d->name, strlen(d->name)+1); ret++; } } } return ret; } //marshalled locals still point to the FIRST_LOCAL range. //this function remaps all the locals back into actual usable defs. void QCC_UnmarshalLocals(void) { QCC_def_t *def; unsigned int ofs; unsigned int maxo; int i; extern int tempsused; ofs = numpr_globals; maxo = ofs+locals_marshalled; for (def = pr.def_head.next ; def ; def = def->next) { if (def->ofs >= FIRST_LOCAL) //unmap defs. { def->ofs = def->ofs + ofs - FIRST_LOCAL; if (maxo < def->ofs + def->type->size) maxo = def->ofs + def->type->size; } } for (i = 0; i < numfunctions; i++) { if (functions[i].parm_start == FIRST_LOCAL) functions[i].parm_start = ofs; } QCC_RemapOffsets(0, numstatements, FIRST_LOCAL, FIRST_LOCAL+(maxo-ofs), ofs); QCC_RemapOffsets(0, numstatements, FIRST_TEMP, FIRST_TEMP+tempsused, maxo); numpr_globals = maxo+tempsused; if (numpr_globals > MAX_REGS) QCC_Error(ERR_TOOMANYGLOBALS, "Too many globals are in use to unmarshal all locals"); if (verbose) printf("Total of %i locals, %i temps\n", maxo-ofs, tempsused); } CompilerConstant_t *QCC_PR_CheckCompConstDefined(char *def); pbool QCC_WriteData (int crc) { char element[MAX_NAME]; QCC_def_t *def, *comp_x, *comp_y, *comp_z; QCC_ddef_t *dd; dprograms_t progs; int h; int i, len; pbool debugtarget = false; pbool types = false; int outputsttype = PST_DEFAULT; int warnedunref = 0; int *statement_linenums; if (numstatements==1 && numfunctions==1 && numglobaldefs==1 && numfielddefs==1) { 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 qcc.cfg", numstatements, (numstatements+32768)&~32767); if (strofs > MAX_STRINGS) QCC_Error(ERR_TOOMANYSTRINGS, "Too many strings - %i\nAdd \"MAX_STRINGS\" \"%i\" to qcc.cfg", strofs, (strofs+32768)&~32767); QCC_UnmarshalLocals(); switch (qcc_targetformat) { case QCF_HEXEN2: case QCF_STANDARD: if (bodylessfuncs) printf("Warning: There are some functions without bodies.\n"); if (numpr_globals > 65530 ) { printf("Forcing target to FTE32 due to numpr_globals\n"); outputsttype = PST_FTE32; } else if (qcc_targetformat == QCF_FTEH2) { printf("Progs execution will require FTE\n"); break; } else if (qcc_targetformat == QCF_HEXEN2) { printf("Progs execution requires a Hexen2 compatible HCVM\n"); break; } else { if (numpr_globals >= 32768) //not much of a different format. Rewrite output to get it working on original executors? printf("Globals exceeds 32k - an enhanced QCVM will be required\n"); else printf("Progs should run on any QuakeC VM\n"); break; } qcc_targetformat = (qcc_targetformat==QCF_HEXEN2)?QCF_FTEH2:QCF_FTE; //intentional fallthrough case QCF_FTEDEBUG: case QCF_FTE: case QCF_FTEH2: case QCF_DARKPLACES: if (qcc_targetformat == QCF_FTEDEBUG) debugtarget = true; if (numpr_globals > 65530) { printf("Using 32 bit target due to numpr_globals\n"); outputsttype = PST_FTE32; } if (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 printf("AMD64 builds cannot write typeinfo structures\n"); types = false; } if (verbose) { if (qcc_targetformat == QCF_DARKPLACES) printf("DarkPlaces or FTE will be required\n"); else printf("An FTE executor will be required\n"); } break; case QCF_KK7: if (bodylessfuncs) printf("Warning: There are some functions without bodies.\n"); if (numpr_globals > 65530) printf("Warning: Saving is not supported. Ensure all engine read fields and globals are defined early on.\n"); printf("A KK compatible executor will be required (FTE/KK)\n"); outputsttype = PST_KKQWSV; break; case QCF_QTEST: printf("Compiled QTest progs will most likely not work at all. YOU'VE BEEN WARNED!\n"); outputsttype = PST_QTEST; break; default: Sys_Error("invalid progs type chosen!"); } //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->references++; def = QCC_PR_GetDef(NULL, "end_sys_fields", NULL, false, 0, false); if (def) def->references++; for (def = pr.def_head.next ; def ; def = def->next) { if ((def->type->type == ev_struct || def->type->type == ev_union || def->arraysize) && def->deftail) { QCC_def_t *d; d = def; while (d != def->deftail) { d = d->next; h = d->references; d->references += def->references; def->references += h; } } 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; sprintf(element, "%s_x", def->name); comp_x = QCC_PR_GetDef(NULL, element, def->scope, false, 0, false); sprintf(element, "%s_y", def->name); comp_y = QCC_PR_GetDef(NULL, element, def->scope, false, 0, false); sprintf(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; } } if (def->references<=0) { int wt = def->constant?WARN_NOTREFERENCEDCONST:WARN_NOTREFERENCED; pr_scope = def->scope; if (strcmp(def->name, "IMMEDIATE")) { char typestr[256]; if (warnedunref >= 10) { if (qccwarningaction[wt]) pr_warning_count++; } else if (QCC_PR_Warning(wt, strings + def->s_file, def->s_line, "%s %s no references.", TypeName(def->type, typestr, sizeof(typestr)), def->name)) { warnedunref++; if (warnedunref == 10) QCC_PR_Note(wt, NULL, 0, "Not reporting other unreferenced variables. You can use the noref prefix or pragma to silence these messages as you clearly don't care."); } } pr_scope = NULL; if (opt_unreferenced && def->type->type != ev_field) { optres_unreferenced++; continue; } } if (def->strip) continue; if (def->type->type == ev_function) { if (opt_function_names && def->initialized && functions[G_FUNCTION(def->ofs)].first_statement<0) { optres_function_names++; def->name = ""; } 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->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. continue; } // df = &functions[numfunctions]; // numfunctions++; } else if (def->type->type == ev_field && def->constant && (!def->scope || def->isstatic || def->initialized)) { if (numfielddefs >= MAX_FIELDS) QCC_PR_ParseError(0, "Too many fields. Limit is %u\n", MAX_FIELDS); dd = &fields[numfielddefs]; numfielddefs++; dd->type = def->type->aux_type->type; dd->s_name = QCC_CopyString (def->name); dd->ofs = G_INT(def->ofs); } 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); 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->initialized || 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 || !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; } 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)", strings+dd->s_name, strings+fields[h].s_name); } } } } } if (numglobaldefs > MAX_GLOBALS) QCC_Error(ERR_TOOMANYGLOBALS, "Too many globals - %i\nAdd \"MAX_GLOBALS\" \"%i\" to qcc.cfg", numglobaldefs, (numglobaldefs+32768)&~32767); for (i = 0; i < nummodels; i++) { if (!precache_model[i].used) QCC_PR_Warning(WARN_EXTRAPRECACHE, precache_model[i].filename, precache_model[i].fileline, "Model %s was precached but not directly used", precache_model[i].name); else if (!precache_model[i].block) QCC_PR_Warning(WARN_NOTPRECACHED, precache_model[i].filename, precache_model[i].fileline, "Model %s was used but not precached", precache_model[i].name); } //PrintStrings (); //PrintFunctions (); //PrintFields (); //PrintGlobals (); strofs = (strofs+3)&~3; if (verbose) { printf ("%6i strofs (of %i)\n", strofs, MAX_STRINGS); printf ("%6i numstatements (of %i)\n", numstatements, MAX_STATEMENTS); printf ("%6i numfunctions (of %i)\n", numfunctions, MAX_FUNCTIONS); printf ("%6i numglobaldefs (of %i)\n", numglobaldefs, MAX_GLOBALS); printf ("%6i numfielddefs (%i unique) (of %i)\n", numfielddefs, pr.size_fields, MAX_FIELDS); printf ("%6i numpr_globals (of %i)\n", numpr_globals, MAX_REGS); printf ("%6i nummodels\n", nummodels); printf ("%6i numsounds\n", numsounds); } if (!*destfile) strcpy(destfile, "progs.dat"); if (verbose) 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_FTEH2) { for (i=0 ; i= 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_KKQWSV: case PST_FTE32: #define statements32 ((QCC_dstatement32_t*) statements) for (i=0 ; iMAX_PARMS)?MAX_PARMS:functions[i].numparms); qtestfuncs[i].locals = PRLittleLong (functions[i].locals); for (j = 0; j < MAX_PARMS; j++) qtestfuncs[i].parm_size[j] = PRLittleLong((int)functions[i].parm_size[j]); } SafeWrite (h, qtestfuncs, numfunctions*sizeof(qtest_function_t)); } break; case PST_DEFAULT: case PST_KKQWSV: case PST_FTE32: for (i=0 ; iMAX_PARMS)?MAX_PARMS:functions[i].numparms); functions[i].locals = PRLittleLong (functions[i].locals); } if (progs.blockscompressed&8) { SafeWrite (h, &len, sizeof(int)); //save for later len = QC_encode(progfuncs, numfunctions*sizeof(QCC_dfunction_t), 2, (char *)functions, 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, functions, numfunctions*sizeof(QCC_dfunction_t)); break; default: Sys_Error("structtype error"); } if (verbose >= 2) QCC_PrintFields(); 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 60) { *s++ = '.'; *s++ = '.'; *s++ = '.'; break; } } *s++ = '"'; *s++ = 0; return buf; } 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_dfunction_t *f; switch (type) { case ev_string: sprintf (line, "%s", QCC_PR_String(strings + *(int *)val)); break; case ev_entity: sprintf (line, "entity %i", *(int *)val); break; case ev_function: f = functions + *(int *)val; if (!f) sprintf (line, "undefined function"); else sprintf (line, "%s()", strings + f->s_name); break; case ev_field: def = QCC_PR_DefForFieldOfs ( *(int *)val ); sprintf (line, ".%s", def->name); break; case ev_void: sprintf (line, "void"); break; case ev_float: sprintf (line, "%5.1f", *(float *)val); break; case ev_integer: sprintf (line, "%i", *(int *)val); break; case ev_vector: sprintf (line, "'%5.1f %5.1f %5.1f'", ((float *)val)[0], ((float *)val)[1], ((float *)val)[2]); break; case ev_pointer: sprintf (line, "pointer"); break; default: sprintf (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); sprintf (line,"%i(?""?""?)", ofs); else sprintf (line,"%i(%s)", ofs, def->name); i = strlen(line); for ( ; i<16 ; i++) strcat (line," "); strcat (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]); sprintf (line,"%i(%s)", ofs, s); } else sprintf (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) { printf ("%s\n",QCC_PR_GlobalString(ofs)); }*/ /* ================= PR_PrintStatement ================= */ /*void QCC_PR_PrintStatement (QCC_dstatement_t *s) { int i; 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++) printf (" "); if (s->op == OP_IF || s->op == OP_IFNOT) printf ("%sbranch %i",QCC_PR_GlobalString(s->a),s->b); else if (s->op == OP_GOTO) { printf ("branch %i",s->a); } else if ( (unsigned)(s->op - OP_STORE_F) < 6) { printf ("%s",QCC_PR_GlobalString(s->a)); printf ("%s", QCC_PR_GlobalStringNoContents(s->b)); } else { if (s->a) printf ("%s",QCC_PR_GlobalString(s->a)); if (s->b) printf ("%s",QCC_PR_GlobalString(s->b)); if (s->c) printf ("%s", QCC_PR_GlobalStringNoContents(s->c)); } 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 (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; numtypeinfos++; return &qcc_typeinfo[numtypeinfos-1]; } /* ============== PR_BeginCompilation called before compiling a batch of files, clears the pr struct ============== */ 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; QCC_PR_ResetErrorScope(); pr_scope = NULL; /* numpr_globals = RESERVED_OFS; for (i=0 ; iaux_type = type_void; type_pointer = QCC_PR_NewType("__pointer", ev_pointer, false); type_integer = QCC_PR_NewType("__integer", ev_integer, true); type_variant = QCC_PR_NewType("variant", ev_variant, true); type_variant = QCC_PR_NewType("__variant", ev_variant, true); 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_field->aux_type = type_float; type_integer = QCC_PR_NewType("integer", ev_integer, keyword_integer?true:false); type_integer = QCC_PR_NewType("int", ev_integer, keyword_integer?true:false); 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)->references++; for (i = 0; i < MAX_PARMS; i++) { sprintf(name, "PARM%i", i); QCC_PR_GetDef(type_vector, name, NULL, true, 0, false)->references++; } } else { numpr_globals = RESERVED_OFS; // for (i=0 ; inext = NULL; pr_error_count = 0; pr_warning_count = 0; recursivefunctiontype = 0; } /* ============== PR_FinishCompilation called after all files are compiled to check for errors Returns false if errors were detected. ============== */ int QCC_PR_FinishCompilation (void) { QCC_def_t *d; QCC_type_t *t; int errors; errors = false; // check to make sure all functions prototyped have code for (d=pr.def_head.next ; d ; d=d->next) { if (d->type->type == ev_function && d->constant)// function parms are ok { // f = G_FUNCTION(d->ofs); // if (!f || (!f->code && !f->builtin) ) if (d->initialized==0) { 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) { QCC_PR_EmitClassFromFunction(d, t); pr_scope = NULL; continue; } } QCC_PR_Warning(ERR_NOFUNC, strings + d->s_file, d->s_line, "function %s has no body",d->name); QCC_PR_ParsePrintDef(ERR_NOFUNC, d); bodylessfuncs = true; errors = true; } else if (d->initialized==2) bodylessfuncs = 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 }; void QCC_CRC_Init(unsigned short *crcvalue) { *crcvalue = CRC_INIT_VALUE; } void QCC_CRC_ProcessByte(unsigned short *crcvalue, qbyte data) { *crcvalue = ((*crcvalue << 8) ^ QCC_crctable[(*crcvalue >> 8) ^ data]) & 0xffff; } 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(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(p) Add(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 Add3(char *p, unsigned short *crc, char *file) { char *s; for(s=p;*s;s++) QCC_CRC_ProcessByte(crc, *s); } #define ADD3(p) Add3(p, &crc, file) unsigned short QCC_PR_WriteProgdefs (char *filename) { #define ADD2(p) strncat(file, p, PROGDEFS_MAX_SIZE-1 - strlen(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 and dump //ADD2: dump but don't crc //ADD3: crc but don't dump ADD("\n/* "); if (qcc_targetformat == QCF_HEXEN2 || qcc_targetformat == QCF_FTEH2) ADD3("generated by hcc, do not modify"); else ADD3("file generated by qcc, do not modify"); ADD2("File generated by FTEQCC, relevent for engine modding only, the generated crc must be the same as your engine expects."); ADD(" */\n\ntypedef struct"); ADD2(" globalvars_s"); ADD(qcva("\n{")); ADD2("\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"); ADD3(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->ofstype->type) { case ev_float: ADD(qcva("\tfloat\t%s;\n",d->name)); break; case ev_vector: ADD(qcva("\tvec3_t\t%s;\n",d->name)); if (d->deftail) d=d->deftail; // skip the elements break; case ev_string: ADD(qcva("\tstring_t\t%s;\n",d->name)); break; case ev_function: ADD(qcva("\tfunc_t\t%s;\n",d->name)); break; case ev_entity: ADD(qcva("\tint\t%s;\n",d->name)); break; case ev_integer: ADD(qcva("\tint\t%s;\n",d->name)); break; default: ADD(qcva("\tint\t%s;\n",d->name)); break; } } ADD("} globalvars_t;\n\n"); // print all fields ADD("typedef struct"); ADD2(" entvars_s"); ADD("\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(qcva("\tfloat\t%s;\n",d->name)); break; case ev_vector: ADD(qcva("\tvec3_t\t%s;\n",d->name)); if (d->deftail) d=d->deftail; // skip the elements break; case ev_string: ADD(qcva("\tstring_t\t%s;\n",d->name)); break; case ev_function: ADD(qcva("\tfunc_t\t%s;\n",d->name)); break; case ev_entity: ADD(qcva("\tint\t%s;\n",d->name)); break; case ev_integer: ADD(qcva("\tint\t%s;\n",d->name)); break; default: ADD(qcva("\tint\t%s;\n",d->name)); break; } } ADD("} entvars_t;\n\n"); /* ///temp ADD2("//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) ADD2(",\n"); ADD2(qcva("\t{%i,\t%i,\t\"%s\"}",G_INT(d->ofs), d->type->aux_type->type, d->name)); f = 1; } ADD2("\n};\n#endif\n\n"); //end temp */ ADD2(qcva("#define PROGHEADER_CRC %i\n", crc)); if (QCC_CheckParm("-progdefs")) { printf ("writing %s\n", filename); f = SafeOpenWrite(filename, 16384); SafeWrite(f, file, strlen(file)); SafeClose(f); } if (ForcedCRC) crc = ForcedCRC; switch (crc) { case 12923: //#pragma sourcefile usage break; case 54730: if (verbose) printf("Recognised progs as QuakeWorld\n"); break; case 5927: if (verbose) printf("Recognised progs as NetQuake server gamecode\n"); break; case 26940: if (verbose) printf("Recognised progs as Quake pre-release...\n"); break; case 38488: if (verbose) printf("Recognised progs as original Hexen2\n"); break; case 26905: if (verbose) printf("Recognised progs as Hexen2 Mission Pack\n"); break; case 14046: if (verbose) printf("Recognised progs as Hexen2 (demo)\n"); break; case 22390: //EXT_CSQC_1 if (verbose) printf("Recognised progs as an EXT_CSQC_1 module\n"); break; case 17105: case 32199: //outdated ext_csqc QCC_PR_Warning(WARN_SYSTEMCRC, NULL, 0, "Recognised progs as outdated CSQC module\n"); 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. printf("Recognised progs as DP-specific CSQC module\n"); break; case 10020: if (verbose) printf("Recognised progs as a DP/FTE Menu module\n"); break; case 32401: QCC_PR_Warning(WARN_SYSTEMCRC, NULL, 0, "Warning: please update your tenebrae system defs.\n"); break; default: QCC_PR_Warning(WARN_SYSTEMCRC, NULL, 0, "Warning: progs CRC not recognised from quake nor clones\n"); break; } return crc; } /*void QCC_PrintFunction (char *name) { int i; QCC_dstatement_t *ds; QCC_dfunction_t *df; for (i=0 ; ifirst_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 ; ifirst_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); printf(" \n \n"); } } } */ /* ============================================================================== DIRECTORY COPYING / PACKFILE CREATION ============================================================================== */ typedef struct { char name[56]; int filepos, filelen; } packfile_t; typedef struct { char id[4]; int dirofs; int dirlen; } packheader_t; packfile_t pfiles[4096], *pf; int packhandle; int packbytes; /* ============ CreatePath ============ */ void QCC_CreatePath (char *path) { /* char *ofs; for (ofs = path+1 ; *ofs ; ofs++) { if (*ofs == '/') { // create the directory *ofs = 0; #ifdef QCC mkdir(path); #else QCC_mkdir (path); #endif *ofs = '/'; } } */ } /* =========== PackFile Copy a file into the pak file =========== */ void QCC_PackFile (char *src, char *name) { int remaining; #if 1 char *f; #else int in; int count; char buf[4096]; #endif if ( (qbyte *)pf - (qbyte *)pfiles > sizeof(pfiles) ) QCC_Error (ERR_TOOMANYPAKFILES, "Too many files in pak file"); #if 1 f = FS_ReadToMem(src, NULL, &remaining); if (!f) { 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); printf ("%64s : %7i\n", pf->name, 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); printf ("%64s : %7i\n", pf->name, 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 =========== */ 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 =========== */ 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 0) printf ("%3i unique precache_sounds\n", numsounds); if (nummodels > 0) printf ("%3i unique precache_models\n", nummodels); if (numtextures > 0) printf ("%3i unique precache_textures\n", numtextures); if (numfiles > 0) 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 < 5; 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 value = qccHunkAlloc(strlen(val)+1); memcpy(cnst->value, val, strlen(val)+1); } } //optimisations. else if ( !strnicmp(myargv[i], "-O", 2) || !strnicmp(myargv[i], "/O", 2) ) { qcc_nopragmaoptimise = true; p = 0; if (myargv[i][2] >= '0' && myargv[i][2] <= '3') { } else if (!strnicmp(myargv[i]+2, "no-", 3)) { if (myargv[i][5]) { for (p = 0; optimisations[p].enabled; p++) { if ((*optimisations[p].abbrev && !stricmp(myargv[i]+5, optimisations[p].abbrev)) || !stricmp(myargv[i]+5, optimisations[p].fullname)) { *optimisations[p].enabled = false; break; } } } } else { if (myargv[i][2]) for (p = 0; optimisations[p].enabled; p++) if ((*optimisations[p].abbrev && !stricmp(myargv[i]+2, optimisations[p].abbrev)) || !stricmp(myargv[i]+2, optimisations[p].fullname)) { *optimisations[p].enabled = true; break; } } if (!optimisations[p].enabled) QCC_PR_Warning(0, NULL, WARN_BADPARAMS, "Unrecognised optimisation parameter (%s)", myargv[i]); } else if ( !strnicmp(myargv[i], "-K", 2) || !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(0, NULL, WARN_BADPARAMS, "Unrecognised keyword parameter (%s)", myargv[i]); } else if ( !strnicmp(myargv[i], "-F", 2) || !strnicmp(myargv[i], "/F", 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(0, NULL, WARN_BADPARAMS, "Unrecognised flag parameter (%s)", myargv[i]); } else if ( !strncmp(myargv[i], "-T", 2) || !strncmp(myargv[i], "/T", 2) ) { p = 0; for (p = 0; targets[p].name; p++) if (!stricmp(myargv[i]+2, targets[p].name)) { qcc_targetformat = targets[p].target; break; } if (!targets[p].name) QCC_PR_Warning(0, NULL, WARN_BADPARAMS, "Unrecognised target parameter (%s)", myargv[i]); } else if ( !strnicmp(myargv[i], "-W", 2) || !strnicmp(myargv[i], "/W", 2) ) { if (!stricmp(myargv[i]+2, "all")) { for (j = 0; j < ERR_PARSEERRORS; j++) if (qccwarningaction[j] == WA_IGNORE) { if (j == WARN_FIXEDRETURNVALUECONFLICT) continue; qccwarningaction[j] = WA_WARN; } } else if (!stricmp(myargv[i]+2, "extra")) { for (j = 0; j < ERR_PARSEERRORS; j++) if (qccwarningaction[j] == WA_IGNORE) qccwarningaction[j] = WA_WARN; } else if (!stricmp(myargv[i]+2, "none")) { for (j = 0; j < ERR_PARSEERRORS; j++) qccwarningaction[j] = WA_IGNORE; } else if(!stricmp(myargv[i]+2, "error")) { werror = true; } else if (!stricmp(myargv[i]+2, "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_FIXEDRETURNVALUECONFLICT] = WA_IGNORE; qccwarningaction[WARN_EXTRAPRECACHE] = WA_IGNORE; qccwarningaction[WARN_CORRECTEDRETURNTYPE] = WA_IGNORE; qccwarningaction[WARN_NOTUTF8] = WA_IGNORE; qccwarningaction[WARN_SELFNOTTHIS] = WA_IGNORE; } else { p = -1; if (!strnicmp(myargv[i]+2, "error-", 6)) { p = QCC_WarningForName(myargv[i]+8); if (p >= 0) qccwarningaction[p] = WA_ERROR; } else if (!strnicmp(myargv[i]+2, "no-", 3)) { p = QCC_WarningForName(myargv[i]+5); if (p >= 0) qccwarningaction[p] = WA_IGNORE; } else { p = QCC_WarningForName(myargv[i]+2); if (p >= 0) qccwarningaction[p] = WA_WARN; } if (p < 0) QCC_PR_Warning(0, NULL, WARN_BADPARAMS, "Unrecognised warning parameter (%s)", myargv[i]); } } } if (werror) { for (j = 0; j < ERR_PARSEERRORS; j++) if (qccwarningaction[j]) qccwarningaction[j] = WA_ERROR; } } /* ============ main ============ */ char *qccmsrc; char *qccmsrc2; char qccmfilename[1024]; char qccmprogsdat[1024]; char qccmsourcedir[1024]; void QCC_FinishCompile(void); void SetEndian(void); void QCC_SetDefaultProperties (void) { int level; int i; Hash_InitTable(&compconstantstable, MAX_CONSTANTS, qccHunkAlloc(Hash_BytesForBuckets(MAX_CONSTANTS))); ForcedCRC = 0; defaultstatic = 0; QCC_PR_DefineName("FTEQCC"); if (QCC_CheckParm("/Oz")) { qcc_targetformat = QCF_FTE; QCC_PR_DefineName("OP_COMP_STATEMENTS"); QCC_PR_DefineName("OP_COMP_DEFS"); QCC_PR_DefineName("OP_COMP_FIELDS"); QCC_PR_DefineName("OP_COMP_FUNCTIONS"); QCC_PR_DefineName("OP_COMP_STRINGS"); QCC_PR_DefineName("OP_COMP_GLOBALS"); QCC_PR_DefineName("OP_COMP_LINES"); QCC_PR_DefineName("OP_COMP_TYPES"); } if (QCC_CheckParm("/O0") || QCC_CheckParm("-O0")) level = 0; else if (QCC_CheckParm("/O1") || QCC_CheckParm("-O1")) level = 1; else if (QCC_CheckParm("/O2") || QCC_CheckParm("-O2")) level = 2; else if (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; } } if (QCC_CheckParm ("-h2")) qcc_targetformat = QCF_HEXEN2; else if (QCC_CheckParm ("-fte")) qcc_targetformat = QCF_FTE; else if (QCC_CheckParm ("-fteh2")) qcc_targetformat = QCF_FTEH2; else if (QCC_CheckParm ("-dp")) qcc_targetformat = QCF_DARKPLACES; else qcc_targetformat = QCF_STANDARD; //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_FIXEDRETURNVALUECONFLICT] = WA_IGNORE; qccwarningaction[WARN_EXTRAPRECACHE] = WA_IGNORE; qccwarningaction[WARN_DEADCODE] = WA_IGNORE; qccwarningaction[WARN_FTE_SPECIFIC] = 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_WARN;//FIXME: make into WA_ERROR; if (QCC_CheckParm("-h2")) qccwarningaction[WARN_CASEINSENSITIVEFRAMEMACRO] = WA_IGNORE; //Check the command line QCC_PR_CommandLinePrecompilerOptions(); if (qcc_targetformat == QCF_HEXEN2 || qcc_targetformat == QCF_FTEH2) //force on the thinktime keyword if hexen2 progs. { keyword_thinktime = true; keyword_until = true; keyword_loop = true; } if (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! 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 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; } int qcc_compileactive = false; extern int accglobalsblock; extern int qcc_debugflag; char *originalqccmsrc; //for autoprototype. pbool QCC_main (int argc, char **argv) //as part of the quake engine { extern int pr_bracelevel, tempsused; time_t long_time; extern QCC_type_t *pr_classtype; int p; #ifndef QCCONLY char destfile2[1024], *s2; #endif char *s; if (numsourcefiles && currentsourcefile == numsourcefiles) { numsourcefiles = 0; return false; } 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 = 1000000; MAX_GLOBALS = 1<<17; MAX_FIELDS = 1<<12; MAX_STATEMENTS = 0x80000; MAX_FUNCTIONS = 16384; maxtypeinfos = 16384; MAX_CONSTANTS = 2048; compressoutput = 0; p = externs->FileSize("qcc.cfg"); if (p < 0) p = externs->FileSize("src/qcc.cfg"); if (p>0) { s = qccHunkAlloc(p+1); s = externs->ReadFile("qcc.cfg", s, p); 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 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]", localtime( &long_time )); 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; } autoprototyped = autoprototype = false; QCC_SetDefaultProperties(); 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_test1 = 0; optres_test2 = 0; accglobalsblock = 0; qcc_debugflag = false; numtemps = 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_dfunction_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(&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 for (p = 1; p\n"); // printf ("to build a clean data tree: qcc -copy \n"); // printf ("to build a clean pak file: qcc -pak \n"); // printf ("to bsp all bmodels: qcc -bspmodels \n"); printf ("-Fwasm causes FTEQCC to dump all asm to qc.asm\n"); printf ("-O0 to disable optimisations\n"); printf ("-O1 to optimise for size\n"); printf ("-O2 to optimise more - some behaviours may change\n"); printf ("-O3 to optimise lots - experimental or non-future-proof\n"); printf ("-Oname to enable an optimisation\n"); printf ("-Ono-name to disable optimisations\n"); printf ("-Kkeyword to activate keyword\n"); printf ("-Kno-keyword to disable keyword\n"); printf ("-Wall to give a stupid number of warnings\n"); printf ("-Ttarget to set a output format\n"); printf ("-Fautoproto to enable automatic prototyping\n"); printf ("-Fsubscope to make locals specific to their subscope\n"); qcc_compileactive = false; return true; } if (flag_caseinsensitive) { printf("Compiling without case sensitivity\n"); pHash_Get = &Hash_GetInsensitive; pHash_GetNext = &Hash_GetNextInsensitive; pHash_Add = &Hash_AddInsensitive; pHash_RemoveData = &Hash_RemoveDataInsensitive; } p = QCC_CheckParm ("-src"); if (p && p < argc-1 ) { strcpy (qccmsourcedir, argv[p+1]); strcat (qccmsourcedir, "/"); printf ("Source directory: %s\n", qccmsourcedir); } else *qccmsourcedir = '\0'; QCC_InitData (); QCC_PR_BeginCompilation ((void *)qccHunkAlloc (0x100000), 0x100000); QCC_PR_ClearGrabMacros (false); if (flag_acc) { 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 || argv[p+1][0] == '-') p = QCC_CheckParm ("-srcfile"); if (p && p < argc-1 ) sprintf (qccmprogsdat, "%s", argv[p+1]); else { //look for a preprogs.src... :o) sprintf (qccmprogsdat, "%spreprogs.src", qccmsourcedir); if (externs->FileSize(qccmprogsdat) <= 0) sprintf (qccmprogsdat, "progs.src"); else sprintf (qccmprogsdat, "preprogs.src"); } numsourcefiles = 0; strcpy(sourcefileslist[numsourcefiles++], qccmprogsdat); currentsourcefile = 0; } else if (currentsourcefile == numsourcefiles) { //no more. qcc_compileactive = false; numsourcefiles = 0; currentsourcefile = 0; return true; } if (currentsourcefile) printf("-------------------------------------\n"); sprintf (qccmprogsdat, "%s%s", qccmsourcedir, sourcefileslist[currentsourcefile++]); printf ("Source file: %s\n", qccmprogsdat); 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; 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; } compilingfile = qccmprogsdat; if (*qcc_token == '#') { void StartNewStyleCompile(void); newstyle: newstylesource = true; originalqccmsrc = qccmsrc; StartNewStyleCompile(); return true; } 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; if (!qccmsrc) QCC_Error (ERR_NOOUTPUT, "No destination filename. qcc -help for info."); strcpy (destfile, qcc_token); #ifndef QCCONLY p=0; s2 = strcpy(destfile2, destfile); if (!strncmp(s2, "./", 2)) s2+=2; else { while(!strncmp(s2, "../", 3)) { s2+=3; p++; } } strcpy(qccmfilename, qccmsourcedir); for (s=qccmfilename+strlen(qccmfilename);p && s>=qccmfilename; s--) { if (*s == '/' || *s == '\\') { *(s+1) = '\0'; p--; } } sprintf(destfile, "%s", s2); while (p>0) { memmove(destfile+3, destfile, strlen(destfile)+1); destfile[0] = '.'; destfile[1] = '.'; destfile[2] = '/'; p--; } #endif 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) { printf("%s changed\n", qcc_token); modified = true; break; } } if (!modified) { printf("No changes\n"); qcc_compileactive = false; return true; } else { pr_file_p = qccmsrc; } } } 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) { char *s, *s2; currentchunk = NULL; 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; } qccmsrc = QCC_COM_Parse(qccmsrc); if (!qccmsrc) { 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; } s = qcc_token; strcpy (qccmfilename, qccmsourcedir); while(1) { if (!strncmp(s, "..\\", 3) || !strncmp(s, "../", 3)) { s2 = qccmfilename + strlen(qccmfilename)-2; while (s2>=qccmfilename) { if (*s2 == '/' || *s2 == '\\') { s2[1] = '\0'; break; } s2--; } if (s2>=qccmfilename) { s+=3; continue; } } if (!strncmp(s, ".\\", 2) || !strncmp(s, "./", 2)) { s+=2; continue; } break; } strcat (qccmfilename, s); if (autoprototype) printf ("prototyping %s\n", qccmfilename); else { 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, ""); if (!QCC_PR_FinishCompilation ()) { QCC_Error (ERR_PARSEERRORS, "compilation errors"); } /* p = QCC_CheckParm ("-asm"); if (p) { for (p++ ; p MAX_ERRORS) return; if (setjmp(pr_parse_abort)) return; QCC_PR_SkipToSemicolon (); if (pr_token_type == tt_eof) return; } compilingfile = qccmprogsdat; pr_file_p = qccmsrc; s_file = s_file2 = QCC_CopyString (compilingfile); 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) { qccmsrc = originalqccmsrc; pr_file_p = qccmsrc; s_file = s_file2 = QCC_CopyString (compilingfile); autoprototyped = autoprototype; QCC_SetDefaultProperties(); autoprototype = false; QCC_PR_NewLine(false); QCC_PR_Lex(); return; } else { QCC_FinishCompile(); PostCompile(); if (!QCC_main(myargc, myargv)) return; return; } } pr_scope = NULL; // outside all functions QCC_PR_ParseDefs (NULL); } /*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); // 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