1 /*
2  * Copyright © 2007,2008,2009  Red Hat, Inc.
3  * Copyright © 2012  Google, Inc.
4  *
5  *  This is part of HarfBuzz, a text shaping library.
6  *
7  * Permission is hereby granted, without written agreement and without
8  * license or royalty fees, to use, copy, modify, and distribute this
9  * software and its documentation for any purpose, provided that the
10  * above copyright notice and the following two paragraphs appear in
11  * all copies of this software.
12  *
13  * IN NO EVENT SHALL THE COPYRIGHT HOLDER BE LIABLE TO ANY PARTY FOR
14  * DIRECT, INDIRECT, SPECIAL, INCIDENTAL, OR CONSEQUENTIAL DAMAGES
15  * ARISING OUT OF THE USE OF THIS SOFTWARE AND ITS DOCUMENTATION, EVEN
16  * IF THE COPYRIGHT HOLDER HAS BEEN ADVISED OF THE POSSIBILITY OF SUCH
17  * DAMAGE.
18  *
19  * THE COPYRIGHT HOLDER SPECIFICALLY DISCLAIMS ANY WARRANTIES, INCLUDING,
20  * BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND
21  * FITNESS FOR A PARTICULAR PURPOSE.  THE SOFTWARE PROVIDED HEREUNDER IS
22  * ON AN "AS IS" BASIS, AND THE COPYRIGHT HOLDER HAS NO OBLIGATION TO
23  * PROVIDE MAINTENANCE, SUPPORT, UPDATES, ENHANCEMENTS, OR MODIFICATIONS.
24  *
25  * Red Hat Author(s): Behdad Esfahbod
26  * Google Author(s): Behdad Esfahbod
27  */
28 
29 #ifndef HB_OPEN_FILE_PRIVATE_HH
30 #define HB_OPEN_FILE_PRIVATE_HH
31 
32 #include "hb-open-type-private.hh"
33 
34 
35 namespace OT {
36 
37 
38 /*
39  *
40  * The OpenType Font File
41  *
42  */
43 
44 
45 /*
46  * Organization of an OpenType Font
47  */
48 
49 struct OpenTypeFontFile;
50 struct OffsetTable;
51 struct TTCHeader;
52 
53 
54 typedef struct TableRecord
55 {
sanitizeOT::TableRecord56   inline bool sanitize (hb_sanitize_context_t *c) {
57     TRACE_SANITIZE (this);
58     return TRACE_RETURN (c->check_struct (this));
59   }
60 
61   Tag		tag;		/* 4-byte identifier. */
62   CheckSum	checkSum;	/* CheckSum for this table. */
63   ULONG		offset;		/* Offset from beginning of TrueType font
64 				 * file. */
65   ULONG		length;		/* Length of this table. */
66   public:
67   DEFINE_SIZE_STATIC (16);
68 } OpenTypeTable;
69 
70 typedef struct OffsetTable
71 {
72   friend struct OpenTypeFontFile;
73 
get_table_countOT::OffsetTable74   inline unsigned int get_table_count (void) const
75   { return numTables; }
get_tableOT::OffsetTable76   inline const TableRecord& get_table (unsigned int i) const
77   {
78     if (unlikely (i >= numTables)) return Null(TableRecord);
79     return tables[i];
80   }
find_table_indexOT::OffsetTable81   inline bool find_table_index (hb_tag_t tag, unsigned int *table_index) const
82   {
83     Tag t;
84     t.set (tag);
85     unsigned int count = numTables;
86     for (unsigned int i = 0; i < count; i++)
87     {
88       if (t == tables[i].tag)
89       {
90         if (table_index) *table_index = i;
91         return true;
92       }
93     }
94     if (table_index) *table_index = Index::NOT_FOUND_INDEX;
95     return false;
96   }
get_table_by_tagOT::OffsetTable97   inline const TableRecord& get_table_by_tag (hb_tag_t tag) const
98   {
99     unsigned int table_index;
100     find_table_index (tag, &table_index);
101     return get_table (table_index);
102   }
103 
104   public:
sanitizeOT::OffsetTable105   inline bool sanitize (hb_sanitize_context_t *c) {
106     TRACE_SANITIZE (this);
107     return TRACE_RETURN (c->check_struct (this) && c->check_array (tables, TableRecord::static_size, numTables));
108   }
109 
110   protected:
111   Tag		sfnt_version;	/* '\0\001\0\00' if TrueType / 'OTTO' if CFF */
112   USHORT	numTables;	/* Number of tables. */
113   USHORT	searchRangeZ;	/* (Maximum power of 2 <= numTables) x 16 */
114   USHORT	entrySelectorZ;	/* Log2(maximum power of 2 <= numTables). */
115   USHORT	rangeShiftZ;	/* NumTables x 16-searchRange. */
116   TableRecord	tables[VAR];	/* TableRecord entries. numTables items */
117   public:
118   DEFINE_SIZE_ARRAY (12, tables);
119 } OpenTypeFontFace;
120 
121 
122 /*
123  * TrueType Collections
124  */
125 
126 struct TTCHeaderVersion1
127 {
128   friend struct TTCHeader;
129 
get_face_countOT::TTCHeaderVersion1130   inline unsigned int get_face_count (void) const { return table.len; }
get_faceOT::TTCHeaderVersion1131   inline const OpenTypeFontFace& get_face (unsigned int i) const { return this+table[i]; }
132 
sanitizeOT::TTCHeaderVersion1133   inline bool sanitize (hb_sanitize_context_t *c) {
134     TRACE_SANITIZE (this);
135     return TRACE_RETURN (table.sanitize (c, this));
136   }
137 
138   protected:
139   Tag		ttcTag;		/* TrueType Collection ID string: 'ttcf' */
140   FixedVersion	version;	/* Version of the TTC Header (1.0),
141 				 * 0x00010000u */
142   ArrayOf<OffsetTo<OffsetTable, ULONG>, ULONG>
143 		table;		/* Array of offsets to the OffsetTable for each font
144 				 * from the beginning of the file */
145   public:
146   DEFINE_SIZE_ARRAY (12, table);
147 };
148 
149 struct TTCHeader
150 {
151   friend struct OpenTypeFontFile;
152 
153   private:
154 
get_face_countOT::TTCHeader155   inline unsigned int get_face_count (void) const
156   {
157     switch (u.header.version.major) {
158     case 2: /* version 2 is compatible with version 1 */
159     case 1: return u.version1.get_face_count ();
160     default:return 0;
161     }
162   }
get_faceOT::TTCHeader163   inline const OpenTypeFontFace& get_face (unsigned int i) const
164   {
165     switch (u.header.version.major) {
166     case 2: /* version 2 is compatible with version 1 */
167     case 1: return u.version1.get_face (i);
168     default:return Null(OpenTypeFontFace);
169     }
170   }
171 
sanitizeOT::TTCHeader172   inline bool sanitize (hb_sanitize_context_t *c) {
173     TRACE_SANITIZE (this);
174     if (unlikely (!u.header.version.sanitize (c))) return TRACE_RETURN (false);
175     switch (u.header.version.major) {
176     case 2: /* version 2 is compatible with version 1 */
177     case 1: return TRACE_RETURN (u.version1.sanitize (c));
178     default:return TRACE_RETURN (true);
179     }
180   }
181 
182   protected:
183   union {
184   struct {
185   Tag		ttcTag;		/* TrueType Collection ID string: 'ttcf' */
186   FixedVersion	version;	/* Version of the TTC Header (1.0 or 2.0),
187 				 * 0x00010000u or 0x00020000u */
188   }			header;
189   TTCHeaderVersion1	version1;
190   } u;
191 };
192 
193 
194 /*
195  * OpenType Font File
196  */
197 
198 struct OpenTypeFontFile
199 {
200   static const hb_tag_t tableTag	= HB_TAG ('_','_','_','_'); /* Sanitizer needs this. */
201 
202   static const hb_tag_t CFFTag		= HB_TAG ('O','T','T','O'); /* OpenType with Postscript outlines */
203   static const hb_tag_t TrueTypeTag	= HB_TAG ( 0 , 1 , 0 , 0 ); /* OpenType with TrueType outlines */
204   static const hb_tag_t TTCTag		= HB_TAG ('t','t','c','f'); /* TrueType Collection */
205   static const hb_tag_t TrueTag		= HB_TAG ('t','r','u','e'); /* Obsolete Apple TrueType */
206   static const hb_tag_t Typ1Tag		= HB_TAG ('t','y','p','1'); /* Obsolete Apple Type1 font in SFNT container */
207 
get_tagOT::OpenTypeFontFile208   inline hb_tag_t get_tag (void) const { return u.tag; }
209 
get_face_countOT::OpenTypeFontFile210   inline unsigned int get_face_count (void) const
211   {
212     switch (u.tag) {
213     case CFFTag:	/* All the non-collection tags */
214     case TrueTag:
215     case Typ1Tag:
216     case TrueTypeTag:	return 1;
217     case TTCTag:	return u.ttcHeader.get_face_count ();
218     default:		return 0;
219     }
220   }
get_faceOT::OpenTypeFontFile221   inline const OpenTypeFontFace& get_face (unsigned int i) const
222   {
223     switch (u.tag) {
224     /* Note: for non-collection SFNT data we ignore index.  This is because
225      * Apple dfont container is a container of SFNT's.  So each SFNT is a
226      * non-TTC, but the index is more than zero. */
227     case CFFTag:	/* All the non-collection tags */
228     case TrueTag:
229     case Typ1Tag:
230     case TrueTypeTag:	return u.fontFace;
231     case TTCTag:	return u.ttcHeader.get_face (i);
232     default:		return Null(OpenTypeFontFace);
233     }
234   }
235 
sanitizeOT::OpenTypeFontFile236   inline bool sanitize (hb_sanitize_context_t *c) {
237     TRACE_SANITIZE (this);
238     if (unlikely (!u.tag.sanitize (c))) return TRACE_RETURN (false);
239     switch (u.tag) {
240     case CFFTag:	/* All the non-collection tags */
241     case TrueTag:
242     case Typ1Tag:
243     case TrueTypeTag:	return TRACE_RETURN (u.fontFace.sanitize (c));
244     case TTCTag:	return TRACE_RETURN (u.ttcHeader.sanitize (c));
245     default:		return TRACE_RETURN (true);
246     }
247   }
248 
249   protected:
250   union {
251   Tag			tag;		/* 4-byte identifier. */
252   OpenTypeFontFace	fontFace;
253   TTCHeader		ttcHeader;
254   } u;
255   public:
256   DEFINE_SIZE_UNION (4, tag);
257 };
258 
259 
260 } /* namespace OT */
261 
262 
263 #endif /* HB_OPEN_FILE_PRIVATE_HH */
264