shithub: freetype+ttf2subf

Download patch

ref: 54b332aaf95cf581fc3a967bdf3724a69eba75f8
parent: c07ca2a1b3cb5057b7cb98f5b15c0704374d7a47
author: Werner Lemberg <[email protected]>
date: Wed Jun 13 04:46:27 EDT 2018

[sfnt] Separate `CPAL' and `COLR' table handling.

Later on we want to support the `SVG' table also, which needs `CPAL'
(but not `COLR').

* include/freetype/internal/sfnt.h (SFNT_Interface): Add `load_cpal'
and `free_cpal' fields.
(FT_DEFINE_SFNT_INTERFACE): Updated.

* include/freetype/internal/tttypes.h (TT_FaceRec): Replace
`colr_and_cpal' fields with `cpal' and `colr'.

* src/sfnt/sfdriver.c (sfnt_interface): Updated.

* src/sfnt/sfobjs.c (sfnt_load_face, sfnt_done_face): Updated.

* src/sfnt/ttcolr.c (Colr, Cpal): Add `table' field.
(ColrCpal): Removed.
(tt_face_load_colr): Split off CPAL handling into...
(tt_face_load_cpal): ... this new function.
(tt_face_free_colr): Split off CPAL handling into...
(tt_face_free_cpal): ... this new function.
(tt_face_load_colr_layers, tt_face_palette_set): Updated.

* src/sfnt/ttcolr.h: Updated.

* src/truetype/ttgload.c (TT_Load_Glyph): Updated.

git/fs: mount .git/fs: mount/attach disallowed
--- a/ChangeLog
+++ b/ChangeLog
@@ -1,3 +1,33 @@
+2018-06-13  Werner Lemberg  <[email protected]>
+
+	[sfnt] Separate `CPAL' and `COLR' table handling.
+
+	Later on we want to support the `SVG' table also, which needs `CPAL'
+	(but not `COLR').
+
+	* include/freetype/internal/sfnt.h (SFNT_Interface): Add `load_cpal'
+	and `free_cpal' fields.
+	(FT_DEFINE_SFNT_INTERFACE): Updated.
+
+	* include/freetype/internal/tttypes.h (TT_FaceRec): Replace
+	`colr_and_cpal' fields with `cpal' and `colr'.
+
+	* src/sfnt/sfdriver.c (sfnt_interface): Updated.
+
+	* src/sfnt/sfobjs.c (sfnt_load_face, sfnt_done_face): Updated.
+
+	* src/sfnt/ttcolr.c (Colr, Cpal): Add `table' field.
+	(ColrCpal): Removed.
+	(tt_face_load_colr): Split off CPAL handling into...
+	(tt_face_load_cpal): ... this new function.
+	(tt_face_free_colr): Split off CPAL handling into...
+	(tt_face_free_cpal): ... this new function.
+	(tt_face_load_colr_layers, tt_face_palette_set): Updated.
+
+	* src/sfnt/ttcolr.h: Updated.
+
+	* src/truetype/ttgload.c (TT_Load_Glyph): Updated.
+
 2018-06-12  Werner Lemberg  <[email protected]>
 
 	[sfnt] Fix `sizeof' thinko.
--- a/include/freetype/internal/sfnt.h
+++ b/include/freetype/internal/sfnt.h
@@ -763,7 +763,9 @@
     TT_Set_SBit_Strike_Func      set_sbit_strike;
     TT_Load_Strike_Metrics_Func  load_strike_metrics;
 
+    TT_Load_Table_Func           load_cpal;
     TT_Load_Table_Func           load_colr;
+    TT_Free_Table_Func           free_cpal;
     TT_Free_Table_Func           free_colr;
     TT_Set_Palette_Func          set_palette;
     TT_Load_Colr_Layer_Func      load_colr_layer;
@@ -811,7 +813,9 @@
           free_eblc_,                    \
           set_sbit_strike_,              \
           load_strike_metrics_,          \
+          load_cpal_,                    \
           load_colr_,                    \
+          free_cpal_,                    \
           free_colr_,                    \
           set_palette_,                  \
           load_colr_layer_,              \
@@ -849,7 +853,9 @@
     free_eblc_,                          \
     set_sbit_strike_,                    \
     load_strike_metrics_,                \
+    load_cpal_,                          \
     load_colr_,                          \
+    free_cpal_,                          \
     free_colr_,                          \
     set_palette_,                        \
     load_colr_layer_,                    \
--- a/include/freetype/internal/tttypes.h
+++ b/include/freetype/internal/tttypes.h
@@ -1546,11 +1546,14 @@
    *     exposed by the API and the indices used in
    *     the font's sbit table.
    *
-   *   colr_and_cpal ::
-   *     A pointer to data related to `COLR' and
-   *     `CPAL' tables.  NULL if tables are not
-   *     available.
+   *   cpal ::
+   *     A pointer to data related to the `CPAL' table.  NULL if the table
+   *     is not available.
    *
+   *   colr ::
+   *     A pointer to data related to the `COLR' table.  NULL if the table
+   *     is not available.
+   *
    *   kern_table ::
    *     A pointer to the `kern' table.
    *
@@ -1780,7 +1783,8 @@
 #endif
 
     /* since 2.10 */
-    void*                 colr_and_cpal;
+    void*                 cpal;
+    void*                 colr;
 
   } TT_FaceRec;
 
--- a/src/sfnt/sfdriver.c
+++ b/src/sfnt/sfdriver.c
@@ -1257,8 +1257,12 @@
     PUT_EMBEDDED_BITMAPS( tt_face_load_strike_metrics ),
                    /* TT_Load_Strike_Metrics_Func  load_strike_metrics */
 
+    PUT_COLOR_LAYERS( tt_face_load_cpal ),
+                            /* TT_Load_Table_Func      load_cpal       */
     PUT_COLOR_LAYERS( tt_face_load_colr ),
                             /* TT_Load_Table_Func      load_colr       */
+    PUT_COLOR_LAYERS( tt_face_free_cpal ),
+                            /* TT_Free_Table_Func      free_cpal       */
     PUT_COLOR_LAYERS( tt_face_free_colr ),
                             /* TT_Free_Table_Func      free_colr       */
     PUT_COLOR_LAYERS( tt_face_palette_set ),
--- a/src/sfnt/sfobjs.c
+++ b/src/sfnt/sfobjs.c
@@ -1341,9 +1341,10 @@
     if ( sfnt->load_eblc )
       LOAD_( eblc );
 
-    if ( sfnt->load_colr )
+    /* colored glyph support */
+    if ( sfnt->load_cpal )
     {
-      /* Ignore error.  Missing optional colr/cpal is okay. */
+      LOAD_( cpal );
       LOAD_( colr );
     }
 
@@ -1401,7 +1402,7 @@
        */
       if ( face->sbit_table_type == TT_SBIT_TABLE_TYPE_CBLC ||
            face->sbit_table_type == TT_SBIT_TABLE_TYPE_SBIX ||
-           face->colr_and_cpal                              )
+           face->colr                                       )
         flags |= FT_FACE_FLAG_COLOR;      /* color glyphs */
 
       if ( has_outline == TRUE )
