/work/workdir/UnpackedTarball/harfbuzz/src/hb-shape-plan.cc
Line | Count | Source (jump to first uncovered line) |
1 | | /* |
2 | | * Copyright © 2012 Google, Inc. |
3 | | * |
4 | | * This is part of HarfBuzz, a text shaping library. |
5 | | * |
6 | | * Permission is hereby granted, without written agreement and without |
7 | | * license or royalty fees, to use, copy, modify, and distribute this |
8 | | * software and its documentation for any purpose, provided that the |
9 | | * above copyright notice and the following two paragraphs appear in |
10 | | * all copies of this software. |
11 | | * |
12 | | * IN NO EVENT SHALL THE COPYRIGHT HOLDER BE LIABLE TO ANY PARTY FOR |
13 | | * DIRECT, INDIRECT, SPECIAL, INCIDENTAL, OR CONSEQUENTIAL DAMAGES |
14 | | * ARISING OUT OF THE USE OF THIS SOFTWARE AND ITS DOCUMENTATION, EVEN |
15 | | * IF THE COPYRIGHT HOLDER HAS BEEN ADVISED OF THE POSSIBILITY OF SUCH |
16 | | * DAMAGE. |
17 | | * |
18 | | * THE COPYRIGHT HOLDER SPECIFICALLY DISCLAIMS ANY WARRANTIES, INCLUDING, |
19 | | * BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND |
20 | | * FITNESS FOR A PARTICULAR PURPOSE. THE SOFTWARE PROVIDED HEREUNDER IS |
21 | | * ON AN "AS IS" BASIS, AND THE COPYRIGHT HOLDER HAS NO OBLIGATION TO |
22 | | * PROVIDE MAINTENANCE, SUPPORT, UPDATES, ENHANCEMENTS, OR MODIFICATIONS. |
23 | | * |
24 | | * Google Author(s): Behdad Esfahbod |
25 | | */ |
26 | | |
27 | | #include "hb.hh" |
28 | | #include "hb-shape-plan.hh" |
29 | | #include "hb-shaper.hh" |
30 | | #include "hb-font.hh" |
31 | | #include "hb-buffer.hh" |
32 | | |
33 | | |
34 | | #ifndef HB_NO_SHAPER |
35 | | |
36 | | /** |
37 | | * SECTION:hb-shape-plan |
38 | | * @title: hb-shape-plan |
39 | | * @short_description: Object representing a shaping plan |
40 | | * @include: hb.h |
41 | | * |
42 | | * Shape plans are an internal mechanism. Each plan contains state |
43 | | * describing how HarfBuzz will shape a particular text segment, based on |
44 | | * the combination of segment properties and the capabilities in the |
45 | | * font face in use. |
46 | | * |
47 | | * Shape plans are not used for shaping directly, but can be queried to |
48 | | * access certain information about how shaping will perform, given a set |
49 | | * of specific input parameters (script, language, direction, features, |
50 | | * etc.). |
51 | | * |
52 | | * Most client programs will not need to deal with shape plans directly. |
53 | | **/ |
54 | | |
55 | | |
56 | | /* |
57 | | * hb_shape_plan_key_t |
58 | | */ |
59 | | |
60 | | bool |
61 | | hb_shape_plan_key_t::init (bool copy, |
62 | | hb_face_t *face, |
63 | | const hb_segment_properties_t *props, |
64 | | const hb_feature_t *user_features, |
65 | | unsigned int num_user_features, |
66 | | const int *coords, |
67 | | unsigned int num_coords, |
68 | | const char * const *shaper_list) |
69 | 32.9M | { |
70 | 32.9M | hb_feature_t *features = nullptr; |
71 | 32.9M | if (copy && num_user_features && !(features = (hb_feature_t *) hb_calloc (num_user_features, sizeof (hb_feature_t)))) |
72 | 0 | goto bail; |
73 | | |
74 | 32.9M | this->props = *props; |
75 | 32.9M | this->num_user_features = num_user_features; |
76 | 32.9M | this->user_features = copy ? features : user_features; |
77 | 32.9M | if (copy && num_user_features) |
78 | 7.21k | { |
79 | 7.21k | hb_memcpy (features, user_features, num_user_features * sizeof (hb_feature_t)); |
80 | | /* Make start/end uniform to easier catch bugs. */ |
81 | 38.3k | for (unsigned int i = 0; i < num_user_features; i++) |
82 | 31.1k | { |
83 | 31.1k | if (features[0].start != HB_FEATURE_GLOBAL_START) |
84 | 114 | features[0].start = 1; |
85 | 31.1k | if (features[0].end != HB_FEATURE_GLOBAL_END) |
86 | 172 | features[0].end = 2; |
87 | 31.1k | } |
88 | 7.21k | } |
89 | 32.9M | this->shaper_func = nullptr; |
90 | 32.9M | this->shaper_name = nullptr; |
91 | 32.9M | #ifndef HB_NO_OT_SHAPE |
92 | 32.9M | this->ot.init (face, coords, num_coords); |
93 | 32.9M | #endif |
94 | | |
95 | | /* |
96 | | * Choose shaper. |
97 | | */ |
98 | | |
99 | 32.9M | #define HB_SHAPER_PLAN(shaper) \ |
100 | 65.8M | HB_STMT_START { \ |
101 | 65.8M | if (face->data.shaper) \ |
102 | 65.8M | { \ |
103 | 32.9M | this->shaper_func = _hb_##shaper##_shape; \ |
104 | 32.9M | this->shaper_name = #shaper; \ |
105 | 32.9M | return true; \ |
106 | 32.9M | } \ |
107 | 65.8M | } HB_STMT_END |
108 | | |
109 | 32.9M | if (unlikely (shaper_list)) |
110 | 32.9M | { |
111 | 65.8M | for (; *shaper_list; shaper_list++) |
112 | 65.8M | if (false) |
113 | 0 | ; |
114 | 65.8M | #define HB_SHAPER_IMPLEMENT(shaper) \ |
115 | 98.8M | else if (0 == strcmp (*shaper_list, #shaper)) \ |
116 | 131M | HB_SHAPER_PLAN (shaper); |
117 | 65.8M | #include "hb-shaper-list.hh" |
118 | 32.9M | #undef HB_SHAPER_IMPLEMENT |
119 | 32.9M | } |
120 | 0 | else |
121 | 0 | { |
122 | 0 | const HB_UNUSED hb_shaper_entry_t *shapers = _hb_shapers_get (); |
123 | 0 | for (unsigned int i = 0; i < HB_SHAPERS_COUNT; i++) |
124 | 0 | if (false) |
125 | 0 | ; |
126 | 0 | #define HB_SHAPER_IMPLEMENT(shaper) \ |
127 | 0 | else if (shapers[i].func == _hb_##shaper##_shape) \ |
128 | 0 | HB_SHAPER_PLAN (shaper); |
129 | 0 | #include "hb-shaper-list.hh" |
130 | 0 | #undef HB_SHAPER_IMPLEMENT |
131 | 0 | } |
132 | 0 | #undef HB_SHAPER_PLAN |
133 | | |
134 | 0 | bail: |
135 | 0 | ::hb_free (features); |
136 | 0 | return false; |
137 | 32.9M | } |
138 | | |
139 | | bool |
140 | | hb_shape_plan_key_t::user_features_match (const hb_shape_plan_key_t *other) |
141 | 389M | { |
142 | 389M | if (this->num_user_features != other->num_user_features) |
143 | 355M | return false; |
144 | 59.9M | for (unsigned int i = 0; i < num_user_features; i++) |
145 | 27.0M | { |
146 | 27.0M | if (this->user_features[i].tag != other->user_features[i].tag || |
147 | 27.0M | this->user_features[i].value != other->user_features[i].value || |
148 | 27.0M | (this->user_features[i].start == HB_FEATURE_GLOBAL_START && |
149 | 25.0M | this->user_features[i].end == HB_FEATURE_GLOBAL_END) != |
150 | 25.0M | (other->user_features[i].start == HB_FEATURE_GLOBAL_START && |
151 | 25.0M | other->user_features[i].end == HB_FEATURE_GLOBAL_END)) |
152 | 1.97M | return false; |
153 | 27.0M | } |
154 | 32.9M | return true; |
155 | 34.9M | } |
156 | | |
157 | | bool |
158 | | hb_shape_plan_key_t::equal (const hb_shape_plan_key_t *other) |
159 | 12.8G | { |
160 | 12.8G | return hb_segment_properties_equal (&this->props, &other->props) && |
161 | 12.8G | this->user_features_match (other) && |
162 | 12.8G | #ifndef HB_NO_OT_SHAPE |
163 | 12.8G | this->ot.equal (&other->ot) && |
164 | 12.8G | #endif |
165 | 12.8G | this->shaper_func == other->shaper_func; |
166 | 12.8G | } |
167 | | |
168 | | |
169 | | /* |
170 | | * hb_shape_plan_t |
171 | | */ |
172 | | |
173 | | |
174 | | /** |
175 | | * hb_shape_plan_create: |
176 | | * @face: #hb_face_t to use |
177 | | * @props: The #hb_segment_properties_t of the segment |
178 | | * @user_features: (array length=num_user_features): The list of user-selected features |
179 | | * @num_user_features: The number of user-selected features |
180 | | * @shaper_list: (array zero-terminated=1): List of shapers to try |
181 | | * |
182 | | * Constructs a shaping plan for a combination of @face, @user_features, @props, |
183 | | * and @shaper_list. |
184 | | * |
185 | | * Return value: (transfer full): The shaping plan |
186 | | * |
187 | | * Since: 0.9.7 |
188 | | **/ |
189 | | hb_shape_plan_t * |
190 | | hb_shape_plan_create (hb_face_t *face, |
191 | | const hb_segment_properties_t *props, |
192 | | const hb_feature_t *user_features, |
193 | | unsigned int num_user_features, |
194 | | const char * const *shaper_list) |
195 | 0 | { |
196 | 0 | return hb_shape_plan_create2 (face, props, |
197 | 0 | user_features, num_user_features, |
198 | 0 | nullptr, 0, |
199 | 0 | shaper_list); |
200 | 0 | } |
201 | | |
202 | | /** |
203 | | * hb_shape_plan_create2: |
204 | | * @face: #hb_face_t to use |
205 | | * @props: The #hb_segment_properties_t of the segment |
206 | | * @user_features: (array length=num_user_features): The list of user-selected features |
207 | | * @num_user_features: The number of user-selected features |
208 | | * @coords: (array length=num_coords): The list of variation-space coordinates |
209 | | * @num_coords: The number of variation-space coordinates |
210 | | * @shaper_list: (array zero-terminated=1): List of shapers to try |
211 | | * |
212 | | * The variable-font version of #hb_shape_plan_create. |
213 | | * Constructs a shaping plan for a combination of @face, @user_features, @props, |
214 | | * and @shaper_list, plus the variation-space coordinates @coords. |
215 | | * |
216 | | * Return value: (transfer full): The shaping plan |
217 | | * |
218 | | * Since: 1.4.0 |
219 | | **/ |
220 | | hb_shape_plan_t * |
221 | | hb_shape_plan_create2 (hb_face_t *face, |
222 | | const hb_segment_properties_t *props, |
223 | | const hb_feature_t *user_features, |
224 | | unsigned int num_user_features, |
225 | | const int *coords, |
226 | | unsigned int num_coords, |
227 | | const char * const *shaper_list) |
228 | 8.85k | { |
229 | 8.85k | DEBUG_MSG_FUNC (SHAPE_PLAN, nullptr, |
230 | 8.85k | "face=%p num_features=%u num_coords=%u shaper_list=%p", |
231 | 8.85k | face, |
232 | 8.85k | num_user_features, |
233 | 8.85k | num_coords, |
234 | 8.85k | shaper_list); |
235 | | |
236 | 8.85k | if (unlikely (!HB_DIRECTION_IS_VALID (props->direction))) |
237 | 0 | return hb_shape_plan_get_empty (); |
238 | | |
239 | 8.85k | hb_shape_plan_t *shape_plan; |
240 | | |
241 | 8.85k | if (unlikely (!props)) |
242 | 0 | goto bail; |
243 | 8.85k | if (!(shape_plan = hb_object_create<hb_shape_plan_t> ())) |
244 | 0 | goto bail; |
245 | | |
246 | 8.85k | if (unlikely (!face)) |
247 | 0 | face = hb_face_get_empty (); |
248 | 8.85k | hb_face_make_immutable (face); |
249 | 8.85k | shape_plan->face_unsafe = face; |
250 | | |
251 | 8.85k | if (unlikely (!shape_plan->key.init (true, |
252 | 8.85k | face, |
253 | 8.85k | props, |
254 | 8.85k | user_features, |
255 | 8.85k | num_user_features, |
256 | 8.85k | coords, |
257 | 8.85k | num_coords, |
258 | 8.85k | shaper_list))) |
259 | 0 | goto bail2; |
260 | 8.85k | #ifndef HB_NO_OT_SHAPE |
261 | 8.85k | if (unlikely (!shape_plan->ot.init0 (face, &shape_plan->key))) |
262 | 0 | goto bail3; |
263 | 8.85k | #endif |
264 | | |
265 | 8.85k | return shape_plan; |
266 | | |
267 | 0 | #ifndef HB_NO_OT_SHAPE |
268 | 0 | bail3: |
269 | 0 | #endif |
270 | 0 | shape_plan->key.fini (); |
271 | 0 | bail2: |
272 | 0 | hb_free (shape_plan); |
273 | 0 | bail: |
274 | 0 | return hb_shape_plan_get_empty (); |
275 | 0 | } |
276 | | |
277 | | /** |
278 | | * hb_shape_plan_get_empty: |
279 | | * |
280 | | * Fetches the singleton empty shaping plan. |
281 | | * |
282 | | * Return value: (transfer full): The empty shaping plan |
283 | | * |
284 | | * Since: 0.9.7 |
285 | | **/ |
286 | | hb_shape_plan_t * |
287 | | hb_shape_plan_get_empty () |
288 | 0 | { |
289 | 0 | return const_cast<hb_shape_plan_t *> (&Null (hb_shape_plan_t)); |
290 | 0 | } |
291 | | |
292 | | /** |
293 | | * hb_shape_plan_reference: (skip) |
294 | | * @shape_plan: A shaping plan |
295 | | * |
296 | | * Increases the reference count on the given shaping plan. |
297 | | * |
298 | | * Return value: (transfer full): @shape_plan |
299 | | * |
300 | | * Since: 0.9.7 |
301 | | **/ |
302 | | hb_shape_plan_t * |
303 | | hb_shape_plan_reference (hb_shape_plan_t *shape_plan) |
304 | 32.9M | { |
305 | 32.9M | return hb_object_reference (shape_plan); |
306 | 32.9M | } |
307 | | |
308 | | /** |
309 | | * hb_shape_plan_destroy: (skip) |
310 | | * @shape_plan: A shaping plan |
311 | | * |
312 | | * Decreases the reference count on the given shaping plan. When the |
313 | | * reference count reaches zero, the shaping plan is destroyed, |
314 | | * freeing all memory. |
315 | | * |
316 | | * Since: 0.9.7 |
317 | | **/ |
318 | | void |
319 | | hb_shape_plan_destroy (hb_shape_plan_t *shape_plan) |
320 | 32.9M | { |
321 | 32.9M | if (!hb_object_destroy (shape_plan)) return; |
322 | | |
323 | 0 | hb_free (shape_plan); |
324 | 0 | } |
325 | | |
326 | | /** |
327 | | * hb_shape_plan_set_user_data: (skip) |
328 | | * @shape_plan: A shaping plan |
329 | | * @key: The user-data key to set |
330 | | * @data: A pointer to the user data |
331 | | * @destroy: (nullable): A callback to call when @data is not needed anymore |
332 | | * @replace: Whether to replace an existing data with the same key |
333 | | * |
334 | | * Attaches a user-data key/data pair to the given shaping plan. |
335 | | * |
336 | | * Return value: `true` if success, `false` otherwise. |
337 | | * |
338 | | * Since: 0.9.7 |
339 | | **/ |
340 | | hb_bool_t |
341 | | hb_shape_plan_set_user_data (hb_shape_plan_t *shape_plan, |
342 | | hb_user_data_key_t *key, |
343 | | void * data, |
344 | | hb_destroy_func_t destroy, |
345 | | hb_bool_t replace) |
346 | 0 | { |
347 | 0 | return hb_object_set_user_data (shape_plan, key, data, destroy, replace); |
348 | 0 | } |
349 | | |
350 | | /** |
351 | | * hb_shape_plan_get_user_data: (skip) |
352 | | * @shape_plan: A shaping plan |
353 | | * @key: The user-data key to query |
354 | | * |
355 | | * Fetches the user data associated with the specified key, |
356 | | * attached to the specified shaping plan. |
357 | | * |
358 | | * Return value: (transfer none): A pointer to the user data |
359 | | * |
360 | | * Since: 0.9.7 |
361 | | **/ |
362 | | void * |
363 | | hb_shape_plan_get_user_data (const hb_shape_plan_t *shape_plan, |
364 | | hb_user_data_key_t *key) |
365 | 0 | { |
366 | 0 | return hb_object_get_user_data (shape_plan, key); |
367 | 0 | } |
368 | | |
369 | | /** |
370 | | * hb_shape_plan_get_shaper: |
371 | | * @shape_plan: A shaping plan |
372 | | * |
373 | | * Fetches the shaper from a given shaping plan. |
374 | | * |
375 | | * Return value: (transfer none): The shaper |
376 | | * |
377 | | * Since: 0.9.7 |
378 | | **/ |
379 | | const char * |
380 | | hb_shape_plan_get_shaper (hb_shape_plan_t *shape_plan) |
381 | 0 | { |
382 | 0 | return shape_plan->key.shaper_name; |
383 | 0 | } |
384 | | |
385 | | |
386 | | static bool |
387 | | _hb_shape_plan_execute_internal (hb_shape_plan_t *shape_plan, |
388 | | hb_font_t *font, |
389 | | hb_buffer_t *buffer, |
390 | | const hb_feature_t *features, |
391 | | unsigned int num_features) |
392 | 32.9M | { |
393 | 32.9M | DEBUG_MSG_FUNC (SHAPE_PLAN, shape_plan, |
394 | 32.9M | "num_features=%u shaper_func=%p, shaper_name=%s", |
395 | 32.9M | num_features, |
396 | 32.9M | shape_plan->key.shaper_func, |
397 | 32.9M | shape_plan->key.shaper_name); |
398 | | |
399 | 32.9M | if (unlikely (!buffer->len)) |
400 | 0 | return true; |
401 | | |
402 | 32.9M | assert (!hb_object_is_immutable (buffer)); |
403 | | |
404 | 32.9M | buffer->assert_unicode (); |
405 | | |
406 | 32.9M | if (unlikely (!hb_object_is_valid (shape_plan))) |
407 | 0 | return false; |
408 | | |
409 | 32.9M | assert (shape_plan->face_unsafe == font->face); |
410 | 32.9M | assert (hb_segment_properties_equal (&shape_plan->key.props, &buffer->props)); |
411 | | |
412 | 32.9M | #define HB_SHAPER_EXECUTE(shaper) \ |
413 | 32.9M | HB_STMT_START { \ |
414 | 32.9M | return font->data.shaper && \ |
415 | 32.9M | _hb_##shaper##_shape (shape_plan, font, buffer, features, num_features); \ |
416 | 32.9M | } HB_STMT_END |
417 | | |
418 | 32.9M | if (false) |
419 | 0 | ; |
420 | 32.9M | #define HB_SHAPER_IMPLEMENT(shaper) \ |
421 | 65.8M | else if (shape_plan->key.shaper_func == _hb_##shaper##_shape) \ |
422 | 65.8M | HB_SHAPER_EXECUTE (shaper); |
423 | 32.9M | #include "hb-shaper-list.hh" |
424 | 0 | #undef HB_SHAPER_IMPLEMENT |
425 | | |
426 | 0 | #undef HB_SHAPER_EXECUTE |
427 | | |
428 | 0 | return false; |
429 | 32.9M | } |
430 | | /** |
431 | | * hb_shape_plan_execute: |
432 | | * @shape_plan: A shaping plan |
433 | | * @font: The #hb_font_t to use |
434 | | * @buffer: The #hb_buffer_t to work upon |
435 | | * @features: (array length=num_features): Features to enable |
436 | | * @num_features: The number of features to enable |
437 | | * |
438 | | * Executes the given shaping plan on the specified buffer, using |
439 | | * the given @font and @features. |
440 | | * |
441 | | * Return value: `true` if success, `false` otherwise. |
442 | | * |
443 | | * Since: 0.9.7 |
444 | | **/ |
445 | | hb_bool_t |
446 | | hb_shape_plan_execute (hb_shape_plan_t *shape_plan, |
447 | | hb_font_t *font, |
448 | | hb_buffer_t *buffer, |
449 | | const hb_feature_t *features, |
450 | | unsigned int num_features) |
451 | 32.9M | { |
452 | 32.9M | bool ret = _hb_shape_plan_execute_internal (shape_plan, font, buffer, |
453 | 32.9M | features, num_features); |
454 | | |
455 | 32.9M | if (ret && buffer->content_type == HB_BUFFER_CONTENT_TYPE_UNICODE) |
456 | 0 | buffer->content_type = HB_BUFFER_CONTENT_TYPE_GLYPHS; |
457 | | |
458 | 32.9M | return ret; |
459 | 32.9M | } |
460 | | |
461 | | |
462 | | /* |
463 | | * Caching |
464 | | */ |
465 | | |
466 | | /** |
467 | | * hb_shape_plan_create_cached: |
468 | | * @face: #hb_face_t to use |
469 | | * @props: The #hb_segment_properties_t of the segment |
470 | | * @user_features: (array length=num_user_features): The list of user-selected features |
471 | | * @num_user_features: The number of user-selected features |
472 | | * @shaper_list: (array zero-terminated=1): List of shapers to try |
473 | | * |
474 | | * Creates a cached shaping plan suitable for reuse, for a combination |
475 | | * of @face, @user_features, @props, and @shaper_list. |
476 | | * |
477 | | * Return value: (transfer full): The shaping plan |
478 | | * |
479 | | * Since: 0.9.7 |
480 | | **/ |
481 | | hb_shape_plan_t * |
482 | | hb_shape_plan_create_cached (hb_face_t *face, |
483 | | const hb_segment_properties_t *props, |
484 | | const hb_feature_t *user_features, |
485 | | unsigned int num_user_features, |
486 | | const char * const *shaper_list) |
487 | 0 | { |
488 | 0 | return hb_shape_plan_create_cached2 (face, props, |
489 | 0 | user_features, num_user_features, |
490 | 0 | nullptr, 0, |
491 | 0 | shaper_list); |
492 | 0 | } |
493 | | |
494 | | /** |
495 | | * hb_shape_plan_create_cached2: |
496 | | * @face: #hb_face_t to use |
497 | | * @props: The #hb_segment_properties_t of the segment |
498 | | * @user_features: (array length=num_user_features): The list of user-selected features |
499 | | * @num_user_features: The number of user-selected features |
500 | | * @coords: (array length=num_coords): The list of variation-space coordinates |
501 | | * @num_coords: The number of variation-space coordinates |
502 | | * @shaper_list: (array zero-terminated=1): List of shapers to try |
503 | | * |
504 | | * The variable-font version of #hb_shape_plan_create_cached. |
505 | | * Creates a cached shaping plan suitable for reuse, for a combination |
506 | | * of @face, @user_features, @props, and @shaper_list, plus the |
507 | | * variation-space coordinates @coords. |
508 | | * |
509 | | * Return value: (transfer full): The shaping plan |
510 | | * |
511 | | * Since: 1.4.0 |
512 | | **/ |
513 | | hb_shape_plan_t * |
514 | | hb_shape_plan_create_cached2 (hb_face_t *face, |
515 | | const hb_segment_properties_t *props, |
516 | | const hb_feature_t *user_features, |
517 | | unsigned int num_user_features, |
518 | | const int *coords, |
519 | | unsigned int num_coords, |
520 | | const char * const *shaper_list) |
521 | 32.9M | { |
522 | 32.9M | DEBUG_MSG_FUNC (SHAPE_PLAN, nullptr, |
523 | 32.9M | "face=%p num_features=%u shaper_list=%p", |
524 | 32.9M | face, |
525 | 32.9M | num_user_features, |
526 | 32.9M | shaper_list); |
527 | | |
528 | 32.9M | retry: |
529 | 32.9M | hb_face_t::plan_node_t *cached_plan_nodes = face->shape_plans; |
530 | | |
531 | 32.9M | bool dont_cache = !hb_object_is_valid (face); |
532 | | |
533 | 32.9M | if (likely (!dont_cache)) |
534 | 32.9M | { |
535 | 32.9M | hb_shape_plan_key_t key; |
536 | 32.9M | if (!key.init (false, |
537 | 32.9M | face, |
538 | 32.9M | props, |
539 | 32.9M | user_features, |
540 | 32.9M | num_user_features, |
541 | 32.9M | coords, |
542 | 32.9M | num_coords, |
543 | 32.9M | shaper_list)) |
544 | 0 | return hb_shape_plan_get_empty (); |
545 | | |
546 | 12.8G | for (hb_face_t::plan_node_t *node = cached_plan_nodes; node; node = node->next) |
547 | 12.8G | if (node->shape_plan->key.equal (&key)) |
548 | 32.9M | { |
549 | 32.9M | DEBUG_MSG_FUNC (SHAPE_PLAN, node->shape_plan, "fulfilled from cache"); |
550 | 32.9M | return hb_shape_plan_reference (node->shape_plan); |
551 | 32.9M | } |
552 | 32.9M | } |
553 | | |
554 | 8.85k | hb_shape_plan_t *shape_plan = hb_shape_plan_create2 (face, props, |
555 | 8.85k | user_features, num_user_features, |
556 | 8.85k | coords, num_coords, |
557 | 8.85k | shaper_list); |
558 | | |
559 | 8.85k | if (unlikely (dont_cache)) |
560 | 0 | return shape_plan; |
561 | | |
562 | 8.85k | hb_face_t::plan_node_t *node = (hb_face_t::plan_node_t *) hb_calloc (1, sizeof (hb_face_t::plan_node_t)); |
563 | 8.85k | if (unlikely (!node)) |
564 | 0 | return shape_plan; |
565 | | |
566 | 8.85k | node->shape_plan = shape_plan; |
567 | 8.85k | node->next = cached_plan_nodes; |
568 | | |
569 | 8.85k | if (unlikely (!face->shape_plans.cmpexch (cached_plan_nodes, node))) |
570 | 0 | { |
571 | 0 | hb_shape_plan_destroy (shape_plan); |
572 | 0 | hb_free (node); |
573 | 0 | goto retry; |
574 | 0 | } |
575 | 8.85k | DEBUG_MSG_FUNC (SHAPE_PLAN, shape_plan, "inserted into cache"); |
576 | | |
577 | 8.85k | return hb_shape_plan_reference (shape_plan); |
578 | 8.85k | } |
579 | | |
580 | | |
581 | | #endif |