ref: 3c6cc9377be0d201f63491f2878332a62c4dcaa2
parent: 18b2e53f5039cd7da7541ad622c0923c7d4e2872
author: Sigrid Solveig Haflínudóttir <sigrid@ftrv.se>
date: Thu Jul 25 20:52:46 EDT 2024
extend generator with offset'ed arrays support
--- a/gen.rkt
+++ b/gen.rkt
@@ -112,7 +112,7 @@
['< '>=]
['> '<=]))
-(define-struct field (type name attrs)
+(define-struct field (type name in-struct attrs)
#:transparent
#:methods gen:code
[(define/generic super-c-type c-type)
@@ -119,13 +119,15 @@
(define (gen-h f)
(define cnt (field-count f))
(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 (or is-ptr (and cnt (not fixed-array))) "*" "")
- (field-name f)
- (if fixed-array (~a "[" cnt "];") ";"))))
+ (define is-ptr (and (cmplx? (field-type f)) (field-at f)))
+ (list (if (field-in-struct f)
+ (~a (if (field-unused? f) "// " "")
+ (super-c-type (field-type f))
+ " "
+ (if (or is-ptr (and cnt (not fixed-array))) "*" "")
+ (field-name f)
+ (if fixed-array (~a "[" cnt "];") ";"))
+ empty)))
(define (gen-c f b index)
(define (size t)
(if (type? t) (type-size t) 0))
@@ -137,7 +139,7 @@
(if (<= x 32768) (~r x) (~r x #:base 16 #:sign '("0x" "" "-0x"))))
(define (parse-if-error read)
(define t (field-type f))
- (define is-ptr (and (cmplx? (field-type f)) (field-offset f)))
+ (define is-ptr (and (cmplx? (field-type f)) (field-at f)))
(match (field-count f)
[#f
(if (or declared (not (field-unused? f)))
@@ -191,16 +193,38 @@
" && ")]))
ts))
(define (at lst)
- (define at (field-offset f))
+ (define offset (field-offset f))
+ (define offset-plus (if offset (~a (fmt-expr offset) "+") ""))
+ (define offset-and (if offset (~a (fmt-expr offset) " != 0 && ") ""))
+ (define at (field-at f))
+ (define count (field-count f))
(if (not at)
lst
- (list (~a "if(" (fmt-expr at) " != 0){")
- (~a " if(otfpushrange(o, " (fmt-expr at) ", -1) < 0)")
- (~a " goto err;")
- (indent lst)
- (~a " if(otfpoprange(o) < 0)")
- (~a " goto err;")
- (~a "}"))))
+ (if (field-array-with-offsets f)
+ (list
+ (~a "if(" offset-and (field-count f) " > 0){")
+ (~a " " ref " = calloc(" count ", sizeof(*" ref "));")
+ (~a " for(int i = 0; i < " count "; i++){")
+ (if offset empty (list (~a " if(" (fmt-expr at) "[i] == 0)") (~a " continue;")))
+ (~a " if(otfpushrange(o, " offset-plus (fmt-expr at) "[i], -1))")
+ (~a " goto err;")
+ (~a " int r = read_" (name (field-type f)) "(o, " ref "+i);")
+ (~a " if(otfpoprange(o) < 0)")
+ (~a " goto err;")
+ ; FIXME do this only if the field is allowed to fail being read
+ (~a " if(r < 0){")
+ (~a " memset(" ref "+i, 0, sizeof(*" ref "));")
+ (~a " break;")
+ (~a " }")
+ (~a " }")
+ (~a "}"))
+ (list (~a "if(" (fmt-expr at) " != 0){")
+ (~a " if(otfpushrange(o, " offset-plus (fmt-expr at) ", -1) < 0)")
+ (~a " goto err;")
+ (indent lst)
+ (~a " if(otfpoprange(o) < 0)")
+ (~a " goto err;")
+ (~a "}")))))
(at (list*
(parse-if-error #t)
(if index
@@ -233,9 +257,19 @@
(define (field-verb f)
(field-attr f 'verb))
-(define (field-offset f)
+(define (field-at f)
(field-attr f 'at))
+(define (field-offset f)
+ (field-attr f 'offset))
+
+; returns #t if it's an array with each element having its own
+; offset pointed to by elements of another array field
+(define (field-array-with-offsets f)
+ (define v (assoc 'at (field-attrs f)))
+ (define a (caddr v))
+ (>= (length a) 1))
+
(define (field-cond f)
(define v (assoc 'cond (field-attrs f)))
(and v (rest v)))
@@ -290,7 +324,7 @@
(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 is-ptr (and (cmplx? t) (field-at f) (not cnt)))
(define lst
(flatten
(list
@@ -367,7 +401,7 @@
(or (empty? g)
(and (type? (field-type f))
(type? (field-type (car g)))
- (not (or (field-offset f) (field-offset (car g))))
+ (not (or (field-at f) (field-at (car g))))
(no-vla? f)
(no-vla? (car g))
(equal? (field-cond f) (field-cond (car g))))))
@@ -402,7 +436,7 @@
(if (cmplx? (field-type (car fields)))
(map (λ (f) (super-gen-c f #f #f)) fields)
(let* ([sum (foldr (λ (f accu) (add (field-size f) accu)) 0 fields)]
- [lst (flatten (list (if (field-offset (car fields))
+ [lst (flatten (list (if (field-at (car fields))
empty
(list (~a "if((b = otfreadn(o, " sum ")) == nil)")
(~a " goto err;")
@@ -517,15 +551,20 @@
#:description "field reference"
(pattern ref:id
#:fail-when (not (assoc (syntax-e #'ref) fields)) "no such field"
- #:with type (cadr (assoc (syntax-e #'ref) fields))))
+ #:with type (cadr (assoc (syntax-e #'ref) fields))
+ #:with attrs (caddr (assoc (syntax-e #'ref) fields))))
(syntax-parse stx
[(_ type name {~literal ->o}) #''(->o #t)]
+ [(_ _ _ ({~literal offset} ref:ref))
+ #:fail-when (not (type-offset? (syntax-e #'ref.type))) "can't be used as an offset"
+ #''(offset ref)]
+ [(_ _ _ ({~literal offset} e:expr)) #''(offset e)]
[(_ _ _ ({~literal at} ref:ref))
#:fail-when (not (type-offset? (syntax-e #'ref.type))) "can't be used as an offset"
- #''(at ref)]
+ #''(at ref ref.attrs)]
; FIXME - check fields and ops/numbers
- [(_ _ _ ({~literal at} e:expr)) #''(at e)]
+ [(_ _ _ ({~literal at} e:expr)) #''(at e ())]
[(_ type _ {~literal hex})
#:fail-when (not (type-number? (syntax-e #'type))) "not a number type"
#'(list 'verb (verb-hex (type-bits type)))]
@@ -559,7 +598,9 @@
(define-syntax-class name
#:description "field name"
(pattern name:id
- #:fail-when (assoc (syntax-e #'name) fields) "duplicate field name"))
+ ;#:fail-when (assoc (syntax-e #'name) fields) "duplicate field name"
+ ; FIXME this should fail as duplicate if types are different or no/same condition is used
+ ))
(define-syntax-class type
#:description "field type"
@@ -569,9 +610,11 @@
(syntax-parse stx
[(_ type:type name:name attrs ...)
(begin
- (set! fields (cons (syntax->datum #'(name type)) fields))
- #'(field type
+ (define uniq (not (assoc (syntax-e #'name) fields)))
+ (set! fields (cons (syntax->datum #'(name type (attrs ...))) fields))
+ #`(field type
`name
+ #,uniq
(list (mkattr type `name [~@ attrs]) ...)))]))
(define-syntax (mkfields stx)
--- a/otf.rkt
+++ b/otf.rkt
@@ -455,14 +455,26 @@
(mkcmplx FeatureList {uint16 featureCount} {FeatureRecord featureRecords [featureCount]})
+(define lookupFlags
+ #hash((0 . LOOKUP_FL_RIGHT_TO_LEFT)
+ (1 . LOOKUP_FL_IGNORE_BASE_GLYPHS)
+ (2 . LOOKUP_FL_IGNORE_LIGATURES)
+ (3 . LOOKUP_FL_IGNORE_MARKS)
+ (4 . LOOKUP_FL_USE_MARK_FILTERING_SET)))
+; FIXME 0xff00 is attachment class filter
+
(mkcmplx Lookup
{uint16 lookupType}
- {uint16 lookupFlag}
+ {uint16 lookupFlag hex (bits lookupFlags)}
{uint16 subTableCount}
{Offset16 subtableOffsets [subTableCount]}
{uint16 markFilteringSet})
+; FIXME lookup subtable based on current table (GPOS vs GSUB) and lookupType
-(mkcmplx LookupList {uint16 lookupCount} {Offset16 lookupOffsets [lookupCount]})
+(mkcmplx LookupList
+ {uint16 lookupCount}
+ {Offset16 lookupOffsets [lookupCount]}
+ {Lookup lookups [lookupCount] (at lookupOffsets)})
(mkcmplx TableGPOS
{uint16 majorVersion (== 1) unused}
@@ -584,27 +596,6 @@
{Coverage1 cov1 (== format 1)}
{Coverage2 cov2 (== format 2)})
-(mkcmplx MathVariants
- {UFWORD minConnectorOverlap}
- {Offset16 vertGlyphCoverageOffset}
- {Offset16 horizGlyphCoverageOffset}
- {uint16 vertGlyphCount}
- {uint16 horizGlyphCount}
- {Offset16 vertGlyphConstructionOffsets [vertGlyphCount]}
- {Offset16 horizGlyphConstructionOffsets [horizGlyphCount]}
- {Coverage vertGlyphCoverage (at vertGlyphCoverageOffset)}
- {Coverage horizGlyphCoverage (at horizGlyphCoverageOffset)})
-
-(mkcmplx MathGlyphInfo
- {Offset16 mathItalicsCorrectionInfoOffset}
- {Offset16 mathTopAccentAttachmentOffset}
- {Offset16 extendedShapeCoverageOffset}
- {Offset16 mathKernInfoOffset}
- {MathItalicsCorrectionInfo mathItalicsCorrectionInfo (at mathItalicsCorrectionInfoOffset)}
- {MathTopAccentAttachment mathTopAccentAttachment (at mathTopAccentAttachmentOffset)}
- {MathKernInfo mathKernInfo (at mathKernInfoOffset)}
- {Coverage extendedShapeCoverage (at extendedShapeCoverageOffset)})
-
(mkcmplx MathGlyphVariantRecord {uint16 variantGlyph} {UFWORD advanceMeasurement})
(mkcmplx GlyphPart
@@ -625,6 +616,30 @@
{MathGlyphVariantRecord mathGlyphVariantRecords [variantCount]}
{GlyphAssembly glyphAssembly (at glyphAssemblyOffset)})
+(mkcmplx
+ MathVariants
+ {UFWORD minConnectorOverlap}
+ {Offset16 vertGlyphCoverageOffset}
+ {Offset16 horizGlyphCoverageOffset}
+ {uint16 vertGlyphCount}
+ {uint16 horizGlyphCount}
+ {Offset16 vertGlyphConstructionOffsets [vertGlyphCount]}
+ {Offset16 horizGlyphConstructionOffsets [horizGlyphCount]}
+ {Coverage vertGlyphCoverage (at vertGlyphCoverageOffset)}
+ {Coverage horizGlyphCoverage (at horizGlyphCoverageOffset)}
+ {MathGlyphConstruction vertGlyphConstruction [vertGlyphCount] (at vertGlyphConstructionOffsets)}
+ {MathGlyphConstruction horizGlyphConstruction [horizGlyphCount] (at horizGlyphConstructionOffsets)})
+
+(mkcmplx MathGlyphInfo
+ {Offset16 mathItalicsCorrectionInfoOffset}
+ {Offset16 mathTopAccentAttachmentOffset}
+ {Offset16 extendedShapeCoverageOffset}
+ {Offset16 mathKernInfoOffset}
+ {MathItalicsCorrectionInfo mathItalicsCorrectionInfo (at mathItalicsCorrectionInfoOffset)}
+ {MathTopAccentAttachment mathTopAccentAttachment (at mathTopAccentAttachmentOffset)}
+ {MathKernInfo mathKernInfo (at mathKernInfoOffset)}
+ {Coverage extendedShapeCoverage (at extendedShapeCoverageOffset)})
+
(mkcmplx TableMATH
{uint16 majorVersion (== 1) unused}
{uint16 minorVersion (== 0) unused}
@@ -656,10 +671,18 @@
{KernClass rightClass (at rightClassOffset)}
{FWORD kerningArray [/ rowWidth 2]})
+(define kernCoverageFlags
+ #hash((0 . KERN_FL_HORIZONTAL)
+ (1 . KERN_FL_MINIMUM)
+ (2 . KERN_FL_CROSS_STREAM)
+ (3 . KERN_FL_OVERRIDE)))
+
(mkcmplx KernSubtable
{uint16 version (== 0) unused}
{uint16 length (>= 6)} ; including this header
- {uint16 coverage hex}) ; FIXME the rest depends on the coverage bits (8-15 = 0 or 2)
+ {uint16 coverage hex (bits kernCoverageFlags)}
+ {KernSubtable0 sub0 (== (>> coverage 8) 0)}
+ {KernSubtable2 sub2 (== (>> coverage 8) 2)})
(mkcmplx TableKern
{uint16 version (== 0) unused}
@@ -674,6 +697,24 @@
#:after (list TableHead TableMaxp)
#:optional #t)
+(mkcmplx GlyphData {int16 originOffsetX} {int16 originOffsetY} {Tag graphicType})
+
+(mkcmplx SbixStrike
+ {uint16 ppem}
+ {uint16 ppi}
+ {Offset32 glyphDataOffsets [+ o->numGlyphs 1]}
+ {GlyphData glyphData [o->numGlyphs] (at glyphDataOffsets)})
+
+(define sbixFlags #hash((1 . SBIX_FL_DRAW_OUTLINES)))
+
+(mkcmplx TableSbix
+ {uint16 version (== 1) unused}
+ {uint16 flags hex (bits sbixFlags)}
+ {uint32 numStrikes}
+ {Offset32 strikeOffsets [numStrikes]}
+ {SbixStrike strikes [numStrikes] (at strikeOffsets)}
+ #:tag "sbix")
+
(define variationAxisRecordFlags #hash((0 . VARIATIONAXISRECORD_FL_HIDDEN_AXIS)))
(mkcmplx VariationAxisRecord
@@ -732,10 +773,7 @@
{Offset32 variationRegionListOffset}
{uint16 itemVariationDataCount}
{Offset32 itemVariationDataOffsets [itemVariationDataCount]}
- #;{ItemVariationData
- itemVariationData
- [itemVariationDataCount]
- (at itemVariationDataOffsets)}) ; FIXME offsets array
+ {ItemVariationData itemVariationData [itemVariationDataCount] (at itemVariationDataOffsets)})
(mkcmplx DeltaSetIndexMap
{uint8 format (== 0 1)}
@@ -792,9 +830,44 @@
{uint16 axisOrdering}
{uint8 unused [- o->designAxisSize 8] unused})
+(mkcmplx AxisValueTable1 {uint16 axisIndex} {uint16 flags} {uint16 valueNameID} {Fixed value})
+
+(mkcmplx AxisValueTable2
+ {uint16 axisIndex}
+ {uint16 flags}
+ {uint16 valueNameID}
+ {Fixed nominalValue}
+ {Fixed rangeMinValue}
+ {Fixed rangeMaxValue})
+
+(mkcmplx AxisValueTable3
+ {uint16 axisIndex}
+ {uint16 flags}
+ {uint16 valueNameID}
+ {Fixed value}
+ {Fixed linkedValue})
+
+(mkcmplx AxisValue {uint16 axisIndex} {Fixed value})
+
+(mkcmplx AxisValueTable4
+ {uint16 axisCount}
+ {uint16 flags}
+ {uint16 valueNameID}
+ {AxisValue axisValues [axisCount]})
+
+(mkcmplx AxisValueTable
+ {uint16 format}
+ {AxisValueTable1 f1 (== format 1)} ; FIXME should be a union
+ {AxisValueTable2 f2 (== format 2)}
+ {AxisValueTable3 f3 (== format 3)}
+ {AxisValueTable3 f4 (== format 4)})
+
(mkcmplx DesignAxes
{AxisRecord designAxes [o->designAxisCount]}
- {Offset16 axisValueOffsets [o->axisValueCount]}) ; FIXME array of offsets
+ {Offset16
+ axisValueTableOffsets
+ [o->axisValueCount]} ; FIXME each offset is from the start of this field itself?
+ {AxisValueTable axisValueTables [o->axisValueCount] (at axisValueTableOffsets)})
(mkcmplx TableSTAT
{uint16 majorVersion (== 1) unused}
@@ -803,13 +876,19 @@
{uint16 designAxisCount ->o}
{Offset32 designAxesOffset}
{uint16 axisValueCount ->o}
- {Offset32 offsetToAxisValueOffsets}
+ {Offset32 offsetToAxisValueOffsets} ; FIXME wtf???
{uint16 elidedFallbackNameID (> minorVersion 0)}
{DesignAxes designAxes (at designAxesOffset)}
#:tag "STAT")
-(mkcmplx GaspRange {uint16 rangeMaxPPEM} {uint16 rangeGaspBehavior hex})
+(define rangeGaspBehaviorFlags
+ #hash((0 . GASP_GRIDFIT)
+ (1 . GASP_DOGRAY)
+ (2 . GASP_SYMMETRIC_GRIDFIT)
+ (3 . GASP_SYMMETRIC_SMOOTHING)))
+(mkcmplx GaspRange {uint16 rangeMaxPPEM} {uint16 rangeGaspBehavior hex (bits rangeGaspBehaviorFlags)})
+
(mkcmplx TableGasp
{uint16 version (== 0 1)}
{uint16 numRanges}
@@ -826,9 +905,10 @@
(mkcmplx TupleVariationHeader
{uint16 variationDataSize}
{uint16 tupleIndex (bits tupleIndexFlags)} ; FIXME 0xfff is TUPLE_INDEX_MASK
- {Tuple peakTuple}
- {Tuple intermediateStartTuple}
- {Tuple intermediateEndTuple})
+ {Tuple peakTuple (!= (& tupleIndex TUPLEINDEX_FL_EMBEDDED_PEAK_TUPLE) 0)}
+ {Tuple intermediateStartTuple (!= (& tupleIndex TUPLEINDEX_FL_INTERMEDIATE_REGION) 0)}
+ {Tuple intermediateEndTuple (!= (& tupleIndex TUPLEINDEX_FL_INTERMEDIATE_REGION) 0)})
+; FIXME there is delta packed stuff here
(define glyphVariationDataCountFlags #hash((15 . COUNT_FL_SHARED_POINT_NUMBERS)))
@@ -843,14 +923,26 @@
TableGvar
{uint16 majorVersion (== 1) unused}
{uint16 minorVersion (== 0) unused}
- {uint16 axisCount #;(== o->axisCount)} ; FIXME
+ {uint16 axisCount ->o}
{uint16 sharedTupleCount}
{Offset32 sharedTuplesOffset}
- {uint16 glyphCount #;(== o->glyphCount)} ; FIXME
+ {uint16 glyphCount}
{uint16 flags hex (bits gvarFlags)}
{Offset32 glyphVariationDataArrayOffset}
{Offset16 glyphVariationDataOffsetsShort [+ glyphCount 1] (== (& flags GVAR_FL_LONG_OFFSETS) 0)}
{Offset32 glyphVariationDataOffsetsLong [+ glyphCount 1] (!= (& flags GVAR_FL_LONG_OFFSETS) 0)}
+ {GlyphVariationData
+ glyphVariationData
+ [glyphCount]
+ (offset glyphVariationDataArrayOffset)
+ (at glyphVariationDataOffsetsShort)
+ (== (& flags GVAR_FL_LONG_OFFSETS) 0)}
+ {GlyphVariationData
+ glyphVariationData
+ [glyphCount]
+ (offset glyphVariationDataArrayOffset)
+ (at glyphVariationDataOffsetsLong)
+ (!= (& flags GVAR_FL_LONG_OFFSETS) 0)}
{Tuple sharedTuples [sharedTupleCount] (at sharedTuplesOffset)}
#:tag "gvar"
#:after (list TableFvar))
--- a/plan9/otf.c
+++ b/plan9/otf.c
@@ -3505,7 +3505,7 @@
print_Lookup(Otfile *f, int indent, Otf *o, Lookup *v)
{
f->print(f->aux, "%*s%s: %ud\n", indent, "", "lookupType", v->lookupType);
- f->print(f->aux, "%*s%s: %ud\n", indent, "", "lookupFlag", v->lookupFlag);
+ f->print(f->aux, "%*s%s: %#ux%s%s%s%s%s\n", indent, "", "lookupFlag", v->lookupFlag, (v->lookupFlag&LOOKUP_FL_RIGHT_TO_LEFT)?" LOOKUP_FL_RIGHT_TO_LEFT":"", (v->lookupFlag&LOOKUP_FL_IGNORE_BASE_GLYPHS)?" LOOKUP_FL_IGNORE_BASE_GLYPHS":"", (v->lookupFlag&LOOKUP_FL_IGNORE_LIGATURES)?" LOOKUP_FL_IGNORE_LIGATURES":"", (v->lookupFlag&LOOKUP_FL_IGNORE_MARKS)?" LOOKUP_FL_IGNORE_MARKS":"", (v->lookupFlag&LOOKUP_FL_USE_MARK_FILTERING_SET)?" LOOKUP_FL_USE_MARK_FILTERING_SET":"");
f->print(f->aux, "%*s%s: %ud\n", indent, "", "subTableCount", v->subTableCount);
for(int i = 0; i < v->subTableCount; i++)
f->print(f->aux, "%*s%s[%d]: %ud\n", indent, "", "subtableOffsets", i, v->subtableOffsets[i]);
@@ -3525,6 +3525,22 @@
v->lookupOffsets = malloc(v->lookupCount*sizeof(*v->lookupOffsets));
for(int i = 0; i < v->lookupCount; i++)
v->lookupOffsets[i] = b[0+i*2]<<8 | b[1+i*2];
+ if(v->lookupCount > 0){
+ v->lookups = calloc(v->lookupCount, sizeof(*v->lookups));
+ for(int i = 0; i < v->lookupCount; i++){
+ if(v->lookupOffsets[i] == 0)
+ continue;
+ if(otfpushrange(o, v->lookupOffsets[i], -1))
+ goto err;
+ int r = read_Lookup(o, v->lookups+i);
+ if(otfpoprange(o) < 0)
+ goto err;
+ if(r < 0){
+ memset(v->lookups+i, 0, sizeof(*v->lookups));
+ break;
+ }
+ }
+ }
return 0;
err:
werrstr("%s: %r", "LookupList");
@@ -3537,6 +3553,10 @@
f->print(f->aux, "%*s%s: %ud\n", indent, "", "lookupCount", v->lookupCount);
for(int i = 0; i < v->lookupCount; i++)
f->print(f->aux, "%*s%s[%d]: %ud\n", indent, "", "lookupOffsets", i, v->lookupOffsets[i]);
+ for(int i = 0; i < v->lookupCount; i++){
+ f->print(f->aux, "%*s%s[%d]:\n", indent, "", "lookups", i);
+ print_Lookup(f, indent+indentΔ, o, &v->lookups[i]);
+ }
USED(o);
}
@@ -4328,6 +4348,134 @@
}
int
+read_MathGlyphVariantRecord(Otf *o, MathGlyphVariantRecord *v)
+{
+ u8int *b = nil; USED(b);
+ if((b = otfreadn(o, 4)) == nil)
+ goto err;
+ v->variantGlyph = b[0]<<8 | b[1];
+ v->advanceMeasurement = b[2]<<8 | b[3];
+ return 0;
+err:
+ werrstr("%s: %r", "MathGlyphVariantRecord");
+ return -1;
+}
+
+void
+print_MathGlyphVariantRecord(Otfile *f, int indent, Otf *o, MathGlyphVariantRecord *v)
+{
+ f->print(f->aux, "%*s%s: %ud\n", indent, "", "variantGlyph", v->variantGlyph);
+ f->print(f->aux, "%*s%s: %ud\n", indent, "", "advanceMeasurement", v->advanceMeasurement);
+ USED(o);
+}
+
+int
+read_GlyphPart(Otf *o, GlyphPart *v)
+{
+ u8int *b = nil; USED(b);
+ if((b = otfreadn(o, 10)) == nil)
+ goto err;
+ v->glyphID = b[0]<<8 | b[1];
+ v->startConnectorLength = b[2]<<8 | b[3];
+ v->endConnectorLength = b[4]<<8 | b[5];
+ v->fullAdvance = b[6]<<8 | b[7];
+ v->partFlags = b[8]<<8 | b[9];
+ return 0;
+err:
+ werrstr("%s: %r", "GlyphPart");
+ return -1;
+}
+
+void
+print_GlyphPart(Otfile *f, int indent, Otf *o, GlyphPart *v)
+{
+ f->print(f->aux, "%*s%s: %ud\n", indent, "", "glyphID", v->glyphID);
+ f->print(f->aux, "%*s%s: %ud\n", indent, "", "startConnectorLength", v->startConnectorLength);
+ f->print(f->aux, "%*s%s: %ud\n", indent, "", "endConnectorLength", v->endConnectorLength);
+ f->print(f->aux, "%*s%s: %ud\n", indent, "", "fullAdvance", v->fullAdvance);
+ f->print(f->aux, "%*s%s: %ud\n", indent, "", "partFlags", v->partFlags);
+ USED(o);
+}
+
+int
+read_GlyphAssembly(Otf *o, GlyphAssembly *v)
+{
+ u8int *b = nil; USED(b);
+ if(read_MathValueRecord(o, &v->italicsCorrection) < 0){
+ werrstr("%s: %r", "italicsCorrection");
+ goto err;
+ }
+ if((b = otfreadn(o, 2)) == nil)
+ goto err;
+ v->partCount = b[0]<<8 | b[1];
+ if(otfarray(o, &v->partRecords, read_GlyphPart, sizeof(GlyphPart), v->partCount) < 0){
+ werrstr("%s: %r", "partRecords");
+ goto err;
+ }
+ return 0;
+err:
+ werrstr("%s: %r", "GlyphAssembly");
+ return -1;
+}
+
+void
+print_GlyphAssembly(Otfile *f, int indent, Otf *o, GlyphAssembly *v)
+{
+ f->print(f->aux, "%*s%s:\n", indent, "", "italicsCorrection");
+ print_MathValueRecord(f, indent+indentΔ, o, &v->italicsCorrection);
+ f->print(f->aux, "%*s%s: %ud\n", indent, "", "partCount", v->partCount);
+ for(int i = 0; i < v->partCount; i++){
+ f->print(f->aux, "%*s%s[%d]:\n", indent, "", "partRecords", i);
+ print_GlyphPart(f, indent+indentΔ, o, &v->partRecords[i]);
+ }
+ USED(o);
+}
+
+int
+read_MathGlyphConstruction(Otf *o, MathGlyphConstruction *v)
+{
+ u8int *b = nil; USED(b);
+ if((b = otfreadn(o, 4)) == nil)
+ goto err;
+ v->glyphAssemblyOffset = b[0]<<8 | b[1];
+ v->variantCount = b[2]<<8 | b[3];
+ if(otfarray(o, &v->mathGlyphVariantRecords, read_MathGlyphVariantRecord, sizeof(MathGlyphVariantRecord), v->variantCount) < 0){
+ werrstr("%s: %r", "mathGlyphVariantRecords");
+ goto err;
+ }
+ if(v->glyphAssemblyOffset != 0){
+ if(otfpushrange(o, v->glyphAssemblyOffset, -1) < 0)
+ goto err;
+ v->glyphAssembly = calloc(1, sizeof(*v->glyphAssembly));
+ if(read_GlyphAssembly(o, v->glyphAssembly) < 0){
+ werrstr("%s: %r", "glyphAssembly");
+ goto err;
+ }
+ if(otfpoprange(o) < 0)
+ goto err;
+ }
+ return 0;
+err:
+ werrstr("%s: %r", "MathGlyphConstruction");
+ return -1;
+}
+
+void
+print_MathGlyphConstruction(Otfile *f, int indent, Otf *o, MathGlyphConstruction *v)
+{
+ f->print(f->aux, "%*s%s: %ud\n", indent, "", "glyphAssemblyOffset", v->glyphAssemblyOffset);
+ f->print(f->aux, "%*s%s: %ud\n", indent, "", "variantCount", v->variantCount);
+ for(int i = 0; i < v->variantCount; i++){
+ f->print(f->aux, "%*s%s[%d]:\n", indent, "", "mathGlyphVariantRecords", i);
+ print_MathGlyphVariantRecord(f, indent+indentΔ, o, &v->mathGlyphVariantRecords[i]);
+ }
+ f->print(f->aux, "%*s%s:\n", indent, "", "glyphAssembly");
+ if(v->glyphAssembly != nil)
+ print_GlyphAssembly(f, indent+indentΔ, o, v->glyphAssembly);
+ USED(o);
+}
+
+int
read_MathVariants(Otf *o, MathVariants *v)
{
u8int *b = nil; USED(b);
@@ -4370,6 +4518,38 @@
if(otfpoprange(o) < 0)
goto err;
}
+ if(v->vertGlyphCount > 0){
+ v->vertGlyphConstruction = calloc(v->vertGlyphCount, sizeof(*v->vertGlyphConstruction));
+ for(int i = 0; i < v->vertGlyphCount; i++){
+ if(v->vertGlyphConstructionOffsets[i] == 0)
+ continue;
+ if(otfpushrange(o, v->vertGlyphConstructionOffsets[i], -1))
+ goto err;
+ int r = read_MathGlyphConstruction(o, v->vertGlyphConstruction+i);
+ if(otfpoprange(o) < 0)
+ goto err;
+ if(r < 0){
+ memset(v->vertGlyphConstruction+i, 0, sizeof(*v->vertGlyphConstruction));
+ break;
+ }
+ }
+ }
+ if(v->horizGlyphCount > 0){
+ v->horizGlyphConstruction = calloc(v->horizGlyphCount, sizeof(*v->horizGlyphConstruction));
+ for(int i = 0; i < v->horizGlyphCount; i++){
+ if(v->horizGlyphConstructionOffsets[i] == 0)
+ continue;
+ if(otfpushrange(o, v->horizGlyphConstructionOffsets[i], -1))
+ goto err;
+ int r = read_MathGlyphConstruction(o, v->horizGlyphConstruction+i);
+ if(otfpoprange(o) < 0)
+ goto err;
+ if(r < 0){
+ memset(v->horizGlyphConstruction+i, 0, sizeof(*v->horizGlyphConstruction));
+ break;
+ }
+ }
+ }
return 0;
err:
werrstr("%s: %r", "MathVariants");
@@ -4394,6 +4574,14 @@
f->print(f->aux, "%*s%s:\n", indent, "", "horizGlyphCoverage");
if(v->horizGlyphCoverage != nil)
print_Coverage(f, indent+indentΔ, o, v->horizGlyphCoverage);
+ for(int i = 0; i < v->vertGlyphCount; i++){
+ f->print(f->aux, "%*s%s[%d]:\n", indent, "", "vertGlyphConstruction", i);
+ print_MathGlyphConstruction(f, indent+indentΔ, o, &v->vertGlyphConstruction[i]);
+ }
+ for(int i = 0; i < v->horizGlyphCount; i++){
+ f->print(f->aux, "%*s%s[%d]:\n", indent, "", "horizGlyphConstruction", i);
+ print_MathGlyphConstruction(f, indent+indentΔ, o, &v->horizGlyphConstruction[i]);
+ }
USED(o);
}
@@ -4480,134 +4668,6 @@
}
int
-read_MathGlyphVariantRecord(Otf *o, MathGlyphVariantRecord *v)
-{
- u8int *b = nil; USED(b);
- if((b = otfreadn(o, 4)) == nil)
- goto err;
- v->variantGlyph = b[0]<<8 | b[1];
- v->advanceMeasurement = b[2]<<8 | b[3];
- return 0;
-err:
- werrstr("%s: %r", "MathGlyphVariantRecord");
- return -1;
-}
-
-void
-print_MathGlyphVariantRecord(Otfile *f, int indent, Otf *o, MathGlyphVariantRecord *v)
-{
- f->print(f->aux, "%*s%s: %ud\n", indent, "", "variantGlyph", v->variantGlyph);
- f->print(f->aux, "%*s%s: %ud\n", indent, "", "advanceMeasurement", v->advanceMeasurement);
- USED(o);
-}
-
-int
-read_GlyphPart(Otf *o, GlyphPart *v)
-{
- u8int *b = nil; USED(b);
- if((b = otfreadn(o, 10)) == nil)
- goto err;
- v->glyphID = b[0]<<8 | b[1];
- v->startConnectorLength = b[2]<<8 | b[3];
- v->endConnectorLength = b[4]<<8 | b[5];
- v->fullAdvance = b[6]<<8 | b[7];
- v->partFlags = b[8]<<8 | b[9];
- return 0;
-err:
- werrstr("%s: %r", "GlyphPart");
- return -1;
-}
-
-void
-print_GlyphPart(Otfile *f, int indent, Otf *o, GlyphPart *v)
-{
- f->print(f->aux, "%*s%s: %ud\n", indent, "", "glyphID", v->glyphID);
- f->print(f->aux, "%*s%s: %ud\n", indent, "", "startConnectorLength", v->startConnectorLength);
- f->print(f->aux, "%*s%s: %ud\n", indent, "", "endConnectorLength", v->endConnectorLength);
- f->print(f->aux, "%*s%s: %ud\n", indent, "", "fullAdvance", v->fullAdvance);
- f->print(f->aux, "%*s%s: %ud\n", indent, "", "partFlags", v->partFlags);
- USED(o);
-}
-
-int
-read_GlyphAssembly(Otf *o, GlyphAssembly *v)
-{
- u8int *b = nil; USED(b);
- if(read_MathValueRecord(o, &v->italicsCorrection) < 0){
- werrstr("%s: %r", "italicsCorrection");
- goto err;
- }
- if((b = otfreadn(o, 2)) == nil)
- goto err;
- v->partCount = b[0]<<8 | b[1];
- if(otfarray(o, &v->partRecords, read_GlyphPart, sizeof(GlyphPart), v->partCount) < 0){
- werrstr("%s: %r", "partRecords");
- goto err;
- }
- return 0;
-err:
- werrstr("%s: %r", "GlyphAssembly");
- return -1;
-}
-
-void
-print_GlyphAssembly(Otfile *f, int indent, Otf *o, GlyphAssembly *v)
-{
- f->print(f->aux, "%*s%s:\n", indent, "", "italicsCorrection");
- print_MathValueRecord(f, indent+indentΔ, o, &v->italicsCorrection);
- f->print(f->aux, "%*s%s: %ud\n", indent, "", "partCount", v->partCount);
- for(int i = 0; i < v->partCount; i++){
- f->print(f->aux, "%*s%s[%d]:\n", indent, "", "partRecords", i);
- print_GlyphPart(f, indent+indentΔ, o, &v->partRecords[i]);
- }
- USED(o);
-}
-
-int
-read_MathGlyphConstruction(Otf *o, MathGlyphConstruction *v)
-{
- u8int *b = nil; USED(b);
- if((b = otfreadn(o, 4)) == nil)
- goto err;
- v->glyphAssemblyOffset = b[0]<<8 | b[1];
- v->variantCount = b[2]<<8 | b[3];
- if(otfarray(o, &v->mathGlyphVariantRecords, read_MathGlyphVariantRecord, sizeof(MathGlyphVariantRecord), v->variantCount) < 0){
- werrstr("%s: %r", "mathGlyphVariantRecords");
- goto err;
- }
- if(v->glyphAssemblyOffset != 0){
- if(otfpushrange(o, v->glyphAssemblyOffset, -1) < 0)
- goto err;
- v->glyphAssembly = calloc(1, sizeof(*v->glyphAssembly));
- if(read_GlyphAssembly(o, v->glyphAssembly) < 0){
- werrstr("%s: %r", "glyphAssembly");
- goto err;
- }
- if(otfpoprange(o) < 0)
- goto err;
- }
- return 0;
-err:
- werrstr("%s: %r", "MathGlyphConstruction");
- return -1;
-}
-
-void
-print_MathGlyphConstruction(Otfile *f, int indent, Otf *o, MathGlyphConstruction *v)
-{
- f->print(f->aux, "%*s%s: %ud\n", indent, "", "glyphAssemblyOffset", v->glyphAssemblyOffset);
- f->print(f->aux, "%*s%s: %ud\n", indent, "", "variantCount", v->variantCount);
- for(int i = 0; i < v->variantCount; i++){
- f->print(f->aux, "%*s%s[%d]:\n", indent, "", "mathGlyphVariantRecords", i);
- print_MathGlyphVariantRecord(f, indent+indentΔ, o, &v->mathGlyphVariantRecords[i]);
- }
- f->print(f->aux, "%*s%s:\n", indent, "", "glyphAssembly");
- if(v->glyphAssembly != nil)
- print_GlyphAssembly(f, indent+indentΔ, o, v->glyphAssembly);
- USED(o);
-}
-
-int
read_TableMATH(Otf *o, TableMATH *v)
{
u8int *b = nil; USED(b);
@@ -4848,6 +4908,18 @@
goto err;
}
v->coverage = b[4]<<8 | b[5];
+ if((v->coverage>>8) == 0){
+ if(read_KernSubtable0(o, &v->sub0) < 0){
+ werrstr("%s: %r", "sub0");
+ goto err;
+ }
+ }
+ if((v->coverage>>8) == 2){
+ if(read_KernSubtable2(o, &v->sub2) < 0){
+ werrstr("%s: %r", "sub2");
+ goto err;
+ }
+ }
return 0;
err:
werrstr("%s: %r", "KernSubtable");
@@ -4858,7 +4930,15 @@
print_KernSubtable(Otfile *f, int indent, Otf *o, KernSubtable *v)
{
f->print(f->aux, "%*s%s: %ud\n", indent, "", "length", v->length);
- f->print(f->aux, "%*s%s: %#ux\n", indent, "", "coverage", v->coverage);
+ f->print(f->aux, "%*s%s: %#ux%s%s%s%s\n", indent, "", "coverage", v->coverage, (v->coverage&KERN_FL_HORIZONTAL)?" KERN_FL_HORIZONTAL":"", (v->coverage&KERN_FL_MINIMUM)?" KERN_FL_MINIMUM":"", (v->coverage&KERN_FL_CROSS_STREAM)?" KERN_FL_CROSS_STREAM":"", (v->coverage&KERN_FL_OVERRIDE)?" KERN_FL_OVERRIDE":"");
+ if((v->coverage>>8) == 0){
+ f->print(f->aux, "%*s%s:\n", indent, "", "sub0");
+ print_KernSubtable0(f, indent+indentΔ, o, &v->sub0);
+ }
+ if((v->coverage>>8) == 2){
+ f->print(f->aux, "%*s%s:\n", indent, "", "sub2");
+ print_KernSubtable2(f, indent+indentΔ, o, &v->sub2);
+ }
USED(o);
}
@@ -4934,6 +5014,133 @@
}
int
+read_GlyphData(Otf *o, GlyphData *v)
+{
+ u8int *b = nil; USED(b);
+ if((b = otfreadn(o, 8)) == nil)
+ goto err;
+ v->originOffsetX = b[0]<<8 | b[1];
+ v->originOffsetY = b[2]<<8 | b[3];
+ v->graphicType = b[4]<<24 | b[5]<<16 | b[6]<<8 | b[7];
+ return 0;
+err:
+ werrstr("%s: %r", "GlyphData");
+ return -1;
+}
+
+void
+print_GlyphData(Otfile *f, int indent, Otf *o, GlyphData *v)
+{
+ f->print(f->aux, "%*s%s: %d\n", indent, "", "originOffsetX", v->originOffsetX);
+ f->print(f->aux, "%*s%s: %d\n", indent, "", "originOffsetY", v->originOffsetY);
+ f->print(f->aux, "%*s%s: %c%c%c%c\n", indent, "", "graphicType", v->graphicType>>24, v->graphicType>>16, v->graphicType>>8, v->graphicType>>0);
+ USED(o);
+}
+
+int
+read_SbixStrike(Otf *o, SbixStrike *v)
+{
+ u8int *b = nil; USED(b);
+ if((b = otfreadn(o, 4)) == nil)
+ goto err;
+ v->ppem = b[0]<<8 | b[1];
+ v->ppi = b[2]<<8 | b[3];
+ if((b = otfreadn(o, (o->numGlyphs+1)*4)) == nil)
+ goto err;
+ v->glyphDataOffsets = malloc((o->numGlyphs+1)*sizeof(*v->glyphDataOffsets));
+ for(int i = 0; i < (o->numGlyphs+1); i++)
+ v->glyphDataOffsets[i] = b[0+i*4]<<24 | b[1+i*4]<<16 | b[2+i*4]<<8 | b[3+i*4];
+ if(o->numGlyphs > 0){
+ v->glyphData = calloc(o->numGlyphs, sizeof(*v->glyphData));
+ for(int i = 0; i < o->numGlyphs; i++){
+ if(v->glyphDataOffsets[i] == 0)
+ continue;
+ if(otfpushrange(o, v->glyphDataOffsets[i], -1))
+ goto err;
+ int r = read_GlyphData(o, v->glyphData+i);
+ if(otfpoprange(o) < 0)
+ goto err;
+ if(r < 0){
+ memset(v->glyphData+i, 0, sizeof(*v->glyphData));
+ break;
+ }
+ }
+ }
+ return 0;
+err:
+ werrstr("%s: %r", "SbixStrike");
+ return -1;
+}
+
+void
+print_SbixStrike(Otfile *f, int indent, Otf *o, SbixStrike *v)
+{
+ f->print(f->aux, "%*s%s: %ud\n", indent, "", "ppem", v->ppem);
+ f->print(f->aux, "%*s%s: %ud\n", indent, "", "ppi", v->ppi);
+ for(int i = 0; i < (o->numGlyphs+1); i++)
+ f->print(f->aux, "%*s%s[%d]: %ud\n", indent, "", "glyphDataOffsets", i, v->glyphDataOffsets[i]);
+ for(int i = 0; i < o->numGlyphs; i++){
+ f->print(f->aux, "%*s%s[%d]:\n", indent, "", "glyphData", i);
+ print_GlyphData(f, indent+indentΔ, o, &v->glyphData[i]);
+ }
+ USED(o);
+}
+
+int
+read_TableSbix(Otf *o, TableSbix *v)
+{
+ u8int *b = nil; USED(b);
+ if((b = otfreadn(o, 8)) == nil)
+ goto err;
+ u16int version = b[0]<<8 | b[1];
+ if(version != 1){
+ werrstr("%s: invalid value: %d (%#ux)", "version", version, version);
+ goto err;
+ }
+ v->flags = b[2]<<8 | b[3];
+ v->numStrikes = b[4]<<24 | b[5]<<16 | b[6]<<8 | b[7];
+ if((b = otfreadn(o, v->numStrikes*4)) == nil)
+ goto err;
+ v->strikeOffsets = malloc(v->numStrikes*sizeof(*v->strikeOffsets));
+ for(int i = 0; i < v->numStrikes; i++)
+ v->strikeOffsets[i] = b[0+i*4]<<24 | b[1+i*4]<<16 | b[2+i*4]<<8 | b[3+i*4];
+ if(v->numStrikes > 0){
+ v->strikes = calloc(v->numStrikes, sizeof(*v->strikes));
+ for(int i = 0; i < v->numStrikes; i++){
+ if(v->strikeOffsets[i] == 0)
+ continue;
+ if(otfpushrange(o, v->strikeOffsets[i], -1))
+ goto err;
+ int r = read_SbixStrike(o, v->strikes+i);
+ if(otfpoprange(o) < 0)
+ goto err;
+ if(r < 0){
+ memset(v->strikes+i, 0, sizeof(*v->strikes));
+ break;
+ }
+ }
+ }
+ return 0;
+err:
+ werrstr("%s: %r", "TableSbix");
+ return -1;
+}
+
+void
+print_TableSbix(Otfile *f, int indent, Otf *o, TableSbix *v)
+{
+ f->print(f->aux, "%*s%s: %#ux%s\n", indent, "", "flags", v->flags, (v->flags&SBIX_FL_DRAW_OUTLINES)?" SBIX_FL_DRAW_OUTLINES":"");
+ f->print(f->aux, "%*s%s: %ud\n", indent, "", "numStrikes", v->numStrikes);
+ for(int i = 0; i < v->numStrikes; i++)
+ f->print(f->aux, "%*s%s[%d]: %ud\n", indent, "", "strikeOffsets", i, v->strikeOffsets[i]);
+ for(int i = 0; i < v->numStrikes; i++){
+ f->print(f->aux, "%*s%s[%d]:\n", indent, "", "strikes", i);
+ print_SbixStrike(f, indent+indentΔ, o, &v->strikes[i]);
+ }
+ USED(o);
+}
+
+int
read_VariationAxisRecord(Otf *o, VariationAxisRecord *v)
{
u8int *b = nil; USED(b);
@@ -5239,6 +5446,22 @@
v->itemVariationDataOffsets = malloc(v->itemVariationDataCount*sizeof(*v->itemVariationDataOffsets));
for(int i = 0; i < v->itemVariationDataCount; i++)
v->itemVariationDataOffsets[i] = b[0+i*4]<<24 | b[1+i*4]<<16 | b[2+i*4]<<8 | b[3+i*4];
+ if(v->itemVariationDataCount > 0){
+ v->itemVariationData = calloc(v->itemVariationDataCount, sizeof(*v->itemVariationData));
+ for(int i = 0; i < v->itemVariationDataCount; i++){
+ if(v->itemVariationDataOffsets[i] == 0)
+ continue;
+ if(otfpushrange(o, v->itemVariationDataOffsets[i], -1))
+ goto err;
+ int r = read_ItemVariationData(o, v->itemVariationData+i);
+ if(otfpoprange(o) < 0)
+ goto err;
+ if(r < 0){
+ memset(v->itemVariationData+i, 0, sizeof(*v->itemVariationData));
+ break;
+ }
+ }
+ }
return 0;
err:
werrstr("%s: %r", "ItemVariationStore");
@@ -5252,6 +5475,10 @@
f->print(f->aux, "%*s%s: %ud\n", indent, "", "itemVariationDataCount", v->itemVariationDataCount);
for(int i = 0; i < v->itemVariationDataCount; i++)
f->print(f->aux, "%*s%s[%d]: %ud\n", indent, "", "itemVariationDataOffsets", i, v->itemVariationDataOffsets[i]);
+ for(int i = 0; i < v->itemVariationDataCount; i++){
+ f->print(f->aux, "%*s%s[%d]:\n", indent, "", "itemVariationData", i);
+ print_ItemVariationData(f, indent+indentΔ, o, &v->itemVariationData[i]);
+ }
USED(o);
}
@@ -5556,6 +5783,204 @@
}
int
+read_AxisValueTable1(Otf *o, AxisValueTable1 *v)
+{
+ u8int *b = nil; USED(b);
+ if((b = otfreadn(o, 10)) == nil)
+ goto err;
+ v->axisIndex = b[0]<<8 | b[1];
+ v->flags = b[2]<<8 | b[3];
+ v->valueNameID = b[4]<<8 | b[5];
+ v->value = (b[6]<<24 | b[7]<<16 | b[8]<<8 | b[9])/65536.0f;
+ return 0;
+err:
+ werrstr("%s: %r", "AxisValueTable1");
+ return -1;
+}
+
+void
+print_AxisValueTable1(Otfile *f, int indent, Otf *o, AxisValueTable1 *v)
+{
+ f->print(f->aux, "%*s%s: %ud\n", indent, "", "axisIndex", v->axisIndex);
+ f->print(f->aux, "%*s%s: %ud\n", indent, "", "flags", v->flags);
+ f->print(f->aux, "%*s%s: %ud\n", indent, "", "valueNameID", v->valueNameID);
+ f->print(f->aux, "%*s%s: %g\n", indent, "", "value", v->value);
+ USED(o);
+}
+
+int
+read_AxisValueTable2(Otf *o, AxisValueTable2 *v)
+{
+ u8int *b = nil; USED(b);
+ if((b = otfreadn(o, 18)) == nil)
+ goto err;
+ v->axisIndex = b[0]<<8 | b[1];
+ v->flags = b[2]<<8 | b[3];
+ v->valueNameID = b[4]<<8 | b[5];
+ v->nominalValue = (b[6]<<24 | b[7]<<16 | b[8]<<8 | b[9])/65536.0f;
+ v->rangeMinValue = (b[10]<<24 | b[11]<<16 | b[12]<<8 | b[13])/65536.0f;
+ v->rangeMaxValue = (b[14]<<24 | b[15]<<16 | b[16]<<8 | b[17])/65536.0f;
+ return 0;
+err:
+ werrstr("%s: %r", "AxisValueTable2");
+ return -1;
+}
+
+void
+print_AxisValueTable2(Otfile *f, int indent, Otf *o, AxisValueTable2 *v)
+{
+ f->print(f->aux, "%*s%s: %ud\n", indent, "", "axisIndex", v->axisIndex);
+ f->print(f->aux, "%*s%s: %ud\n", indent, "", "flags", v->flags);
+ f->print(f->aux, "%*s%s: %ud\n", indent, "", "valueNameID", v->valueNameID);
+ f->print(f->aux, "%*s%s: %g\n", indent, "", "nominalValue", v->nominalValue);
+ f->print(f->aux, "%*s%s: %g\n", indent, "", "rangeMinValue", v->rangeMinValue);
+ f->print(f->aux, "%*s%s: %g\n", indent, "", "rangeMaxValue", v->rangeMaxValue);
+ USED(o);
+}
+
+int
+read_AxisValueTable3(Otf *o, AxisValueTable3 *v)
+{
+ u8int *b = nil; USED(b);
+ if((b = otfreadn(o, 14)) == nil)
+ goto err;
+ v->axisIndex = b[0]<<8 | b[1];
+ v->flags = b[2]<<8 | b[3];
+ v->valueNameID = b[4]<<8 | b[5];
+ v->value = (b[6]<<24 | b[7]<<16 | b[8]<<8 | b[9])/65536.0f;
+ v->linkedValue = (b[10]<<24 | b[11]<<16 | b[12]<<8 | b[13])/65536.0f;
+ return 0;
+err:
+ werrstr("%s: %r", "AxisValueTable3");
+ return -1;
+}
+
+void
+print_AxisValueTable3(Otfile *f, int indent, Otf *o, AxisValueTable3 *v)
+{
+ f->print(f->aux, "%*s%s: %ud\n", indent, "", "axisIndex", v->axisIndex);
+ f->print(f->aux, "%*s%s: %ud\n", indent, "", "flags", v->flags);
+ f->print(f->aux, "%*s%s: %ud\n", indent, "", "valueNameID", v->valueNameID);
+ f->print(f->aux, "%*s%s: %g\n", indent, "", "value", v->value);
+ f->print(f->aux, "%*s%s: %g\n", indent, "", "linkedValue", v->linkedValue);
+ USED(o);
+}
+
+int
+read_AxisValue(Otf *o, AxisValue *v)
+{
+ u8int *b = nil; USED(b);
+ if((b = otfreadn(o, 6)) == nil)
+ goto err;
+ v->axisIndex = b[0]<<8 | b[1];
+ v->value = (b[2]<<24 | b[3]<<16 | b[4]<<8 | b[5])/65536.0f;
+ return 0;
+err:
+ werrstr("%s: %r", "AxisValue");
+ return -1;
+}
+
+void
+print_AxisValue(Otfile *f, int indent, Otf *o, AxisValue *v)
+{
+ f->print(f->aux, "%*s%s: %ud\n", indent, "", "axisIndex", v->axisIndex);
+ f->print(f->aux, "%*s%s: %g\n", indent, "", "value", v->value);
+ USED(o);
+}
+
+int
+read_AxisValueTable4(Otf *o, AxisValueTable4 *v)
+{
+ u8int *b = nil; USED(b);
+ if((b = otfreadn(o, 6)) == nil)
+ goto err;
+ v->axisCount = b[0]<<8 | b[1];
+ v->flags = b[2]<<8 | b[3];
+ v->valueNameID = b[4]<<8 | b[5];
+ if(otfarray(o, &v->axisValues, read_AxisValue, sizeof(AxisValue), v->axisCount) < 0){
+ werrstr("%s: %r", "axisValues");
+ goto err;
+ }
+ return 0;
+err:
+ werrstr("%s: %r", "AxisValueTable4");
+ return -1;
+}
+
+void
+print_AxisValueTable4(Otfile *f, int indent, Otf *o, AxisValueTable4 *v)
+{
+ f->print(f->aux, "%*s%s: %ud\n", indent, "", "axisCount", v->axisCount);
+ f->print(f->aux, "%*s%s: %ud\n", indent, "", "flags", v->flags);
+ f->print(f->aux, "%*s%s: %ud\n", indent, "", "valueNameID", v->valueNameID);
+ for(int i = 0; i < v->axisCount; i++){
+ f->print(f->aux, "%*s%s[%d]:\n", indent, "", "axisValues", i);
+ print_AxisValue(f, indent+indentΔ, o, &v->axisValues[i]);
+ }
+ USED(o);
+}
+
+int
+read_AxisValueTable(Otf *o, AxisValueTable *v)
+{
+ u8int *b = nil; USED(b);
+ if((b = otfreadn(o, 2)) == nil)
+ goto err;
+ v->format = b[0]<<8 | b[1];
+ if(v->format == 1){
+ if(read_AxisValueTable1(o, &v->f1) < 0){
+ werrstr("%s: %r", "f1");
+ goto err;
+ }
+ }
+ if(v->format == 2){
+ if(read_AxisValueTable2(o, &v->f2) < 0){
+ werrstr("%s: %r", "f2");
+ goto err;
+ }
+ }
+ if(v->format == 3){
+ if(read_AxisValueTable3(o, &v->f3) < 0){
+ werrstr("%s: %r", "f3");
+ goto err;
+ }
+ }
+ if(v->format == 4){
+ if(read_AxisValueTable3(o, &v->f4) < 0){
+ werrstr("%s: %r", "f4");
+ goto err;
+ }
+ }
+ return 0;
+err:
+ werrstr("%s: %r", "AxisValueTable");
+ return -1;
+}
+
+void
+print_AxisValueTable(Otfile *f, int indent, Otf *o, AxisValueTable *v)
+{
+ f->print(f->aux, "%*s%s: %ud\n", indent, "", "format", v->format);
+ if(v->format == 1){
+ f->print(f->aux, "%*s%s:\n", indent, "", "f1");
+ print_AxisValueTable1(f, indent+indentΔ, o, &v->f1);
+ }
+ if(v->format == 2){
+ f->print(f->aux, "%*s%s:\n", indent, "", "f2");
+ print_AxisValueTable2(f, indent+indentΔ, o, &v->f2);
+ }
+ if(v->format == 3){
+ f->print(f->aux, "%*s%s:\n", indent, "", "f3");
+ print_AxisValueTable3(f, indent+indentΔ, o, &v->f3);
+ }
+ if(v->format == 4){
+ f->print(f->aux, "%*s%s:\n", indent, "", "f4");
+ print_AxisValueTable3(f, indent+indentΔ, o, &v->f4);
+ }
+ USED(o);
+}
+
+int
read_DesignAxes(Otf *o, DesignAxes *v)
{
u8int *b = nil; USED(b);
@@ -5565,9 +5990,25 @@
}
if((b = otfreadn(o, o->axisValueCount*2)) == nil)
goto err;
- v->axisValueOffsets = malloc(o->axisValueCount*sizeof(*v->axisValueOffsets));
+ v->axisValueTableOffsets = malloc(o->axisValueCount*sizeof(*v->axisValueTableOffsets));
for(int i = 0; i < o->axisValueCount; i++)
- v->axisValueOffsets[i] = b[0+i*2]<<8 | b[1+i*2];
+ v->axisValueTableOffsets[i] = b[0+i*2]<<8 | b[1+i*2];
+ if(o->axisValueCount > 0){
+ v->axisValueTables = calloc(o->axisValueCount, sizeof(*v->axisValueTables));
+ for(int i = 0; i < o->axisValueCount; i++){
+ if(v->axisValueTableOffsets[i] == 0)
+ continue;
+ if(otfpushrange(o, v->axisValueTableOffsets[i], -1))
+ goto err;
+ int r = read_AxisValueTable(o, v->axisValueTables+i);
+ if(otfpoprange(o) < 0)
+ goto err;
+ if(r < 0){
+ memset(v->axisValueTables+i, 0, sizeof(*v->axisValueTables));
+ break;
+ }
+ }
+ }
return 0;
err:
werrstr("%s: %r", "DesignAxes");
@@ -5582,7 +6023,11 @@
print_AxisRecord(f, indent+indentΔ, o, &v->designAxes[i]);
}
for(int i = 0; i < o->axisValueCount; i++)
- f->print(f->aux, "%*s%s[%d]: %ud\n", indent, "", "axisValueOffsets", i, v->axisValueOffsets[i]);
+ f->print(f->aux, "%*s%s[%d]: %ud\n", indent, "", "axisValueTableOffsets", i, v->axisValueTableOffsets[i]);
+ for(int i = 0; i < o->axisValueCount; i++){
+ f->print(f->aux, "%*s%s[%d]:\n", indent, "", "axisValueTables", i);
+ print_AxisValueTable(f, indent+indentΔ, o, &v->axisValueTables[i]);
+ }
USED(o);
}
@@ -5670,7 +6115,7 @@
print_GaspRange(Otfile *f, int indent, Otf *o, GaspRange *v)
{
f->print(f->aux, "%*s%s: %ud\n", indent, "", "rangeMaxPPEM", v->rangeMaxPPEM);
- f->print(f->aux, "%*s%s: %#ux\n", indent, "", "rangeGaspBehavior", v->rangeGaspBehavior);
+ f->print(f->aux, "%*s%s: %#ux%s%s%s%s\n", indent, "", "rangeGaspBehavior", v->rangeGaspBehavior, (v->rangeGaspBehavior&GASP_GRIDFIT)?" GASP_GRIDFIT":"", (v->rangeGaspBehavior&GASP_DOGRAY)?" GASP_DOGRAY":"", (v->rangeGaspBehavior&GASP_SYMMETRIC_GRIDFIT)?" GASP_SYMMETRIC_GRIDFIT":"", (v->rangeGaspBehavior&GASP_SYMMETRIC_SMOOTHING)?" GASP_SYMMETRIC_SMOOTHING":"");
USED(o);
}
@@ -5739,17 +6184,23 @@
goto err;
v->variationDataSize = b[0]<<8 | b[1];
v->tupleIndex = b[2]<<8 | b[3];
- if(read_Tuple(o, &v->peakTuple) < 0){
- werrstr("%s: %r", "peakTuple");
- goto err;
+ if((v->tupleIndex&TUPLEINDEX_FL_EMBEDDED_PEAK_TUPLE) != 0){
+ if(read_Tuple(o, &v->peakTuple) < 0){
+ werrstr("%s: %r", "peakTuple");
+ goto err;
+ }
}
- if(read_Tuple(o, &v->intermediateStartTuple) < 0){
- werrstr("%s: %r", "intermediateStartTuple");
- goto err;
+ if((v->tupleIndex&TUPLEINDEX_FL_INTERMEDIATE_REGION) != 0){
+ if(read_Tuple(o, &v->intermediateStartTuple) < 0){
+ werrstr("%s: %r", "intermediateStartTuple");
+ goto err;
+ }
}
- if(read_Tuple(o, &v->intermediateEndTuple) < 0){
- werrstr("%s: %r", "intermediateEndTuple");
- goto err;
+ if((v->tupleIndex&TUPLEINDEX_FL_INTERMEDIATE_REGION) != 0){
+ if(read_Tuple(o, &v->intermediateEndTuple) < 0){
+ werrstr("%s: %r", "intermediateEndTuple");
+ goto err;
+ }
}
return 0;
err:
@@ -5762,12 +6213,18 @@
{
f->print(f->aux, "%*s%s: %ud\n", indent, "", "variationDataSize", v->variationDataSize);
f->print(f->aux, "%*s%s: %ud%s%s%s\n", indent, "", "tupleIndex", v->tupleIndex, (v->tupleIndex&TUPLEINDEX_FL_PRIVATE_POINT_NUMBERS)?" TUPLEINDEX_FL_PRIVATE_POINT_NUMBERS":"", (v->tupleIndex&TUPLEINDEX_FL_INTERMEDIATE_REGION)?" TUPLEINDEX_FL_INTERMEDIATE_REGION":"", (v->tupleIndex&TUPLEINDEX_FL_EMBEDDED_PEAK_TUPLE)?" TUPLEINDEX_FL_EMBEDDED_PEAK_TUPLE":"");
- f->print(f->aux, "%*s%s:\n", indent, "", "peakTuple");
- print_Tuple(f, indent+indentΔ, o, &v->peakTuple);
- f->print(f->aux, "%*s%s:\n", indent, "", "intermediateStartTuple");
- print_Tuple(f, indent+indentΔ, o, &v->intermediateStartTuple);
- f->print(f->aux, "%*s%s:\n", indent, "", "intermediateEndTuple");
- print_Tuple(f, indent+indentΔ, o, &v->intermediateEndTuple);
+ if((v->tupleIndex&TUPLEINDEX_FL_EMBEDDED_PEAK_TUPLE) != 0){
+ f->print(f->aux, "%*s%s:\n", indent, "", "peakTuple");
+ print_Tuple(f, indent+indentΔ, o, &v->peakTuple);
+ }
+ if((v->tupleIndex&TUPLEINDEX_FL_INTERMEDIATE_REGION) != 0){
+ f->print(f->aux, "%*s%s:\n", indent, "", "intermediateStartTuple");
+ print_Tuple(f, indent+indentΔ, o, &v->intermediateStartTuple);
+ }
+ if((v->tupleIndex&TUPLEINDEX_FL_INTERMEDIATE_REGION) != 0){
+ f->print(f->aux, "%*s%s:\n", indent, "", "intermediateEndTuple");
+ print_Tuple(f, indent+indentΔ, o, &v->intermediateEndTuple);
+ }
USED(o);
}
@@ -5818,6 +6275,7 @@
goto err;
}
v->axisCount = b[4]<<8 | b[5];
+ o->axisCount = v->axisCount;
v->sharedTupleCount = b[6]<<8 | b[7];
v->sharedTuplesOffset = b[8]<<24 | b[9]<<16 | b[10]<<8 | b[11];
v->glyphCount = b[12]<<8 | b[13];
@@ -5837,6 +6295,38 @@
for(int i = 0; i < (v->glyphCount+1); i++)
v->glyphVariationDataOffsetsLong[i] = b[0+i*4]<<24 | b[1+i*4]<<16 | b[2+i*4]<<8 | b[3+i*4];
}
+ if((v->flags&GVAR_FL_LONG_OFFSETS) == 0){
+ if(v->glyphVariationDataArrayOffset != 0 && v->glyphCount > 0){
+ v->glyphVariationData = calloc(v->glyphCount, sizeof(*v->glyphVariationData));
+ for(int i = 0; i < v->glyphCount; i++){
+ if(otfpushrange(o, v->glyphVariationDataArrayOffset+v->glyphVariationDataOffsetsShort[i], -1))
+ goto err;
+ int r = read_GlyphVariationData(o, v->glyphVariationData+i);
+ if(otfpoprange(o) < 0)
+ goto err;
+ if(r < 0){
+ memset(v->glyphVariationData+i, 0, sizeof(*v->glyphVariationData));
+ break;
+ }
+ }
+ }
+ }
+ if((v->flags&GVAR_FL_LONG_OFFSETS) != 0){
+ if(v->glyphVariationDataArrayOffset != 0 && v->glyphCount > 0){
+ v->glyphVariationData = calloc(v->glyphCount, sizeof(*v->glyphVariationData));
+ for(int i = 0; i < v->glyphCount; i++){
+ if(otfpushrange(o, v->glyphVariationDataArrayOffset+v->glyphVariationDataOffsetsLong[i], -1))
+ goto err;
+ int r = read_GlyphVariationData(o, v->glyphVariationData+i);
+ if(otfpoprange(o) < 0)
+ goto err;
+ if(r < 0){
+ memset(v->glyphVariationData+i, 0, sizeof(*v->glyphVariationData));
+ break;
+ }
+ }
+ }
+ }
if(v->sharedTuplesOffset != 0){
if(otfpushrange(o, v->sharedTuplesOffset, -1) < 0)
goto err;
@@ -5857,6 +6347,7 @@
print_TableGvar(Otfile *f, int indent, Otf *o, TableGvar *v)
{
f->print(f->aux, "%*s%s: %ud\n", indent, "", "axisCount", v->axisCount);
+ o->axisCount = v->axisCount;
f->print(f->aux, "%*s%s: %ud\n", indent, "", "sharedTupleCount", v->sharedTupleCount);
f->print(f->aux, "%*s%s: %ud\n", indent, "", "sharedTuplesOffset", v->sharedTuplesOffset);
f->print(f->aux, "%*s%s: %ud\n", indent, "", "glyphCount", v->glyphCount);
@@ -5870,6 +6361,18 @@
for(int i = 0; i < (v->glyphCount+1); i++)
f->print(f->aux, "%*s%s[%d]: %ud\n", indent, "", "glyphVariationDataOffsetsLong", i, v->glyphVariationDataOffsetsLong[i]);
}
+ if((v->flags&GVAR_FL_LONG_OFFSETS) == 0){
+ for(int i = 0; i < v->glyphCount; i++){
+ f->print(f->aux, "%*s%s[%d]:\n", indent, "", "glyphVariationData", i);
+ print_GlyphVariationData(f, indent+indentΔ, o, &v->glyphVariationData[i]);
+ }
+ }
+ if((v->flags&GVAR_FL_LONG_OFFSETS) != 0){
+ for(int i = 0; i < v->glyphCount; i++){
+ f->print(f->aux, "%*s%s[%d]:\n", indent, "", "glyphVariationData", i);
+ print_GlyphVariationData(f, indent+indentΔ, o, &v->glyphVariationData[i]);
+ }
+ }
for(int i = 0; i < v->sharedTupleCount; i++){
f->print(f->aux, "%*s%s[%d]:\n", indent, "", "sharedTuples", i);
print_Tuple(f, indent+indentΔ, o, &v->sharedTuples[i]);
@@ -6349,6 +6852,18 @@
}
rec->parsed = v->loca;
rec->print = (void*)print_TableLoca;
+ break;
+ case (u32int)('s'<<24|'b'<<16|'i'<<8|'x'):
+ if(v->sbix != nil)
+ break;
+ v->sbix = calloc(1, sizeof(TableSbix));
+ if(read_TableSbix(o, v->sbix) < 0){
+ free(v->sbix);
+ v->sbix = nil;
+ goto err;
+ }
+ rec->parsed = v->sbix;
+ rec->print = (void*)print_TableSbix;
break;
case (u32int)('f'<<24|'v'<<16|'a'<<8|'r'):
if(v->fvar != nil)
--- a/plan9/otf.h
+++ b/plan9/otf.h
@@ -209,12 +209,12 @@
typedef struct RangeRecord RangeRecord;
typedef struct Coverage2 Coverage2;
typedef struct Coverage Coverage;
-typedef struct MathVariants MathVariants;
-typedef struct MathGlyphInfo MathGlyphInfo;
typedef struct MathGlyphVariantRecord MathGlyphVariantRecord;
typedef struct GlyphPart GlyphPart;
typedef struct GlyphAssembly GlyphAssembly;
typedef struct MathGlyphConstruction MathGlyphConstruction;
+typedef struct MathVariants MathVariants;
+typedef struct MathGlyphInfo MathGlyphInfo;
typedef struct TableMATH TableMATH;
typedef struct KernPair KernPair;
typedef struct KernSubtable0 KernSubtable0;
@@ -223,6 +223,9 @@
typedef struct KernSubtable KernSubtable;
typedef struct TableKern TableKern;
typedef struct TableLoca TableLoca;
+typedef struct GlyphData GlyphData;
+typedef struct SbixStrike SbixStrike;
+typedef struct TableSbix TableSbix;
typedef struct VariationAxisRecord VariationAxisRecord;
typedef struct UserTuple UserTuple;
typedef struct InstanceRecord InstanceRecord;
@@ -240,6 +243,12 @@
typedef struct SignatureRecord SignatureRecord;
typedef struct TableDSIG TableDSIG;
typedef struct AxisRecord AxisRecord;
+typedef struct AxisValueTable1 AxisValueTable1;
+typedef struct AxisValueTable2 AxisValueTable2;
+typedef struct AxisValueTable3 AxisValueTable3;
+typedef struct AxisValue AxisValue;
+typedef struct AxisValueTable4 AxisValueTable4;
+typedef struct AxisValueTable AxisValueTable;
typedef struct DesignAxes DesignAxes;
typedef struct TableSTAT TableSTAT;
typedef struct GaspRange GaspRange;
@@ -909,6 +918,15 @@
int read_FeatureList(Otf *o, FeatureList *v);
void print_FeatureList(Otfile *f, int indent, Otf *o, FeatureList *v);
+enum { // Lookup
+ // lookupFlag
+ LOOKUP_FL_RIGHT_TO_LEFT = 1<<0,
+ LOOKUP_FL_IGNORE_BASE_GLYPHS = 1<<1,
+ LOOKUP_FL_IGNORE_LIGATURES = 1<<2,
+ LOOKUP_FL_IGNORE_MARKS = 1<<3,
+ LOOKUP_FL_USE_MARK_FILTERING_SET = 1<<4,
+};
+
struct Lookup {
u16int lookupType;
u16int lookupFlag;
@@ -923,6 +941,7 @@
struct LookupList {
u16int lookupCount;
u16int *lookupOffsets;
+ Lookup *lookups;
};
int read_LookupList(Otf *o, LookupList *v);
@@ -1107,35 +1126,6 @@
int read_Coverage(Otf *o, Coverage *v);
void print_Coverage(Otfile *f, int indent, Otf *o, Coverage *v);
-struct MathVariants {
- u16int minConnectorOverlap;
- u16int vertGlyphCoverageOffset;
- u16int horizGlyphCoverageOffset;
- u16int vertGlyphCount;
- u16int horizGlyphCount;
- u16int *vertGlyphConstructionOffsets;
- u16int *horizGlyphConstructionOffsets;
- Coverage *vertGlyphCoverage;
- Coverage *horizGlyphCoverage;
-};
-
-int read_MathVariants(Otf *o, MathVariants *v);
-void print_MathVariants(Otfile *f, int indent, Otf *o, MathVariants *v);
-
-struct MathGlyphInfo {
- u16int mathItalicsCorrectionInfoOffset;
- u16int mathTopAccentAttachmentOffset;
- u16int extendedShapeCoverageOffset;
- u16int mathKernInfoOffset;
- MathItalicsCorrectionInfo *mathItalicsCorrectionInfo;
- MathTopAccentAttachment *mathTopAccentAttachment;
- MathKernInfo *mathKernInfo;
- Coverage *extendedShapeCoverage;
-};
-
-int read_MathGlyphInfo(Otf *o, MathGlyphInfo *v);
-void print_MathGlyphInfo(Otfile *f, int indent, Otf *o, MathGlyphInfo *v);
-
struct MathGlyphVariantRecord {
u16int variantGlyph;
u16int advanceMeasurement;
@@ -1174,6 +1164,37 @@
int read_MathGlyphConstruction(Otf *o, MathGlyphConstruction *v);
void print_MathGlyphConstruction(Otfile *f, int indent, Otf *o, MathGlyphConstruction *v);
+struct MathVariants {
+ u16int minConnectorOverlap;
+ u16int vertGlyphCoverageOffset;
+ u16int horizGlyphCoverageOffset;
+ u16int vertGlyphCount;
+ u16int horizGlyphCount;
+ u16int *vertGlyphConstructionOffsets;
+ u16int *horizGlyphConstructionOffsets;
+ Coverage *vertGlyphCoverage;
+ Coverage *horizGlyphCoverage;
+ MathGlyphConstruction *vertGlyphConstruction;
+ MathGlyphConstruction *horizGlyphConstruction;
+};
+
+int read_MathVariants(Otf *o, MathVariants *v);
+void print_MathVariants(Otfile *f, int indent, Otf *o, MathVariants *v);
+
+struct MathGlyphInfo {
+ u16int mathItalicsCorrectionInfoOffset;
+ u16int mathTopAccentAttachmentOffset;
+ u16int extendedShapeCoverageOffset;
+ u16int mathKernInfoOffset;
+ MathItalicsCorrectionInfo *mathItalicsCorrectionInfo;
+ MathTopAccentAttachment *mathTopAccentAttachment;
+ MathKernInfo *mathKernInfo;
+ Coverage *extendedShapeCoverage;
+};
+
+int read_MathGlyphInfo(Otf *o, MathGlyphInfo *v);
+void print_MathGlyphInfo(Otfile *f, int indent, Otf *o, MathGlyphInfo *v);
+
struct TableMATH {
// u16int majorVersion;
// u16int minorVersion;
@@ -1230,10 +1251,20 @@
int read_KernSubtable2(Otf *o, KernSubtable2 *v);
void print_KernSubtable2(Otfile *f, int indent, Otf *o, KernSubtable2 *v);
+enum { // KernSubtable
+ // coverage
+ KERN_FL_HORIZONTAL = 1<<0,
+ KERN_FL_MINIMUM = 1<<1,
+ KERN_FL_CROSS_STREAM = 1<<2,
+ KERN_FL_OVERRIDE = 1<<3,
+};
+
struct KernSubtable {
// u16int version;
u16int length;
u16int coverage;
+ KernSubtable0 sub0;
+ KernSubtable2 sub2;
};
int read_KernSubtable(Otf *o, KernSubtable *v);
@@ -1256,6 +1287,41 @@
int read_TableLoca(Otf *o, TableLoca *v);
void print_TableLoca(Otfile *f, int indent, Otf *o, TableLoca *v);
+struct GlyphData {
+ s16int originOffsetX;
+ s16int originOffsetY;
+ u32int graphicType;
+};
+
+int read_GlyphData(Otf *o, GlyphData *v);
+void print_GlyphData(Otfile *f, int indent, Otf *o, GlyphData *v);
+
+struct SbixStrike {
+ u16int ppem;
+ u16int ppi;
+ u32int *glyphDataOffsets;
+ GlyphData *glyphData;
+};
+
+int read_SbixStrike(Otf *o, SbixStrike *v);
+void print_SbixStrike(Otfile *f, int indent, Otf *o, SbixStrike *v);
+
+enum { // TableSbix
+ // flags
+ SBIX_FL_DRAW_OUTLINES = 1<<1,
+};
+
+struct TableSbix {
+ // u16int version;
+ u16int flags;
+ u32int numStrikes;
+ u32int *strikeOffsets;
+ SbixStrike *strikes;
+};
+
+int read_TableSbix(Otf *o, TableSbix *v);
+void print_TableSbix(Otfile *f, int indent, Otf *o, TableSbix *v);
+
enum { // VariationAxisRecord
// flags
VARIATIONAXISRECORD_FL_HIDDEN_AXIS = 1<<0,
@@ -1353,6 +1419,7 @@
u32int variationRegionListOffset;
u16int itemVariationDataCount;
u32int *itemVariationDataOffsets;
+ ItemVariationData *itemVariationData;
};
int read_ItemVariationStore(Otf *o, ItemVariationStore *v);
@@ -1440,9 +1507,72 @@
int read_AxisRecord(Otf *o, AxisRecord *v);
void print_AxisRecord(Otfile *f, int indent, Otf *o, AxisRecord *v);
+struct AxisValueTable1 {
+ u16int axisIndex;
+ u16int flags;
+ u16int valueNameID;
+ double value;
+};
+
+int read_AxisValueTable1(Otf *o, AxisValueTable1 *v);
+void print_AxisValueTable1(Otfile *f, int indent, Otf *o, AxisValueTable1 *v);
+
+struct AxisValueTable2 {
+ u16int axisIndex;
+ u16int flags;
+ u16int valueNameID;
+ double nominalValue;
+ double rangeMinValue;
+ double rangeMaxValue;
+};
+
+int read_AxisValueTable2(Otf *o, AxisValueTable2 *v);
+void print_AxisValueTable2(Otfile *f, int indent, Otf *o, AxisValueTable2 *v);
+
+struct AxisValueTable3 {
+ u16int axisIndex;
+ u16int flags;
+ u16int valueNameID;
+ double value;
+ double linkedValue;
+};
+
+int read_AxisValueTable3(Otf *o, AxisValueTable3 *v);
+void print_AxisValueTable3(Otfile *f, int indent, Otf *o, AxisValueTable3 *v);
+
+struct AxisValue {
+ u16int axisIndex;
+ double value;
+};
+
+int read_AxisValue(Otf *o, AxisValue *v);
+void print_AxisValue(Otfile *f, int indent, Otf *o, AxisValue *v);
+
+struct AxisValueTable4 {
+ u16int axisCount;
+ u16int flags;
+ u16int valueNameID;
+ AxisValue *axisValues;
+};
+
+int read_AxisValueTable4(Otf *o, AxisValueTable4 *v);
+void print_AxisValueTable4(Otfile *f, int indent, Otf *o, AxisValueTable4 *v);
+
+struct AxisValueTable {
+ u16int format;
+ AxisValueTable1 f1;
+ AxisValueTable2 f2;
+ AxisValueTable3 f3;
+ AxisValueTable3 f4;
+};
+
+int read_AxisValueTable(Otf *o, AxisValueTable *v);
+void print_AxisValueTable(Otfile *f, int indent, Otf *o, AxisValueTable *v);
+
struct DesignAxes {
AxisRecord *designAxes;
- u16int *axisValueOffsets;
+ u16int *axisValueTableOffsets;
+ AxisValueTable *axisValueTables;
};
int read_DesignAxes(Otf *o, DesignAxes *v);
@@ -1463,6 +1593,14 @@
int read_TableSTAT(Otf *o, TableSTAT *v);
void print_TableSTAT(Otfile *f, int indent, Otf *o, TableSTAT *v);
+enum { // GaspRange
+ // rangeGaspBehavior
+ GASP_GRIDFIT = 1<<0,
+ GASP_DOGRAY = 1<<1,
+ GASP_SYMMETRIC_GRIDFIT = 1<<2,
+ GASP_SYMMETRIC_SMOOTHING = 1<<3,
+};
+
struct GaspRange {
u16int rangeMaxPPEM;
u16int rangeGaspBehavior;
@@ -1535,6 +1673,7 @@
u32int glyphVariationDataArrayOffset;
u16int *glyphVariationDataOffsetsShort;
u32int *glyphVariationDataOffsetsLong;
+ GlyphVariationData *glyphVariationData;
Tuple *sharedTuples;
};
@@ -1647,6 +1786,7 @@
TableMATH *math;
TableKern *kern;
TableLoca *loca;
+ TableSbix *sbix;
TableFvar *fvar;
TableHVAR *hvar;
TableFFTM *fftm;