new progs format proposal for engine developers (45% of globals are 0, why write...
authorDale Weiler <killfieldengine@gmail.com>
Tue, 24 Apr 2012 12:19:48 +0000 (08:19 -0400)
committerDale Weiler <killfieldengine@gmail.com>
Tue, 24 Apr 2012 12:19:48 +0000 (08:19 -0400)
code.c
gmqcc.h
main.c
propsal.txt [new file with mode: 0644]

diff --git a/code.c b/code.c
index 66dc9517a68f008abf498748c1f9e0728246bd70..392508832792f25855734249ed03db5280d4f68a 100644 (file)
--- a/code.c
+++ b/code.c
@@ -28,8 +28,11 @@ typedef struct {
 } prog_section;
 
 typedef struct {
-    uint32_t     version;      /* Program version (6)     */
-    uint32_t     crc16;        /* What is this?           */
+    uint16_t     version;      /* Program version (6)     */
+    uint16_t     flags;        /* see propsal.txt         */
+    uint16_t     crc16;        /* What is this?           */
+    uint16_t     skip;         /* see propsal.txt         */
+    
     prog_section statements;   /* prog_section_statement  */
     prog_section defs;         /* prog_section_def        */
     prog_section fields;       /* prog_section_field      */
@@ -90,6 +93,10 @@ int code_strings_add(const char *src) {
 }
 
 void code_init() {
+    /* omit creation of null code */
+    if (opts_omit_nullcode)
+        return;
+        
     /*
      * The way progs.dat is suppose to work is odd, there needs to be
      * some null (empty) statements, functions, and 28 globals
@@ -137,6 +144,13 @@ void code_test() {
 
 void code_write() {
     prog_header code_header={0};
+
+    /* see proposal.txt */
+    if (opts_omit_nullcode) {
+        code_header.skip   = 28;
+        code_header.flags  = 1;
+    }
+    
     code_header.version    = 6;
     code_header.crc16      = 0; /* TODO: */
     code_header.statements = (prog_section){sizeof(prog_header), code_statements_elements };
@@ -150,7 +164,7 @@ void code_write() {
     if (opts_darkplaces_stringtablebug) {
         util_debug("GEN", "Patching stringtable for -fdarkplaces-stringtablebug\n");
 
-        /* >= + padd */
+        /* >= + P */
         code_chars_add('\0'); /* > */
         code_chars_add('\0'); /* = */
         code_chars_add('\0'); /* P */
diff --git a/gmqcc.h b/gmqcc.h
index f27f41ea982eb6edf23f485aded40eedcd29d7cc..5edaeae14bd8172c9f0bcbf98d007c0d94174e7a 100644 (file)
--- a/gmqcc.h
+++ b/gmqcc.h
@@ -485,4 +485,5 @@ enum {
 extern int opts_debug;
 extern int opts_memchk;
 extern int opts_darkplaces_stringtablebug;
+extern int opts_omit_nullcode;
 #endif
diff --git a/main.c b/main.c
index 2c0be85ca4797ac62cb0b4d8bad6f1f511dc724e..e24a7fe6bae9a92b932edb3409fc075e66ab1ff5 100644 (file)
--- a/main.c
+++ b/main.c
@@ -30,6 +30,7 @@ int opts_debug                     = 0;
 int opts_memchk                    = 0;
 int opts_compiler                  = COMPILER_GMQCC;
 int opts_darkplaces_stringtablebug = 0;
+int opts_omit_nullcode             = 0;
 
 static const int usage(const char *const app) {
     printf("usage:\n");
@@ -48,8 +49,9 @@ static const int usage(const char *const app) {
     printf("            -std=ftqecc  -- fteqcc QuakeC\n");
     printf("            -std=qccx    -- qccx QuakeC\n");
     printf("            -std=gmqcc   -- this compiler QuakeC (default selection)\n");
-    printf("    code flags -f*\n");
-    printf("        -fdarkplaces-stringtablebug -- patches the string table to work with bugged versions of darkplaces\n");
+    printf("    codegen flags:\n");
+    printf("        -fdarkplaces-string-table-bug -- patches the string table to work with bugged versions of darkplaces\n");
+    printf("        -fomit-nullcode               -- omits the generation of null code (will break everywhere see propsal.txt)\n");
     return -1;
 }
 
@@ -92,10 +94,14 @@ int main(int argc, char **argv) {
                 }
 
                 /* code specific switches */
-                if (!strncmp(&argv[1][1], "fdarkplaces-stringtablebug", 26)) {
+                if (!strcmp(&argv[1][1], "fdarkplaces-string-table-bug")) {
                     opts_darkplaces_stringtablebug = 1;
                     break;
                 }
+                if (!strcmp(&argv[1][1], "fomit-nullcode")) {
+                    opts_omit_nullcode = 1;
+                    break;
+                }
                 return usage(app);
                 
         }
diff --git a/propsal.txt b/propsal.txt
new file mode 100644 (file)
index 0000000..88658e1
--- /dev/null
@@ -0,0 +1,35 @@
+This is a propsal to extend the progs.dat file format without breaking
+backwards compatability.  Currently the progs file format header has a
+description simaler to this:
+
+struct {
+    uint32_t     version;
+    uint32_t     crc16;
+    ....
+    uint32_t     entfield;
+}
+
+The obvious notable issue here is version and crc16 are larger than they
+essentially need to be, if we made version and crc16 both uint16_t we can
+give ourselfs 32 bytes (2x16) to store additional data that can be used
+to make smaller progs.dat files.
+
+I propose a new structual layout like this:
+struct {
+    uint16_t     version;
+    uint16_t     flags;   /* contains a skip field */
+    uint16_t     crc16;
+    uint16_t     skip;    /* skiped globals        */
+    ....
+    uint32_t     entfield;
+}
+
+about 45% of globals are zero, if we could order them at the top of the
+globals array we can essentially use the skip field to specify how much
+zero globals the engine would have to populate (instead of being stored
+in the actual file itself) flags can specify if the progs.dat file skiped
+globals on the write of the progs.dat file.
+
+Of course only one bit in the flags would have to be set to specify if the
+file contains a skip field.  Which lends itself to the fact that flags could
+later be extended for other things.