@@ -1746,8 +1747,11 @@
         sfnt->free_eblc( face );
 
       /* destroy color table data if it is loaded */
-      if ( sfnt->free_colr )
+      if ( sfnt->free_cpal )
+      {
+        sfnt->free_cpal( face );
         sfnt->free_colr( face );
+      }
     }
 
 #ifdef TT_CONFIG_OPTION_BDF
--- a/src/sfnt/ttcolr.c
+++ b/src/sfnt/ttcolr.c
@@ -57,18 +57,6 @@
   } BaseGlyphRecord;
 
 
-  typedef struct Colr_
-  {
-    FT_UShort  version;
-    FT_UShort  num_base_glyphs;
-    FT_UShort  num_layers;
-
-    FT_Byte*  base_glyphs;
-    FT_Byte*  layers;
-
-  } Colr;
-
-
   /* all data from `CPAL' not covered in FT_Palette_Data */
   typedef struct Cpal_
   {
@@ -79,22 +67,27 @@
     FT_Byte*  color_indices; /* Index of each palette's first color record */
                              /* in the combined color record array.        */
 
+    /* The memory which backs up the `CPAL' table. */
+    void*  table;
+
   } Cpal;
 
 
-  typedef struct ColrCpal_
+  typedef struct Colr_
   {
-    /* Accessors into the colr/cpal tables. */
-    Colr  colr;
-    Cpal  cpal;
+    FT_UShort  version;
+    FT_UShort  num_base_glyphs;
+    FT_UShort  num_layers;
 
-    /* The memory which backs up colr/cpal tables. */
-    void*  colr_table;
-    void*  cpal_table;
+    FT_Byte*  base_glyphs;
+    FT_Byte*  layers;
 
-  } ColrCpal;
+    /* The memory which backs up the `COLR' table. */
+    void*  table;
 
