1""" TSI{0,1,2,3,5} are private tables used by Microsoft Visual TrueType (VTT) 2tool to store its hinting source data. 3 4TSI1 contains the text of the glyph programs in the form of low-level assembly 5code, as well as the 'extra' programs 'fpgm', 'ppgm' (i.e. 'prep'), and 'cvt'. 6""" 7from fontTools.misc.py23 import strjoin, tobytes, tostr 8from . import DefaultTable 9from fontTools.misc.loggingTools import LogMixin 10 11 12class table_T_S_I__1(LogMixin, DefaultTable.DefaultTable): 13 14 extras = {0xfffa: "ppgm", 0xfffb: "cvt", 0xfffc: "reserved", 0xfffd: "fpgm"} 15 16 indextable = "TSI0" 17 18 def decompile(self, data, ttFont): 19 totalLength = len(data) 20 indextable = ttFont[self.indextable] 21 for indices, isExtra in zip( 22 (indextable.indices, indextable.extra_indices), (False, True)): 23 programs = {} 24 for i, (glyphID, textLength, textOffset) in enumerate(indices): 25 if isExtra: 26 name = self.extras[glyphID] 27 else: 28 name = ttFont.getGlyphName(glyphID) 29 if textOffset > totalLength: 30 self.log.warning("textOffset > totalLength; %r skipped" % name) 31 continue 32 if textLength < 0x8000: 33 # If the length stored in the record is less than 32768, then use 34 # that as the length of the record. 35 pass 36 elif textLength == 0x8000: 37 # If the length is 32768, compute the actual length as follows: 38 isLast = i == (len(indices)-1) 39 if isLast: 40 if isExtra: 41 # For the last "extra" record (the very last record of the 42 # table), the length is the difference between the total 43 # length of the TSI1 table and the textOffset of the final 44 # record. 45 nextTextOffset = totalLength 46 else: 47 # For the last "normal" record (the last record just prior 48 # to the record containing the "magic number"), the length 49 # is the difference between the textOffset of the record 50 # following the "magic number" (0xFFFE) record (i.e. the 51 # first "extra" record), and the textOffset of the last 52 # "normal" record. 53 nextTextOffset = indextable.extra_indices[0][2] 54 else: 55 # For all other records with a length of 0x8000, the length is 56 # the difference between the textOffset of the record in 57 # question and the textOffset of the next record. 58 nextTextOffset = indices[i+1][2] 59 assert nextTextOffset >= textOffset, "entries not sorted by offset" 60 if nextTextOffset > totalLength: 61 self.log.warning( 62 "nextTextOffset > totalLength; %r truncated" % name) 63 nextTextOffset = totalLength 64 textLength = nextTextOffset - textOffset 65 else: 66 from fontTools import ttLib 67 raise ttLib.TTLibError( 68 "%r textLength (%d) must not be > 32768" % (name, textLength)) 69 text = data[textOffset:textOffset+textLength] 70 assert len(text) == textLength 71 text = tostr(text, encoding='utf-8') 72 if text: 73 programs[name] = text 74 if isExtra: 75 self.extraPrograms = programs 76 else: 77 self.glyphPrograms = programs 78 79 def compile(self, ttFont): 80 if not hasattr(self, "glyphPrograms"): 81 self.glyphPrograms = {} 82 self.extraPrograms = {} 83 data = b'' 84 indextable = ttFont[self.indextable] 85 glyphNames = ttFont.getGlyphOrder() 86 87 indices = [] 88 for i in range(len(glyphNames)): 89 if len(data) % 2: 90 data = data + b"\015" # align on 2-byte boundaries, fill with return chars. Yum. 91 name = glyphNames[i] 92 if name in self.glyphPrograms: 93 text = tobytes(self.glyphPrograms[name], encoding="utf-8") 94 else: 95 text = b"" 96 textLength = len(text) 97 if textLength >= 0x8000: 98 textLength = 0x8000 99 indices.append((i, textLength, len(data))) 100 data = data + text 101 102 extra_indices = [] 103 codes = sorted(self.extras.items()) 104 for i in range(len(codes)): 105 if len(data) % 2: 106 data = data + b"\015" # align on 2-byte boundaries, fill with return chars. 107 code, name = codes[i] 108 if name in self.extraPrograms: 109 text = tobytes(self.extraPrograms[name], encoding="utf-8") 110 else: 111 text = b"" 112 textLength = len(text) 113 if textLength >= 0x8000: 114 textLength = 0x8000 115 extra_indices.append((code, textLength, len(data))) 116 data = data + text 117 indextable.set(indices, extra_indices) 118 return data 119 120 def toXML(self, writer, ttFont): 121 names = sorted(self.glyphPrograms.keys()) 122 writer.newline() 123 for name in names: 124 text = self.glyphPrograms[name] 125 if not text: 126 continue 127 writer.begintag("glyphProgram", name=name) 128 writer.newline() 129 writer.write_noindent(text.replace("\r", "\n")) 130 writer.newline() 131 writer.endtag("glyphProgram") 132 writer.newline() 133 writer.newline() 134 extra_names = sorted(self.extraPrograms.keys()) 135 for name in extra_names: 136 text = self.extraPrograms[name] 137 if not text: 138 continue 139 writer.begintag("extraProgram", name=name) 140 writer.newline() 141 writer.write_noindent(text.replace("\r", "\n")) 142 writer.newline() 143 writer.endtag("extraProgram") 144 writer.newline() 145 writer.newline() 146 147 def fromXML(self, name, attrs, content, ttFont): 148 if not hasattr(self, "glyphPrograms"): 149 self.glyphPrograms = {} 150 self.extraPrograms = {} 151 lines = strjoin(content).replace("\r", "\n").split("\n") 152 text = '\r'.join(lines[1:-1]) 153 if name == "glyphProgram": 154 self.glyphPrograms[attrs["name"]] = text 155 elif name == "extraProgram": 156 self.extraPrograms[attrs["name"]] = text 157