hb-subset-plan.cc 6.6 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
 */

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

29
#include "hb-subset-plan.hh"
30
#include "hb-ot-cmap-table.hh"
31
#include "hb-ot-glyf-table.hh"
32

B
Behdad Esfahbod 已提交
33 34 35 36 37 38 39
static int
_hb_codepoint_t_cmp (const void *pa, const void *pb)
{
  hb_codepoint_t a = * (hb_codepoint_t *) pa;
  hb_codepoint_t b = * (hb_codepoint_t *) pb;

  return a < b ? -1 : a > b ? +1 : 0;
40 41
}

42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59
hb_bool_t
hb_subset_plan_new_gid_for_codepoint (hb_subset_plan_t *plan,
                                      hb_codepoint_t codepoint,
                                      hb_codepoint_t *new_gid)
{
  // TODO actual map, delete this garbage.
  for (unsigned int i = 0; i < plan->codepoints.len; i++)
  {
    if (plan->codepoints[i] != codepoint) continue;
    if (!hb_subset_plan_new_gid_for_old_id(plan, plan->gids_to_retain[i], new_gid))
    {
      return false;
    }
    return true;
  }
  return false;
}

60
hb_bool_t
G
Garret Rieger 已提交
61 62 63 64
hb_subset_plan_new_gid_for_old_id (hb_subset_plan_t *plan,
                                   hb_codepoint_t old_gid,
                                   hb_codepoint_t *new_gid)
{
65
  // the index in old_gids is the new gid; only up to codepoints.len are valid
66 67 68 69 70
  for (unsigned int i = 0; i < plan->gids_to_retain_sorted.len; i++)
  {
    if (plan->gids_to_retain_sorted[i] == old_gid)
    {
      *new_gid = i;
71 72 73 74 75 76
      return true;
    }
  }
  return false;
}

R
Rod Sheeter 已提交
77 78 79 80 81
hb_bool_t
hb_subset_plan_add_table (hb_subset_plan_t *plan,
                          hb_tag_t tag,
                          hb_blob_t *contents)
{
82 83 84
  hb_blob_t *source_blob = plan->source->reference_table (tag);
  DEBUG_MSG(SUBSET, nullptr, "add table %c%c%c%c, dest %d bytes, source %d bytes", HB_UNTAG(tag), hb_blob_get_length (contents), hb_blob_get_length (source_blob));
  hb_blob_destroy (source_blob);
R
Rod Sheeter 已提交
85 86 87
  return hb_subset_face_add_table(plan->dest, tag, contents);
}

88
static void
G
Garret Rieger 已提交
89
_populate_codepoints (hb_set_t *input_codepoints,
90
                      hb_vector_t<hb_codepoint_t>& plan_codepoints)
G
Garret Rieger 已提交
91 92
{
  plan_codepoints.alloc (hb_set_get_population (input_codepoints));
93
  hb_codepoint_t cp = -1;
G
Garret Rieger 已提交
94
  while (hb_set_next (input_codepoints, &cp)) {
95 96 97
    hb_codepoint_t *wr = plan_codepoints.push();
    *wr = cp;
  }
98
  plan_codepoints.qsort (_hb_codepoint_t_cmp);
99 100
}

101
static void
102 103 104
_add_gid_and_children (const OT::glyf::accelerator_t &glyf,
		       hb_codepoint_t gid,
		       hb_set_t *gids_to_retain)
105 106 107 108 109 110 111
{
  if (hb_set_has (gids_to_retain, gid))
    // Already visited this gid, ignore.
    return;

  hb_set_add (gids_to_retain, gid);

112
  OT::glyf::CompositeGlyphHeader::Iterator composite;
113 114 115 116
  if (glyf.get_composite (gid, &composite))
  {
    do
    {
117 118
      _add_gid_and_children (glyf, (hb_codepoint_t) composite.current->glyphIndex, gids_to_retain);
    } while (composite.move_to_next());
119 120 121
  }
}

122
static void
G
Garret Rieger 已提交
123
_populate_gids_to_retain (hb_face_t *face,
124 125 126
                          hb_vector_t<hb_codepoint_t>& codepoints,
                          hb_vector_t<hb_codepoint_t>& old_gids,
                          hb_vector_t<hb_codepoint_t>& old_gids_sorted)