+  } Colr;
 
+
   /**************************************************************************
    *
    * The macro FT_COMPONENT is used in trace mode.  It is an implicit
@@ -106,101 +99,55 @@
 
 
   FT_LOCAL_DEF( FT_Error )
-  tt_face_load_colr( TT_Face    face,
+  tt_face_load_cpal( TT_Face    face,
                      FT_Stream  stream )
   {
     FT_Error   error;
     FT_Memory  memory = face->root.memory;
 
-    FT_Byte*  colr_table = NULL;
-    FT_Byte*  cpal_table = NULL;
-    FT_Byte*  p          = NULL;
+    FT_Byte*  table = NULL;
+    FT_Byte*  p     = NULL;
 
-    Colr       colr;
-    Cpal       cpal;
-    ColrCpal*  cc = NULL;
+    Cpal*  cpal = NULL;
 
-    FT_ULong  base_glyph_offset, layer_offset, colors_offset;
+    FT_ULong  colors_offset;
     FT_ULong  table_size;
 
 
-    /*
-     * COLR
-     */
-
-    error = face->goto_table( face, TTAG_COLR, stream, &table_size );
-    if ( error )
-      goto NoColor;
-
-    if ( table_size < COLR_HEADER_SIZE )
-      goto InvalidTable;
-
-    if ( FT_FRAME_EXTRACT( table_size, colr_table ) )
-      goto NoColor;
-
-    p = colr_table;
-
-    FT_ZERO( &colr );
-    colr.version = FT_NEXT_USHORT( p );
-    if ( colr.version != 0 )
-      goto InvalidTable;
-
-    colr.num_base_glyphs = FT_NEXT_USHORT( p );
-    base_glyph_offset    = FT_NEXT_ULONG( p );
-
-    if ( base_glyph_offset >= table_size )
-      goto InvalidTable;
-    if ( colr.num_base_glyphs * BASE_GLYPH_SIZE >
-           table_size - base_glyph_offset )
-      goto InvalidTable;
-
-    layer_offset    = FT_NEXT_ULONG( p );
-    colr.num_layers = FT_NEXT_USHORT( p );
-
-    if ( layer_offset >= table_size )
-      goto InvalidTable;
-    if ( colr.num_layers * LAYER_SIZE > table_size - layer_offset )
-      goto InvalidTable;
-
-    colr.base_glyphs = (FT_Byte*)( colr_table + base_glyph_offset );
-    colr.layers      = (FT_Byte*)( colr_table + layer_offset      );
-
-    /*
-     * CPAL
-     */
-
     error = face->goto_table( face, TTAG_CPAL, stream, &table_size );
     if ( error )
-      goto NoColor;
+      goto NoCpal;
 
     if ( table_size < CPAL_V0_HEADER_BASE_SIZE )
       goto InvalidTable;
 
-    if ( FT_FRAME_EXTRACT( table_size, cpal_table ) )
-      goto NoColor;
+    if ( FT_FRAME_EXTRACT( table_size, table ) )
+      goto NoCpal;
 
-    p = cpal_table;
+    p = table;
 
-    FT_ZERO( &cpal );
-    cpal.version = FT_NEXT_USHORT( p );
-    if ( cpal.version > 1 )
+    if ( FT_NEW( cpal ) )
+      goto NoCpal;
+
+    cpal->version = FT_NEXT_USHORT( p );
+    if ( cpal->version > 1 )
       goto InvalidTable;
 
     face->palette_data.num_palette_entries = FT_NEXT_USHORT( p );
     face->palette_data.num_palettes        = FT_NEXT_USHORT( p );
 
-    cpal.num_colors = FT_NEXT_USHORT( p );
-    colors_offset   = FT_NEXT_ULONG( p );
+    cpal->num_colors = FT_NEXT_USHORT( p );
+    colors_offset    = FT_NEXT_ULONG( p );
 
     if ( colors_offset >= table_size )
       goto InvalidTable;
