shithub: fnt

Download patch

ref: 5f2d73e0f7149c12aeab5c7d06bc9c91de044df6
parent: a24a5759afa209af707c750660236f0719ad8b71
author: Sigrid Solveig Haflínudóttir <sigrid@ftrv.se>
date: Sun Jun 16 19:02:52 EDT 2024

implement conditional fields

--- a/otf.rkt
+++ b/otf.rkt
@@ -66,6 +66,17 @@
 (define (field-verb f)
   (assoc 'verb (field-attrs f)))
 
+(define (field-cond f)
+  (assoc 'cond (field-attrs f)))
+
+(define (block stmt lst)
+  (if (= (length lst) 1) (list* stmt lst) (list (string-append stmt "{") lst "}")))
+
+(define (wrap-cond-c cond lst)
+  (match cond
+    [#f lst]
+    [(list 'cond op ref n) (block (~a "if(v->" ref " " op " " n ")") (indent lst))]))
+
 (define (field-fprint-c f)
   (define t (field-type f))
   (define count (field-count f))
@@ -75,7 +86,7 @@
   (define verb (if (type? t) (if (field-verb f) (cadr (field-verb f)) (type-verb t)) ""))
   (define fprint-index
     (if fixed-array
-        (~a "\tfprint(f, \"%*s%s[%d]: "
+        (~a "fprint(f, \"%*s%s[%d]: "
             verb
             "\\n\", indent, \"\", \""
             (field-name f)
@@ -83,34 +94,26 @@
             (field-name f)
             array-index
             ");")
-        (~a "\tfprint(f, \"%*s%s[%d]:\\n\", indent, \"\", \"" (field-name f) "\", i);")))
+        (~a "fprint(f, \"%*s%s[%d]:\\n\", indent, \"\", \"" (field-name f) "\", i);")))
   (define array-loop
     (if count
         (λ (lst)
-          (if fixed-array
-              (list (~a "for(int i = 0; i < " (cadr count) "; i++)") fprint-index)
-              (list (~a "for(int i = 0; i < v->" (cadr count) "; i++){")
-                    fprint-index
-                    (indent lst)
-                    (~a "}"))))
+          (block (~a "for(int i = 0; i < " (if fixed-array "" "v->") (cadr count) "; i++)")
+                 (indent (list* fprint-index (if fixed-array empty lst)))))
         identity))
-  (if (field-unused? f)
-      empty
-      (array-loop (list (if (type? t)
-                            (~a "fprint(f, \"%*s%s: "
-                                verb
-                                "\\n\", indent, \"\", \""
-                                (field-name f)
-                                "\", v->"
-                                (field-name f)
-                                array-index
-                                ");")
-                            (~a "fprint_"
-                                (cmplx-name t)
-                                "(f, indent+indentΔ, &v->"
-                                (field-name f)
-                                array-index
-                                ");"))))))
+  (define lst
+    (list
+     (if (type? t)
+         (~a "fprint(f, \"%*s%s: "
+             verb
+             "\\n\", indent, \"\", \""
+             (field-name f)
+             "\", v->"
+             (field-name f)
+             array-index
+             ");")
+         (~a "fprint_" (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-struct field (type name attrs)
   #:transparent
@@ -198,33 +201,48 @@
                       (list (~a "};")
                             (~a "int read_" (cmplx-name c) "(Ctx *ctx, " (cmplx-name c) " *v);")))))
    (define (gen-c c b index)
-     (define field-groups (group-by (λ (f) (cmplx? (field-type f))) (cmplx-fields c)))
+     ; group fields to minimize number of reads
+     ; complex types are always alone
+     ; simple types can be combined unless versioned
+     ; versioned simple types are combined if the check is the same
+     (define (group-fields fields)
+       (define (combine? f g)
+         (or (empty? g)
+             (and (type? (field-type f))
+                  (type? (field-type (car g)))
+                  (equal? (field-cond f) (field-cond (car g))))))
+       (define (group- g fields)
+         (match fields
+           [(list-rest f tail)
+            (if (combine? f g) (group- (cons f g) tail) (cons (reverse g) (group- (list f) tail)))]
+           [_ (list (reverse g))]))
+       (group- empty fields))
+     (define (field-size f)
+       (match (field-count f)
+         [#f (type-size (field-type f))]
+         [count
+          #:when (type? (field-type f))
+          (* (cadr count) (type-size (field-type f)))]))
      (define (parse-group g)
        (define (p fs index)
          (if (pair? fs)
-             (cons (super-gen-c (car fs) "b" index)
-                   (p (cdr fs) (+ index (type-size (field-type (car fs))))))
+             (cons (super-gen-c (car fs) "b" index) (p (cdr fs) (+ index (field-size (car fs)))))
              empty))
        (p g 0))
      (define (gen-group-c fields)
-       (define (field-size f)
-         (match (field-count f)
-           [#f (type-size (field-type f))]
-           [count
-            #:when (type? (field-type f))
-            (* (cadr count) (type-size (field-type f)))]))
        (if (cmplx? (field-type (car fields)))
            (map (λ (f) (super-gen-c f #f #f)) fields)
-           (let ([sum (apply + (map (λ (f) (field-size f)) fields))])
-             (list* (~a "if((b = ctxreadn(ctx, " sum ")) == nil)")
-                    (~a "\tgoto err;")
-                    (parse-group fields)))))
+           (let* ([sum (apply + (map (λ (f) (field-size f)) fields))]
+                  [lst (flatten (list* (~a "if((b = ctxreadn(ctx, " sum ")) == nil)")
+                                       (~a "\tgoto err;")
+                                       (parse-group fields)))])
+             (wrap-cond-c (field-cond (car fields)) lst))))
      (flatten
       (append (list (~a "int")
                     (~a "read_" (cmplx-name c) "(Ctx *ctx, " (cmplx-name c) " *v)")
                     (~a "{")
                     (~a "\tu8int *b;"))
-              (indent (flatten (map gen-group-c field-groups)))
+              (indent (flatten (map gen-group-c (group-fields (cmplx-fields c)))))
               (indent (filter-extra (cmplx-extra c) 'read))
               (list (~a "\treturn 0;")
                     (~a "err:")
@@ -281,6 +299,11 @@
     [(_ (n:number)) #''(count n)]
     [(_ {~literal hex}) #''(verb "%#ux")]
     [(_ ({~literal =} vs:number ...+)) #''(= vs ...)]
+    [(_ (p:expr ref:id n:number))
+     #''(cond
+          p
+          ref
+          n)]
     [(_ ({~literal count} n:id))
      (begin
        (define counter (assoc (syntax->datum #`n) fields))
@@ -308,7 +331,7 @@
 
 (define-syntax (mkfields stx)
   (syntax-parse stx
-    [(_ x ...)
+    [(_ x ...+)
      (begin
        (set! fields '())
        #'(list (mkfield [~@ . x]) ...))]))
@@ -408,8 +431,15 @@
                    {FWORD sTypoLineGap}
                    {UFWORD usWinAscent}
                    {UFWORD usWinDescent}
-                   ; FIXME more fields depending on version
-                   )
+                   {uint32 ulCodePageRange1 (>= version 1) hex}
+                   {uint32 ulCodePageRange2 (>= version 1) hex}
+                   {FWORD sxHeight (>= version 2)}
+                   {FWORD sCapHeight (>= version 2)}
+                   {uint16 usDefaultChar (>= version 2) hex}
+                   {uint16 usBreakChar (>= version 2) hex}
+                   {uint16 usMaxContext (>= version 2)}
+                   {uint16 usLowerOpticalPointSize (>= version 5)}
+                   {uint16 usUpperOpticalPointSize (>= version 5)})
          #:tag "OS/2")
 
 (mkcmplx TableRecord