1
/***************************************************************************/
5
/* TrueTypeGX/AAT mort table validation (body). */
7
/* Copyright 2005 by suzuki toshiya, Masatake YAMATO, Red Hat K.K., */
8
/* David Turner, Robert Wilhelm, and Werner Lemberg. */
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. */
16
/***************************************************************************/
18
/***************************************************************************/
20
/* gxvalid is derived from both gxlayout module and otvalid module. */
21
/* Development of gxlayout is supported by the Information-technology */
22
/* Promotion Agency(IPA), Japan. */
24
/***************************************************************************/
31
/*************************************************************************/
33
/* The macro FT_COMPONENT is used in trace mode. It is an implicit */
34
/* parameter of the FT_TRACE() and FT_ERROR() macros, used to print/log */
35
/* messages during execution. */
38
#define FT_COMPONENT trace_gxvmort
42
gxv_mort_feature_validate( GXV_mort_feature f,
45
if ( f->featureType >= gxv_feat_registry_length )
47
GXV_TRACE(( "featureType %d is out of registered range, "
48
"setting %d is unchecked\n",
49
f->featureType, f->featureSetting ));
50
if ( valid->root->level >= FT_VALIDATE_PARANOID )
53
else if ( !gxv_feat_registry[f->featureType].existence )
55
GXV_TRACE(( "featureType %d is within registered area "
56
"but undefined, setting %d is unchecked\n",
57
f->featureType, f->featureSetting ));
58
if ( valid->root->level >= FT_VALIDATE_PARANOID )
63
FT_Byte nSettings_max;
66
/* nSettings in gxvfeat.c is halved for exclusive on/off settings */
67
nSettings_max = gxv_feat_registry[f->featureType].nSettings;
68
if ( gxv_feat_registry[f->featureType].exclusive )
69
nSettings_max = (FT_Byte)( 2 * nSettings_max );
71
GXV_TRACE(( "featureType %d is registered", f->featureType ));
72
GXV_TRACE(( "setting %d", f->featureSetting ));
74
if ( f->featureSetting > nSettings_max )
76
GXV_TRACE(( "out of defined range %d", nSettings_max ));
77
if ( valid->root->level >= FT_VALIDATE_PARANOID )
83
/* TODO: enableFlags must be unique value in specified chain? */
88
* nFeatureFlags is typed to FT_ULong to accept that in
89
* mort (typed FT_UShort) and morx (typed FT_ULong).
92
gxv_mort_featurearray_validate( FT_Bytes table,
94
FT_ULong nFeatureFlags,
100
GXV_mort_featureRec f = GXV_MORT_FEATURE_OFF;
103
GXV_NAME_ENTER( "mort feature list" );
104
for ( i = 0; i < nFeatureFlags; i++ )
106
GXV_LIMIT_CHECK( 2 + 2 + 4 + 4 );
107
f.featureType = FT_NEXT_USHORT( p );
108
f.featureSetting = FT_NEXT_USHORT( p );
109
f.enableFlags = FT_NEXT_ULONG( p );
110
f.disableFlags = FT_NEXT_ULONG( p );
112
gxv_mort_feature_validate( &f, valid );
115
if ( !IS_GXV_MORT_FEATURE_OFF( f ) )
118
valid->subtable_length = p - table;
124
gxv_mort_coverage_validate( FT_UShort coverage,
125
GXV_Validator valid )
129
if ( coverage & 0x8000U )
130
GXV_TRACE(( " this subtable is for vertical text only\n" ));
132
GXV_TRACE(( " this subtable is for horizontal text only\n" ));
134
if ( coverage & 0x4000 )
135
GXV_TRACE(( " this subtable is applied to glyph array "
136
"in descending order\n" ));
138
GXV_TRACE(( " this subtable is applied to glyph array "
139
"in ascending order\n" ));
141
if ( coverage & 0x2000 )
142
GXV_TRACE(( " this subtable is forcibly applied to "
143
"vertical/horizontal text\n" ));
145
if ( coverage & 0x1FF8 )
146
GXV_TRACE(( " coverage has non-zero bits in reserved area\n" ));
151
gxv_mort_subtables_validate( FT_Bytes table,
153
FT_UShort nSubtables,
154
GXV_Validator valid )
158
GXV_Validate_Func fmt_funcs_table[] =
160
gxv_mort_subtable_type0_validate, /* 0 */
161
gxv_mort_subtable_type1_validate, /* 1 */
162
gxv_mort_subtable_type2_validate, /* 2 */
164
gxv_mort_subtable_type4_validate, /* 4 */
165
gxv_mort_subtable_type5_validate, /* 5 */
169
GXV_Validate_Func func;
173
GXV_NAME_ENTER( "subtables in a chain" );
175
for ( i = 0; i < nSubtables; i++ )
179
FT_ULong subFeatureFlags;
184
GXV_LIMIT_CHECK( 2 + 2 + 4 );
185
length = FT_NEXT_USHORT( p );
186
coverage = FT_NEXT_USHORT( p );
187
subFeatureFlags = FT_NEXT_ULONG( p );
189
GXV_TRACE(( "validating chain subtable %d/%d (%d bytes)\n",
190
i + 1, nSubtables, length ));
191
type = coverage & 0x0007;
192
rest = length - ( 2 + 2 + 4 );
194
GXV_LIMIT_CHECK( rest );
195
gxv_mort_coverage_validate( coverage, valid );
200
func = fmt_funcs_table[type];
202
GXV_TRACE(( "morx type %d is reserved\n", type ));
204
func( p, p + rest, valid );
209
valid->subtable_length = p - table;
216
gxv_mort_chain_validate( FT_Bytes table,
218
GXV_Validator valid )
221
FT_ULong defaultFlags;
222
FT_ULong chainLength;
223
FT_UShort nFeatureFlags;
224
FT_UShort nSubtables;
227
GXV_NAME_ENTER( "mort chain header" );
229
GXV_LIMIT_CHECK( 4 + 4 + 2 + 2 );
230
defaultFlags = FT_NEXT_ULONG( p );
231
chainLength = FT_NEXT_ULONG( p );
232
nFeatureFlags = FT_NEXT_USHORT( p );
233
nSubtables = FT_NEXT_USHORT( p );
235
gxv_mort_featurearray_validate( p, table + chainLength,
236
nFeatureFlags, valid );
237
p += valid->subtable_length;
238
gxv_mort_subtables_validate( p, table + chainLength, nSubtables, valid );
239
valid->subtable_length = chainLength;
246
gxv_mort_validate( FT_Bytes table,
248
FT_Validator ftvalid )
250
GXV_ValidatorRec validrec;
251
GXV_Validator valid = &validrec;
259
valid->root = ftvalid;
261
limit = valid->root->limit;
263
FT_TRACE3(( "validating `mort' table\n" ));
266
GXV_LIMIT_CHECK( 4 + 4 );
267
version = FT_NEXT_ULONG( p );
268
nChains = FT_NEXT_ULONG( p );
270
if (version != 0x00010000UL)
273
for ( i = 0; i < nChains; i++ )
275
GXV_TRACE(( "validating chain %d/%d\n", i + 1, nChains ));
276
GXV_32BIT_ALIGNMENT_VALIDATE( p - table );
277
gxv_mort_chain_validate( p, limit, valid );
278
p += valid->subtable_length;