Edit

kc3-lang/freetype/src/autofit/afglobal.c

Branch :

  • Show log

    Commit

  • Author : Werner Lemberg
    Date : 2025-07-01 22:15:07
    Hash : 0fd0cf33
    Message : [autofit] Remove `globals->gsub_length`. We completely validate the accessed data from the 'GSUB' table, making this field unnecessary. * src/autofit/afglobal.h (AF_FaceGlobalsRec): Remove `gsub_length` field. * src/autofit/afglobal.c (af_face_globals_new), src/autofit/afgsub.c (af_parse_gsub): Updated.

  • src/autofit/afglobal.c
  • /****************************************************************************
     *
     * afglobal.c
     *
     *   Auto-fitter routines to compute global hinting values (body).
     *
     * Copyright (C) 2003-2024 by
     * David Turner, Robert Wilhelm, and Werner Lemberg.
     *
     * This file is part of the FreeType project, and may only be used,
     * modified, and distributed under the terms of the FreeType project
     * license, LICENSE.TXT.  By continuing to use, modify, or distribute
     * this file you indicate that you have read the license and
     * understand and accept it fully.
     *
     */
    
    
    #include "afglobal.h"
    #include "afranges.h"
    #include "afshaper.h"
    #include "afws-decl.h"
    #include <freetype/internal/ftdebug.h>
    
    #ifdef FT_CONFIG_OPTION_USE_HARFBUZZ
    #  include "afgsub.h"
    #  include "ft-hb-ft.h"
    #endif
    
    
      /**************************************************************************
       *
       * The macro FT_COMPONENT is used in trace mode.  It is an implicit
       * parameter of the FT_TRACE() and FT_ERROR() macros, used to print/log
       * messages during execution.
       */
    #undef  FT_COMPONENT
    #define FT_COMPONENT  afglobal
    
    
    #include "aferrors.h"
    
    
    #undef  SCRIPT
    #define SCRIPT( s, S, d, h, H, ss )         \
              AF_DEFINE_SCRIPT_CLASS(           \
                af_ ## s ## _script_class,      \
                AF_SCRIPT_ ## S,                \
                af_ ## s ## _uniranges,         \
                af_ ## s ## _nonbase_uniranges, \
                AF_ ## H,                       \
                ss )
    
    #include "afscript.h"
    
    
    #undef  STYLE
    #define STYLE( s, S, d, ws, sc, ss, c )  \
              AF_DEFINE_STYLE_CLASS(         \
                af_ ## s ## _style_class,    \
                AF_STYLE_ ## S,              \
                ws,                          \
                sc,                          \
                ss,                          \
                c )
    
    #include "afstyles.h"
    
    
    #undef  WRITING_SYSTEM
    #define WRITING_SYSTEM( ws, WS )               \
              &af_ ## ws ## _writing_system_class,
    
      FT_LOCAL_ARRAY_DEF( AF_WritingSystemClass )
      af_writing_system_classes[] =
      {
    
    #include "afws-iter.h"
    
        NULL  /* do not remove */
      };
    
    
    #undef  SCRIPT
    #define SCRIPT( s, S, d, h, H, ss )   \
              &af_ ## s ## _script_class,
    
      FT_LOCAL_ARRAY_DEF( AF_ScriptClass )
      af_script_classes[] =
      {
    
    #include "afscript.h"
    
        NULL  /* do not remove */
      };
    
    
    #undef  STYLE
    #define STYLE( s, S, d, ws, sc, ss, c ) \
              &af_ ## s ## _style_class,
    
      FT_LOCAL_ARRAY_DEF( AF_StyleClass )
      af_style_classes[] =
      {
    
    #include "afstyles.h"
    
        NULL  /* do not remove */
      };
    
    
    #ifdef FT_DEBUG_LEVEL_TRACE
    
    #undef  STYLE
    #define STYLE( s, S, d, ws, sc, ss, c )  #s,
    
      FT_LOCAL_ARRAY_DEF( char* )
      af_style_names[] =
      {
    
    #include "afstyles.h"
    
      };
    
    #endif /* FT_DEBUG_LEVEL_TRACE */
    
    
      /* Compute the style index of each glyph within a given face. */
    
      static FT_Error
      af_face_globals_compute_style_coverage( AF_FaceGlobals  globals )
      {
        FT_Error    error;
        FT_Face     face        = globals->face;
        FT_CharMap  old_charmap = face->charmap;
        FT_UShort*  gstyles     = globals->glyph_styles;
        FT_UShort   ss;
        FT_UShort   dflt        = 0xFFFFU; /* a non-valid value */
        FT_UInt     i;
    
    
        /* the value AF_STYLE_UNASSIGNED means `uncovered glyph' */
        for ( i = 0; i < globals->glyph_count; i++ )
          gstyles[i] = AF_STYLE_UNASSIGNED;
    
        error = FT_Select_Charmap( face, FT_ENCODING_UNICODE );
        if ( error )
        {
          /*
           * Ignore this error; we simply use the fallback style.
           * XXX: Shouldn't we rather disable hinting?
           */
          error = FT_Err_Ok;
          goto Exit;
        }
    
        /* scan each style in a Unicode charmap */
        for ( ss = 0; af_style_classes[ss]; ss++ )
        {
          AF_StyleClass       style_class =
                                af_style_classes[ss];
          AF_ScriptClass      script_class =
                                af_script_classes[style_class->script];
          AF_Script_UniRange  range;
    
    
          if ( !script_class->script_uni_ranges )
            continue;
    
          /*
           * Scan all Unicode points in the range and set the corresponding
           * glyph style index.
           */
          if ( style_class->coverage == AF_COVERAGE_DEFAULT )
          {
            if ( style_class->script == globals->module->default_script )
              dflt = ss;
    
            for ( range = script_class->script_uni_ranges;
                  range->first != 0;
                  range++ )
            {
              FT_ULong  charcode = range->first;
              FT_UInt   gindex;
    
    
              gindex = FT_Get_Char_Index( face, charcode );
    
              if ( gindex != 0                                                &&
                   gindex < globals->glyph_count                              &&
                   ( gstyles[gindex] & AF_STYLE_MASK ) == AF_STYLE_UNASSIGNED )
                gstyles[gindex] = ss | AF_HAS_CMAP_ENTRY;
    
              for (;;)
              {
                charcode = FT_Get_Next_Char( face, charcode, &gindex );
    
                if ( gindex == 0 || charcode > range->last )
                  break;
    
                if ( gindex < globals->glyph_count                              &&
                     ( gstyles[gindex] & AF_STYLE_MASK ) == AF_STYLE_UNASSIGNED )
                  gstyles[gindex] = ss | AF_HAS_CMAP_ENTRY;
              }
            }
    
            /* do the same for the script's non-base characters */
            for ( range = script_class->script_uni_nonbase_ranges;
                  range->first != 0;
                  range++ )
            {
              FT_ULong  charcode = range->first;
              FT_UInt   gindex;
    
    
              gindex = FT_Get_Char_Index( face, charcode );
    
              if ( gindex != 0                               &&
                   gindex < globals->glyph_count             &&
                   ( gstyles[gindex] & AF_STYLE_MASK ) == ss )
                gstyles[gindex] |= AF_NONBASE;
    
              for (;;)
              {
                charcode = FT_Get_Next_Char( face, charcode, &gindex );
    
                if ( gindex == 0 || charcode > range->last )
                  break;
    
                if ( gindex < globals->glyph_count             &&
                     ( gstyles[gindex] & AF_STYLE_MASK ) == ss )
                  gstyles[gindex] |= AF_NONBASE;
              }
            }
          }
          else
          {
            /* get glyphs not directly addressable by cmap */
            af_shaper_get_coverage( globals, style_class, gstyles, 0 );
          }
        }
    
        /* handle the remaining default OpenType features ... */
        for ( ss = 0; af_style_classes[ss]; ss++ )
        {
          AF_StyleClass  style_class = af_style_classes[ss];
    
    
          if ( style_class->coverage == AF_COVERAGE_DEFAULT )
            af_shaper_get_coverage( globals, style_class, gstyles, 0 );
        }
    
        /* ... and finally the default OpenType features of the default script */
        af_shaper_get_coverage( globals, af_style_classes[dflt], gstyles, 1 );
    
        /* mark ASCII digits */
        for ( i = 0x30; i <= 0x39; i++ )
        {
          FT_UInt  gindex = FT_Get_Char_Index( face, i );
    
    
          if ( gindex != 0 && gindex < globals->glyph_count )
            gstyles[gindex] |= AF_DIGIT;
        }
    
      Exit:
        /*
         * By default, all uncovered glyphs are set to the fallback style.
         * XXX: Shouldn't we disable hinting or do something similar?
         */
        if ( globals->module->fallback_style != AF_STYLE_UNASSIGNED )
        {
          FT_UInt  nn;
    
    
          for ( nn = 0; nn < globals->glyph_count; nn++ )
          {
            if ( ( gstyles[nn] & AF_STYLE_MASK ) == AF_STYLE_UNASSIGNED )
            {
              gstyles[nn] &= ~AF_STYLE_MASK;
              gstyles[nn] |= globals->module->fallback_style;
            }
          }
        }
    
    #ifdef FT_DEBUG_LEVEL_TRACE
    
        FT_TRACE4(( "\n" ));
        FT_TRACE4(( "style coverage\n" ));
        FT_TRACE4(( "==============\n" ));
        FT_TRACE4(( "\n" ));
    
        for ( ss = 0; af_style_classes[ss]; ss++ )
        {
          AF_StyleClass  style_class = af_style_classes[ss];
          FT_UInt        count       = 0;
          FT_UInt        idx;
    
    
          FT_TRACE4(( "%s:\n", af_style_names[style_class->style] ));
    
          for ( idx = 0; idx < globals->glyph_count; idx++ )
          {
            if ( ( gstyles[idx] & AF_STYLE_MASK ) == style_class->style )
            {
              if ( !( count % 10 ) )
                FT_TRACE4(( " " ));
    
              FT_TRACE4(( " %u", idx ));
              count++;
    
              if ( !( count % 10 ) )
                FT_TRACE4(( "\n" ));
            }
          }
    
          if ( !count )
            FT_TRACE4(( "  (none)\n" ));
          if ( count % 10 )
            FT_TRACE4(( "\n" ));
        }
    
    #endif /* FT_DEBUG_LEVEL_TRACE */
    
        face->charmap = old_charmap;
        return error;
      }
    
    
      FT_LOCAL_DEF( FT_Error )
      af_face_globals_new( FT_Face          face,
                           AF_FaceGlobals  *aglobals,
                           AF_Module        module )
      {
        FT_Error        error;
        FT_Memory       memory;
        AF_FaceGlobals  globals = NULL;
    
    
        memory = face->memory;
    
        /* we allocate an AF_FaceGlobals structure together */
        /* with the glyph_styles array                      */
        if ( FT_QALLOC( globals,
                        sizeof ( *globals ) +
                          (FT_ULong)face->num_glyphs * sizeof ( FT_UShort ) ) )
          goto Exit;
    
        FT_ZERO( &globals->metrics );
    
        globals->face                      = face;
        globals->glyph_count               = (FT_UInt)face->num_glyphs;
        /* right after the globals structure come the glyph styles */
        globals->glyph_styles              = (FT_UShort*)( globals + 1 );
        globals->module                    = module;
        globals->stem_darkening_for_ppem   = 0;
        globals->darken_x                  = 0;
        globals->darken_y                  = 0;
        globals->standard_vertical_width   = 0;
        globals->standard_horizontal_width = 0;
        globals->scale_down_factor         = 0;
    
    #ifdef FT_CONFIG_OPTION_USE_HARFBUZZ
        if ( ft_hb_enabled ( globals ) )
        {
          globals->hb_font = ft_hb_ft_font_create( globals );
          globals->hb_buf  = hb( buffer_create )();
    
          af_parse_gsub( globals );
        }
        else
        {
          globals->hb_font = NULL;
          globals->hb_buf  = NULL;
    
          globals->gsub                          = NULL;
          globals->gsub_lookups_single_alternate = NULL;
        }
    #endif
    
        error = af_face_globals_compute_style_coverage( globals );
        if ( error )
        {
          af_face_globals_free( globals );
          globals = NULL;
        }
        else
          globals->increase_x_height = AF_PROP_INCREASE_X_HEIGHT_MAX;
    
      Exit:
        *aglobals = globals;
        return error;
      }
    
    
      FT_LOCAL_DEF( void )
      af_face_globals_free( void*  globals_ )
      {
        AF_FaceGlobals  globals = (AF_FaceGlobals)globals_;
    
    
        if ( globals )
        {
          FT_Memory  memory = globals->face->memory;
          FT_UInt    nn;
    
    
          for ( nn = 0; nn < AF_STYLE_MAX; nn++ )
          {
            if ( globals->metrics[nn] )
            {
              AF_StyleClass          style_class =
                af_style_classes[nn];
              AF_WritingSystemClass  writing_system_class =
                af_writing_system_classes[style_class->writing_system];
    
    
              if ( writing_system_class->style_metrics_done )
                writing_system_class->style_metrics_done( globals->metrics[nn] );
    
              FT_FREE( globals->metrics[nn] );
            }
          }
    
    #ifdef FT_CONFIG_OPTION_USE_HARFBUZZ
          if ( ft_hb_enabled ( globals ) )
          {
            hb( font_destroy )( globals->hb_font );
            hb( buffer_destroy )( globals->hb_buf );
    
            FT_FREE( globals->gsub );
            FT_FREE( globals->gsub_lookups_single_alternate );
          }
    #endif
    
          /* no need to free `globals->glyph_styles'; */
          /* it is part of the `globals' array        */
          FT_FREE( globals );
        }
      }
    
    
      FT_LOCAL_DEF( FT_Error )
      af_face_globals_get_metrics( AF_FaceGlobals    globals,
                                   FT_UInt           gindex,
                                   FT_UInt           options,
                                   AF_StyleMetrics  *ametrics )
      {
        AF_StyleMetrics  metrics = NULL;
    
        AF_Style               style = (AF_Style)options;
        AF_WritingSystemClass  writing_system_class;
        AF_StyleClass          style_class;
    
        FT_Error  error = FT_Err_Ok;
    
    
        if ( gindex >= globals->glyph_count )
        {
          error = FT_THROW( Invalid_Argument );
          goto Exit;
        }
    
        /* if we have a forced style (via `options'), use it, */
        /* otherwise look into `glyph_styles' array           */
        if ( style == AF_STYLE_NONE_DFLT || style + 1 >= AF_STYLE_MAX )
          style = (AF_Style)( globals->glyph_styles[gindex] &
                              AF_STYLE_UNASSIGNED           );
    
      Again:
        style_class          = af_style_classes[style];
        writing_system_class = af_writing_system_classes
                                 [style_class->writing_system];
    
        metrics = globals->metrics[style];
        if ( !metrics )
        {
          /* create the global metrics object if necessary */
          FT_Memory  memory = globals->face->memory;
    
    
          if ( FT_ALLOC( metrics, writing_system_class->style_metrics_size ) )
            goto Exit;
    
          metrics->style_class = style_class;
          metrics->globals     = globals;
    
          if ( writing_system_class->style_metrics_init )
          {
            error = writing_system_class->style_metrics_init( metrics,
                                                              globals->face );
            if ( error )
            {
              if ( writing_system_class->style_metrics_done )
                writing_system_class->style_metrics_done( metrics );
    
              FT_FREE( metrics );
    
              /* internal error code -1 indicates   */
              /* that no blue zones have been found */
              if ( error == -1 )
              {
                style = (AF_Style)( globals->glyph_styles[gindex] &
                                    AF_STYLE_UNASSIGNED           );
                /* IMPORTANT: Clear the error code, see
                 * https://gitlab.freedesktop.org/freetype/freetype/-/issues/1063
                 */
                error = FT_Err_Ok;
                goto Again;
              }
    
              goto Exit;
            }
          }
    
          globals->metrics[style] = metrics;
        }
    
      Exit:
        *ametrics = metrics;
    
        return error;
      }
    
    
      FT_LOCAL_DEF( FT_Bool )
      af_face_globals_is_digit( AF_FaceGlobals  globals,
                                FT_UInt         gindex )
      {
        if ( gindex < globals->glyph_count )
          return FT_BOOL( globals->glyph_styles[gindex] & AF_DIGIT );
    
        return FT_BOOL( 0 );
      }
    
    
    /* END */