shithub: fnt

Download patch

ref: 720585ccc54165aa6d41a0fe85374223ac3a9e37
parent: 207f8c591b3e831f4a35185b99482cfb34fccf4b
author: Sigrid Solveig Haflínudóttir <sigrid@ftrv.se>
date: Wed Jun 19 21:41:38 EDT 2024

handle offsets; clean up a bit; add more types

--- a/fmt.sh
+++ b/fmt.sh
@@ -1,2 +1,2 @@
 #!/bin/sh
-raco fmt --width 100 --indent 2 -i *.rkt
+raco fmt --width 102 --indent 2 -i *.rkt
--- a/otf.rkt
+++ b/otf.rkt
@@ -19,8 +19,8 @@
   (make-contract #:name 'tag/c #:first-order (λ (x) (and (string? x) (= 4 (string-length x))))))
 
 (define/contract (indent lst)
-  (-> (listof string?) (listof string?))
-  (map (λ (str) (string-append "\t" str)) lst))
+  (-> (listof any/c) (listof string?))
+  (map (λ (str) (string-append "\t" str)) (flatten lst)))
 
 (define/contract (c-typedef? s)
   (-> string? boolean?)
@@ -51,24 +51,6 @@
   (-> type? positive?)
   (/ (type-bits t) 8))
 
