hb-subset-plan.cc 4.9 KB
Newer Older
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23
/*
 * Copyright © 2018  Google, Inc.
 *
 *  This is part of HarfBuzz, a text shaping library.
 *
 * Permission is hereby granted, without written agreement and without
 * license or royalty fees, to use, copy, modify, and distribute this
 * software and its documentation for any purpose, provided that the
 * above copyright notice and the following two paragraphs appear in
 * all copies of this software.
 *
 * IN NO EVENT SHALL THE COPYRIGHT HOLDER BE LIABLE TO ANY PARTY FOR
 * DIRECT, INDIRECT, SPECIAL, INCIDENTAL, OR CONSEQUENTIAL DAMAGES
 * ARISING OUT OF THE USE OF THIS SOFTWARE AND ITS DOCUMENTATION, EVEN
 * IF THE COPYRIGHT HOLDER HAS BEEN ADVISED OF THE POSSIBILITY OF SUCH
 * DAMAGE.
 *
 * THE COPYRIGHT HOLDER SPECIFICALLY DISCLAIMS ANY WARRANTIES, INCLUDING,
 * BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND
 * FITNESS FOR A PARTICULAR PURPOSE.  THE SOFTWARE PROVIDED HEREUNDER IS
 * ON AN "AS IS" BASIS, AND THE COPYRIGHT HOLDER HAS NO OBLIGATION TO
 * PROVIDE MAINTENANCE, SUPPORT, UPDATES, ENHANCEMENTS, OR MODIFICATIONS.
 *
24
 * Google Author(s): Garret Rieger, Roderick Sheeter
25 26
 */

27
#include "hb-map-private.hh"
G
Garret Rieger 已提交
28
#include "hb-subset-private.hh"
29
#include "hb-set-private.hh"
30

31
#include "hb-subset-plan.hh"
32
#include "hb-ot-cmap-table.hh"
33
#include "hb-ot-glyf-table.hh"
34

35
static void
36 37 38
_add_gid_and_children (const OT::glyf::accelerator_t &glyf,
		       hb_codepoint_t gid,
		       hb_set_t *gids_to_retain)
39 40 41 42 43 44 45
{
  if (hb_set_has (gids_to_retain, gid))
    // Already visited this gid, ignore.
    return;

  hb_set_add (gids_to_retain, gid);

46
  OT::glyf::CompositeGlyphHeader::Iterator composite;
47 48 49 50
  if (glyf.get_composite (gid, &composite))
  {
    do
    {
51 52
      _add_gid_and_children (glyf, (hb_codepoint_t) composite.current->glyphIndex, gids_to_retain);
    } while (composite.move_to_next());
53 54 55
  }
}

56
static void
G
Garret Rieger 已提交
57
_populate_gids_to_retain (hb_face_t *face,
58 59 60 61
                          const hb_set_t *unicodes,
                          hb_set_t *unicodes_to_retain,
                          hb_map_t *codepoint_to_glyph,
                          hb_vector_t<hb_codepoint_t> *glyphs)
G
Garret Rieger 已提交
62
{
63
  OT::cmap::accelerator_t cmap;
64
  OT::glyf::accelerator_t glyf;
65
  cmap.init (face);
66
  glyf.init (face);
67

68 69
  hb_set_t *initial_gids_to_retain = hb_set_create ();
  initial_gids_to_retain->add (0); // Not-def
70

71 72
  hb_codepoint_t cp = HB_SET_VALUE_INVALID;
  while (unicodes->next (&cp))
B
Behdad Esfahbod 已提交
73
  {
74
    hb_codepoint_t gid;
75
    if (!cmap.get_nominal_glyph (cp, &gid))
B
Behdad Esfahbod 已提交
76
    {
77 78
      DEBUG_MSG(SUBSET, nullptr, "Drop U+%04X; no gid", cp);
      continue;
79
    }
80 81 82
    unicodes_to_retain->add (cp);
    codepoint_to_glyph->set (cp, gid);
    initial_gids_to_retain->add (gid);
83
  }
84

85 86 87 88
  // Populate a full set of glyphs to retain by adding all referenced
  // composite glyphs.
  // TODO expand with glyphs reached by G*

B
Behdad Esfahbod 已提交
89
  hb_codepoint_t gid = HB_SET_VALUE_INVALID;
90 91 92 93 94 95 96 97 98 99 100
  hb_set_t *all_gids_to_retain = hb_set_create ();
  while (initial_gids_to_retain->next (&gid))
  {
    _add_gid_and_children (glyf, gid, all_gids_to_retain);
  }
  hb_set_destroy (initial_gids_to_retain);

  glyphs->alloc (all_gids_to_retain->get_population ());
  gid = HB_SET_VALUE_INVALID;
  while (all_gids_to_retain->next (&gid))
    glyphs->push (gid);
101

102
  hb_set_destroy (all_gids_to_retain);
103
  glyf.fini ();
104
  cmap.fini ();
G
Garret Rieger 已提交
105
}
106

107 108 109 110 111 112 113 114 115
static void
_create_old_gid_to_new_gid_map (const hb_vector_t<hb_codepoint_t> &glyphs,
                                hb_map_t *glyph_map)
{
  for (unsigned int i = 0; i < glyphs.len; i++) {
    glyph_map->set (glyphs[i], i);
  }
}

116 117 118 119 120 121
/**
 * hb_subset_plan_create:
 * Computes a plan for subsetting the supplied face according
 * to a provide profile and input. The plan describes
 * which tables and glyphs should be retained.
 *
122
 * Return value: New subset plan.
123 124 125 126
 *
 * Since: 1.7.5
 **/
hb_subset_plan_t *
127
hb_subset_plan_create (hb_face_t           *face,
128 129 130
                       hb_subset_profile_t *profile,
                       hb_subset_input_t   *input)
{
G
Garret Rieger 已提交
131
  hb_subset_plan_t *plan = hb_object_create<hb_subset_plan_t> ();
132

133
  plan->drop_hints = input->drop_hints;
134
  plan->drop_ot_layout = input->drop_ot_layout;
135 136
  plan->unicodes = hb_set_create();
  plan->glyphs.init();
137
  plan->source = hb_face_reference (face);
R
Rod Sheeter 已提交
138
  plan->dest = hb_subset_face_create ();
139 140
  plan->codepoint_to_glyph = hb_map_create();
  plan->glyph_map = hb_map_create();
141

142
  _populate_gids_to_retain (face,
143 144 145 146 147 148
                            input->unicodes,
                            plan->unicodes,
                            plan->codepoint_to_glyph,
                            &plan->glyphs);
  _create_old_gid_to_new_gid_map (plan->glyphs,
                                  plan->glyph_map);
R
Rod Sheeter 已提交
149

G
Garret Rieger 已提交
150
  return plan;
151 152 153 154 155 156 157 158 159 160 161 162
}

/**
 * hb_subset_plan_destroy:
 *
 * Since: 1.7.5
 **/
void
hb_subset_plan_destroy (hb_subset_plan_t *plan)
{
  if (!hb_object_destroy (plan)) return;

163 164
  hb_set_destroy (plan->unicodes);
  plan->glyphs.fini();
165 166
  hb_face_destroy (plan->source);
  hb_face_destroy (plan->dest);
167 168
  hb_map_destroy (plan->codepoint_to_glyph);
  hb_map_destroy (plan->glyph_map);
169

170 171
  free (plan);
}