Subversion Repositories Kolibri OS

Rev

Blame | Last modification | View Log | RSS feed

  1. /***************************************************************************/
  2. /*                                                                         */
  3. /*  gxvfeat.c                                                              */
  4. /*                                                                         */
  5. /*    TrueTypeGX/AAT feat table validation (body).                         */
  6. /*                                                                         */
  7. /*  Copyright 2004, 2005, 2008, 2012 by                                    */
  8. /*  suzuki toshiya, Masatake YAMATO, Red Hat K.K.,                         */
  9. /*  David Turner, Robert Wilhelm, and Werner Lemberg.                      */
  10. /*                                                                         */
  11. /*  This file is part of the FreeType project, and may only be used,       */
  12. /*  modified, and distributed under the terms of the FreeType project      */
  13. /*  license, LICENSE.TXT.  By continuing to use, modify, or distribute     */
  14. /*  this file you indicate that you have read the license and              */
  15. /*  understand and accept it fully.                                        */
  16. /*                                                                         */
  17. /***************************************************************************/
  18.  
  19. /***************************************************************************/
  20. /*                                                                         */
  21. /* gxvalid is derived from both gxlayout module and otvalid module.        */
  22. /* Development of gxlayout is supported by the Information-technology      */
  23. /* Promotion Agency(IPA), Japan.                                           */
  24. /*                                                                         */
  25. /***************************************************************************/
  26.  
  27.  
  28. #include "gxvalid.h"
  29. #include "gxvcommn.h"
  30. #include "gxvfeat.h"
  31.  
  32.  
  33.   /*************************************************************************/
  34.   /*                                                                       */
  35.   /* The macro FT_COMPONENT is used in trace mode.  It is an implicit      */
  36.   /* parameter of the FT_TRACE() and FT_ERROR() macros, used to print/log  */
  37.   /* messages during execution.                                            */
  38.   /*                                                                       */
  39. #undef  FT_COMPONENT
  40. #define FT_COMPONENT  trace_gxvfeat
  41.  
  42.  
  43.   /*************************************************************************/
  44.   /*************************************************************************/
  45.   /*****                                                               *****/
  46.   /*****                      Data and Types                           *****/
  47.   /*****                                                               *****/
  48.   /*************************************************************************/
  49.   /*************************************************************************/
  50.  
  51.   typedef struct  GXV_feat_DataRec_
  52.   {
  53.     FT_UInt    reserved_size;
  54.     FT_UShort  feature;
  55.     FT_UShort  setting;
  56.  
  57.   } GXV_feat_DataRec, *GXV_feat_Data;
  58.  
  59.  
  60. #define GXV_FEAT_DATA( field )  GXV_TABLE_DATA( feat, field )
  61.  
  62.  
  63.   typedef enum  GXV_FeatureFlagsMask_
  64.   {
  65.     GXV_FEAT_MASK_EXCLUSIVE_SETTINGS = 0x8000U,
  66.     GXV_FEAT_MASK_DYNAMIC_DEFAULT    = 0x4000,
  67.     GXV_FEAT_MASK_UNUSED             = 0x3F00,
  68.     GXV_FEAT_MASK_DEFAULT_SETTING    = 0x00FF
  69.  
  70.   } GXV_FeatureFlagsMask;
  71.  
  72.  
  73.   /*************************************************************************/
  74.   /*************************************************************************/
  75.   /*****                                                               *****/
  76.   /*****                      UTILITY FUNCTIONS                        *****/
  77.   /*****                                                               *****/
  78.   /*************************************************************************/
  79.   /*************************************************************************/
  80.  
  81.   static void
  82.   gxv_feat_registry_validate( FT_UShort      feature,
  83.                               FT_UShort      nSettings,
  84.                               FT_Bool        exclusive,
  85.                               GXV_Validator  valid )
  86.   {
  87.     GXV_NAME_ENTER( "feature in registry" );
  88.  
  89.     GXV_TRACE(( " (feature = %u)\n", feature ));
  90.  
  91.     if ( feature >= gxv_feat_registry_length )
  92.     {
  93.       GXV_TRACE(( "feature number %d is out of range %d\n",
  94.                   feature, gxv_feat_registry_length ));
  95.       GXV_SET_ERR_IF_PARANOID( FT_INVALID_DATA );
  96.       goto Exit;
  97.     }
  98.  
  99.     if ( gxv_feat_registry[feature].existence == 0 )
  100.     {
  101.       GXV_TRACE(( "feature number %d is in defined range but doesn't exist\n",
  102.                   feature ));
  103.       GXV_SET_ERR_IF_PARANOID( FT_INVALID_DATA );
  104.       goto Exit;
  105.     }
  106.  
  107.     if ( gxv_feat_registry[feature].apple_reserved )
  108.     {
  109.       /* Don't use here. Apple is reserved. */
  110.       GXV_TRACE(( "feature number %d is reserved by Apple\n", feature ));
  111.       if ( valid->root->level >= FT_VALIDATE_TIGHT )
  112.         FT_INVALID_DATA;
  113.     }
  114.  
  115.     if ( nSettings != gxv_feat_registry[feature].nSettings )
  116.     {
  117.       GXV_TRACE(( "feature %d: nSettings %d != defined nSettings %d\n",
  118.                   feature, nSettings,
  119.                   gxv_feat_registry[feature].nSettings ));
  120.       if ( valid->root->level >= FT_VALIDATE_TIGHT )
  121.         FT_INVALID_DATA;
  122.     }
  123.  
  124.     if ( exclusive != gxv_feat_registry[feature].exclusive )
  125.     {
  126.       GXV_TRACE(( "exclusive flag %d differs from predefined value\n",
  127.                   exclusive ));
  128.       if ( valid->root->level >= FT_VALIDATE_TIGHT )
  129.         FT_INVALID_DATA;
  130.     }
  131.  
  132.   Exit:
  133.     GXV_EXIT;
  134.   }
  135.  
  136.  
  137.   static void
  138.   gxv_feat_name_index_validate( FT_Bytes       table,
  139.                                 FT_Bytes       limit,
  140.                                 GXV_Validator  valid )
  141.   {
  142.     FT_Bytes  p = table;
  143.  
  144.     FT_Short  nameIndex;
  145.  
  146.  
  147.     GXV_NAME_ENTER( "nameIndex" );
  148.  
  149.     GXV_LIMIT_CHECK( 2 );
  150.     nameIndex = FT_NEXT_SHORT ( p );
  151.     GXV_TRACE(( " (nameIndex = %d)\n", nameIndex ));
  152.  
  153.     gxv_sfntName_validate( (FT_UShort)nameIndex,
  154.                            255,
  155.                            32768U,
  156.                            valid );
  157.  
  158.     GXV_EXIT;
  159.   }
  160.  
  161.  
  162.   static void
  163.   gxv_feat_setting_validate( FT_Bytes       table,
  164.                              FT_Bytes       limit,
  165.                              FT_Bool        exclusive,
  166.                              GXV_Validator  valid )
  167.   {
  168.     FT_Bytes   p = table;
  169.     FT_UShort  setting;
  170.  
  171.  
  172.     GXV_NAME_ENTER( "setting" );
  173.  
  174.     GXV_LIMIT_CHECK( 2 );
  175.  
  176.     setting = FT_NEXT_USHORT( p );
  177.  
  178.     /* If we have exclusive setting, the setting should be odd. */
  179.     if ( exclusive && ( setting & 1 ) == 0 )
  180.       FT_INVALID_DATA;
  181.  
  182.     gxv_feat_name_index_validate( p, limit, valid );
  183.  
  184.     GXV_FEAT_DATA( setting ) = setting;
  185.  
  186.     GXV_EXIT;
  187.   }
  188.  
  189.  
  190.   static void
  191.   gxv_feat_name_validate( FT_Bytes       table,
  192.                           FT_Bytes       limit,
  193.                           GXV_Validator  valid )
  194.   {
  195.     FT_Bytes   p             = table;
  196.     FT_UInt    reserved_size = GXV_FEAT_DATA( reserved_size );
  197.  
  198.     FT_UShort  feature;
  199.     FT_UShort  nSettings;
  200.     FT_ULong   settingTable;
  201.     FT_UShort  featureFlags;
  202.  
  203.     FT_Bool    exclusive;
  204.     FT_Int     last_setting;
  205.     FT_UInt    i;
  206.  
  207.  
  208.     GXV_NAME_ENTER( "name" );
  209.  
  210.     /* feature + nSettings + settingTable + featureFlags */
  211.     GXV_LIMIT_CHECK( 2 + 2 + 4 + 2 );
  212.  
  213.     feature = FT_NEXT_USHORT( p );
  214.     GXV_FEAT_DATA( feature ) = feature;
  215.  
  216.     nSettings    = FT_NEXT_USHORT( p );
  217.     settingTable = FT_NEXT_ULONG ( p );
  218.     featureFlags = FT_NEXT_USHORT( p );
  219.  
  220.     if ( settingTable < reserved_size )
  221.       FT_INVALID_OFFSET;
  222.  
  223.     if ( ( featureFlags & GXV_FEAT_MASK_UNUSED ) == 0 )
  224.       GXV_SET_ERR_IF_PARANOID( FT_INVALID_DATA );
  225.  
  226.     exclusive = FT_BOOL( featureFlags & GXV_FEAT_MASK_EXCLUSIVE_SETTINGS );
  227.     if ( exclusive )
  228.     {
  229.       FT_Byte  dynamic_default;
  230.  
  231.  
  232.       if ( featureFlags & GXV_FEAT_MASK_DYNAMIC_DEFAULT )
  233.         dynamic_default = (FT_Byte)( featureFlags &
  234.                                      GXV_FEAT_MASK_DEFAULT_SETTING );
  235.       else
  236.         dynamic_default = 0;
  237.  
  238.       /* If exclusive, check whether default setting is in the range. */
  239.       if ( !( dynamic_default < nSettings ) )
  240.         FT_INVALID_FORMAT;
  241.     }
  242.  
  243.     gxv_feat_registry_validate( feature, nSettings, exclusive, valid );
  244.  
  245.     gxv_feat_name_index_validate( p, limit, valid );
  246.  
  247.     p = valid->root->base + settingTable;
  248.     for ( last_setting = -1, i = 0; i < nSettings; i++ )
  249.     {
  250.       gxv_feat_setting_validate( p, limit, exclusive, valid );
  251.  
  252.       if ( (FT_Int)GXV_FEAT_DATA( setting ) <= last_setting )
  253.         GXV_SET_ERR_IF_PARANOID( FT_INVALID_FORMAT );
  254.  
  255.       last_setting = (FT_Int)GXV_FEAT_DATA( setting );
  256.       /* setting + nameIndex */
  257.       p += ( 2 + 2 );
  258.     }
  259.  
  260.     GXV_EXIT;
  261.   }
  262.  
  263.  
  264.   /*************************************************************************/
  265.   /*************************************************************************/
  266.   /*****                                                               *****/
  267.   /*****                         feat TABLE                            *****/
  268.   /*****                                                               *****/
  269.   /*************************************************************************/
  270.   /*************************************************************************/
  271.  
  272.   FT_LOCAL_DEF( void )
  273.   gxv_feat_validate( FT_Bytes      table,
  274.                      FT_Face       face,
  275.                      FT_Validator  ftvalid )
  276.   {
  277.     GXV_ValidatorRec  validrec;
  278.     GXV_Validator     valid = &validrec;
  279.  
  280.     GXV_feat_DataRec  featrec;
  281.     GXV_feat_Data     feat = &featrec;
  282.  
  283.     FT_Bytes          p     = table;
  284.     FT_Bytes          limit = 0;
  285.  
  286.     FT_UInt           featureNameCount;
  287.  
  288.     FT_UInt           i;
  289.     FT_Int            last_feature;
  290.  
  291.  
  292.     valid->root       = ftvalid;
  293.     valid->table_data = feat;
  294.     valid->face       = face;
  295.  
  296.     FT_TRACE3(( "validating `feat' table\n" ));
  297.     GXV_INIT;
  298.  
  299.     feat->reserved_size = 0;
  300.  
  301.     /* version + featureNameCount + none_0 + none_1  */
  302.     GXV_LIMIT_CHECK( 4 + 2 + 2 + 4 );
  303.     feat->reserved_size += 4 + 2 + 2 + 4;
  304.  
  305.     if ( FT_NEXT_ULONG( p ) != 0x00010000UL ) /* Version */
  306.       FT_INVALID_FORMAT;
  307.  
  308.     featureNameCount = FT_NEXT_USHORT( p );
  309.     GXV_TRACE(( " (featureNameCount = %d)\n", featureNameCount ));
  310.  
  311.     if ( !( IS_PARANOID_VALIDATION ) )
  312.       p += 6; /* skip (none) and (none) */
  313.     else
  314.     {
  315.       if ( FT_NEXT_USHORT( p ) != 0 )
  316.         FT_INVALID_DATA;
  317.  
  318.       if ( FT_NEXT_ULONG( p )  != 0 )
  319.         FT_INVALID_DATA;
  320.     }
  321.  
  322.     feat->reserved_size += featureNameCount * ( 2 + 2 + 4 + 2 + 2 );
  323.  
  324.     for ( last_feature = -1, i = 0; i < featureNameCount; i++ )
  325.     {
  326.       gxv_feat_name_validate( p, limit, valid );
  327.  
  328.       if ( (FT_Int)GXV_FEAT_DATA( feature ) <= last_feature )
  329.         GXV_SET_ERR_IF_PARANOID( FT_INVALID_FORMAT );
  330.  
  331.       last_feature = GXV_FEAT_DATA( feature );
  332.       p += 2 + 2 + 4 + 2 + 2;
  333.     }
  334.  
  335.     FT_TRACE4(( "\n" ));
  336.   }
  337.  
  338.  
  339. /* END */
  340.