Home | History | Annotate | Download | only in layout
      1 
      2 /*
      3  *
      4  * (C) Copyright IBM Corp. and others 1998-2013 - All Rights Reserved
      5  *
      6  */
      7 
      8 #include "LETypes.h"
      9 #include "LEScripts.h"
     10 #include "LELanguages.h"
     11 
     12 #include "LayoutEngine.h"
     13 #include "CanonShaping.h"
     14 #include "OpenTypeLayoutEngine.h"
     15 #include "ScriptAndLanguageTags.h"
     16 #include "CharSubstitutionFilter.h"
     17 
     18 #include "GlyphSubstitutionTables.h"
     19 #include "GlyphDefinitionTables.h"
     20 #include "GlyphPositioningTables.h"
     21 
     22 #include "LEGlyphStorage.h"
     23 #include "GlyphPositionAdjustments.h"
     24 
     25 #include "GDEFMarkFilter.h"
     26 
     27 #include "KernTable.h"
     28 
     29 U_NAMESPACE_BEGIN
     30 
     31 UOBJECT_DEFINE_RTTI_IMPLEMENTATION(OpenTypeLayoutEngine)
     32 
     33 #define ccmpFeatureTag LE_CCMP_FEATURE_TAG
     34 #define ligaFeatureTag LE_LIGA_FEATURE_TAG
     35 #define cligFeatureTag LE_CLIG_FEATURE_TAG
     36 #define kernFeatureTag LE_KERN_FEATURE_TAG
     37 #define markFeatureTag LE_MARK_FEATURE_TAG
     38 #define mkmkFeatureTag LE_MKMK_FEATURE_TAG
     39 #define loclFeatureTag LE_LOCL_FEATURE_TAG
     40 #define caltFeatureTag LE_CALT_FEATURE_TAG
     41 
     42 #define dligFeatureTag LE_DLIG_FEATURE_TAG
     43 #define rligFeatureTag LE_RLIG_FEATURE_TAG
     44 #define paltFeatureTag LE_PALT_FEATURE_TAG
     45 
     46 #define hligFeatureTag LE_HLIG_FEATURE_TAG
     47 #define smcpFeatureTag LE_SMCP_FEATURE_TAG
     48 #define fracFeatureTag LE_FRAC_FEATURE_TAG
     49 #define afrcFeatureTag LE_AFRC_FEATURE_TAG
     50 #define zeroFeatureTag LE_ZERO_FEATURE_TAG
     51 #define swshFeatureTag LE_SWSH_FEATURE_TAG
     52 #define cswhFeatureTag LE_CSWH_FEATURE_TAG
     53 #define saltFeatureTag LE_SALT_FEATURE_TAG
     54 #define naltFeatureTag LE_NALT_FEATURE_TAG
     55 #define rubyFeatureTag LE_RUBY_FEATURE_TAG
     56 #define ss01FeatureTag LE_SS01_FEATURE_TAG
     57 #define ss02FeatureTag LE_SS02_FEATURE_TAG
     58 #define ss03FeatureTag LE_SS03_FEATURE_TAG
     59 #define ss04FeatureTag LE_SS04_FEATURE_TAG
     60 #define ss05FeatureTag LE_SS05_FEATURE_TAG
     61 #define ss06FeatureTag LE_SS06_FEATURE_TAG
     62 #define ss07FeatureTag LE_SS07_FEATURE_TAG
     63 
     64 #define ccmpFeatureMask 0x80000000UL
     65 #define ligaFeatureMask 0x40000000UL
     66 #define cligFeatureMask 0x20000000UL
     67 #define kernFeatureMask 0x10000000UL
     68 #define paltFeatureMask 0x08000000UL
     69 #define markFeatureMask 0x04000000UL
     70 #define mkmkFeatureMask 0x02000000UL
     71 #define loclFeatureMask 0x01000000UL
     72 #define caltFeatureMask 0x00800000UL
     73 
     74 #define dligFeatureMask 0x00400000UL
     75 #define rligFeatureMask 0x00200000UL
     76 #define hligFeatureMask 0x00100000UL
     77 #define smcpFeatureMask 0x00080000UL
     78 #define fracFeatureMask 0x00040000UL
     79 #define afrcFeatureMask 0x00020000UL
     80 #define zeroFeatureMask 0x00010000UL
     81 #define swshFeatureMask 0x00008000UL
     82 #define cswhFeatureMask 0x00004000UL
     83 #define saltFeatureMask 0x00002000UL
     84 #define naltFeatureMask 0x00001000UL
     85 #define rubyFeatureMask 0x00000800UL
     86 #define ss01FeatureMask 0x00000400UL
     87 #define ss02FeatureMask 0x00000200UL
     88 #define ss03FeatureMask 0x00000100UL
     89 #define ss04FeatureMask 0x00000080UL
     90 #define ss05FeatureMask 0x00000040UL
     91 #define ss06FeatureMask 0x00000020UL
     92 #define ss07FeatureMask 0x00000010UL
     93 
     94 #define minimalFeatures     (ccmpFeatureMask | markFeatureMask | mkmkFeatureMask | loclFeatureMask | caltFeatureMask)
     95 
     96 static const FeatureMap featureMap[] =
     97 {
     98     {ccmpFeatureTag, ccmpFeatureMask},
     99     {ligaFeatureTag, ligaFeatureMask},
    100     {cligFeatureTag, cligFeatureMask},
    101     {kernFeatureTag, kernFeatureMask},
    102     {paltFeatureTag, paltFeatureMask},
    103     {markFeatureTag, markFeatureMask},
    104     {mkmkFeatureTag, mkmkFeatureMask},
    105     {loclFeatureTag, loclFeatureMask},
    106     {caltFeatureTag, caltFeatureMask},
    107     {hligFeatureTag, hligFeatureMask},
    108     {smcpFeatureTag, smcpFeatureMask},
    109     {fracFeatureTag, fracFeatureMask},
    110     {afrcFeatureTag, afrcFeatureMask},
    111     {zeroFeatureTag, zeroFeatureMask},
    112     {swshFeatureTag, swshFeatureMask},
    113     {cswhFeatureTag, cswhFeatureMask},
    114     {saltFeatureTag, saltFeatureMask},
    115     {naltFeatureTag, naltFeatureMask},
    116     {rubyFeatureTag, rubyFeatureMask},
    117     {ss01FeatureTag, ss01FeatureMask},
    118     {ss02FeatureTag, ss02FeatureMask},
    119     {ss03FeatureTag, ss03FeatureMask},
    120     {ss04FeatureTag, ss04FeatureMask},
    121     {ss05FeatureTag, ss05FeatureMask},
    122     {ss06FeatureTag, ss06FeatureMask},
    123     {ss07FeatureTag, ss07FeatureMask}
    124 };
    125 
    126 static const le_int32 featureMapCount = LE_ARRAY_SIZE(featureMap);
    127 
    128 OpenTypeLayoutEngine::OpenTypeLayoutEngine(const LEFontInstance *fontInstance, le_int32 scriptCode, le_int32 languageCode,
    129                      le_int32 typoFlags, const LEReferenceTo<GlyphSubstitutionTableHeader> &gsubTable, LEErrorCode &success)
    130     : LayoutEngine(fontInstance, scriptCode, languageCode, typoFlags, success), fFeatureMask(minimalFeatures),
    131       fFeatureMap(featureMap), fFeatureMapCount(featureMapCount), fFeatureOrder(FALSE),
    132       fGSUBTable(gsubTable),
    133       fGDEFTable(fontInstance, LE_GDEF_TABLE_TAG, success),
    134       fGPOSTable(fontInstance, LE_GPOS_TABLE_TAG, success), fSubstitutionFilter(NULL)
    135 {
    136     applyTypoFlags();
    137 
    138     setScriptAndLanguageTags();
    139 
    140 // JK patch, 2008-05-30 - see Sinhala bug report and LKLUG font
    141 //    if (gposTable != NULL && gposTable->coversScriptAndLanguage(fScriptTag, fLangSysTag)) {
    142     if (!fGPOSTable.isEmpty()&& !fGPOSTable->coversScript(fGPOSTable, fScriptTag, success)) {
    143       fGPOSTable.clear(); // already loaded
    144     }
    145 }
    146 
    147 void OpenTypeLayoutEngine::applyTypoFlags() {
    148     const le_int32& typoFlags = fTypoFlags;
    149     const LEFontInstance *fontInstance = fFontInstance;
    150 
    151     switch (typoFlags & (LE_SS01_FEATURE_FLAG
    152                          | LE_SS02_FEATURE_FLAG
    153                          | LE_SS03_FEATURE_FLAG
    154                          | LE_SS04_FEATURE_FLAG
    155                          | LE_SS05_FEATURE_FLAG
    156                          | LE_SS06_FEATURE_FLAG
    157                          | LE_SS07_FEATURE_FLAG)) {
    158         case LE_SS01_FEATURE_FLAG:
    159             fFeatureMask |= ss01FeatureMask;
    160             break;
    161         case LE_SS02_FEATURE_FLAG:
    162             fFeatureMask |= ss02FeatureMask;
    163             break;
    164         case LE_SS03_FEATURE_FLAG:
    165             fFeatureMask |= ss03FeatureMask;
    166             break;
    167         case LE_SS04_FEATURE_FLAG:
    168             fFeatureMask |= ss04FeatureMask;
    169             break;
    170         case LE_SS05_FEATURE_FLAG:
    171             fFeatureMask |= ss05FeatureMask;
    172             break;
    173         case LE_SS06_FEATURE_FLAG:
    174             fFeatureMask |= ss06FeatureMask;
    175             break;
    176         case LE_SS07_FEATURE_FLAG:
    177             fFeatureMask |= ss07FeatureMask;
    178             break;
    179     }
    180 
    181     if (typoFlags & LE_Kerning_FEATURE_FLAG) {
    182       fFeatureMask |= (kernFeatureMask | paltFeatureMask);
    183       // Convenience.
    184     }
    185     if (typoFlags & LE_Ligatures_FEATURE_FLAG) {
    186       fFeatureMask |= (ligaFeatureMask | cligFeatureMask);
    187       // Convenience TODO: should add: .. dligFeatureMask | rligFeatureMask ?
    188     }
    189     if (typoFlags & LE_CLIG_FEATURE_FLAG) fFeatureMask |= cligFeatureMask;
    190     if (typoFlags & LE_DLIG_FEATURE_FLAG) fFeatureMask |= dligFeatureMask;
    191     if (typoFlags & LE_HLIG_FEATURE_FLAG) fFeatureMask |= hligFeatureMask;
    192     if (typoFlags & LE_LIGA_FEATURE_FLAG) fFeatureMask |= ligaFeatureMask;
    193     if (typoFlags & LE_RLIG_FEATURE_FLAG) fFeatureMask |= rligFeatureMask;
    194     if (typoFlags & LE_SMCP_FEATURE_FLAG) fFeatureMask |= smcpFeatureMask;
    195     if (typoFlags & LE_FRAC_FEATURE_FLAG) fFeatureMask |= fracFeatureMask;
    196     if (typoFlags & LE_AFRC_FEATURE_FLAG) fFeatureMask |= afrcFeatureMask;
    197     if (typoFlags & LE_ZERO_FEATURE_FLAG) fFeatureMask |= zeroFeatureMask;
    198     if (typoFlags & LE_SWSH_FEATURE_FLAG) fFeatureMask |= swshFeatureMask;
    199     if (typoFlags & LE_CSWH_FEATURE_FLAG) fFeatureMask |= cswhFeatureMask;
    200     if (typoFlags & LE_SALT_FEATURE_FLAG) fFeatureMask |= saltFeatureMask;
    201     if (typoFlags & LE_RUBY_FEATURE_FLAG) fFeatureMask |= rubyFeatureMask;
    202     if (typoFlags & LE_NALT_FEATURE_FLAG) {
    203       // Mutually exclusive with ALL other features. http://www.microsoft.com/typography/otspec/features_ko.htm
    204       fFeatureMask = naltFeatureMask;
    205     }
    206 
    207     if (typoFlags & LE_CHAR_FILTER_FEATURE_FLAG) {
    208       // This isn't a font feature, but requests a Char Substitution Filter
    209       fSubstitutionFilter = new CharSubstitutionFilter(fontInstance);
    210     }
    211 
    212 }
    213 
    214 void OpenTypeLayoutEngine::reset()
    215 {
    216     // NOTE: if we're called from
    217     // the destructor, LayoutEngine;:reset()
    218     // will have been called already by
    219     // LayoutEngine::~LayoutEngine()
    220     LayoutEngine::reset();
    221 }
    222 
    223 OpenTypeLayoutEngine::OpenTypeLayoutEngine(const LEFontInstance *fontInstance, le_int32 scriptCode, le_int32 languageCode,
    224                        le_int32 typoFlags, LEErrorCode &success)
    225     : LayoutEngine(fontInstance, scriptCode, languageCode, typoFlags, success), fFeatureOrder(FALSE),
    226       fGSUBTable(), fGDEFTable(), fGPOSTable(), fSubstitutionFilter(NULL)
    227 {
    228   applyTypoFlags();
    229   setScriptAndLanguageTags();
    230 }
    231 
    232 OpenTypeLayoutEngine::~OpenTypeLayoutEngine()
    233 {
    234     if (fTypoFlags & LE_CHAR_FILTER_FEATURE_FLAG) {
    235         delete fSubstitutionFilter;
    236         fSubstitutionFilter = NULL;
    237     }
    238 
    239     reset();
    240 }
    241 
    242 LETag OpenTypeLayoutEngine::getScriptTag(le_int32 scriptCode)
    243 {
    244     if (scriptCode < 0 || scriptCode >= scriptCodeCount) {
    245         return 0xFFFFFFFF;
    246     }
    247     return scriptTags[scriptCode];
    248 }
    249 
    250 LETag OpenTypeLayoutEngine::getV2ScriptTag(le_int32 scriptCode)
    251 {
    252 	switch (scriptCode) {
    253 		case bengScriptCode :    return bng2ScriptTag;
    254 		case devaScriptCode :    return dev2ScriptTag;
    255 		case gujrScriptCode :    return gjr2ScriptTag;
    256 		case guruScriptCode :    return gur2ScriptTag;
    257 		case kndaScriptCode :    return knd2ScriptTag;
    258 		case mlymScriptCode :    return mlm2ScriptTag;
    259 		case oryaScriptCode :    return ory2ScriptTag;
    260 		case tamlScriptCode :    return tml2ScriptTag;
    261 		case teluScriptCode :    return tel2ScriptTag;
    262 		default:                 return nullScriptTag;
    263 	}
    264 }
    265 
    266 LETag OpenTypeLayoutEngine::getLangSysTag(le_int32 languageCode)
    267 {
    268     if (languageCode < 0 || languageCode >= languageCodeCount) {
    269         return 0xFFFFFFFF;
    270     }
    271 
    272     return languageTags[languageCode];
    273 }
    274 
    275 void OpenTypeLayoutEngine::setScriptAndLanguageTags()
    276 {
    277     fScriptTag  = getScriptTag(fScriptCode);
    278     fScriptTagV2 = getV2ScriptTag(fScriptCode);
    279     fLangSysTag = getLangSysTag(fLanguageCode);
    280 }
    281 
    282 le_int32 OpenTypeLayoutEngine::characterProcessing(const LEUnicode chars[], le_int32 offset, le_int32 count, le_int32 max, le_bool rightToLeft,
    283                 LEUnicode *&outChars, LEGlyphStorage &glyphStorage, LEErrorCode &success)
    284 {
    285     if (LE_FAILURE(success)) {
    286         return 0;
    287     }
    288 
    289     if (offset < 0 || count < 0 || max < 0 || offset >= max || offset + count > max) {
    290         success = LE_ILLEGAL_ARGUMENT_ERROR;
    291         return 0;
    292     }
    293 
    294     // This is the cheapest way to get mark reordering only for Hebrew.
    295     // We could just do the mark reordering for all scripts, but most
    296     // of them probably don't need it... Another option would be to
    297     // add a HebrewOpenTypeLayoutEngine subclass, but the only thing it
    298     // would need to do is mark reordering, so that seems like overkill.
    299     if (fScriptCode == hebrScriptCode) {
    300         outChars = LE_NEW_ARRAY(LEUnicode, count);
    301 
    302         if (outChars == NULL) {
    303             success = LE_MEMORY_ALLOCATION_ERROR;
    304             return 0;
    305         }
    306 
    307         if (LE_FAILURE(success)) {
    308             LE_DELETE_ARRAY(outChars);
    309             return 0;
    310         }
    311 
    312         CanonShaping::reorderMarks(&chars[offset], count, rightToLeft, outChars, glyphStorage);
    313     }
    314 
    315     if (LE_FAILURE(success)) {
    316         return 0;
    317     }
    318 
    319     glyphStorage.allocateGlyphArray(count, rightToLeft, success);
    320     glyphStorage.allocateAuxData(success);
    321 
    322     for (le_int32 i = 0; i < count; i += 1) {
    323         glyphStorage.setAuxData(i, fFeatureMask, success);
    324     }
    325 
    326     return count;
    327 }
    328 
    329 // Input: characters, tags
    330 // Output: glyphs, char indices
    331 le_int32 OpenTypeLayoutEngine::glyphProcessing(const LEUnicode chars[], le_int32 offset, le_int32 count, le_int32 max, le_bool rightToLeft,
    332                                                LEGlyphStorage &glyphStorage, LEErrorCode &success)
    333 {
    334     if (LE_FAILURE(success)) {
    335         return 0;
    336     }
    337 
    338     if (chars == NULL || offset < 0 || count < 0 || max < 0 || offset >= max || offset + count > max) {
    339         success = LE_ILLEGAL_ARGUMENT_ERROR;
    340         return 0;
    341     }
    342 
    343     mapCharsToGlyphs(chars, offset, count, rightToLeft, rightToLeft, glyphStorage, success);
    344 
    345     if (LE_FAILURE(success)) {
    346         return 0;
    347     }
    348 
    349     if (fGSUBTable.isValid()) {
    350       if (fScriptTagV2 != nullScriptTag && fGSUBTable->coversScriptAndLanguage(fGSUBTable, fScriptTagV2, fLangSysTag, success)) {
    351           count = fGSUBTable->process(fGSUBTable, glyphStorage, rightToLeft, fScriptTagV2, fLangSysTag, fGDEFTable, fSubstitutionFilter,
    352                                     fFeatureMap, fFeatureMapCount, fFeatureOrder, success);
    353 
    354         } else {
    355           count = fGSUBTable->process(fGSUBTable, glyphStorage, rightToLeft, fScriptTag, fLangSysTag, fGDEFTable, fSubstitutionFilter,
    356                                     fFeatureMap, fFeatureMapCount, fFeatureOrder, success);
    357         }
    358     }
    359 
    360     return count;
    361 }
    362 // Input: characters, tags
    363 // Output: glyphs, char indices
    364 le_int32 OpenTypeLayoutEngine::glyphSubstitution(le_int32 count, le_int32 max, le_bool rightToLeft,
    365                                                LEGlyphStorage &glyphStorage, LEErrorCode &success)
    366 {
    367     if (LE_FAILURE(success)) {
    368         return 0;
    369     }
    370 
    371     if ( count < 0 || max < 0 ) {
    372         success = LE_ILLEGAL_ARGUMENT_ERROR;
    373         return 0;
    374     }
    375 
    376     if (fGSUBTable.isValid()) {
    377        if (fScriptTagV2 != nullScriptTag && fGSUBTable->coversScriptAndLanguage(fGSUBTable,fScriptTagV2,fLangSysTag,success)) {
    378           count = fGSUBTable->process(fGSUBTable, glyphStorage, rightToLeft, fScriptTagV2, fLangSysTag, fGDEFTable, fSubstitutionFilter,
    379                                     fFeatureMap, fFeatureMapCount, fFeatureOrder, success);
    380 
    381         } else {
    382           count = fGSUBTable->process(fGSUBTable, glyphStorage, rightToLeft, fScriptTag, fLangSysTag, fGDEFTable, fSubstitutionFilter,
    383                                     fFeatureMap, fFeatureMapCount, fFeatureOrder, success);
    384         }
    385     }
    386 
    387     return count;
    388 }
    389 le_int32 OpenTypeLayoutEngine::glyphPostProcessing(LEGlyphStorage &tempGlyphStorage, LEGlyphStorage &glyphStorage, LEErrorCode &success)
    390 {
    391     if (LE_FAILURE(success)) {
    392         return 0;
    393     }
    394 
    395     glyphStorage.adoptGlyphArray(tempGlyphStorage);
    396     glyphStorage.adoptCharIndicesArray(tempGlyphStorage);
    397     glyphStorage.adoptAuxDataArray(tempGlyphStorage);
    398     glyphStorage.adoptGlyphCount(tempGlyphStorage);
    399 
    400     return glyphStorage.getGlyphCount();
    401 }
    402 
    403 le_int32 OpenTypeLayoutEngine::computeGlyphs(const LEUnicode chars[], le_int32 offset, le_int32 count, le_int32 max, le_bool rightToLeft, LEGlyphStorage &glyphStorage, LEErrorCode &success)
    404 {
    405     LEUnicode *outChars = NULL;
    406     LEGlyphStorage fakeGlyphStorage;
    407     le_int32 outCharCount, outGlyphCount;
    408 
    409     if (LE_FAILURE(success)) {
    410         return 0;
    411     }
    412 
    413     if (chars == NULL || offset < 0 || count < 0 || max < 0 || offset >= max || offset + count > max) {
    414         success = LE_ILLEGAL_ARGUMENT_ERROR;
    415         return 0;
    416     }
    417 
    418     outCharCount = characterProcessing(chars, offset, count, max, rightToLeft, outChars, fakeGlyphStorage, success);
    419 
    420     if (LE_FAILURE(success)) {
    421         return 0;
    422     }
    423 
    424     if (outChars != NULL) {
    425         // le_int32 fakeGlyphCount =
    426         glyphProcessing(outChars, 0, outCharCount, outCharCount, rightToLeft, fakeGlyphStorage, success);
    427         LE_DELETE_ARRAY(outChars); // FIXME: a subclass may have allocated this, in which case this delete might not work...
    428         //adjustGlyphs(outChars, 0, outCharCount, rightToLeft, fakeGlyphs, fakeGlyphCount);
    429     } else {
    430         // le_int32 fakeGlyphCount =
    431         glyphProcessing(chars, offset, count, max, rightToLeft, fakeGlyphStorage, success);
    432         //adjustGlyphs(chars, offset, count, rightToLeft, fakeGlyphs, fakeGlyphCount);
    433     }
    434 
    435     if (LE_FAILURE(success)) {
    436         return 0;
    437     }
    438 
    439     outGlyphCount = glyphPostProcessing(fakeGlyphStorage, glyphStorage, success);
    440 
    441     return outGlyphCount;
    442 }
    443 
    444 // apply GPOS table, if any
    445 void OpenTypeLayoutEngine::adjustGlyphPositions(const LEUnicode chars[], le_int32 offset, le_int32 count, le_bool reverse,
    446                                                 LEGlyphStorage &glyphStorage, LEErrorCode &success)
    447 {
    448     if (LE_FAILURE(success)) {
    449         return;
    450     }
    451 
    452     if (chars == NULL || offset < 0 || count < 0) {
    453         success = LE_ILLEGAL_ARGUMENT_ERROR;
    454         return;
    455     }
    456 
    457     le_int32 glyphCount = glyphStorage.getGlyphCount();
    458     if (glyphCount == 0) {
    459         return;
    460     }
    461 
    462     if (!fGPOSTable.isEmpty()) {
    463         GlyphPositionAdjustments *adjustments = new GlyphPositionAdjustments(glyphCount);
    464         le_int32 i;
    465 
    466         if (adjustments == NULL) {
    467             success = LE_MEMORY_ALLOCATION_ERROR;
    468             return;
    469         }
    470 
    471 #if 0
    472         // Don't need to do this if we allocate
    473         // the adjustments array w/ new...
    474         for (i = 0; i < glyphCount; i += 1) {
    475             adjustments->setXPlacement(i, 0);
    476             adjustments->setYPlacement(i, 0);
    477 
    478             adjustments->setXAdvance(i, 0);
    479             adjustments->setYAdvance(i, 0);
    480 
    481             adjustments->setBaseOffset(i, -1);
    482         }
    483 #endif
    484 
    485         if (!fGPOSTable.isEmpty()) {
    486             if (fScriptTagV2 != nullScriptTag &&
    487                 fGPOSTable->coversScriptAndLanguage(fGPOSTable, fScriptTagV2,fLangSysTag,success)) {
    488               fGPOSTable->process(fGPOSTable, glyphStorage, adjustments, reverse, fScriptTagV2, fLangSysTag,
    489                                   fGDEFTable, success, fFontInstance, fFeatureMap, fFeatureMapCount, fFeatureOrder);
    490 
    491             } else {
    492               fGPOSTable->process(fGPOSTable, glyphStorage, adjustments, reverse, fScriptTag, fLangSysTag,
    493                                   fGDEFTable, success, fFontInstance, fFeatureMap, fFeatureMapCount, fFeatureOrder);
    494             }
    495         } else if (fTypoFlags & LE_Kerning_FEATURE_FLAG) { /* kerning enabled */
    496           LETableReference kernTable(fFontInstance, LE_KERN_TABLE_TAG, success);
    497           KernTable kt(kernTable, success);
    498           kt.process(glyphStorage, success);
    499         }
    500 
    501         float xAdjust = 0, yAdjust = 0;
    502 
    503         for (i = 0; i < glyphCount; i += 1) {
    504             float xAdvance   = adjustments->getXAdvance(i);
    505             float yAdvance   = adjustments->getYAdvance(i);
    506             float xPlacement = 0;
    507             float yPlacement = 0;
    508 
    509 
    510 #if 0
    511             // This is where separate kerning adjustments
    512             // should get applied.
    513             xAdjust += xKerning;
    514             yAdjust += yKerning;
    515 #endif
    516 
    517             for (le_int32 base = i; base >= 0; base = adjustments->getBaseOffset(base)) {
    518                 xPlacement += adjustments->getXPlacement(base);
    519                 yPlacement += adjustments->getYPlacement(base);
    520             }
    521 
    522             xPlacement = fFontInstance->xUnitsToPoints(xPlacement);
    523             yPlacement = fFontInstance->yUnitsToPoints(yPlacement);
    524             glyphStorage.adjustPosition(i, xAdjust + xPlacement, -(yAdjust + yPlacement), success);
    525 
    526             xAdjust += fFontInstance->xUnitsToPoints(xAdvance);
    527             yAdjust += fFontInstance->yUnitsToPoints(yAdvance);
    528         }
    529 
    530         glyphStorage.adjustPosition(glyphCount, xAdjust, -yAdjust, success);
    531 
    532         delete adjustments;
    533     } else {
    534         // if there was no GPOS table, maybe there's non-OpenType kerning we can use
    535         LayoutEngine::adjustGlyphPositions(chars, offset, count, reverse, glyphStorage, success);
    536     }
    537 
    538     LEGlyphID zwnj  = fFontInstance->mapCharToGlyph(0x200C);
    539 
    540     if (zwnj != 0x0000) {
    541         for (le_int32 g = 0; g < glyphCount; g += 1) {
    542             LEGlyphID glyph = glyphStorage[g];
    543 
    544             if (glyph == zwnj) {
    545                 glyphStorage[g] = LE_SET_GLYPH(glyph, 0xFFFF);
    546             }
    547         }
    548     }
    549 
    550 #if 0
    551     // Don't know why this is here...
    552     LE_DELETE_ARRAY(fFeatureTags);
    553     fFeatureTags = NULL;
    554 #endif
    555 }
    556 
    557 U_NAMESPACE_END
    558