X-Git-Url: https://pintos-os.org/cgi-bin/gitweb.cgi?a=blobdiff_plain;f=dump.c;h=f478131555e2ef126837096f65f4a292b2d02741;hb=815d7313445b7bf351178ecb5171ad5f46a4e83f;hp=96611d976e2da517fa48dbbbc3d2d46aa321dd9a;hpb=f5abc4d9b95aed4e1ab2734eec9e40cccb0547d8;p=pspp diff --git a/dump.c b/dump.c index 96611d976e..f478131555 100644 --- a/dump.c +++ b/dump.c @@ -104,7 +104,7 @@ match_u32_assert(uint32_t x, const char *where) static bool match_byte(uint8_t b) { - if (data[pos] == b) + if (pos < n && data[pos] == b) { pos++; return true; @@ -148,278 +148,22 @@ get_string(const char *where) #define get_string() get_string(WHERE) static void -dump_value(int level) +dump_value_31(void) { - for (int i = 0; i <= level; i++) - printf (" "); - - match_byte (0); - match_byte (0); - match_byte (0); - match_byte (0); - if (match_byte (3)) + if (match_byte (0x31)) { - get_string(); - if (match_byte (0x31)) + if (match_u32 (0)) { if (match_u32 (1)) - { - printf("(footnote %d) ", get_u32()); - match_byte_assert (0); - match_byte_assert (0); - int subn = get_u32 (); - printf ("nested %d bytes", subn); - pos += subn; - } - else if (match_u32 (2)) - { - printf("(special 2)"); - match_byte_assert(0); - match_byte_assert(0); - match_u32_assert(1); - match_byte_assert(0); - match_byte_assert(0); - int subn = get_u32 (); - printf ("nested %d bytes", subn); - pos += subn; - } + get_string(); else - { - match_u32_assert(3); - printf("(special 3)"); - match_byte_assert(0); - match_byte_assert(0); - match_byte_assert(1); - match_byte_assert(0); - int subn = get_u32 (); - printf ("nested %d bytes, ", subn); - pos += subn; - subn = get_u32 (); - printf ("nested %d bytes, ", subn); - pos += subn; - } - } - else - match_byte_assert (0x58); - get_string(); - printf("string \"%s\"", get_string()); - match_byte (0); - match_byte (0); - match_byte (0); - match_byte (1); - match_byte (1); - match_byte (0); - match_byte (0); - match_byte (0); - match_byte (1); - } - else if (match_byte (5)) - { - match_byte_assert (0x58); - printf ("variable \"%s\"", get_string()); - get_string(); - if (!match_byte(1) && !match_byte(2)) - match_byte_assert(3); - match_byte (0); - match_byte (0); - match_byte (0); - match_byte (0); - } - else if (match_byte (2)) - { - unsigned int format; - char *var, *vallab; - double value; - - match_byte_assert (0x58); - format = get_u32 (); - value = get_double (); - var = get_string (); - vallab = get_string (); - printf ("value %g format %d(%d.%d) var \"%s\" vallab \"%s\"", - value, format >> 16, (format >> 8) & 0xff, format & 0xff, var, vallab); - if (!match_byte (1) && !match_byte(2)) - match_byte_assert (3); - match_byte (0); - match_byte (0); - match_byte (0); - match_byte (0); - match_byte (0); - match_byte (0); - match_byte (0); - } - else if (match_byte (4)) - { - unsigned int format; - char *var, *vallab, *value; - - match_byte_assert (0x58); - format = get_u32 (); - vallab = get_string (); - var = get_string (); - if (!match_byte(1) && !match_byte(2)) - match_byte_assert (3); - value = get_string (); - printf ("value \"%s\" format %d(%d.%d) var \"%s\" vallab \"%s\"", - value, format >> 16, (format >> 8) & 0xff, format & 0xff, var, vallab); - match_byte (0); - match_byte (0); - match_byte (0); - match_byte (0); - } - else if (match_byte (1)) - { - unsigned int format; - double value; - - if (match_byte (0x31)) - { - if (match_u32 (1)) - { - printf("(footnote %d) ", get_u32()); - match_byte_assert (0); - match_byte_assert (0); - int subn = get_u32 (); - printf ("nested %d bytes", subn); - pos += subn; - } - } - else - match_byte_assert (0x58); - format = get_u32 (); - value = get_double (); - printf ("value %g format %d(%d.%d)", value, format >> 16, (format >> 8) & 0xff, format & 0xff); - match_byte (1); - match_byte (0); - match_byte (0); - match_byte (0); - match_byte (1); - } - else if (match_byte (0x31)) - { - if (match_u32 (1)) - { - printf("(footnote %d) ", get_u32()); - match_byte_assert (0); - match_byte_assert (0); - int subn = get_u32 (); - printf ("nested %d bytes", subn); - pos += subn; - } - else - { - match_u32_assert (0); - match_u32_assert (0); + match_u32_assert (0); int subn = get_u32 (); printf ("nested %d bytes", subn); pos += subn; } - char *base = get_string(); - int x = get_u32(); - printf ("\"%s\"; %d variables:\n", base, x); - if (match_u32(0)) - { - for (int i = 0; i < x; i++) - { - dump_value (level+1); - putchar('\n'); - } - } - else - { - for (int i = 0; i < x; i++) - { - int y = get_u32(); - match_u32_assert(0); - for (int j = 0; j <= level; j++) - printf (" "); - printf("variable %d has %d values:\n", i, y); - for (int j = 0; j < y; j++) - { - if (match_byte(3)) - { - char *a = get_string(); - match_byte_assert(0x58); - char *b = get_string(); - char *c = get_string(); - for (int k = 0; k <= level + 1; k++) - printf (" "); - printf ("\"%s\", \"%s\", \"%s\"", a, b, c); - match_byte(0); - match_byte(0); - match_byte(0); - match_byte(0); - match_byte(0); - } - else - dump_value (level+1); - putchar('\n'); - } - } - } - } - else - { - - match_byte_assert (0x58); - char *base = get_string(); - int x = get_u32(); - printf ("\"%s\" with %d variables:\n", base, x); - if (match_u32(0)) - { - for (int i = 0; i < x; i++) - { - dump_value (level+1); - putchar('\n'); - } - } - else - { - for (int i = 0; i < x; i++) - { - int y = get_u32(); - match_u32_assert(0); - for (int j = 0; j <= level; j++) - printf (" "); - printf("variable %d has %d values:\n", i, y); - for (int j = 0; j < y; j++) - { - if (match_byte(3)) - { - char *a = get_string(); - match_byte_assert(0x58); - char *b = get_string(); - char *c = get_string(); - for (int k = 0; k <= level + 1; k++) - printf (" "); - printf ("\"%s\", \"%s\", \"%s\"", a, b, c); - match_byte(0); - match_byte(0); - match_byte(0); - match_byte(0); - match_byte(0); - } - else - dump_value (level+1); - putchar('\n'); - } - } - } - } -} - -static void -dump_dim_value(int level) -{ - for (int i = 0; i <= level; i++) - printf (" "); - - if (match_byte (3)) - { - get_string(); - if (match_byte (0x31)) + else if (match_u32 (1)) { - match_u32 (1); printf("(footnote %d) ", get_u32()); match_byte_assert (0); match_byte_assert (0); @@ -427,666 +171,68 @@ dump_dim_value(int level) printf ("nested %d bytes", subn); pos += subn; } - else - match_byte_assert (0x58); - get_string(); - printf("string \"%s\"", get_string()); - match_byte (0); - match_byte_assert (1); - match_byte (0); - match_byte (0); - match_byte (0); - match_byte (1); - } - else if (match_byte (5)) - { - match_byte_assert (0x58); - printf ("variable \"%s\"", get_string()); - get_string(); - match_byte_assert (2); - } - else if (match_byte (2)) - { - unsigned int format; - char *var, *vallab; - double value; - - match_byte_assert (0x58); - format = get_u32 (); - value = get_double (); - var = get_string (); - vallab = get_string (); - printf ("value %g format %d(%d.%d) var \"%s\" vallab \"%s\"", - value, format >> 16, (format >> 8) & 0xff, format & 0xff, var, vallab); - if (!match_u32 (3)) - match_u32_assert (2); - match_byte (0); - } - else if (match_byte (1)) - { - unsigned int format; - double value; - - match_byte_assert (0x58); - format = get_u32 (); - value = get_double (); - printf ("value %g format %d(%d.%d)", value, format >> 16, (format >> 8) & 0xff, format & 0xff); - match_byte (1); - match_byte (0); - match_byte (0); - match_byte (0); - match_byte (1); - } - else - { - int subn; - - match_byte (0); - match_byte_assert (0x31); - match_u32_assert (0); - match_u32_assert (0); - subn = get_u32 (); - printf ("nested %d bytes", subn); - pos += subn; - printf ("; \"%s\", substitutions:", get_string()); - int total_subs = get_u32(); - int x = get_u32(); - if (x) - { - total_subs = (total_subs - 1) + x; - match_u32_assert (0); - } - printf (" (total %d)", total_subs); - - for (int i = 0; i < total_subs; i++) - { - putc ('\n', stdout); - dump_value (level + 1); - } - } -} - -static void -dump_category(int level) -{ - match_byte (0); - match_byte (0); - match_byte (0); - match_byte (0); - dump_value (level); - - if (match_u32 (2)) - get_u32 (); - else if (match_u32 (1)) - { - match_byte (0); - match_byte (0); - match_byte (0); - get_u32 (); - } - else if (match_byte (1)) - { - match_byte (0); - if (!match_u32 (2)) - match_u32_assert (1); - match_byte (0); - get_u32(); - } - else - { - match_u32_assert (0); - get_u32 (); - } - - int n_categories = get_u32(); - if (n_categories > 0) - printf (", %d subcategories:", n_categories); - printf("\n"); - for (int i = 0; i < n_categories; i++) - dump_category (level + 1); -} - -static void -dump_dim(void) -{ - int n_categories; - printf("next dim\n"); - match_byte(0); - if (match_byte(3)) - { - get_string(); - match_byte_assert(0x58); - get_string(); - printf("string \"%s\": ", get_string()); - match_byte(1) || match_byte(0); - } - else if (match_byte(5)) - { - match_byte_assert(0x58); - printf("variable \"%s\": ", get_string()); - get_string(); - if (!match_byte(2)) - match_byte_assert(3); - } - else if (match_byte(0x31)) - { - int subn; - int total_subs = 1; - - match_u32_assert (0); - match_u32_assert (0); - subn = get_u32 (); - printf ("nested %d bytes", subn); - pos += subn; - printf ("; \"%s\", substitutions:", get_string()); - for (;;) - { - int n_subst = get_u32(); - if (!n_subst) - break; - printf (" %d", n_subst); - total_subs *= n_subst; - } - - for (int i = 0; i < total_subs; i++) - { - putc ('\n', stdout); - dump_dim_value (0); - } - } - else - { - int total_subs = 1; - - match_byte_assert (0x58); - printf ("\"%s\" with substitutions:", get_string()); - for (;;) - { - int n_subst = get_u32(); - if (!n_subst) - break; - printf (" %d", n_subst); - total_subs *= n_subst; - } - - for (int i = 0; i < total_subs; i++) - { - putc ('\n', stdout); - dump_dim_value (0); - } - } - - /* This byte is usually 0x02 but 0x00 and 0x75 (!) have also been spotted. */ - pos++; - - if (!match_byte(0) && !match_byte(1)) - match_byte_assert(2); - if (!match_u32(0)) - match_u32_assert(2); - if (!match_byte(0)) - match_byte_assert(1); - match_byte(0); - match_byte(0); - match_byte(0); - match_byte(0); - get_u32(); - match_byte(0); - match_byte(0); - match_byte(0); - match_byte(0); - n_categories = get_u32(); - printf("%d nested categories\n", n_categories); - for (int i = 0; i < n_categories; i++) - dump_category (0); -} - -int n_dims; -static void -dump_dims(void) -{ - n_dims = get_u32(); - printf ("%u dimensions\n", n_dims); - for (int i = 0; i < n_dims; i++) - { - printf("\n"); - dump_dim (); - } -} - -static void -dump_data(void) -{ -#if 1 - int a[16]; - for (int i = 0; i < 3 + n_dims; i++) - a[i] = get_u32(); - printf ("data intro:"); - for (int i = 0; i < 3 + n_dims; i++) - printf(" %d", a[i]); - printf("\n"); -#else - fprintf (stderr,"data intro (%d dims):", n_dims); - for (int i = 0; i < 3+n_dims; i++) - fprintf (stderr," %d", get_u32()); - fprintf(stderr,"\n"); -#endif - int x = get_u32(); - printf ("%d data values, starting at %08x\n", x, pos); - for (int i = 0; i < x; i++) - { - printf("%08x, index %d:\n", pos, get_u32()); - match_u32_assert(0); - match_byte(0); - match_byte(0); - match_byte(0); - match_byte(0); - if (match_byte (1)) - { - unsigned int format; - double value; - - if (match_byte (0x31)) - { - if (match_u32 (0)) - { - if (match_u32 (1)) - get_string(); - else - match_u32_assert (0); - int subn = get_u32 (); - printf ("nested %d bytes", subn); - pos += subn; - } - else if (match_u32 (1)) - { - printf("(footnote %d) ", get_u32()); - match_byte_assert (0); - match_byte_assert (0); - int subn = get_u32 (); - printf ("nested %d bytes", subn); - pos += subn; - } - else - { - match_u32_assert(2); - printf("(special 2)"); - match_byte_assert(0); - match_byte_assert(0); - match_u32_assert(1); - match_byte_assert(0); - match_byte_assert(0); - int subn = get_u32 (); - printf ("nested %d bytes", subn); - pos += subn; - } - } - else - match_byte_assert (0x58); - format = get_u32 (); - value = get_double (); - printf (" value %g format %d(%d.%d)", value, format >> 16, (format >> 8) & 0xff, format & 0xff); - } - else if (match_byte (3)) - { - get_string(); - if (match_byte (0x31)) - { - if (match_u32 (0)) - { - match_u32_assert (1); - get_string (); - int subn = get_u32 (); - printf ("nested %d bytes", subn); - pos += subn; - } - else if (match_u32 (1)) - { - printf("(footnote %d) ", get_u32()); - match_byte_assert (0); - match_byte_assert (0); - int subn = get_u32 (); - printf ("nested %d bytes", subn); - pos += subn; - } - else if (match_u32 (2)) - { - printf("(special 2)"); - match_byte_assert(0); - match_byte_assert(0); - match_u32_assert(1); - match_byte_assert(0); - match_byte_assert(0); - int subn = get_u32 (); - printf ("nested %d bytes", subn); - pos += subn; - } - else - { - match_u32_assert(3); - printf("(special 3)"); - match_byte_assert(0); - match_byte_assert(0); - match_byte_assert(1); - match_byte_assert(0); - int subn = get_u32 (); - printf ("nested %d bytes, ", subn); - pos += subn; - subn = get_u32 (); - printf ("nested %d bytes, ", subn); - pos += subn; - } - } - else - match_byte_assert (0x58); - get_string(); - printf("string \"%s\"", get_string()); - match_byte (0); - } - else if (match_byte (2)) - { - unsigned int format; - char *var, *vallab; - double value; - - match_byte_assert (0x58); - format = get_u32 (); - value = get_double (); - var = get_string (); - vallab = get_string (); - printf ("value %g format %d(%d.%d) var \"%s\" vallab \"%s\"", - value, format >> 16, (format >> 8) & 0xff, format & 0xff, var, vallab); - if (!match_byte (1) && !match_byte(2)) - match_byte_assert (3); - } - else if (match_byte (4)) - { - unsigned int format; - char *var, *vallab, *value; - - match_byte_assert (0x58); - format = get_u32 (); - vallab = get_string (); - var = get_string (); - if (!match_byte(1) && !match_byte(2)) - match_byte_assert (3); - value = get_string (); - printf ("value \"%s\" format %d(%d.%d) var \"%s\" vallab \"%s\"", - value, format >> 16, (format >> 8) & 0xff, format & 0xff, var, vallab); - } - else if (match_byte (5)) - { - match_byte_assert (0x58); - printf ("variable \"%s\"", get_string()); - get_string(); - if (!match_byte(1) && !match_byte(2)) - match_byte_assert(3); - match_byte (0); - match_byte (0); - match_byte (0); - match_byte (0); - } - else if (match_byte(0x31)) + else if (match_u32 (2)) { - if (match_u32 (1)) - { - printf("(footnote %d) ", get_u32()); - match_byte_assert (0); - match_byte_assert (0); - int subn = get_u32 (); - printf ("nested %d bytes", subn); - pos += subn; - } - else - { - match_u32_assert (0); - match_u32_assert (0); - int subn = get_u32 (); - printf ("nested %d bytes", subn); - pos += subn; - } - char *base = get_string(); - int x = get_u32(); - printf ("\"%s\"; %d variables:\n", base, x); - for (int i = 0; i < x; i++) - { - int y = get_u32(); - if (!y) - y = 1; - else - match_u32_assert(0); - for (int j = 0; j <= 0; j++) - printf (" "); - printf("variable %d has %d values:\n", i, y); - for (int j = 0; j < y; j++) - { - if (match_byte (1)) - { - unsigned int format; - double value; - - if (match_byte (0x31)) - { - if (match_u32 (0)) - { - if (match_u32 (1)) - get_string(); - else - match_u32_assert (0); - int subn = get_u32 (); - printf ("nested %d bytes", subn); - pos += subn; - } - else if (match_u32 (1)) - { - printf("(footnote %d) ", get_u32()); - match_byte_assert (0); - match_byte_assert (0); - int subn = get_u32 (); - printf ("nested %d bytes", subn); - pos += subn; - } - else - { - match_u32_assert(2); - printf("(special 2)"); - match_byte_assert(0); - match_byte_assert(0); - match_u32_assert(1); - match_byte_assert(0); - match_byte_assert(0); - int subn = get_u32 (); - printf ("nested %d bytes", subn); - pos += subn; - } - } - else - match_byte_assert (0x58); - format = get_u32 (); - value = get_double (); - printf (" value %g format %d(%d.%d)", value, format >> 16, (format >> 8) & 0xff, format & 0xff); - } - else if (match_byte(3)) - { - char *a = get_string(); - match_byte_assert(0x58); - char *b = get_string(); - char *c = get_string(); - for (int k = 0; k <= 1; k++) - printf (" "); - printf ("\"%s\", \"%s\", \"%s\"", a, b, c); - match_byte(0); - match_byte(0); - match_byte(0); - match_byte(0); - } - else if (match_byte(5)) - { - match_byte_assert (0x58); - printf ("variable \"%s\"", get_string()); - get_string(); - if (!match_byte(1) && !match_byte(2)) - match_byte_assert(3); - match_byte (0); - match_byte (0); - match_byte (0); - match_byte (0); - } - else - dump_value (0); - putchar('\n'); - } - } + printf("(special 2)"); + match_byte_assert(0); + match_byte_assert(0); + if (!match_u32 (2)) + match_u32_assert(1); + match_byte_assert(0); + match_byte_assert(0); + int subn = get_u32 (); + printf ("nested %d bytes", subn); + pos += subn; } else { - match_byte_assert (0x58); - char *base = get_string(); - int x = get_u32(); - printf ("\"%s\" with %d variables:\n", base, x); - for (int i = 0; i < x; i++) - { - int y = get_u32(); - if (!y) - y = 1; - else - match_u32_assert(0); - for (int j = 0; j <= 0; j++) - printf (" "); - printf("variable %d has %d values:\n", i, y); - for (int j = 0; j < y; j++) - { - if (match_byte(3)) - { - char *a = get_string(); - match_byte_assert(0x58); - char *b = get_string(); - char *c = get_string(); - for (int k = 0; k <= 1; k++) - printf (" "); - printf ("\"%s\", \"%s\", \"%s\"", a, b, c); - match_byte(0); - match_byte(0); - match_byte(0); - match_byte(0); - match_byte(0); - } - else if (match_byte (1)) - { - unsigned int format; - double value; - - if (match_byte (0x31)) - { - if (match_u32 (1)) - { - printf("(footnote %d) ", get_u32()); - match_byte_assert (0); - match_byte_assert (0); - int subn = get_u32 (); - printf ("nested %d bytes", subn); - pos += subn; - } - } - else - match_byte_assert (0x58); - format = get_u32 (); - value = get_double (); - printf ("value %g format %d(%d.%d)", value, format >> 16, (format >> 8) & 0xff, format & 0xff); - } - else - dump_value (1); - putchar('\n'); - } - } + match_u32_assert(3); + printf("(special 3)"); + match_byte_assert(0); + match_byte_assert(0); + match_byte_assert(1); + match_byte_assert(0); + int subn = get_u32 (); + printf ("nested %d bytes, ", subn); + pos += subn; + subn = get_u32 (); + printf ("nested %d bytes, ", subn); + pos += subn; } - putchar('\n'); } + else + match_byte_assert (0x58); } static void -dump_title_value(int level) +dump_value__(int level, bool match1) { for (int i = 0; i <= level; i++) printf (" "); - match_byte (0); - match_byte (0); - match_byte (0); - match_byte (0); - match_byte (0); + match_byte(0); + match_byte(0); + match_byte(0); + match_byte(0); + if (match_byte (3)) { - get_string(); - if (match_byte (0x31)) - { - if (match_u32 (1)) - { - printf("(footnote %d) ", get_u32()); - match_byte_assert (0); - match_byte_assert (0); - int subn = get_u32 (); - printf ("nested %d bytes", subn); - pos += subn; - } - else if (match_u32 (2)) - { - printf("(special 2)"); - match_byte_assert(0); - match_byte_assert(0); - if (!match_u32(2)) - match_u32_assert(1); - match_byte_assert(0); - match_byte_assert(0); - int subn = get_u32 (); - printf ("nested %d bytes", subn); - pos += subn; - } - else - { - match_u32_assert(3); - printf("(special 3)"); - match_byte_assert(0); - match_byte_assert(0); - match_byte_assert(1); - match_byte_assert(0); - int subn = get_u32 (); - printf ("nested %d bytes, ", subn); - pos += subn; - subn = get_u32 (); - printf ("nested %d bytes, ", subn); - pos += subn; - } - } + char *s1 = get_string(); + dump_value_31(); + char *s2 = get_string(); + char *s3 = get_string(); + if (strcmp(s1, s3)) + printf("strings \"%s\", \"%s\" and \"%s\"", s1, s2, s3); else - match_byte_assert (0x58); - get_string(); - printf("string \"%s\"", get_string()); - match_byte (0); - match_byte (0); - match_byte (0); - match_byte (1); - match_byte (1); - match_byte (0); - match_byte (0); - match_byte (0); - match_byte (1); + printf("string \"%s\" and \"%s\"", s1, s2); + if (!match_byte (0)) + match_byte_assert(1); + if (match1) + match_byte (1); } else if (match_byte (5)) { - if (match_byte (0x31)) - { - if (match_u32 (1)) - { - printf("(footnote %d) ", get_u32()); - match_byte_assert (0); - match_byte_assert (0); - int subn = get_u32 (); - printf ("nested %d bytes", subn); - pos += subn; - } - } - else - match_byte_assert (0x58); + dump_value_31(); printf ("variable \"%s\"", get_string()); get_string(); if (!match_byte(1) && !match_byte(2)) @@ -1107,13 +253,6 @@ dump_title_value(int level) value, format >> 16, (format >> 8) & 0xff, format & 0xff, var, vallab); if (!match_byte (1) && !match_byte(2)) match_byte_assert (3); - match_byte (0); - match_byte (0); - match_byte (0); - match_byte (0); - match_byte (0); - match_byte (0); - match_byte (0); } else if (match_byte (4)) { @@ -1129,76 +268,26 @@ dump_title_value(int level) value = get_string (); printf ("value \"%s\" format %d(%d.%d) var \"%s\" vallab \"%s\"", value, format >> 16, (format >> 8) & 0xff, format & 0xff, var, vallab); - match_byte (0); - match_byte (0); - match_byte (0); - match_byte (0); } else if (match_byte (1)) { unsigned int format; double value; - if (match_byte (0x31)) - { - if (match_u32 (1)) - { - printf("(footnote %d) ", get_u32()); - match_byte_assert (0); - match_byte_assert (0); - int subn = get_u32 (); - printf ("nested %d bytes", subn); - pos += subn; - } - } - else - match_byte_assert (0x58); + dump_value_31(); format = get_u32 (); value = get_double (); printf ("value %g format %d(%d.%d)", value, format >> 16, (format >> 8) & 0xff, format & 0xff); - match_byte (1); - match_byte (0); - match_byte (0); - match_byte (0); - match_byte (1); + if (match1) + match_byte (1); } - else if (match_byte (0x31)) + else { - if (match_u32 (1)) - { - printf("(footnote %d) ", get_u32()); - match_byte_assert (0); - match_byte_assert (0); - int subn = get_u32 (); - printf ("nested %d bytes", subn); - pos += subn; - } - else if (match_u32 (0)) - { - match_u32_assert (0); - int subn = get_u32 (); - printf ("nested %d bytes", subn); - pos += subn; - } - else - { - match_u32_assert(3); - printf("(special 3)"); - match_byte_assert(0); - match_byte_assert(0); - match_byte_assert(1); - match_byte_assert(0); - int subn = get_u32 (); - printf ("nested %d bytes, ", subn); - pos += subn; - subn = get_u32 (); - printf ("nested %d bytes, ", subn); - pos += subn; - } + dump_value_31(); char *base = get_string(); int x = get_u32(); - printf ("\"%s\"; %d variables:\n", base, x); + printf ("\"%s\" with %d variables:\n", base, x); for (int i = 0; i < x; i++) { int y = get_u32(); @@ -1211,56 +300,144 @@ dump_title_value(int level) printf("variable %d has %d values:\n", i, y); for (int j = 0; j < y; j++) { - if (match_byte(3)) - { - char *a = get_string(); - match_byte_assert(0x58); - char *b = get_string(); - char *c = get_string(); - for (int k = 0; k <= level + 1; k++) - printf (" "); - printf ("\"%s\", \"%s\", \"%s\"", a, b, c); - match_byte(0); - } - else - dump_title_value (level+1); + dump_value__ (level + 1, false); putchar('\n'); } } } - else +} + +static int +compare_int(const void *a_, const void *b_) +{ + const int *a = a_; + const int *b = b_; + return *a < *b ? -1 : *a > *b; +} + +static void +check_permutation(int *a, int n, const char *name) +{ + int b[n]; + memcpy(b, a, n * sizeof *a); + qsort(b, n, sizeof *b, compare_int); + for (int i = 0; i < n; i++) + if (b[i] != i) + { + fprintf(stderr, "bad %s permutation:", name); + for (int i = 0; i < n; i++) + fprintf(stderr, " %d", a[i]); + putc('\n', stderr); + exit(1); + } +} + +static void +dump_category(int level, int *indexes, int *n_indexes) +{ + dump_value__ (level, true); + match_byte(0); + match_byte(0); + match_byte(0); + + if (match_u32 (1)) + match_byte (0); + else if (match_byte (1)) { - match_byte_assert (0x58); - char *base = get_string(); - int x = get_u32(); - printf ("\"%s\" with %d variables:\n", base, x); - for (int i = 0; i < x; i++) + match_byte (0); + if (!match_u32 (2)) + match_u32_assert (1); + match_byte (0); + } + else if (!match_u32(2)) + match_u32_assert (0); + + int indx = get_u32(); + int n_categories = get_u32(); + if (indx != -1) + { + if (n_categories != 0) { - int y = get_u32(); - if (!y) - y = 1; - else - match_u32_assert(0); - for (int j = 0; j <= level; j++) - printf (" "); - printf("variable %d has %d values:\n", i, y); - for (int j = 0; j < y; j++) - { - if (match_byte(3)) - { - char *a = get_string(); - match_byte_assert(0x58); - char *b = get_string(); - char *c = get_string(); - for (int k = 0; k <= level + 1; k++) - printf (" "); - printf ("\"%s\", \"%s\", \"%s\"", a, b, c); - } - else - dump_title_value (level+1); - putchar('\n'); - } + fprintf(stderr, "index not -1 but subcategories\n"); + exit(1); } + indexes[(*n_indexes)++] = indx; + } + if (n_categories > 0) + printf (", %d subcategories:", n_categories); + else + printf (", index %d", indx); + printf("\n"); + for (int i = 0; i < n_categories; i++) + dump_category (level + 1, indexes, n_indexes); +} + +static void +dump_dim(void) +{ + int n_categories; + printf("next dim\n"); + dump_value__ (0, false); + + /* This byte is usually 0x02 but 0x00 and 0x75 (!) have also been spotted. */ + pos++; + + if (!match_byte(0) && !match_byte(1)) + match_byte_assert(2); + if (!match_u32(0)) + match_u32_assert(2); + if (!match_byte(0)) + match_byte_assert(1); + if (!match_byte(0)) + match_byte_assert(1); + match_byte_assert(1); + static int dim_indx = 0; + match_u32_assert(dim_indx++); + n_categories = get_u32(); + printf("%d nested categories\n", n_categories); + + int indexes[1024]; + int n_indexes = 0; + for (int i = 0; i < n_categories; i++) + dump_category (0, indexes, &n_indexes); + check_permutation(indexes, n_indexes, "categories"); +} + +int n_dims; +static void +dump_dims(void) +{ + n_dims = get_u32(); + printf ("%u dimensions\n", n_dims); + for (int i = 0; i < n_dims; i++) + { + printf("\n"); + dump_dim (); + } +} + +static void +dump_data(void) +{ + /* The first three numbers add to the number of dimensions. */ + int t = get_u32(); + t += get_u32(); + match_u32_assert(n_dims - t); + + /* The next n_dims numbers are a permutation of the dimension numbers. */ + int a[n_dims]; + for (int i = 0; i < n_dims; i++) + a[i] = get_u32(); + check_permutation(a, n_dims, "dimensions"); + + int x = get_u32(); + printf ("%d data values, starting at %08x\n", x, pos); + for (int i = 0; i < x; i++) + { + printf("%08x, index %d:\n", pos, get_u32()); + match_u32_assert(0); + dump_value__(0, false); + putchar('\n'); } } @@ -1268,18 +445,140 @@ static void dump_title(void) { pos = 0x27; - dump_title_value(0); putchar('\n'); putchar('\n'); - dump_title_value(0); putchar('\n'); putchar('\n'); + dump_value__(0, true); putchar('\n'); + dump_value__(0, true); putchar('\n'); match_byte_assert(0x31); - dump_title_value(0); putchar('\n'); putchar('\n'); + dump_value__(0, true); putchar('\n'); match_byte(0); match_byte_assert(0x58); if (match_byte(0x31)) { - dump_title_value(0); putchar('\n'); + dump_value__(0, false); putchar('\n'); } else match_byte_assert(0x58); + + + int n_footnotes = get_u32(); + if (n_footnotes >= 20) + { + fprintf(stderr, "%08x: %d footnotes\n", pos - 4, n_footnotes); + exit(1); + } + + printf("------\n%d footnotes\n", n_footnotes); + if (n_footnotes < 20) + { + for (int i = 0; i < n_footnotes; i++) + { + printf("footnote %d:\n", i); + dump_value__(0, false); + if (match_byte (0x31)) + { + /* Custom footnote marker string. */ + match_byte_assert(3); + get_string(); + match_byte_assert(0x58); + match_u32_assert(0); + get_string(); + } + else + match_byte_assert (0x58); + printf("(%d)\n", get_u32()); + } + } +} + +static int +find_dimensions(void) +{ + { + const char dimensions[] = "-,,,.\0"; + int x = try_find_tail(dimensions, sizeof dimensions - 1); + if (x) + return x; + } + + const char dimensions[] = "-,,, .\0"; + return find_tail(dimensions, sizeof dimensions - 1); +} + +static void +dump_fonts(void) +{ + printf("fonts: offset=%08x\n", pos); + match_byte(0); + for (int i = 1; i <= 8; i++) + { + printf("%08x: font %d, ", pos, i); + match_byte_assert(i); + match_byte_assert(0x31); + printf("%s, ", get_string()); + match_byte_assert(0); + match_byte_assert(0); + if (!match_byte(0x40) && !match_byte(0x20) && !match_byte(0x80) && !match_byte(0x10)) + match_byte_assert(0x50); + if (!match_byte(0x41)) + match_byte_assert(0x51); + pos += 13; + printf ("%s, ", get_string()); + printf ("%s, ", get_string()); + match_u32_assert(0); + match_u32_assert(0); + pos++; + get_u32(); + get_u32(); + get_u32(); + get_u32(); + putchar('\n'); + } + + match_u32_assert(240); + pos += 240; + + match_u32_assert(18); + pos += 18; + + if (match_u32(117)) + pos += 117; + else + { + match_u32_assert(142); + pos += 142; + } + + int count = get_u32(); + pos += 4 * count; + + char *encoding = get_string(); + printf("encoding=%s\n", encoding); + + if (!match_u32(0)) + match_u32_assert(UINT32_MAX); + if (!match_byte(0)) + match_byte_assert(1); + match_byte_assert(0); + if (!match_byte(0)) + match_byte_assert(1); + if (!match_byte(0x99) && !match_byte(0x98)) + match_byte_assert(0x97); + match_byte_assert(7); + match_byte_assert(0); + match_byte_assert(0); + if (match_byte('.')) + match_byte_assert(','); + else + { + match_byte_assert(','); + if (!match_byte('.')) + match_byte_assert(' '); + } + match_u32_assert(5); + for (int i = 0; i < 5; i++) + get_string(); + pos += get_u32(); + if (pos != find_dimensions()) + fprintf (stderr, "%08x / %08x\n", pos, find_dimensions()); } int @@ -1350,22 +649,32 @@ main(int argc, char *argv[]) } else if (!strcmp(argv[1], "dimensions") || !strcmp(argv[1], "all")) { - { - const char dimensions[] = "-,,,.\0"; - start = try_find_tail(dimensions, sizeof dimensions - 1); - } - - if (!start) - { - const char dimensions[] = "-,,, .\0"; - start = find_tail(dimensions, sizeof dimensions - 1); - } - - pos = start; + pos = 0; + match_byte_assert(1); + match_byte_assert(0); + match_u32_assert(3); + match_byte_assert(1); + if (!match_byte(0)) + match_byte_assert(1); + match_byte_assert(0); + match_byte_assert(0); + if (!match_byte(0)) + match_byte_assert(1); + pos++; + match_byte_assert(0); + match_byte_assert(0); + match_byte_assert(0); + dump_title (); + dump_fonts(); dump_dims (); + printf("\n\ndata:\n"); dump_data (); - if (!strcmp(argv[1], "all")) - dump_title (); + match_byte (1); + if (pos != n) + { + fprintf (stderr, "%x / %x\n", pos, n); + exit(1); + } exit(0); } else