G
Garret Rieger 已提交
127
{
128
  OT::cmap::accelerator_t cmap;
129
  OT::glyf::accelerator_t glyf;
130
  cmap.init (face);
131
  glyf.init (face);
132

133
  hb_auto_array_t<unsigned int> bad_indices;
134

G
Garret Rieger 已提交
135
  old_gids.alloc (codepoints.len);
B
Behdad Esfahbod 已提交
136 137
  for (unsigned int i = 0; i < codepoints.len; i++)
  {
138
    hb_codepoint_t gid;
B
Behdad Esfahbod 已提交
139 140
    if (!cmap.get_nominal_glyph (codepoints[i], &gid))
    {
141
      gid = -1;
G
Garret Rieger 已提交
142
      *(bad_indices.push ()) = i;
143
    }
G
Garret Rieger 已提交
144
    *(old_gids.push ()) = gid;
145 146
  }

R
Rod Sheeter 已提交
147
  /* Generally there shouldn't be any */
B
Behdad Esfahbod 已提交
148 149
  while (bad_indices.len > 0)
  {
150
    unsigned int i = bad_indices[bad_indices.len - 1];
G
Garret Rieger 已提交
151
    bad_indices.pop ();
152
    DEBUG_MSG(SUBSET, nullptr, "Drop U+%04X; no gid", codepoints[i]);
G
Garret Rieger 已提交
153 154
    codepoints.remove (i);
    old_gids.remove (i);
155
  }
156

157 158 159 160 161 162 163 164 165
  // Populate a full set of glyphs to retain by adding all referenced
  // composite glyphs.
  // TODO expand with glyphs reached by G*
  hb_set_t * all_gids_to_retain = hb_set_create ();
  _add_gid_and_children (glyf, 0, all_gids_to_retain);
  for (unsigned int i = 0; i < old_gids.len; i++)
    _add_gid_and_children (glyf, old_gids[i], all_gids_to_retain);

  // Transfer to a sorted list.
166
  old_gids_sorted.alloc (hb_set_get_population (all_gids_to_retain));
B
Behdad Esfahbod 已提交
167
  hb_codepoint_t gid = HB_SET_VALUE_INVALID;
168 169
  while (hb_set_next (all_gids_to_retain, &gid))
    *(old_gids_sorted.push ()) = gid;
170

171
  hb_set_destroy (all_gids_to_retain);
172
  glyf.fini ();
173
  cmap.fini ();
G
Garret Rieger 已提交
174
}
175 176 177 178 179 180 181

/**
 * 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.
 *
182
 * Return value: New subset plan.
183 184 185 186
 *
 * Since: 1.7.5
 **/
hb_subset_plan_t *
187
hb_subset_plan_create (hb_face_t           *face,
188 189 190
                       hb_subset_profile_t *profile,
                       hb_subset_input_t   *input)
{
G
Garret Rieger 已提交
191
  hb_subset_plan_t *plan = hb_object_create<hb_subset_plan_t> ();
192 193 194 195

  plan->codepoints.init();
  plan->gids_to_retain.init();
  plan->gids_to_retain_sorted.init();
196
  plan->source = hb_face_reference (face);
R
Rod Sheeter 已提交
197
  plan->dest = hb_subset_face_create ();
198
  plan->drop_hints = input->drop_hints;
199

200
  _populate_codepoints (input->unicodes, plan->codepoints);
201 202 203 204
  _populate_gids_to_retain (face,
                            plan->codepoints,
                            plan->gids_to_retain,
                            plan->gids_to_retain_sorted);
R
Rod Sheeter 已提交
205

G
Garret Rieger 已提交
206
  return plan;
207 208 209 210 211 212 213 214 215 216 217 218
}

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

B
Behdad Esfahbod 已提交
219 220 221
  plan->codepoints.fini ();
  plan->gids_to_retain.fini ();
  plan->gids_to_retain_sorted.fini ();
222

223 224 225
  hb_face_destroy (plan->source);
  hb_face_destroy (plan->dest);

226 227
  free (plan);
}