1 /***************************************************************************/ 2 /* */ 3 /* afshaper.c */ 4 /* */ 5 /* HarfBuzz interface for accessing OpenType features (body). */ 6 /* */ 7 /* Copyright 2013-2017 by */ 8 /* David Turner, Robert Wilhelm, and Werner Lemberg. */ 9 /* */ 10 /* This file is part of the FreeType project, and may only be used, */ 11 /* modified, and distributed under the terms of the FreeType project */ 12 /* license, LICENSE.TXT. By continuing to use, modify, or distribute */ 13 /* this file you indicate that you have read the license and */ 14 /* understand and accept it fully. */ 15 /* */ 16 /***************************************************************************/ 17 18 19 #include <ft2build.h> 20 #include FT_FREETYPE_H 21 #include "afglobal.h" 22 #include "aftypes.h" 23 #include "afshaper.h" 24 25 #ifdef FT_CONFIG_OPTION_USE_HARFBUZZ 26 27 28 /*************************************************************************/ 29 /* */ 30 /* The macro FT_COMPONENT is used in trace mode. It is an implicit */ 31 /* parameter of the FT_TRACE() and FT_ERROR() macros, used to print/log */ 32 /* messages during execution. */ 33 /* */ 34 #undef FT_COMPONENT 35 #define FT_COMPONENT trace_afshaper 36 37 38 /* 39 * We use `sets' (in the HarfBuzz sense, which comes quite near to the 40 * usual mathematical meaning) to manage both lookups and glyph indices. 41 * 42 * 1. For each coverage, collect lookup IDs in a set. Note that an 43 * auto-hinter `coverage' is represented by one `feature', and a 44 * feature consists of an arbitrary number of (font specific) `lookup's 45 * that actually do the mapping job. Please check the OpenType 46 * specification for more details on features and lookups. 47 * 48 * 2. Create glyph ID sets from the corresponding lookup sets. 49 * 50 * 3. The glyph set corresponding to AF_COVERAGE_DEFAULT is computed 51 * with all lookups specific to the OpenType script activated. It 52 * relies on the order of AF_DEFINE_STYLE_CLASS entries so that 53 * special coverages (like `oldstyle figures') don't get overwritten. 54 * 55 */ 56 57 58 /* load coverage tags */ 59 #undef COVERAGE 60 #define COVERAGE( name, NAME, description, \ 61 tag1, tag2, tag3, tag4 ) \ 62 static const hb_tag_t name ## _coverage[] = \ 63 { \ 64 HB_TAG( tag1, tag2, tag3, tag4 ), \ 65 HB_TAG_NONE \ 66 }; 67 68 69 #include "afcover.h" 70 71 72 /* define mapping between coverage tags and AF_Coverage */ 73 #undef COVERAGE 74 #define COVERAGE( name, NAME, description, \ 75 tag1, tag2, tag3, tag4 ) \ 76 name ## _coverage, 77 78 79 static const hb_tag_t* coverages[] = 80 { 81 #include "afcover.h" 82 83 NULL /* AF_COVERAGE_DEFAULT */ 84 }; 85 86 87 /* load HarfBuzz script tags */ 88 #undef SCRIPT 89 #define SCRIPT( s, S, d, h, H, ss ) h, 90 91 92 static const hb_script_t scripts[] = 93 { 94 #include "afscript.h" 95 }; 96 97 98 FT_Error af_shaper_get_coverage(AF_FaceGlobals globals,AF_StyleClass style_class,FT_UShort * gstyles,FT_Bool default_script)99 af_shaper_get_coverage( AF_FaceGlobals globals, 100 AF_StyleClass style_class, 101 FT_UShort* gstyles, 102 FT_Bool default_script ) 103 { 104 hb_face_t* face; 105 106 hb_set_t* gsub_lookups; /* GSUB lookups for a given script */ 107 hb_set_t* gsub_glyphs; /* glyphs covered by GSUB lookups */ 108 hb_set_t* gpos_lookups; /* GPOS lookups for a given script */ 109 hb_set_t* gpos_glyphs; /* glyphs covered by GPOS lookups */ 110 111 hb_script_t script; 112 const hb_tag_t* coverage_tags; 113 hb_tag_t script_tags[] = { HB_TAG_NONE, 114 HB_TAG_NONE, 115 HB_TAG_NONE, 116 HB_TAG_NONE }; 117 118 hb_codepoint_t idx; 119 #ifdef FT_DEBUG_LEVEL_TRACE 120 int count; 121 #endif 122 123 124 if ( !globals || !style_class || !gstyles ) 125 return FT_THROW( Invalid_Argument ); 126 127 face = hb_font_get_face( globals->hb_font ); 128 129 gsub_lookups = hb_set_create(); 130 gsub_glyphs = hb_set_create(); 131 gpos_lookups = hb_set_create(); 132 gpos_glyphs = hb_set_create(); 133 134 coverage_tags = coverages[style_class->coverage]; 135 script = scripts[style_class->script]; 136 137 /* Convert a HarfBuzz script tag into the corresponding OpenType */ 138 /* tag or tags -- some Indic scripts like Devanagari have an old */ 139 /* and a new set of features. */ 140 hb_ot_tags_from_script( script, 141 &script_tags[0], 142 &script_tags[1] ); 143 144 /* `hb_ot_tags_from_script' usually returns HB_OT_TAG_DEFAULT_SCRIPT */ 145 /* as the second tag. We change that to HB_TAG_NONE except for the */ 146 /* default script. */ 147 if ( default_script ) 148 { 149 if ( script_tags[0] == HB_TAG_NONE ) 150 script_tags[0] = HB_OT_TAG_DEFAULT_SCRIPT; 151 else 152 { 153 if ( script_tags[1] == HB_TAG_NONE ) 154 script_tags[1] = HB_OT_TAG_DEFAULT_SCRIPT; 155 else if ( script_tags[1] != HB_OT_TAG_DEFAULT_SCRIPT ) 156 script_tags[2] = HB_OT_TAG_DEFAULT_SCRIPT; 157 } 158 } 159 else 160 { 161 /* we use non-standard tags like `khms' for special purposes; */ 162 /* HarfBuzz maps them to `DFLT', which we don't want to handle here */ 163 if ( script_tags[0] == HB_OT_TAG_DEFAULT_SCRIPT ) 164 goto Exit; 165 166 if ( script_tags[1] == HB_OT_TAG_DEFAULT_SCRIPT ) 167 script_tags[1] = HB_TAG_NONE; 168 } 169 170 hb_ot_layout_collect_lookups( face, 171 HB_OT_TAG_GSUB, 172 script_tags, 173 NULL, 174 coverage_tags, 175 gsub_lookups ); 176 177 if ( hb_set_is_empty( gsub_lookups ) ) 178 goto Exit; /* nothing to do */ 179 180 hb_ot_layout_collect_lookups( face, 181 HB_OT_TAG_GPOS, 182 script_tags, 183 NULL, 184 coverage_tags, 185 gpos_lookups ); 186 187 FT_TRACE4(( "GSUB lookups (style `%s'):\n" 188 " ", 189 af_style_names[style_class->style] )); 190 191 #ifdef FT_DEBUG_LEVEL_TRACE 192 count = 0; 193 #endif 194 195 for ( idx = HB_SET_VALUE_INVALID; hb_set_next( gsub_lookups, &idx ); ) 196 { 197 #ifdef FT_DEBUG_LEVEL_TRACE 198 FT_TRACE4(( " %d", idx )); 199 count++; 200 #endif 201 202 /* get output coverage of GSUB feature */ 203 hb_ot_layout_lookup_collect_glyphs( face, 204 HB_OT_TAG_GSUB, 205 idx, 206 NULL, 207 NULL, 208 NULL, 209 gsub_glyphs ); 210 } 211 212 #ifdef FT_DEBUG_LEVEL_TRACE 213 if ( !count ) 214 FT_TRACE4(( " (none)" )); 215 FT_TRACE4(( "\n\n" )); 216 #endif 217 218 FT_TRACE4(( "GPOS lookups (style `%s'):\n" 219 " ", 220 af_style_names[style_class->style] )); 221 222 #ifdef FT_DEBUG_LEVEL_TRACE 223 count = 0; 224 #endif 225 226 for ( idx = HB_SET_VALUE_INVALID; hb_set_next( gpos_lookups, &idx ); ) 227 { 228 #ifdef FT_DEBUG_LEVEL_TRACE 229 FT_TRACE4(( " %d", idx )); 230 count++; 231 #endif 232 233 /* get input coverage of GPOS feature */ 234 hb_ot_layout_lookup_collect_glyphs( face, 235 HB_OT_TAG_GPOS, 236 idx, 237 NULL, 238 gpos_glyphs, 239 NULL, 240 NULL ); 241 } 242 243 #ifdef FT_DEBUG_LEVEL_TRACE 244 if ( !count ) 245 FT_TRACE4(( " (none)" )); 246 FT_TRACE4(( "\n\n" )); 247 #endif 248 249 /* 250 * We now check whether we can construct blue zones, using glyphs 251 * covered by the feature only. In case there is not a single zone 252 * (this is, not a single character is covered), we skip this coverage. 253 * 254 */ 255 if ( style_class->coverage != AF_COVERAGE_DEFAULT ) 256 { 257 AF_Blue_Stringset bss = style_class->blue_stringset; 258 const AF_Blue_StringRec* bs = &af_blue_stringsets[bss]; 259 260 FT_Bool found = 0; 261 262 263 for ( ; bs->string != AF_BLUE_STRING_MAX; bs++ ) 264 { 265 const char* p = &af_blue_strings[bs->string]; 266 267 268 while ( *p ) 269 { 270 hb_codepoint_t ch; 271 272 273 GET_UTF8_CHAR( ch, p ); 274 275 for ( idx = HB_SET_VALUE_INVALID; hb_set_next( gsub_lookups, 276 &idx ); ) 277 { 278 hb_codepoint_t gidx = FT_Get_Char_Index( globals->face, ch ); 279 280 281 if ( hb_ot_layout_lookup_would_substitute( face, idx, 282 &gidx, 1, 1 ) ) 283 { 284 found = 1; 285 break; 286 } 287 } 288 } 289 } 290 291 if ( !found ) 292 { 293 FT_TRACE4(( " no blue characters found; style skipped\n" )); 294 goto Exit; 295 } 296 } 297 298 /* 299 * Various OpenType features might use the same glyphs at different 300 * vertical positions; for example, superscript and subscript glyphs 301 * could be the same. However, the auto-hinter is completely 302 * agnostic of OpenType features after the feature analysis has been 303 * completed: The engine then simply receives a glyph index and returns a 304 * hinted and usually rendered glyph. 305 * 306 * Consider the superscript feature of font `pala.ttf': Some of the 307 * glyphs are `real', this is, they have a zero vertical offset, but 308 * most of them are small caps glyphs shifted up to the superscript 309 * position (this is, the `sups' feature is present in both the GSUB and 310 * GPOS tables). The code for blue zones computation actually uses a 311 * feature's y offset so that the `real' glyphs get correct hints. But 312 * later on it is impossible to decide whether a glyph index belongs to, 313 * say, the small caps or superscript feature. 314 * 315 * For this reason, we don't assign a style to a glyph if the current 316 * feature covers the glyph in both the GSUB and the GPOS tables. This 317 * is quite a broad condition, assuming that 318 * 319 * (a) glyphs that get used in multiple features are present in a 320 * feature without vertical shift, 321 * 322 * and 323 * 324 * (b) a feature's GPOS data really moves the glyph vertically. 325 * 326 * Not fulfilling condition (a) makes a font larger; it would also 327 * reduce the number of glyphs that could be addressed directly without 328 * using OpenType features, so this assumption is rather strong. 329 * 330 * Condition (b) is much weaker, and there might be glyphs which get 331 * missed. However, the OpenType features we are going to handle are 332 * primarily located in GSUB, and HarfBuzz doesn't provide an API to 333 * directly get the necessary information from the GPOS table. A 334 * possible solution might be to directly parse the GPOS table to find 335 * out whether a glyph gets shifted vertically, but this is something I 336 * would like to avoid if not really necessary. 337 * 338 * Note that we don't follow this logic for the default coverage. 339 * Complex scripts like Devanagari have mandatory GPOS features to 340 * position many glyph elements, using mark-to-base or mark-to-ligature 341 * tables; the number of glyphs missed due to condition (b) would be far 342 * too large. 343 * 344 */ 345 if ( style_class->coverage != AF_COVERAGE_DEFAULT ) 346 hb_set_subtract( gsub_glyphs, gpos_glyphs ); 347 348 #ifdef FT_DEBUG_LEVEL_TRACE 349 FT_TRACE4(( " glyphs without GPOS data (`*' means already assigned)" )); 350 count = 0; 351 #endif 352 353 for ( idx = HB_SET_VALUE_INVALID; hb_set_next( gsub_glyphs, &idx ); ) 354 { 355 #ifdef FT_DEBUG_LEVEL_TRACE 356 if ( !( count % 10 ) ) 357 FT_TRACE4(( "\n" 358 " " )); 359 360 FT_TRACE4(( " %d", idx )); 361 count++; 362 #endif 363 364 /* glyph indices returned by `hb_ot_layout_lookup_collect_glyphs' */ 365 /* can be arbitrary: some fonts use fake indices for processing */ 366 /* internal to GSUB or GPOS, which is fully valid */ 367 if ( idx >= (hb_codepoint_t)globals->glyph_count ) 368 continue; 369 370 if ( gstyles[idx] == AF_STYLE_UNASSIGNED ) 371 gstyles[idx] = (FT_UShort)style_class->style; 372 #ifdef FT_DEBUG_LEVEL_TRACE 373 else 374 FT_TRACE4(( "*" )); 375 #endif 376 } 377 378 #ifdef FT_DEBUG_LEVEL_TRACE 379 if ( !count ) 380 FT_TRACE4(( "\n" 381 " (none)" )); 382 FT_TRACE4(( "\n\n" )); 383 #endif 384 385 Exit: 386 hb_set_destroy( gsub_lookups ); 387 hb_set_destroy( gsub_glyphs ); 388 hb_set_destroy( gpos_lookups ); 389 hb_set_destroy( gpos_glyphs ); 390 391 return FT_Err_Ok; 392 } 393 394 395 /* construct HarfBuzz features */ 396 #undef COVERAGE 397 #define COVERAGE( name, NAME, description, \ 398 tag1, tag2, tag3, tag4 ) \ 399 static const hb_feature_t name ## _feature[] = \ 400 { \ 401 { \ 402 HB_TAG( tag1, tag2, tag3, tag4 ), \ 403 1, 0, (unsigned int)-1 \ 404 } \ 405 }; 406 407 408 #include "afcover.h" 409 410 411 /* define mapping between HarfBuzz features and AF_Coverage */ 412 #undef COVERAGE 413 #define COVERAGE( name, NAME, description, \ 414 tag1, tag2, tag3, tag4 ) \ 415 name ## _feature, 416 417 418 static const hb_feature_t* features[] = 419 { 420 #include "afcover.h" 421 422 NULL /* AF_COVERAGE_DEFAULT */ 423 }; 424 425 426 void* af_shaper_buf_create(FT_Face face)427 af_shaper_buf_create( FT_Face face ) 428 { 429 FT_UNUSED( face ); 430 431 return (void*)hb_buffer_create(); 432 } 433 434 435 void af_shaper_buf_destroy(FT_Face face,void * buf)436 af_shaper_buf_destroy( FT_Face face, 437 void* buf ) 438 { 439 FT_UNUSED( face ); 440 441 hb_buffer_destroy( (hb_buffer_t*)buf ); 442 } 443 444 445 const char* af_shaper_get_cluster(const char * p,AF_StyleMetrics metrics,void * buf_,unsigned int * count)446 af_shaper_get_cluster( const char* p, 447 AF_StyleMetrics metrics, 448 void* buf_, 449 unsigned int* count ) 450 { 451 AF_StyleClass style_class; 452 const hb_feature_t* feature; 453 FT_Int upem; 454 const char* q; 455 int len; 456 457 hb_buffer_t* buf = (hb_buffer_t*)buf_; 458 hb_font_t* font; 459 hb_codepoint_t dummy; 460 461 462 upem = (FT_Int)metrics->globals->face->units_per_EM; 463 style_class = metrics->style_class; 464 feature = features[style_class->coverage]; 465 466 font = metrics->globals->hb_font; 467 468 /* we shape at a size of units per EM; this means font units */ 469 hb_font_set_scale( font, upem, upem ); 470 471 while ( *p == ' ' ) 472 p++; 473 474 /* count bytes up to next space (or end of buffer) */ 475 q = p; 476 while ( !( *q == ' ' || *q == '\0' ) ) 477 GET_UTF8_CHAR( dummy, q ); 478 len = (int)( q - p ); 479 480 /* feed character(s) to the HarfBuzz buffer */ 481 hb_buffer_clear_contents( buf ); 482 hb_buffer_add_utf8( buf, p, len, 0, len ); 483 484 /* we let HarfBuzz guess the script and writing direction */ 485 hb_buffer_guess_segment_properties( buf ); 486 487 /* shape buffer, which means conversion from character codes to */ 488 /* glyph indices, possibly applying a feature */ 489 hb_shape( font, buf, feature, feature ? 1 : 0 ); 490 491 if ( feature ) 492 { 493 hb_buffer_t* hb_buf = metrics->globals->hb_buf; 494 495 unsigned int gcount; 496 hb_glyph_info_t* ginfo; 497 498 unsigned int hb_gcount; 499 hb_glyph_info_t* hb_ginfo; 500 501 502 /* we have to check whether applying a feature does actually change */ 503 /* glyph indices; otherwise the affected glyph or glyphs aren't */ 504 /* available at all in the feature */ 505 506 hb_buffer_clear_contents( hb_buf ); 507 hb_buffer_add_utf8( hb_buf, p, len, 0, len ); 508 hb_buffer_guess_segment_properties( hb_buf ); 509 hb_shape( font, hb_buf, NULL, 0 ); 510 511 ginfo = hb_buffer_get_glyph_infos( buf, &gcount ); 512 hb_ginfo = hb_buffer_get_glyph_infos( hb_buf, &hb_gcount ); 513 514 if ( gcount == hb_gcount ) 515 { 516 unsigned int i; 517 518 519 for (i = 0; i < gcount; i++ ) 520 if ( ginfo[i].codepoint != hb_ginfo[i].codepoint ) 521 break; 522 523 if ( i == gcount ) 524 { 525 /* both buffers have identical glyph indices */ 526 hb_buffer_clear_contents( buf ); 527 } 528 } 529 } 530 531 *count = hb_buffer_get_length( buf ); 532 533 #ifdef FT_DEBUG_LEVEL_TRACE 534 if ( feature && *count > 1 ) 535 FT_TRACE1(( "af_shaper_get_cluster:" 536 " input character mapped to multiple glyphs\n" )); 537 #endif 538 539 return q; 540 } 541 542 543 FT_ULong af_shaper_get_elem(AF_StyleMetrics metrics,void * buf_,unsigned int idx,FT_Long * advance,FT_Long * y_offset)544 af_shaper_get_elem( AF_StyleMetrics metrics, 545 void* buf_, 546 unsigned int idx, 547 FT_Long* advance, 548 FT_Long* y_offset ) 549 { 550 hb_buffer_t* buf = (hb_buffer_t*)buf_; 551 hb_glyph_info_t* ginfo; 552 hb_glyph_position_t* gpos; 553 unsigned int gcount; 554 555 FT_UNUSED( metrics ); 556 557 558 ginfo = hb_buffer_get_glyph_infos( buf, &gcount ); 559 gpos = hb_buffer_get_glyph_positions( buf, &gcount ); 560 561 if ( idx >= gcount ) 562 return 0; 563 564 if ( advance ) 565 *advance = gpos[idx].x_advance; 566 if ( y_offset ) 567 *y_offset = gpos[idx].y_offset; 568 569 return ginfo[idx].codepoint; 570 } 571 572 573 #else /* !FT_CONFIG_OPTION_USE_HARFBUZZ */ 574 575 576 FT_Error af_shaper_get_coverage(AF_FaceGlobals globals,AF_StyleClass style_class,FT_UShort * gstyles,FT_Bool default_script)577 af_shaper_get_coverage( AF_FaceGlobals globals, 578 AF_StyleClass style_class, 579 FT_UShort* gstyles, 580 FT_Bool default_script ) 581 { 582 FT_UNUSED( globals ); 583 FT_UNUSED( style_class ); 584 FT_UNUSED( gstyles ); 585 FT_UNUSED( default_script ); 586 587 return FT_Err_Ok; 588 } 589 590 591 void* af_shaper_buf_create(FT_Face face)592 af_shaper_buf_create( FT_Face face ) 593 { 594 FT_Error error; 595 FT_Memory memory = face->memory; 596 FT_ULong* buf; 597 598 599 FT_MEM_ALLOC( buf, sizeof ( FT_ULong ) ); 600 601 return (void*)buf; 602 } 603 604 605 void af_shaper_buf_destroy(FT_Face face,void * buf)606 af_shaper_buf_destroy( FT_Face face, 607 void* buf ) 608 { 609 FT_Memory memory = face->memory; 610 611 612 FT_FREE( buf ); 613 } 614 615 616 const char* af_shaper_get_cluster(const char * p,AF_StyleMetrics metrics,void * buf_,unsigned int * count)617 af_shaper_get_cluster( const char* p, 618 AF_StyleMetrics metrics, 619 void* buf_, 620 unsigned int* count ) 621 { 622 FT_Face face = metrics->globals->face; 623 FT_ULong ch, dummy = 0; 624 FT_ULong* buf = (FT_ULong*)buf_; 625 626 627 while ( *p == ' ' ) 628 p++; 629 630 GET_UTF8_CHAR( ch, p ); 631 632 /* since we don't have an engine to handle clusters, */ 633 /* we scan the characters but return zero */ 634 while ( !( *p == ' ' || *p == '\0' ) ) 635 GET_UTF8_CHAR( dummy, p ); 636 637 if ( dummy ) 638 { 639 *buf = 0; 640 *count = 0; 641 } 642 else 643 { 644 *buf = FT_Get_Char_Index( face, ch ); 645 *count = 1; 646 } 647 648 return p; 649 } 650 651 652 FT_ULong af_shaper_get_elem(AF_StyleMetrics metrics,void * buf_,unsigned int idx,FT_Long * advance,FT_Long * y_offset)653 af_shaper_get_elem( AF_StyleMetrics metrics, 654 void* buf_, 655 unsigned int idx, 656 FT_Long* advance, 657 FT_Long* y_offset ) 658 { 659 FT_Face face = metrics->globals->face; 660 FT_ULong glyph_index = *(FT_ULong*)buf_; 661 662 FT_UNUSED( idx ); 663 664 665 if ( advance ) 666 FT_Get_Advance( face, 667 glyph_index, 668 FT_LOAD_NO_SCALE | 669 FT_LOAD_NO_HINTING | 670 FT_LOAD_IGNORE_TRANSFORM, 671 advance ); 672 673 if ( y_offset ) 674 *y_offset = 0; 675 676 return glyph_index; 677 } 678 679 680 #endif /* !FT_CONFIG_OPTION_USE_HARFBUZZ */ 681 682 683 /* END */ 684