hb-subset-plan.cc 5.7 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 31
#include "hb-ot-cmap-table.hh"

B
Behdad Esfahbod 已提交
32 33 34 35 36 37 38
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;
39 40
}

41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58
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;
}

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

R
Rod Sheeter 已提交
75 76 77 78 79 80 81 82
hb_bool_t
hb_subset_plan_add_table (hb_subset_plan_t *plan,
                          hb_tag_t tag,
                          hb_blob_t *contents)
{
  return hb_subset_face_add_table(plan->dest, tag, contents);
}

83
static void
G
Garret Rieger 已提交
84
_populate_codepoints (hb_set_t *input_codepoints,
85
                      hb_prealloced_array_t<hb_codepoint_t>& plan_codepoints)
G
Garret Rieger 已提交
86 87
{
  plan_codepoints.alloc (hb_set_get_population (input_codepoints));
88
  hb_codepoint_t cp = -1;
G
Garret Rieger 已提交
89
  while (hb_set_next (input_codepoints, &cp)) {
90 91 92
    hb_codepoint_t *wr = plan_codepoints.push();
    *wr = cp;
  }
93
  plan_codepoints.qsort (_hb_codepoint_t_cmp);
94 95
}

96
static void
G
Garret Rieger 已提交
97
_populate_gids_to_retain (hb_face_t *face,
98 99 100
                          hb_prealloced_array_t<hb_codepoint_t>& codepoints,
                          hb_prealloced_array_t<hb_codepoint_t>& old_gids,
                          hb_prealloced_array_t<hb_codepoint_t>& old_gids_sorted)
G
Garret Rieger 已提交
101
{
102 103
  OT::cmap::accelerator_t cmap;
  cmap.init (face);
104

105
  hb_auto_array_t<unsigned int> bad_indices;
106

G
Garret Rieger 已提交
107
  old_gids.alloc (codepoints.len);
108
  bool has_zero = false;
109
  for (unsigned int i = 0; i < codepoints.len; i++) {
110
    hb_codepoint_t gid;
G
Garret Rieger 已提交
111
    if (!cmap.get_nominal_glyph (codepoints[i], &gid)) {
112
      gid = -1;
G
Garret Rieger 已提交
113
      *(bad_indices.push ()) = i;
114
    }
115 116 117
    if (gid == 0) {
      has_zero = true;
    }
G
Garret Rieger 已提交
118
    *(old_gids.push ()) = gid;
119 120
  }

R
Rod Sheeter 已提交
121
  /* Generally there shouldn't be any */
122 123
  while (bad_indices.len > 0) {
    unsigned int i = bad_indices[bad_indices.len - 1];
G
Garret Rieger 已提交
124
    bad_indices.pop ();
125
    DEBUG_MSG(SUBSET, nullptr, "Drop U+%04X; no gid", codepoints[i]);
G
Garret Rieger 已提交
126 127
    codepoints.remove (i);
    old_gids.remove (i);
128
  }
129

130 131 132 133 134 135 136 137 138 139
  // Populate a second glyph id array that is sorted by glyph id
  // and is gauranteed to contain 0.
  old_gids_sorted.alloc (old_gids.len + (has_zero ? 0 : 1));
  for (unsigned int i = 0; i < old_gids.len; i++) {
    *(old_gids_sorted.push ()) = old_gids[i];
  }
  if (!has_zero)
    *(old_gids_sorted.push ()) = 0;
  old_gids_sorted.qsort (_hb_codepoint_t_cmp);

140 141
  // TODO(Q1) expand with glyphs that make up complex glyphs
  // TODO expand with glyphs reached by G*
142 143
  //
  cmap.fini ();
G
Garret Rieger 已提交
144
}
145 146 147 148 149 150 151

/**
 * 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.
 *
152
 * Return value: New subset plan.
153 154 155 156
 *
 * Since: 1.7.5
 **/
hb_subset_plan_t *
157
hb_subset_plan_create (hb_face_t           *face,
158 159 160
                       hb_subset_profile_t *profile,
                       hb_subset_input_t   *input)
{
G
Garret Rieger 已提交
161
  hb_subset_plan_t *plan = hb_object_create<hb_subset_plan_t> ();
162 163 164 165

  plan->codepoints.init();
  plan->gids_to_retain.init();
  plan->gids_to_retain_sorted.init();
166
  plan->source = hb_face_reference (face);
R
Rod Sheeter 已提交
167
  plan->dest = hb_subset_face_create ();
168

169
  _populate_codepoints (input->unicodes, plan->codepoints);
170 171 172 173
  _populate_gids_to_retain (face,
                            plan->codepoints,
                            plan->gids_to_retain,
                            plan->gids_to_retain_sorted);
R
Rod Sheeter 已提交
174

G
Garret Rieger 已提交
175
  return plan;
176 177 178 179 180 181 182 183 184 185 186 187
}

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

G
Garret Rieger 已提交
188 189
  plan->codepoints.finish ();
  plan->gids_to_retain.finish ();
190
  plan->gids_to_retain_sorted.finish ();
191

192 193 194
  hb_face_destroy (plan->source);
  hb_face_destroy (plan->dest);

195 196
  free (plan);
}