-    if ( cpal.num_colors * COLOR_SIZE > table_size - colors_offset )
+    if ( cpal->num_colors * COLOR_SIZE > table_size - colors_offset )
       goto InvalidTable;
 
-    cpal.color_indices = p;
-    cpal.colors        = (FT_Byte*)( cpal_table + colors_offset );
+    cpal->color_indices = p;
+    cpal->colors        = (FT_Byte*)( table + colors_offset );
 
-    if ( cpal.version == 1 )
+    if ( cpal->version == 1 )
     {
       FT_ULong    type_offset, label_offset, entry_label_offset;
       FT_UShort*  array = NULL;
@@ -223,9 +170,9 @@
           goto InvalidTable;
 
         if ( FT_QNEW_ARRAY( array, face->palette_data.num_palettes ) )
-          goto NoColor;
+          goto NoCpal;
 
-        p     = cpal_table + type_offset;
+        p     = table + type_offset;
         q     = array;
         limit = q + face->palette_data.num_palettes;
 
@@ -244,9 +191,9 @@
           goto InvalidTable;
 
         if ( FT_QNEW_ARRAY( array, face->palette_data.num_palettes ) )
-          goto NoColor;
+          goto NoCpal;
 
-        p     = cpal_table + label_offset;
+        p     = table + label_offset;
         q     = array;
         limit = q + face->palette_data.num_palettes;
 
@@ -265,9 +212,9 @@
           goto InvalidTable;
 
         if ( FT_QNEW_ARRAY( array, face->palette_data.num_palette_entries ) )
-          goto NoColor;
+          goto NoCpal;
 
-        p     = cpal_table + entry_label_offset;
+        p     = table + entry_label_offset;
         q     = array;
         limit = q + face->palette_data.num_palette_entries;
 
@@ -278,20 +225,14 @@
       }
     }
 
-    if ( FT_NEW( cc ) )
-      goto NoColor;
+    cpal->table = table;
 
-    cc->colr       = colr;
-    cc->cpal       = cpal;
-    cc->colr_table = colr_table;
-    cc->cpal_table = cpal_table;
+    face->cpal = cpal;
 
-    face->colr_and_cpal = cc;
-
     /* set up default palette */
     if ( FT_NEW_ARRAY( face->palette,
                        face->palette_data.num_palette_entries ) )
-      goto NoColor;
+      goto NoCpal;
 
     tt_face_palette_set( face, 0 );
 
@@ -300,12 +241,10 @@
   InvalidTable:
     error = FT_THROW( Invalid_Table );
 
-  NoColor:
-    FT_FRAME_RELEASE( colr_table );
-    FT_FRAME_RELEASE( cpal_table );
+  NoCpal:
+    FT_FRAME_RELEASE( table );
+    FT_FREE( cpal );
 
-    FT_FREE( cc );
-
     /* arrays in `face->palette_data' and `face->palette' */
     /* are freed in `sfnt_done_face'                      */
 
@@ -313,21 +252,111 @@
   }
 
 
+  FT_LOCAL_DEF( FT_Error )
+  tt_face_load_colr( TT_Face    face,
+                     FT_Stream  stream )
+  {
+    FT_Error   error;
+    FT_Memory  memory = face->root.memory;
+
+    FT_Byte*  table = NULL;
+    FT_Byte*  p     = NULL;
+
+    Colr*  colr = NULL;
+
+    FT_ULong  base_glyph_offset, layer_offset;
+    FT_ULong  table_size;
+
+
+    /* `COLR' always needs `CPAL' */
+    if ( !face->cpal )
+      return FT_THROW( Invalid_File_Format );
+
+    error = face->goto_table( face, TTAG_COLR, stream, &table_size );
+    if ( error )
+      goto NoColr;
+
+    if ( table_size < COLR_HEADER_SIZE )
+      goto InvalidTable;
+
+    if ( FT_FRAME_EXTRACT( table_size, table ) )
+      goto NoColr;
+
+    p = table;
+
+    if ( FT_NEW( colr ) )
+      goto NoColr;
+
+    colr->version = FT_NEXT_USHORT( p );
+    if ( colr->version != 0 )
+      goto InvalidTable;
+
+    colr->num_base_glyphs = FT_NEXT_USHORT( p );
+    base_glyph_offset     = FT_NEXT_ULONG( p );
+
+    if ( base_glyph_offset >= table_size )
+      goto InvalidTable;
+    if ( colr->num_base_glyphs * BASE_GLYPH_SIZE >
+           table_size - base_glyph_offset )
+      goto InvalidTable;
+
+    layer_offset     = FT_NEXT_ULONG( p );
+    colr->num_layers = FT_NEXT_USHORT( p );
+
+    if ( layer_offset >= table_size )
+      goto InvalidTable;
+    if ( colr->num_layers * LAYER_SIZE > table_size - layer_offset )
+      goto InvalidTable;
+
+    colr->base_glyphs = (FT_Byte*)( table + base_glyph_offset );
+    colr->layers      = (FT_Byte*)( table + layer_offset      );
+    colr->table       = table;
+
+    face->colr = colr;
+
+    return FT_Err_Ok;
+
+  InvalidTable:
+    error = FT_THROW( Invalid_Table );
+
+  NoColr:
+    FT_FRAME_RELEASE( table );
+    FT_FREE( colr );
+
+    return error;
+  }
+
+
   FT_LOCAL_DEF( void )