-(define (field-unused? f)
-  (assoc 'unused (field-attrs f)))
-
-(define (field-count f)
-  (assoc 'count (field-attrs f)))
-
-(define (field-test f)
-  (filter (λ (t) (eq? (car t) 'test)) (field-attrs f)))
-
-(define (field-verb f)
-  (assoc 'verb (field-attrs f)))
-
-(define (field-cond f)
-  (assoc 'cond (field-attrs f)))
-
-(define (field-ptr f)
-  (assoc 'ptr (field-attrs f)))
-
 (define (block stmt lst)
   (if (= (length lst) 1) (list* stmt lst) (list (string-append stmt "{") lst "}")))
 
@@ -75,49 +57,10 @@
 (define (wrap-cond-c cond lst)
   (match cond
     [#f lst]
-    [(list 'cond op ref n ...)
+    [(list op ref n ...)
      (block (~a "if(" (string-join (map (λ (n) (~a "v->" ref " " op " " n)) n) " || ") ")")
-            (indent (flatten lst)))]))
+            (indent lst))]))
 
-(define (field-print-c f)
-  (define t (field-type f))
-  (define count (field-count f))
-  (define end (and count (if (cmplx? t) (~a "v->" (cadr count)) (cadr count))))
-  (define basic-array (and (type? t) end))
-  (define fixed-array (and basic-array (number? end)))
-  (define array-index (if (field-count f) "[i]" ""))
-  (define verb (if (type? t) (if (field-verb f) (cadr (field-verb f)) (type-verb t)) ""))
-  (define print-index
-    (if basic-array
-        (~a "Bprint(f, \"%*s%s[%d]: "
-            verb
-            "\\n\", indent, \"\", \""
-            (field-name f)
-            "\", i, v->"
-            (field-name f)
-            array-index
-            ");")
-        (~a "Bprint(f, \"%*s%s[%d]:\\n\", indent, \"\", \"" (field-name f) "\", i);")))
-  (define array-loop
-    (if count
-        (λ (lst)
-          (block (~a "for(int i = 0; i < " (if fixed-array "" "v->") (cadr count) "; i++)")
-                 (indent (list* print-index (if basic-array empty lst)))))
-        identity))
-  (define lst
-    (list
-     (if (type? t)
-         (~a "Bprint(f, \"%*s%s: "
-             verb
-             "\\n\", indent, \"\", \""
-             (field-name f)
-             "\", v->"
-             (field-name f)
-             array-index
-             ");")
-         (~a "print_" (cmplx-name t) "(f, indent+indentΔ, &v->" (field-name f) array-index ");"))))
-  (if (field-unused? f) empty (wrap-cond-c (field-cond f) (array-loop lst))))
-
 (define (invert-c op)
   (match op
     ['== '!=]
@@ -132,14 +75,14 @@
   [(define/generic super-c-type c-type)
    (define (gen-h f)
      (define cnt (field-count f))
-     (define fixed-array (and cnt (number? (cadr cnt))))
+     (define fixed-array (number? cnt))
+     (define is-ptr (and (cmplx? (field-type f)) (field-offset f)))
      (list (~a (if (field-unused? f) "// " "")
                (super-c-type (field-type f))
                " "
-               (if (and cnt (not fixed-array)) "*" "")
+               (if (or is-ptr (and cnt (not fixed-array))) "*" "")
                (field-name f)
-               (if fixed-array (~a "[" (cadr cnt) "]") "")
-               ";")))
+               (if fixed-array (~a "[" cnt "];") ";"))))
    (define (gen-c f b index)
      (define (size t)
        (if (type? t) (type-size t) 0))
@@ -150,6 +93,7 @@
      (define (format-number x)
        (if (<= x 32768) (~r x) (~r x #:base 16 #:sign '("0x" "" "-0x"))))
      (define (parse-if-error read)
+       (define is-ptr (and (cmplx? (field-type f)) (field-offset f)))
        (match (field-count f)
          [#f
           (if (or declared (not (field-unused? f)))
@@ -159,20 +103,23 @@
                       " = "
                       ((type-parse (field-type f)) b index)
                       ";")
-                  (list (if declared (~a (name (field-type f)) " " ref ";") empty)
-                        (~a "if(read_" (name (field-type f)) "(ctx, &" ref ") < 0){")))
-              (if index empty (~a "if(skip_bytes(ctx, " (size (field-type f)) ") < 0){")))]
+                  (list
+                   (if declared
+                       (~a (name (field-type f)) " " ref ";")
+                       (if is-ptr (~a ref " = calloc(1, sizeof(*" ref "));") empty))
+                   (~a "if(read_" (name (field-type f)) "(ctx, " (if is-ptr "" "&") ref ") < 0){")))
+              (if index empty (~a "if(ctxreadn(ctx, " (size (field-type f)) ") == nil){")))]
          [count
           #:when (type? (field-type f))
           (if (field-unused? f)
               empty
-              (list
-               (~a "for(int i = 0; i < " (if (number? (cadr count)) "" "v->") (cadr count) "; i++)")
-               (~a "\t"
-                   ref
-                   "[i] = "
-                   ((type-parse (field-type f)) b index (~a "i*" (size (field-type f))))
-                   ";")))]
+              (list (if (number? count) empty (~a ref " = malloc(" count "*sizeof(*" ref "));"))
+                    (~a "for(int i = 0; i < " count "; i++)")
+                    (~a "\t"
+                        ref
+                        "[i] = "
+                        ((type-parse (field-type f)) b index (~a "i*" (size (field-type f))))
+                        ";")))]
          [count
           (~a "if(ctxarray(ctx, &"
               ref
@@ -180,8 +127,8 @@
               (super-c-type (field-type f))
               ", sizeof("
               (super-c-type (field-type f))
-              "), v->"
-              (cadr count)
+              "), "
+              count
               ") < 0){")]))
      (define (test-cond ts)
        (map (λ (t)
@@ -190,25 +137,120 @@
                  (string-join (map (λ (v) (~a ref " " (invert-c op) " " (format-number v))) a)
                               " && ")]))
             ts))
-     (list*
-      (parse-if-error #t)
-      (if index
-          empty
-          (list (~a "\twerrstr(\"%s: %r\", \"" (field-name f) "\");") (~a "\tgoto err;") (~a "}")))
-      (match (test-cond (field-test f))
-        [(list) empty]
-        [(list a ...)
-         (list (~a "if(" (string-join a " || ") "){")
-               (~a "\twerrstr(\"%s: invalid value: %d (0x%ux)\", \""
-                   (field-name f)
-                   "\", "
-                   ref
-                   ", "
-                   ref
-                   ");")
-               (~a "\tgoto err;")
-               (~a "}"))])))])
+     (define (at lst)
+       (define at (field-offset f))
+       (if (not at)
+           lst
+           (list (~a "if(v->" at " != 0){")
+                 (~a "\tif(ctxpushrange(ctx, v->" at ", -1) < 0)")
+                 (~a "\t\tgoto err;")
+                 (indent lst)
+                 (~a "\tif(ctxpoprange(ctx) < 0)")
+                 (~a "\t\tgoto err;")
+                 (~a "}"))))
+     (at (list* (parse-if-error #t)
+                (if index
+                    empty
+                    (list (~a "\twerrstr(\"%s: %r\", \"" (field-name f) "\");")
+                          (~a "\tgoto err;")
+                          (~a "}")))
+                (match (test-cond (field-test f))
+                  [(list) empty]
+                  [(list a ...)
+                   (list (~a "if(" (string-join a " || ") "){")
+                         (~a "\twerrstr(\"%s: invalid value: %d (0x%ux)\", \""
+                             (field-name f)
+                             "\", "
+                             ref
+                             ", "
+                             ref
+                             ");")
+                         (~a "\tgoto err;")
+                         (~a "}"))]))))])
 
+(define/contract (field-attr f a)
+  (-> field? symbol? any)
+  (define v (assoc a (field-attrs f)))
+  (and v (cadr v)))
+
+(define (field-unused? f)
+  (field-attr f 'unused))
+
+(define (field-verb f)
+  (field-attr f 'verb))
+
+(define (field-offset f)
+  (field-attr f 'at))
+
+(define (field-cond f)
+  (define v (assoc 'cond (field-attrs f)))
+  (and v (rest v)))
+
+(define/contract (field-count f)
+  (-> field? (or/c false/c number? string?))
+  (define (fmt-expr e)
+    (cond
+      [(number? e) e]
+      [(list? e)
+       (match e
+         [(list op x y) (~a (fmt-expr x) op (fmt-expr y))])]
+      [(symbol? e) (~a "v->" e)]))
+  (define e (field-attr f 'count))
+  (and e (fmt-expr e)))
+
+(define (field-test f)
+  (filter (λ (t) (eq? (car t) 'test)) (field-attrs f)))
+
+(define (field-print-c f)
+  (define t (field-type f))
+  (define cnt (field-count f))
+  (define basic-array (and cnt (type? t)))
+  (define fixed-array (and basic-array (number? cnt)))
+  (define array-index (if cnt "[i]" ""))
+  (define verb (if (type? t) (or (field-verb f) (type-verb t)) ""))
+  (define print-index
+    (if basic-array
+        (~a "Bprint(f, \"%*s%s[%d]: "
+            verb
+            "\\n\", indent, \"\", \""
+            (field-name f)
+            "\", i, v->"
+            (field-name f)
+            array-index
+            ");")
+        (~a "Bprint(f, \"%*s%s[%d]:\\n\", indent, \"\", \"" (field-name f) "\", i);")))
+  (define array-loop
+    (if cnt
+        (λ (lst)
+          (block (~a "for(int i = 0; i < " cnt "; i++)")
+                 (indent (list* print-index (if basic-array empty lst)))))
+        identity))
+  (define is-ptr (and (cmplx? t) (field-offset f) (not cnt)))
+  (define lst
+    (flatten
+     (list
+      (if (type? t)
+          (~a "Bprint(f, \"%*s%s: "
+              verb
+              "\\n\", indent, \"\", \""
+              (field-name f)
+              "\", v->"
+              (field-name f)
+              array-index
+              ");")
+          (list (if cnt empty (~a "Bprint(f, \"%*s%s:\\n\", indent, \"\", \"" (field-name f) "\");"))
+                (if is-ptr (~a "if(v->" (field-name f) " != nil)") empty)
+                (~a (if is-ptr "\t" "")
+                    "print_"
+                    (cmplx-name t)
+                    "(f, indent+indentΔ, "
+                    (if is-ptr "" "&")
+                    "v->"
+                    (field-name f)
+                    array-index
+                    ");"))))))
+  (if (field-unused? f) empty (wrap-cond-c (field-cond f) (array-loop lst))))
+
 (define (filter-extra extra key)
   (flatten (filter-map (λ (e) (and (eq? (car e) key) (cdr e))) extra)))
 
@@ -221,7 +263,7 @@
      (flatten (append (list (~a "typedef struct " (cmplx-name c) " " (cmplx-name c) ";")
                             (~a "")
                             (~a "struct " (cmplx-name c) " {"))
-                      (indent (flatten (map super-gen-h (cmplx-fields c))))
+                      (indent (map super-gen-h (cmplx-fields c)))
                       (indent (filter-extra (cmplx-extra c) 'field))
                       (list (~a "};")
                             (~a "")
@@ -229,7 +271,7 @@
    (define (gen-c c b index)
      (define (no-vla? f)
        (define cnt (field-count f))
-       (or (not cnt) (number? (cadr cnt))))
+       (or (not cnt) (number? cnt)))
      ; group fields to minimize number of reads
      ; complex types are always alone
      ; simple types can be combined unless versioned
@@ -253,9 +295,7 @@
          [#f (type-size (field-type f))]
          [count
           #:when (type? (field-type f))
-          ((if (number? (cadr count))
-               (λ (sz) (* (cadr count) sz))
-               (λ (sz) (~a "v->" (cadr count) "*" sz)))
+          ((if (number? count) (λ (sz) (* count sz)) (λ (sz) (~a count "*" sz)))
            (type-size (field-type f)))]))
      (define (add x y)
        (match (list x y)
@@ -269,6 +309,7 @@
              empty))
        (p g 0))
      (define (gen-group-c fields)
+       (define unused (andmap field-unused? fields))
        (wrap-cond-c (field-cond (car fields))
                     (if (cmplx? (field-type (car fields)))
                         (map (λ (f) (super-gen-c f #f #f)) fields)
@@ -275,6 +316,7 @@
                         (let* ([sum (foldr (λ (f accu) (add (field-size f) accu)) 0 fields)]
                                [lst (flatten (list* (~a "if((b = ctxreadn(ctx, " sum ")) == nil)")
                                                     (~a "\tgoto err;")
+                                                    (if unused "USED(b);" empty)
                                                     (parse-group fields)))])
                           lst))))
      (flatten
@@ -283,7 +325,7 @@
                     (~a "read_" (cmplx-name c) "(Ctx *ctx, " (cmplx-name c) " *v)")
                     (~a "{")
                     (~a "\tu8int *b;"))
-              (indent (flatten (map gen-group-c (group-fields (cmplx-fields c)))))
+              (indent (map gen-group-c (group-fields (cmplx-fields c))))
               (indent (filter-extra (cmplx-extra c) 'read))
               (list (~a "\treturn 0;")
                     (~a "err:")
@@ -294,7 +336,7 @@
                     (~a "void")
                     (~a "print_" (cmplx-name c) "(Biobuf *f, int indent, " (cmplx-name c) " *v)")
                     (~a "{")
-                    (indent (flatten (map field-print-c (cmplx-fields c))))
+                    (indent (map field-print-c (cmplx-fields c)))
                     (indent (filter-extra (cmplx-extra c) 'print))
                     (~a "}")))))
    (define (c-type c)
@@ -381,11 +423,14 @@
 
 (define-syntax (mkattr stx)
   (syntax-parse stx
-    [(_ offType:id ({~literal ptr} type:id name:id))
-     (begin
-       (when (not (type-offset? (syntax->datum #`offType)))
-         (raise-syntax-error #f "can't be used as an offset" stx #'offType))
-       #''(ptr type name))]
+    [(_ _ ({~literal at} ref:id))
+     (let ([rf (assoc (syntax->datum #`ref) fields)])
+       (begin
+         (when (not rf)
+           (raise-syntax-error #f "no such field" stx #'ref))
+         (when (not (type-offset? (cadr rf)))
+           (raise-syntax-error #f (~a (cadr rf) " can't be used as an offset") stx #'ref))
+         #''(at ref)))]
     [(_ _ (n:number)) #''(count n)]
     [(_ type {~literal hex})
      (begin
@@ -417,6 +462,7 @@
                                   stx
                                   #'n))]
          [else (raise-syntax-error #f "no such field" stx #'n)]))]
+    [(_ _ ({~literal count} e:expr)) #''(count e)] ; FIXME - check fields and ops/numbers
     [(_ _ {~literal unused}) #''(unused #t)]))
 
 (define-syntax (mkfield stx)
@@ -439,11 +485,6 @@
        (set! fields '())
        #'(list (mkfield [~@ . x]) ...))]))
 
-(mkcmplx EncodingRecord
-         (mkfields {uint16 platformID (<= 4)}
-                   {uint16 encodingID}
-                   {Offset32 subtableOffset (ptr SubTableCmap subtable)}))
-
 (mkcmplx SubHeader
          (mkfields {uint16 firstCode} {uint16 entryCode} {int16 idDelta} {uint16 idRangeOffset}))
 
@@ -453,27 +494,24 @@
 (mkcmplx SubtableCmap0 (mkfields {uint16 length} {uint16 language} {uint8 glyphIdArray [256]}))
 
 ; FIXME
-#;(mkcmplx SubtableCmap2
-           (mkfields {uint16 length}
-                     {uint16 language}
-                     {uint16 subHeaderKeys [256]}
-                     {SubHeader subHeaders (count ?)}
-                     {uint16 glyphIdArray (count ?)}))
+(mkcmplx SubtableCmap2 (mkfields {uint16 length} {uint16 language} {uint16 subHeaderKeys [256]}))
+#|                     {SubHeader subHeaders (count ?)}
+                       {uint16 glyphIdArray (count ?)}))|#
 
 ; FIXME
-#;(mkcmplx SubtableCmap4
-           (mkfields {uint16 length}
-                     {uint16 language}
-                     {uint16 segCountX2}
-                     {uint16 searchRange}
-                     {uint16 entrySelector}
-                     {uint16 rangeShift}
-                     {uint16 endCode (count (/ segCountX2 2))}
-                     {uint16 reservedPad unused}
-                     {uint16 startCode (count (/ segCountX2 2))}
-                     {int16 idDelta (count (/ segCountX2 2))}
-                     {int16 idRangeOffset (count (/ segCountX2 2))}
-                     {uint16 glyphIdArray (count ?)}))
+(mkcmplx SubtableCmap4
+         (mkfields {uint16 length}
+                   {uint16 language}
+                   {uint16 segCountX2}
+                   {uint16 searchRange}
+                   {uint16 entrySelector}
+                   {uint16 rangeShift}
+                   {uint16 endCode (count (/ segCountX2 2))}
+                   {uint16 reservedPad unused}
+                   {uint16 startCode (count (/ segCountX2 2))}
+                   {int16 idDelta (count (/ segCountX2 2))}
+                   {int16 idRangeOffset (count (/ segCountX2 2))}
+                   #;{uint16 glyphIdArray (count ?)}))
 
 (mkcmplx SubtableCmap6
          (mkfields {uint16 length}
@@ -485,19 +523,18 @@
 (mkcmplx SubtableCmap8
          (mkfields {uint16 length}
                    {uint16 language}
-                   ; FIXME this crosses an uncomfortable threshold - allocate dynamically
                    {uint8 is32 [8192]}
                    {uint32 numGroups}
                    {MapGroup groups (count numGroups)}))
 
 ; FIXME
-#;(mkcmplx SubtableCmap10
-           (mkfields {uint16 reserved}
-                     {uint32 length}
-                     {uint32 language}
-                     {uint32 startCharCode}
-                     {uint32 numChars}
-                     {uint16 glyphIdArray (count ?)}))
+(mkcmplx SubtableCmap10
+         (mkfields {uint16 reserved}
+                   {uint32 length}
+                   {uint32 language}
+                   {uint32 startCharCode}
+                   {uint32 numChars}
+                   #;{uint16 glyphIdArray (count ?)}))
 
 (mkcmplx SubtableCmap12or13
          (mkfields {uint16 reserved}
@@ -518,7 +555,11 @@
          (mkfields {uint32 numUVSMappings} {UVSMapping uvsMappings (count numUVSMappings)}))
 
 (mkcmplx VariationSelector
-         (mkfields {uint24 varSelector} {Offset32 defaultUVSOffset} {Offset32 nonDefaultUVSOffset}))
+         (mkfields {uint24 varSelector}
+                   {Offset32 defaultUVSOffset}
+                   {Offset32 nonDefaultUVSOffset}
+                   {DefaultUVS defaultUVS (at defaultUVSOffset)}
+                   {NonDefaultUVS nonDefaultUVS (at nonDefaultUVSOffset)}))
 
 (mkcmplx SubtableCmap14
          (mkfields {uint32 length}
@@ -526,17 +567,23 @@
                    {VariationSelector varSelector (count numVarSelectorRecords)}))
 
 (mkcmplx SubtableCmap
-         (mkfields {uint16 format (== 0 #|2 4|# 6 8 #|10|# 12 13 14)}
+         (mkfields {uint16 format (== 0 2 4 6 8 10 12 13 14)}
                    ; FIXME these fields should be put into a union automatically
                    {SubtableCmap0 sub0 (== format 0)}
-                   ;{SubtableCmap2 sub2 (== format 2)}
-                   ;{SubtableCmap4 sub4 (== format 4)}
+                   {SubtableCmap2 sub2 (== format 2)}
+                   {SubtableCmap4 sub4 (== format 4)}
                    {SubtableCmap6 sub6 (== format 6)}
                    {SubtableCmap8 sub8 (== format 8)}
-                   ;{SubtableCmap10 sub10 (== format 10)}
+                   {SubtableCmap10 sub10 (== format 10)}
                    {SubtableCmap12or13 sub12or13 (== format 12 13)}
                    {SubtableCmap14 sub14 (== format 14)}))
 
+(mkcmplx EncodingRecord
+         (mkfields {uint16 platformID (<= 4)}
+                   {uint16 encodingID}
+                   {Offset32 subtableOffset}
+                   {SubtableCmap subtable (at subtableOffset)}))
+
 (mkcmplx TableCmap
          (mkfields {uint16 version unused (== 0)}
                    {uint16 numTables}
@@ -653,6 +700,42 @@
                    {int8 minAfterBL}
                    {int8 pad [2] unused}))
 
+(mkcmplx IndexSubtable1
+         (mkfields {Offset32 sbitOffsets} #;(count (+ (- lastGlyphIndex firstGlyphIndex) 2))))
+
+(mkcmplx IndexSubtable2 (mkfields {uint32 imageSize} {BigGlyphMetrics bigMetrics}))
+
+; FIXME
+(mkcmplx IndexSubtable3
+         (mkfields {Offset16 sbitOffsets} #;(count (+ (- lastGlyphIndex firstGlyphIndex) 2))))
+
+(mkcmplx GlyphIdOffsetPair (mkfields {uint16 glyphID} {Offset16 sbitOffset}))
+
+(mkcmplx IndexSubtable4
+         (mkfields {uint32 numGlyphs} {GlyphIdOffsetPair glyphArray (count (+ numGlyphs 1))}))
+
+(mkcmplx IndexSubtable5
+         (mkfields {uint32 imageSize}
+                   {BigGlyphMetrics bigMetrics}
+                   {uint32 numGlyphs}
+                   {uint16 glyphIdArray (count numGlyphs)}))
+
+(mkcmplx IndexSubtable
+         (mkfields {uint16 indexFormat (>= 1) (<= 5)}
+                   {uint16 imageFormat}
+                   {Offset32 imageDataOffset}
+                   {IndexSubtable1 sub1 (== indexFormat 1)}
+                   {IndexSubtable2 sub2 (== indexFormat 2)}
+                   {IndexSubtable3 sub3 (== indexFormat 3)}
+                   {IndexSubtable4 sub4 (== indexFormat 4)}
+                   {IndexSubtable5 sub5 (== indexFormat 5)}))
+
+(mkcmplx IndexSubtableRecord
+         (mkfields {uint16 firstGlyphIndex}
+                   {uint16 lastGlyphIndex}
+                   {Offset32 indexSubtableOffset}
+                   {IndexSubtable indexSubtable (at indexSubtableOffset)}))
+
 (mkcmplx BitmapSize
          (mkfields {Offset32 indexSubtableListOffset}
                    {uint32 indexSubtableListSize}
@@ -665,8 +748,16 @@
                    {uint8 ppemX}
                    {uint8 ppemY}
                    {uint8 bitDepth}
-                   {int8 flags}))
+                   {int8 flags}
+                   {IndexSubtableRecord
+                    indexSubtableList
+                    (count numberOfIndexSubtables)
+                    (at indexSubtableListOffset)}))
 
+(mkcmplx TableEBDT
+         (mkfields {uint16 majorVersion (== 2)} {uint16 minorVersion (== 0) unused})
+         #:tag "EBDT")
+
 (mkcmplx TableEBLC
          (mkfields {uint16 majorVersion (== 2) unused}
                    {uint16 minorVersion (== 0) unused}
@@ -723,6 +814,46 @@
                    {Offset32 itemVarStoreOffset (>= minorVersion 3)})
          #:tag "GDEF")
 
+(mkcmplx LangSys
+         (mkfields {Offset16 lookupOrderOffset unused}
+                   {uint16 requiredFeatureIndex}
+                   {uint16 featureIndexCount}
+                   {uint16 featureIndices (count featureIndexCount)}))
+
+(mkcmplx LangSysRecord
+         (mkfields {Tag langSysTag} {Offset16 langSysOffset} {LangSys langSys (at langSysOffset)}))
+
+(mkcmplx Script
+         (mkfields {Offset16 defaultLangSysOffset}
+                   {uint16 langSysCount}
+                   {LangSysRecord langSysRecords (count langSysCount)}
+                   {LangSys defaultLangSys (at defaultLangSysOffset)}))
+
+(mkcmplx ScriptRecord
+         (mkfields {Tag scriptTag} {Offset16 scriptOffset} {Script script (at scriptOffset)}))
+
+(mkcmplx ScriptList (mkfields {uint16 scriptCount} {ScriptRecord scriptRecords (count scriptCount)}))
+
+(mkcmplx Feature
+         (mkfields {Offset16 featureParamsOffset}
+                   {uint16 lookupIndexCount}
+                   {uint16 lookupListIndices (count lookupIndexCount)}))
+
+(mkcmplx FeatureRecord
+         (mkfields {Tag featureTag} {Offset16 featureOffset} {Feature feature (at featureOffset)}))
+
+(mkcmplx FeatureList
+         (mkfields {uint16 featureCount} {FeatureRecord featureRecords (count featureCount)}))
+
+(mkcmplx Lookup
+         (mkfields {uint16 lookupType}
+                   {uint16 lookupFlag}
+                   {uint16 subTableCount}
+                   {Offset16 subtableOffsets (count subTableCount)}
+                   {uint16 markFilteringSet}))
+
+(mkcmplx LookupList (mkfields {uint16 lookupCount} {Offset16 lookupOffsets (count lookupCount)}))
+
 (mkcmplx TableGPOS
          (mkfields {uint16 majorVersion (== 1) unused}
                    {uint16 minorVersion (<= 1)}
@@ -729,7 +860,10 @@
                    {Offset16 scriptListOffset}
                    {Offset16 featureListOffset}
                    {Offset16 lookupListOffset}
-                   {Offset32 featureVariationsOffset (== minorVersion 1)})
+                   {Offset32 featureVariationsOffset (== minorVersion 1)}
+                   {ScriptList scriptList (at scriptListOffset)}
+                   {FeatureList featureList (at featureListOffset)}
+                   {LookupList lookupList (at lookupListOffset)})
          #:tag "GPOS")
 
 (mkcmplx TableGSUB
@@ -738,17 +872,11 @@
                    {Offset16 scriptListOffset}
                    {Offset16 featureListOffset}
                    {Offset16 lookupListOffset}
-                   {Offset32 featureVariationsOffset (== minorVersion 1)})
+                   {Offset32 featureVariationsOffset (== minorVersion 1)}
+                   {ScriptList scriptList (at scriptListOffset)}
+                   {FeatureList featureList (at featureListOffset)})
          #:tag "GSUB")
 
-(mkcmplx TableMATH
-         (mkfields {uint16 majorVersion (== 1) unused}
-                   {uint16 minorVersion (== 0) unused}
-                   {Offset16 mathConstantsOffset}
-                   {Offset16 mathGlyphInfoOffset}
-                   {Offset16 mathVariantsOffset})
-         #:tag "MATH")
-
 (mkcmplx MathValueRecord (mkfields {FWORD value} {Offset16 deviceOffset}))
 
 (mkcmplx MathConstants
@@ -809,12 +937,6 @@
                    {MathValueRecord radicalKernAfterDegree}
                    {int16 radicalDegreeBottomRaisePercent}))
 
-(mkcmplx MathGlyphInfo
-         (mkfields {Offset16 mathItalicsCorrectionInfoOffset}
-                   {Offset16 mathTopAccentAttachmentOffset}
-                   {Offset16 extendedShapeCoverageOffset}
-                   {Offset16 mathKernInfoOffset}))
-
 (mkcmplx MathItalicsCorrectionInfo
          (mkfields {Offset16 italicsCorrectionCoverageOffset}
                    {uint16 italicsCorrectionCount}
@@ -841,6 +963,16 @@
                    {MathValueRecord correctionHeight (count heightCount)}
                    {MathValueRecord kernValues (count heightCount)}))
 
+(mkcmplx Coverage1 (mkfields {uint16 glyphCount} {uint16 glyphArray (count glyphCount)}))
+
+(mkcmplx RangeRecord (mkfields {uint16 startGlyphID} {uint16 endGlyphID} {uint16 startCoverageIndex}))
+
+(mkcmplx Coverage2 (mkfields {uint16 rangeCount} {RangeRecord rangeRecords (count rangeCount)}))
+
+(mkcmplx
+ Coverage
+ (mkfields {uint16 format (== 1 2)} {Coverage1 cov1 (== format 1)} {Coverage2 cov2 (== format 2)}))
+
 (mkcmplx MathVariants
          (mkfields {UFWORD minConnectorOverlap}
                    {Offset16 vertGlyphCoverageOffset}
@@ -848,15 +980,22 @@
                    {uint16 vertGlyphCount}
                    {uint16 horizGlyphCount}
                    {Offset16 vertGlyphConstructionOffsets (count vertGlyphCount)}
-                   {Offset16 horizGlyphConstructionOffsets (count horizGlyphCount)}))
+                   {Offset16 horizGlyphConstructionOffsets (count horizGlyphCount)}
+                   {Coverage vertGlyphCoverage (at vertGlyphCoverageOffset)}
+                   {Coverage horizGlyphCoverage (at horizGlyphCoverageOffset)}))
 
+(mkcmplx MathGlyphInfo
+         (mkfields
+          {Offset16 mathItalicsCorrectionInfoOffset}
+          {Offset16 mathTopAccentAttachmentOffset}
+          {Offset16 extendedShapeCoverageOffset} ; FIXME WHERE is this shit defined???
+          {Offset16 mathKernInfoOffset}
+          {MathItalicsCorrectionInfo mathItalicsCorrectionInfo (at mathItalicsCorrectionInfoOffset)}
+          {MathTopAccentAttachment mathTopAccentAttachment (at mathTopAccentAttachmentOffset)}
+          {MathKernInfo mathKernInfo (at mathKernInfoOffset)}))
+
 (mkcmplx MathGlyphVariantRecord (mkfields {uint16 variantGlyph} {UFWORD advanceMeasurement}))
 
-(mkcmplx MathGlyphConstruction
-         (mkfields {Offset16 glyphAssemblyOffset}
-                   {uint16 variantCount}
-                   {MathGlyphVariantRecord mathGlyphVariantRecords (count variantCount)}))
-
 (mkcmplx GlyphPart
          (mkfields {uint16 glyphID}
                    {UFWORD startConnectorLength}
@@ -869,6 +1008,23 @@
                    {uint16 partCount}
                    {GlyphPart partRecords (count partCount)}))
 
+(mkcmplx MathGlyphConstruction
+         (mkfields {Offset16 glyphAssemblyOffset}
+                   {uint16 variantCount}
+                   {MathGlyphVariantRecord mathGlyphVariantRecords (count variantCount)}
+                   {GlyphAssembly glyphAssembly (at glyphAssemblyOffset)}))
+
+(mkcmplx TableMATH
+         (mkfields {uint16 majorVersion (== 1) unused}
+                   {uint16 minorVersion (== 0) unused}
+                   {Offset16 mathConstantsOffset}
+                   {Offset16 mathGlyphInfoOffset}
+                   {Offset16 mathVariantsOffset}
+                   {MathConstants mathConstants (at mathConstantsOffset)}
+                   {MathGlyphInfo mathGlyphInfo (at mathGlyphInfoOffset)}
+                   {MathVariants mathVariants (at mathVariantsOffset)})
+         #:tag "MATH")
+
 (mkcmplx TableOS∕2
          (mkfields {uint16 version (<= 5)}
                    {FWORD xAvgCharWidth}
@@ -934,7 +1090,6 @@
     (list (~a "\tcase " case-tag ":")
           (~a "\t\tv->" (ptr c) " = calloc(1, sizeof(" (cmplx-name c) "));")
           (~a "\t\tif(read_" (cmplx-name c) "(ctx, v->" (ptr c) ") < 0){")
-          (~a "\t\t\twerrstr(\"%s: %r\", \"" tag "\");")
           (~a "\t\t\tfree(v->" (ptr c) ");")
           (~a "\t\t\tgoto err;")
           (~a "\t\t}")
--- a/test.c
+++ b/test.c
@@ -28,11 +28,15 @@
 
 	r = nil;
 	if(ctx->r != nil){
-		if(off+len > ctx->r->len){
-			werrstr("range overflow by %d bytes", off+len - ctx->r->len);
+		if(len < 0)
+			len = ctx->r->len - off;
+		if(len < 0 || off+len > ctx->r->len){
+			werrstr("range overflow (len %d) by %d bytes", len, off+len - ctx->r->len);
 			goto err;
 		}
 		off += ctx->r->start;
+	}else if(len < 0){
+		len = 0x7fffffff;
 	}
 	if((r = malloc(sizeof(*r))) == nil){
 		werrstr("no memory");
@@ -51,7 +55,6 @@
 	return 0;
 err:
 	free(r);
-	werrstr("ctxpushrange: %r");
 	return -1;
 }
 
@@ -75,7 +78,6 @@
 	free(r);
 	return 0;
 err:
-	werrstr("ctxpoprange: %r");
 	return -1;
 }
 
@@ -88,7 +90,7 @@
 
 	r = ctx->r;
 	if(r != nil && ctx->off+n > r->start+r->len){
-		werrstr("short read: need %d at %d, have %d at %d", n, ctx->off, r->len, r->start);
+		werrstr("need %d at %d, have %d at %d", n, ctx->off, r->len, r->start);
 		goto err;
 	}
 	if(n > ctx->bufsz){
@@ -100,7 +102,7 @@
 		ctx->bufsz = n;
 	}
 	if((x = Bread(ctx->f, ctx->buf, n)) != n){
-		werrstr("short read: need %d, got %d; off %d", n, x, ctx->off);
+		werrstr("need %d, got %d; off %d", n, x, ctx->off);
 		goto err;
 	}
 	ctx->off += n;
@@ -107,7 +109,6 @@
 
 	return ctx->buf;
 err:
-	werrstr("ctxreadn: %r");
 	return nil;
 }
 
@@ -131,7 +132,6 @@
 	return 0;
 err:
 	free(arr);
-	werrstr("ctxarray: %r");
 	return -1;
 }
 
@@ -154,7 +154,7 @@
 		}else if(read_TableDirectory(&ctx, &td) != 0){
 			fprint(2, "%s: %r\n", argv[i]);
 		} else {
-			print_TableDirectory(out, 0, &td);
+			print_TableDirectory(out, indentΔ, &td);
 		}
 		if(ctx.f != nil)
 			Bterm(ctx.f);