1 /****************************************************************************
2  *
3  * otvgdef.c
4  *
5  *   OpenType GDEF table validation (body).
6  *
7  * Copyright 2004-2018 by
8  * David Turner, Robert Wilhelm, and Werner Lemberg.
9  *
10  * This file is part of the FreeType project, and may only be used,
11  * modified, and distributed under the terms of the FreeType project
12  * license, LICENSE.TXT.  By continuing to use, modify, or distribute
13  * this file you indicate that you have read the license and
14  * understand and accept it fully.
15  *
16  */
17 
18 
19 #include "otvalid.h"
20 #include "otvcommn.h"
21 
22 
23   /**************************************************************************
24    *
25    * The macro FT_COMPONENT is used in trace mode.  It is an implicit
26    * parameter of the FT_TRACE() and FT_ERROR() macros, used to print/log
27    * messages during execution.
28    */
29 #undef  FT_COMPONENT
30 #define FT_COMPONENT  trace_otvgdef
31 
32 
33   /*************************************************************************/
34   /*************************************************************************/
35   /*****                                                               *****/
36   /*****                      UTILITY FUNCTIONS                        *****/
37   /*****                                                               *****/
38   /*************************************************************************/
39   /*************************************************************************/
40 
41 #define AttachListFunc    otv_O_x_Ox
42 #define LigCaretListFunc  otv_O_x_Ox
43 
44   /* sets valid->extra1 (0)           */
45 
46   static void
otv_O_x_Ox(FT_Bytes table,OTV_Validator otvalid)47   otv_O_x_Ox( FT_Bytes       table,
48               OTV_Validator  otvalid )
49   {
50     FT_Bytes           p = table;
51     FT_Bytes           Coverage;
52     FT_UInt            GlyphCount;
53     OTV_Validate_Func  func;
54 
55 
56     OTV_ENTER;
57 
58     OTV_LIMIT_CHECK( 4 );
59     Coverage   = table + FT_NEXT_USHORT( p );
60     GlyphCount = FT_NEXT_USHORT( p );
61 
62     OTV_TRACE(( " (GlyphCount = %d)\n", GlyphCount ));
63 
64     otv_Coverage_validate( Coverage, otvalid, (FT_Int)GlyphCount );
65     if ( GlyphCount != otv_Coverage_get_count( Coverage ) )
66       FT_INVALID_DATA;
67 
68     OTV_LIMIT_CHECK( GlyphCount * 2 );
69 
70     otvalid->nesting_level++;
71     func            = otvalid->func[otvalid->nesting_level];
72     otvalid->extra1 = 0;
73 
74     for ( ; GlyphCount > 0; GlyphCount-- )
75       func( table + FT_NEXT_USHORT( p ), otvalid );
76 
77     otvalid->nesting_level--;
78 
79     OTV_EXIT;
80   }
81 
82 
83   /*************************************************************************/
84   /*************************************************************************/
85   /*****                                                               *****/
86   /*****                       LIGATURE CARETS                         *****/
87   /*****                                                               *****/
88   /*************************************************************************/
89   /*************************************************************************/
90 
91 #define CaretValueFunc  otv_CaretValue_validate
92 
93   static void
otv_CaretValue_validate(FT_Bytes table,OTV_Validator otvalid)94   otv_CaretValue_validate( FT_Bytes       table,
95                            OTV_Validator  otvalid )
96   {
97     FT_Bytes  p = table;
98     FT_UInt   CaretValueFormat;
99 
100 
101     OTV_ENTER;
102 
103     OTV_LIMIT_CHECK( 4 );
104 
105     CaretValueFormat = FT_NEXT_USHORT( p );
106 
107     OTV_TRACE(( " (format = %d)\n", CaretValueFormat ));
108 
109     switch ( CaretValueFormat )
110     {
111     case 1:     /* CaretValueFormat1 */
112       /* skip Coordinate, no test */
113       break;
114 
115     case 2:     /* CaretValueFormat2 */
116       /* skip CaretValuePoint, no test */
117       break;
118 
119     case 3:     /* CaretValueFormat3 */
120       p += 2;   /* skip Coordinate */
121 
122       OTV_LIMIT_CHECK( 2 );
123 
124       /* DeviceTable */
125       otv_Device_validate( table + FT_NEXT_USHORT( p ), otvalid );
126       break;
127 
128     default:
129       FT_INVALID_FORMAT;
130     }
131 
132     OTV_EXIT;
133   }
134 
135 
136   /*************************************************************************/
137   /*************************************************************************/
138   /*****                                                               *****/
139   /*****                       MARK GLYPH SETS                         *****/
140   /*****                                                               *****/
141   /*************************************************************************/
142   /*************************************************************************/
143 
144   static void
otv_MarkGlyphSets_validate(FT_Bytes table,OTV_Validator otvalid)145   otv_MarkGlyphSets_validate( FT_Bytes       table,
146                               OTV_Validator  otvalid )
147   {
148     FT_Bytes  p = table;
149     FT_UInt   MarkGlyphSetCount;
150 
151 
152     OTV_NAME_ENTER( "MarkGlyphSets" );
153 
154     p += 2;     /* skip Format */
155 
156     OTV_LIMIT_CHECK( 2 );
157     MarkGlyphSetCount = FT_NEXT_USHORT( p );
158 
159     OTV_TRACE(( " (MarkGlyphSetCount = %d)\n", MarkGlyphSetCount ));
160 
161     OTV_LIMIT_CHECK( MarkGlyphSetCount * 4 );      /* CoverageOffsets */
162 
163     for ( ; MarkGlyphSetCount > 0; MarkGlyphSetCount-- )
164       otv_Coverage_validate( table + FT_NEXT_ULONG( p ), otvalid, -1 );
165 
166     OTV_EXIT;
167   }
168 
169 
170   /*************************************************************************/
171   /*************************************************************************/
172   /*****                                                               *****/
173   /*****                         GDEF TABLE                            *****/
174   /*****                                                               *****/
175   /*************************************************************************/
176   /*************************************************************************/
177 
178   /* sets otvalid->glyph_count */
179 
180   FT_LOCAL_DEF( void )
otv_GDEF_validate(FT_Bytes table,FT_Bytes gsub,FT_Bytes gpos,FT_UInt glyph_count,FT_Validator ftvalid)181   otv_GDEF_validate( FT_Bytes      table,
182                      FT_Bytes      gsub,
183                      FT_Bytes      gpos,
184                      FT_UInt       glyph_count,
185                      FT_Validator  ftvalid )
186   {
187     OTV_ValidatorRec  otvalidrec;
188     OTV_Validator     otvalid = &otvalidrec;
189     FT_Bytes          p       = table;
190     FT_UInt           table_size;
191     FT_UShort         version;
192     FT_Bool           need_MarkAttachClassDef = 1;
193 
194     OTV_OPTIONAL_TABLE( GlyphClassDef );
195     OTV_OPTIONAL_TABLE( AttachListOffset );
196     OTV_OPTIONAL_TABLE( LigCaretListOffset );
197     OTV_OPTIONAL_TABLE( MarkAttachClassDef );
198     OTV_OPTIONAL_TABLE( MarkGlyphSetsDef );
199 
200     OTV_OPTIONAL_TABLE32( itemVarStore );
201 
202 
203     otvalid->root = ftvalid;
204 
205     FT_TRACE3(( "validating GDEF table\n" ));
206     OTV_INIT;
207 
208     OTV_LIMIT_CHECK( 4 );
209 
210     if ( FT_NEXT_USHORT( p ) != 1 )  /* majorVersion */
211       FT_INVALID_FORMAT;
212 
213     version = FT_NEXT_USHORT( p );   /* minorVersion */
214 
215     table_size = 10;
216     switch ( version )
217     {
218     case 0:
219       /* MarkAttachClassDef has been added to the OpenType */
220       /* specification without increasing GDEF's version,  */
221       /* so we use this ugly hack to find out whether the  */
222       /* table is needed actually.                         */
223 
224       need_MarkAttachClassDef = FT_BOOL(
225         otv_GSUBGPOS_have_MarkAttachmentType_flag( gsub ) ||
226         otv_GSUBGPOS_have_MarkAttachmentType_flag( gpos ) );
227 
228       if ( need_MarkAttachClassDef )
229       {
230         OTV_LIMIT_CHECK( 8 );
231         table_size += 2;
232       }
233       else
234         OTV_LIMIT_CHECK( 6 );  /* OpenType < 1.2 */
235 
236       break;
237 
238     case 2:
239       OTV_LIMIT_CHECK( 10 );
240       table_size += 4;
241       break;
242 
243     case 3:
244       OTV_LIMIT_CHECK( 14 );
245       table_size += 8;
246       break;
247 
248     default:
249       FT_INVALID_FORMAT;
250     }
251 
252     otvalid->glyph_count = glyph_count;
253 
254     OTV_OPTIONAL_OFFSET( GlyphClassDef );
255     OTV_SIZE_CHECK( GlyphClassDef );
256     if ( GlyphClassDef )
257       otv_ClassDef_validate( table + GlyphClassDef, otvalid );
258 
259     OTV_OPTIONAL_OFFSET( AttachListOffset );
260     OTV_SIZE_CHECK( AttachListOffset );
261     if ( AttachListOffset )
262     {
263       OTV_NEST2( AttachList, AttachPoint );
264       OTV_RUN( table + AttachListOffset, otvalid );
265     }
266 
267     OTV_OPTIONAL_OFFSET( LigCaretListOffset );
268     OTV_SIZE_CHECK( LigCaretListOffset );
269     if ( LigCaretListOffset )
270     {
271       OTV_NEST3( LigCaretList, LigGlyph, CaretValue );
272       OTV_RUN( table + LigCaretListOffset, otvalid );
273     }
274 
275     if ( need_MarkAttachClassDef )
276     {
277       OTV_OPTIONAL_OFFSET( MarkAttachClassDef );
278       OTV_SIZE_CHECK( MarkAttachClassDef );
279       if ( MarkAttachClassDef )
280         otv_ClassDef_validate( table + MarkAttachClassDef, otvalid );
281     }
282 
283     if ( version > 0 )
284     {
285       OTV_OPTIONAL_OFFSET( MarkGlyphSetsDef );
286       OTV_SIZE_CHECK( MarkGlyphSetsDef );
287       if ( MarkGlyphSetsDef )
288         otv_MarkGlyphSets_validate( table + MarkGlyphSetsDef, otvalid );
289     }
290 
291     if ( version > 2 )
292     {
293       OTV_OPTIONAL_OFFSET32( itemVarStore );
294       OTV_SIZE_CHECK32( itemVarStore );
295       if ( itemVarStore )
296         OTV_TRACE(( "  [omitting itemVarStore validation]\n" )); /* XXX */
297     }
298 
299     FT_TRACE4(( "\n" ));
300   }
301 
302 
303 /* END */
304