PageRenderTime 39ms CodeModel.GetById 15ms RepoModel.GetById 0ms app.codeStats 1ms

/modules/freetype2/src/otvalid/otvgsub.c

https://bitbucket.org/soko/mozilla-central
C | 585 lines | 313 code | 160 blank | 112 comment | 26 complexity | a3ffb11e5de3447f041dc612a880e83d MD5 | raw file
Possible License(s): GPL-2.0, JSON, 0BSD, LGPL-3.0, AGPL-1.0, MIT, MPL-2.0-no-copyleft-exception, BSD-3-Clause, LGPL-2.1, Apache-2.0
  1. /***************************************************************************/
  2. /* */
  3. /* otvgsub.c */
  4. /* */
  5. /* OpenType GSUB table validation (body). */
  6. /* */
  7. /* Copyright 2004, 2005, 2007 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. #include "otvalid.h"
  18. #include "otvcommn.h"
  19. /*************************************************************************/
  20. /* */
  21. /* The macro FT_COMPONENT is used in trace mode. It is an implicit */
  22. /* parameter of the FT_TRACE() and FT_ERROR() macros, used to print/log */
  23. /* messages during execution. */
  24. /* */
  25. #undef FT_COMPONENT
  26. #define FT_COMPONENT trace_otvgsub
  27. /*************************************************************************/
  28. /*************************************************************************/
  29. /***** *****/
  30. /***** GSUB LOOKUP TYPE 1 *****/
  31. /***** *****/
  32. /*************************************************************************/
  33. /*************************************************************************/
  34. /* uses valid->glyph_count */
  35. static void
  36. otv_SingleSubst_validate( FT_Bytes table,
  37. OTV_Validator valid )
  38. {
  39. FT_Bytes p = table;
  40. FT_UInt SubstFormat;
  41. OTV_NAME_ENTER( "SingleSubst" );
  42. OTV_LIMIT_CHECK( 2 );
  43. SubstFormat = FT_NEXT_USHORT( p );
  44. OTV_TRACE(( " (format %d)\n", SubstFormat ));
  45. switch ( SubstFormat )
  46. {
  47. case 1: /* SingleSubstFormat1 */
  48. {
  49. FT_Bytes Coverage;
  50. FT_Int DeltaGlyphID;
  51. FT_Long idx;
  52. OTV_LIMIT_CHECK( 4 );
  53. Coverage = table + FT_NEXT_USHORT( p );
  54. DeltaGlyphID = FT_NEXT_SHORT( p );
  55. otv_Coverage_validate( Coverage, valid, -1 );
  56. idx = otv_Coverage_get_first( Coverage ) + DeltaGlyphID;
  57. if ( idx < 0 )
  58. FT_INVALID_DATA;
  59. idx = otv_Coverage_get_last( Coverage ) + DeltaGlyphID;
  60. if ( (FT_UInt)idx >= valid->glyph_count )
  61. FT_INVALID_DATA;
  62. }
  63. break;
  64. case 2: /* SingleSubstFormat2 */
  65. {
  66. FT_UInt Coverage, GlyphCount;
  67. OTV_LIMIT_CHECK( 4 );
  68. Coverage = FT_NEXT_USHORT( p );
  69. GlyphCount = FT_NEXT_USHORT( p );
  70. OTV_TRACE(( " (GlyphCount = %d)\n", GlyphCount ));
  71. otv_Coverage_validate( table + Coverage, valid, GlyphCount );
  72. OTV_LIMIT_CHECK( GlyphCount * 2 );
  73. /* Substitute */
  74. for ( ; GlyphCount > 0; GlyphCount-- )
  75. if ( FT_NEXT_USHORT( p ) >= valid->glyph_count )
  76. FT_INVALID_GLYPH_ID;
  77. }
  78. break;
  79. default:
  80. FT_INVALID_FORMAT;
  81. }
  82. OTV_EXIT;
  83. }
  84. /*************************************************************************/
  85. /*************************************************************************/
  86. /***** *****/
  87. /***** GSUB LOOKUP TYPE 2 *****/
  88. /***** *****/
  89. /*************************************************************************/
  90. /*************************************************************************/
  91. /* sets valid->extra1 (glyph count) */
  92. static void
  93. otv_MultipleSubst_validate( FT_Bytes table,
  94. OTV_Validator valid )
  95. {
  96. FT_Bytes p = table;
  97. FT_UInt SubstFormat;
  98. OTV_NAME_ENTER( "MultipleSubst" );
  99. OTV_LIMIT_CHECK( 2 );
  100. SubstFormat = FT_NEXT_USHORT( p );
  101. OTV_TRACE(( " (format %d)\n", SubstFormat ));
  102. switch ( SubstFormat )
  103. {
  104. case 1:
  105. valid->extra1 = valid->glyph_count;
  106. OTV_NEST2( MultipleSubstFormat1, Sequence );
  107. OTV_RUN( table, valid );
  108. break;
  109. default:
  110. FT_INVALID_FORMAT;
  111. }
  112. OTV_EXIT;
  113. }
  114. /*************************************************************************/
  115. /*************************************************************************/
  116. /***** *****/
  117. /***** GSUB LOOKUP TYPE 3 *****/
  118. /***** *****/
  119. /*************************************************************************/
  120. /*************************************************************************/
  121. /* sets valid->extra1 (glyph count) */
  122. static void
  123. otv_AlternateSubst_validate( FT_Bytes table,
  124. OTV_Validator valid )
  125. {
  126. FT_Bytes p = table;
  127. FT_UInt SubstFormat;
  128. OTV_NAME_ENTER( "AlternateSubst" );
  129. OTV_LIMIT_CHECK( 2 );
  130. SubstFormat = FT_NEXT_USHORT( p );
  131. OTV_TRACE(( " (format %d)\n", SubstFormat ));
  132. switch ( SubstFormat )
  133. {
  134. case 1:
  135. valid->extra1 = valid->glyph_count;
  136. OTV_NEST2( AlternateSubstFormat1, AlternateSet );
  137. OTV_RUN( table, valid );
  138. break;
  139. default:
  140. FT_INVALID_FORMAT;
  141. }
  142. OTV_EXIT;
  143. }
  144. /*************************************************************************/
  145. /*************************************************************************/
  146. /***** *****/
  147. /***** GSUB LOOKUP TYPE 4 *****/
  148. /***** *****/
  149. /*************************************************************************/
  150. /*************************************************************************/
  151. #define LigatureFunc otv_Ligature_validate
  152. /* uses valid->glyph_count */
  153. static void
  154. otv_Ligature_validate( FT_Bytes table,
  155. OTV_Validator valid )
  156. {
  157. FT_Bytes p = table;
  158. FT_UInt LigatureGlyph, CompCount;
  159. OTV_ENTER;
  160. OTV_LIMIT_CHECK( 4 );
  161. LigatureGlyph = FT_NEXT_USHORT( p );
  162. if ( LigatureGlyph >= valid->glyph_count )
  163. FT_INVALID_DATA;
  164. CompCount = FT_NEXT_USHORT( p );
  165. OTV_TRACE(( " (CompCount = %d)\n", CompCount ));
  166. if ( CompCount == 0 )
  167. FT_INVALID_DATA;
  168. CompCount--;
  169. OTV_LIMIT_CHECK( CompCount * 2 ); /* Component */
  170. /* no need to check the Component glyph indices */
  171. OTV_EXIT;
  172. }
  173. static void
  174. otv_LigatureSubst_validate( FT_Bytes table,
  175. OTV_Validator valid )
  176. {
  177. FT_Bytes p = table;
  178. FT_UInt SubstFormat;
  179. OTV_NAME_ENTER( "LigatureSubst" );
  180. OTV_LIMIT_CHECK( 2 );
  181. SubstFormat = FT_NEXT_USHORT( p );
  182. OTV_TRACE(( " (format %d)\n", SubstFormat ));
  183. switch ( SubstFormat )
  184. {
  185. case 1:
  186. OTV_NEST3( LigatureSubstFormat1, LigatureSet, Ligature );
  187. OTV_RUN( table, valid );
  188. break;
  189. default:
  190. FT_INVALID_FORMAT;
  191. }
  192. OTV_EXIT;
  193. }
  194. /*************************************************************************/
  195. /*************************************************************************/
  196. /***** *****/
  197. /***** GSUB LOOKUP TYPE 5 *****/
  198. /***** *****/
  199. /*************************************************************************/
  200. /*************************************************************************/
  201. /* sets valid->extra1 (lookup count) */
  202. static void
  203. otv_ContextSubst_validate( FT_Bytes table,
  204. OTV_Validator valid )
  205. {
  206. FT_Bytes p = table;
  207. FT_UInt SubstFormat;
  208. OTV_NAME_ENTER( "ContextSubst" );
  209. OTV_LIMIT_CHECK( 2 );
  210. SubstFormat = FT_NEXT_USHORT( p );
  211. OTV_TRACE(( " (format %d)\n", SubstFormat ));
  212. switch ( SubstFormat )
  213. {
  214. case 1:
  215. /* no need to check glyph indices/classes used as input for these */
  216. /* context rules since even invalid glyph indices/classes return */
  217. /* meaningful results */
  218. valid->extra1 = valid->lookup_count;
  219. OTV_NEST3( ContextSubstFormat1, SubRuleSet, SubRule );
  220. OTV_RUN( table, valid );
  221. break;
  222. case 2:
  223. /* no need to check glyph indices/classes used as input for these */
  224. /* context rules since even invalid glyph indices/classes return */
  225. /* meaningful results */
  226. OTV_NEST3( ContextSubstFormat2, SubClassSet, SubClassRule );
  227. OTV_RUN( table, valid );
  228. break;
  229. case 3:
  230. OTV_NEST1( ContextSubstFormat3 );
  231. OTV_RUN( table, valid );
  232. break;
  233. default:
  234. FT_INVALID_FORMAT;
  235. }
  236. OTV_EXIT;
  237. }
  238. /*************************************************************************/
  239. /*************************************************************************/
  240. /***** *****/
  241. /***** GSUB LOOKUP TYPE 6 *****/
  242. /***** *****/
  243. /*************************************************************************/
  244. /*************************************************************************/
  245. /* sets valid->extra1 (lookup count) */
  246. static void
  247. otv_ChainContextSubst_validate( FT_Bytes table,
  248. OTV_Validator valid )
  249. {
  250. FT_Bytes p = table;
  251. FT_UInt SubstFormat;
  252. OTV_NAME_ENTER( "ChainContextSubst" );
  253. OTV_LIMIT_CHECK( 2 );
  254. SubstFormat = FT_NEXT_USHORT( p );
  255. OTV_TRACE(( " (format %d)\n", SubstFormat ));
  256. switch ( SubstFormat )
  257. {
  258. case 1:
  259. /* no need to check glyph indices/classes used as input for these */
  260. /* context rules since even invalid glyph indices/classes return */
  261. /* meaningful results */
  262. valid->extra1 = valid->lookup_count;
  263. OTV_NEST3( ChainContextSubstFormat1,
  264. ChainSubRuleSet, ChainSubRule );
  265. OTV_RUN( table, valid );
  266. break;
  267. case 2:
  268. /* no need to check glyph indices/classes used as input for these */
  269. /* context rules since even invalid glyph indices/classes return */
  270. /* meaningful results */
  271. OTV_NEST3( ChainContextSubstFormat2,
  272. ChainSubClassSet, ChainSubClassRule );
  273. OTV_RUN( table, valid );
  274. break;
  275. case 3:
  276. OTV_NEST1( ChainContextSubstFormat3 );
  277. OTV_RUN( table, valid );
  278. break;
  279. default:
  280. FT_INVALID_FORMAT;
  281. }
  282. OTV_EXIT;
  283. }
  284. /*************************************************************************/
  285. /*************************************************************************/
  286. /***** *****/
  287. /***** GSUB LOOKUP TYPE 7 *****/
  288. /***** *****/
  289. /*************************************************************************/
  290. /*************************************************************************/
  291. /* uses valid->type_funcs */
  292. static void
  293. otv_ExtensionSubst_validate( FT_Bytes table,
  294. OTV_Validator valid )
  295. {
  296. FT_Bytes p = table;
  297. FT_UInt SubstFormat;
  298. OTV_NAME_ENTER( "ExtensionSubst" );
  299. OTV_LIMIT_CHECK( 2 );
  300. SubstFormat = FT_NEXT_USHORT( p );
  301. OTV_TRACE(( " (format %d)\n", SubstFormat ));
  302. switch ( SubstFormat )
  303. {
  304. case 1: /* ExtensionSubstFormat1 */
  305. {
  306. FT_UInt ExtensionLookupType;
  307. FT_ULong ExtensionOffset;
  308. OTV_Validate_Func validate;
  309. OTV_LIMIT_CHECK( 6 );
  310. ExtensionLookupType = FT_NEXT_USHORT( p );
  311. ExtensionOffset = FT_NEXT_ULONG( p );
  312. if ( ExtensionLookupType == 0 ||
  313. ExtensionLookupType == 7 ||
  314. ExtensionLookupType > 8 )
  315. FT_INVALID_DATA;
  316. validate = valid->type_funcs[ExtensionLookupType - 1];
  317. validate( table + ExtensionOffset, valid );
  318. }
  319. break;
  320. default:
  321. FT_INVALID_FORMAT;
  322. }
  323. OTV_EXIT;
  324. }
  325. /*************************************************************************/
  326. /*************************************************************************/
  327. /***** *****/
  328. /***** GSUB LOOKUP TYPE 8 *****/
  329. /***** *****/
  330. /*************************************************************************/
  331. /*************************************************************************/
  332. /* uses valid->glyph_count */
  333. static void
  334. otv_ReverseChainSingleSubst_validate( FT_Bytes table,
  335. OTV_Validator valid )
  336. {
  337. FT_Bytes p = table, Coverage;
  338. FT_UInt SubstFormat;
  339. FT_UInt BacktrackGlyphCount, LookaheadGlyphCount, GlyphCount;
  340. OTV_NAME_ENTER( "ReverseChainSingleSubst" );
  341. OTV_LIMIT_CHECK( 2 );
  342. SubstFormat = FT_NEXT_USHORT( p );
  343. OTV_TRACE(( " (format %d)\n", SubstFormat ));
  344. switch ( SubstFormat )
  345. {
  346. case 1: /* ReverseChainSingleSubstFormat1 */
  347. OTV_LIMIT_CHECK( 4 );
  348. Coverage = table + FT_NEXT_USHORT( p );
  349. BacktrackGlyphCount = FT_NEXT_USHORT( p );
  350. OTV_TRACE(( " (BacktrackGlyphCount = %d)\n", BacktrackGlyphCount ));
  351. otv_Coverage_validate( Coverage, valid, -1 );
  352. OTV_LIMIT_CHECK( BacktrackGlyphCount * 2 + 2 );
  353. for ( ; BacktrackGlyphCount > 0; BacktrackGlyphCount-- )
  354. otv_Coverage_validate( table + FT_NEXT_USHORT( p ), valid, -1 );
  355. LookaheadGlyphCount = FT_NEXT_USHORT( p );
  356. OTV_TRACE(( " (LookaheadGlyphCount = %d)\n", LookaheadGlyphCount ));
  357. OTV_LIMIT_CHECK( LookaheadGlyphCount * 2 + 2 );
  358. for ( ; LookaheadGlyphCount > 0; LookaheadGlyphCount-- )
  359. otv_Coverage_validate( table + FT_NEXT_USHORT( p ), valid, -1 );
  360. GlyphCount = FT_NEXT_USHORT( p );
  361. OTV_TRACE(( " (GlyphCount = %d)\n", GlyphCount ));
  362. if ( GlyphCount != otv_Coverage_get_count( Coverage ) )
  363. FT_INVALID_DATA;
  364. OTV_LIMIT_CHECK( GlyphCount * 2 );
  365. /* Substitute */
  366. for ( ; GlyphCount > 0; GlyphCount-- )
  367. if ( FT_NEXT_USHORT( p ) >= valid->glyph_count )
  368. FT_INVALID_DATA;
  369. break;
  370. default:
  371. FT_INVALID_FORMAT;
  372. }
  373. OTV_EXIT;
  374. }
  375. static const OTV_Validate_Func otv_gsub_validate_funcs[8] =
  376. {
  377. otv_SingleSubst_validate,
  378. otv_MultipleSubst_validate,
  379. otv_AlternateSubst_validate,
  380. otv_LigatureSubst_validate,
  381. otv_ContextSubst_validate,
  382. otv_ChainContextSubst_validate,
  383. otv_ExtensionSubst_validate,
  384. otv_ReverseChainSingleSubst_validate
  385. };
  386. /*************************************************************************/
  387. /*************************************************************************/
  388. /***** *****/
  389. /***** GSUB TABLE *****/
  390. /***** *****/
  391. /*************************************************************************/
  392. /*************************************************************************/
  393. /* sets valid->type_count */
  394. /* sets valid->type_funcs */
  395. /* sets valid->glyph_count */
  396. FT_LOCAL_DEF( void )
  397. otv_GSUB_validate( FT_Bytes table,
  398. FT_UInt glyph_count,
  399. FT_Validator ftvalid )
  400. {
  401. OTV_ValidatorRec validrec;
  402. OTV_Validator valid = &validrec;
  403. FT_Bytes p = table;
  404. FT_UInt ScriptList, FeatureList, LookupList;
  405. valid->root = ftvalid;
  406. FT_TRACE3(( "validating GSUB table\n" ));
  407. OTV_INIT;
  408. OTV_LIMIT_CHECK( 10 );
  409. if ( FT_NEXT_ULONG( p ) != 0x10000UL ) /* Version */
  410. FT_INVALID_FORMAT;
  411. ScriptList = FT_NEXT_USHORT( p );
  412. FeatureList = FT_NEXT_USHORT( p );
  413. LookupList = FT_NEXT_USHORT( p );
  414. valid->type_count = 8;
  415. valid->type_funcs = (OTV_Validate_Func*)otv_gsub_validate_funcs;
  416. valid->glyph_count = glyph_count;
  417. otv_LookupList_validate( table + LookupList,
  418. valid );
  419. otv_FeatureList_validate( table + FeatureList, table + LookupList,
  420. valid );
  421. otv_ScriptList_validate( table + ScriptList, table + FeatureList,
  422. valid );
  423. FT_TRACE4(( "\n" ));
  424. }
  425. /* END */