-  tt_face_free_colr( TT_Face  face )
+  tt_face_free_cpal( TT_Face  face )
   {
     FT_Stream  stream = face->root.stream;
     FT_Memory  memory = face->root.memory;
 
-    ColrCpal*  colr_and_cpal = (ColrCpal*)face->colr_and_cpal;
+    Cpal*  cpal = (Cpal*)face->cpal;
 
 
-    if ( colr_and_cpal )
+    if ( cpal )
     {
-      FT_FRAME_RELEASE( colr_and_cpal->colr_table );
-      FT_FRAME_RELEASE( colr_and_cpal->cpal_table );
+      FT_FRAME_RELEASE( cpal->table );
+      FT_FREE( cpal );
+    }
+  }
 
-      FT_FREE( face->colr_and_cpal );
+
+  FT_LOCAL_DEF( void )
+  tt_face_free_colr( TT_Face  face )
+  {
+    FT_Stream  stream = face->root.stream;
+    FT_Memory  memory = face->root.memory;
+
+    Colr*  colr = (Colr*)face->colr;
+
+
+    if ( colr )
+    {
+      FT_FRAME_RELEASE( colr->table );
+      FT_FREE( colr );
     }
   }
 
@@ -377,8 +406,7 @@
     FT_Error   error;
     FT_Memory  memory = face->root.memory;
 
-    ColrCpal*  colr_and_cpal = (ColrCpal *)face->colr_and_cpal;
-    Colr*      colr          = &colr_and_cpal->colr;
+    Colr*  colr = (Colr*)face->colr;
 
     BaseGlyphRecord  glyph_record;
     FT_Glyph_Layer   layers = NULL;
@@ -441,8 +469,7 @@
   tt_face_palette_set( TT_Face  face,
                        FT_UInt  palette_index )
   {
-    ColrCpal*  colr_and_cpal = (ColrCpal *)face->colr_and_cpal;
-    Cpal*      cpal          = &colr_and_cpal->cpal;
+    Cpal*  cpal = (Cpal*)face->cpal;
 
     FT_Byte*   offset;
     FT_Byte*   p;
--- a/src/sfnt/ttcolr.h
+++ b/src/sfnt/ttcolr.h
@@ -30,8 +30,15 @@
 
 
   FT_LOCAL( FT_Error )
+  tt_face_load_cpal( TT_Face    face,
+                     FT_Stream  stream );
+
+  FT_LOCAL( FT_Error )
   tt_face_load_colr( TT_Face    face,
                      FT_Stream  stream );
+
+  FT_LOCAL( void )
+  tt_face_free_cpal( TT_Face  face );
 
   FT_LOCAL( void )
   tt_face_free_colr( TT_Face  face );
--- a/src/truetype/ttgload.c
+++ b/src/truetype/ttgload.c
@@ -2906,8 +2906,8 @@
 
     /* The outline based algorithm took care of metrics. */
     /* Read additional color info if requested.          */
-    if ( ( load_flags & FT_LOAD_COLOR )          &&
-         ( (TT_Face)glyph->face )->colr_and_cpal )
+    if ( ( load_flags & FT_LOAD_COLOR ) &&
+         ( (TT_Face)glyph->face )->colr )
     {
       TT_Face       face   = (TT_Face)glyph->face;
       FT_Memory     memory = face->root.memory;