Home | History | Annotate | Download | only in src
      1 /*
      2  * Copyright (C) 2008 Nokia Corporation and/or its subsidiary(-ies)
      3  *
      4  * This is part of HarfBuzz, an OpenType Layout engine 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 
     25 #include "harfbuzz-shaper.h"
     26 #include "harfbuzz-shaper-private.h"
     27 
     28 #include <assert.h>
     29 
     30 /*
     31  tibetan syllables are of the form:
     32     head position consonant
     33     first sub-joined consonant
     34     ....intermediate sub-joined consonants (if any)
     35     last sub-joined consonant
     36     sub-joined vowel (a-chung U+0F71)
     37     standard or compound vowel sign (or 'virama' for devanagari transliteration)
     38 */
     39 
     40 typedef enum {
     41     TibetanOther,
     42     TibetanHeadConsonant,
     43     TibetanSubjoinedConsonant,
     44     TibetanSubjoinedVowel,
     45     TibetanVowel
     46 } TibetanForm;
     47 
     48 /* this table starts at U+0f40 */
     49 static const unsigned char tibetanForm[0x80] = {
     50     TibetanHeadConsonant, TibetanHeadConsonant, TibetanHeadConsonant, TibetanHeadConsonant,
     51     TibetanHeadConsonant, TibetanHeadConsonant, TibetanHeadConsonant, TibetanHeadConsonant,
     52     TibetanHeadConsonant, TibetanHeadConsonant, TibetanHeadConsonant, TibetanHeadConsonant,
     53     TibetanHeadConsonant, TibetanHeadConsonant, TibetanHeadConsonant, TibetanHeadConsonant,
     54 
     55     TibetanHeadConsonant, TibetanHeadConsonant, TibetanHeadConsonant, TibetanHeadConsonant,
     56     TibetanHeadConsonant, TibetanHeadConsonant, TibetanHeadConsonant, TibetanHeadConsonant,
     57     TibetanHeadConsonant, TibetanHeadConsonant, TibetanHeadConsonant, TibetanHeadConsonant,
     58     TibetanHeadConsonant, TibetanHeadConsonant, TibetanHeadConsonant, TibetanHeadConsonant,
     59 
     60     TibetanHeadConsonant, TibetanHeadConsonant, TibetanHeadConsonant, TibetanHeadConsonant,
     61     TibetanHeadConsonant, TibetanHeadConsonant, TibetanHeadConsonant, TibetanHeadConsonant,
     62     TibetanHeadConsonant, TibetanHeadConsonant, TibetanHeadConsonant, TibetanHeadConsonant,
     63     TibetanOther, TibetanOther, TibetanOther, TibetanOther,
     64 
     65     TibetanOther, TibetanVowel, TibetanVowel, TibetanVowel,
     66     TibetanVowel, TibetanVowel, TibetanVowel, TibetanVowel,
     67     TibetanVowel, TibetanVowel, TibetanVowel, TibetanVowel,
     68     TibetanVowel, TibetanVowel, TibetanVowel, TibetanVowel,
     69 
     70     TibetanVowel, TibetanVowel, TibetanVowel, TibetanVowel,
     71     TibetanVowel, TibetanVowel, TibetanVowel, TibetanVowel,
     72     TibetanOther, TibetanOther, TibetanOther, TibetanOther,
     73     TibetanOther, TibetanOther, TibetanOther, TibetanOther,
     74 
     75     TibetanSubjoinedConsonant, TibetanSubjoinedConsonant, TibetanSubjoinedConsonant, TibetanSubjoinedConsonant,
     76     TibetanSubjoinedConsonant, TibetanSubjoinedConsonant, TibetanSubjoinedConsonant, TibetanSubjoinedConsonant,
     77     TibetanSubjoinedConsonant, TibetanSubjoinedConsonant, TibetanSubjoinedConsonant, TibetanSubjoinedConsonant,
     78     TibetanSubjoinedConsonant, TibetanSubjoinedConsonant, TibetanSubjoinedConsonant, TibetanSubjoinedConsonant,
     79 
     80     TibetanSubjoinedConsonant, TibetanSubjoinedConsonant, TibetanSubjoinedConsonant, TibetanSubjoinedConsonant,
     81     TibetanSubjoinedConsonant, TibetanSubjoinedConsonant, TibetanSubjoinedConsonant, TibetanSubjoinedConsonant,
     82     TibetanSubjoinedConsonant, TibetanSubjoinedConsonant, TibetanSubjoinedConsonant, TibetanSubjoinedConsonant,
     83     TibetanSubjoinedConsonant, TibetanSubjoinedConsonant, TibetanSubjoinedConsonant, TibetanSubjoinedConsonant,
     84 
     85     TibetanSubjoinedConsonant, TibetanSubjoinedConsonant, TibetanSubjoinedConsonant, TibetanSubjoinedConsonant,
     86     TibetanSubjoinedConsonant, TibetanSubjoinedConsonant, TibetanSubjoinedConsonant, TibetanSubjoinedConsonant,
     87     TibetanSubjoinedConsonant, TibetanSubjoinedConsonant, TibetanSubjoinedConsonant, TibetanSubjoinedConsonant,
     88     TibetanSubjoinedConsonant, TibetanOther, TibetanOther, TibetanOther
     89 };
     90 
     91 
     92 #define tibetan_form(c) \
     93     ((c) >= 0x0f40 && (c) < 0x0fc0 ? (TibetanForm)tibetanForm[(c) - 0x0f40] : TibetanOther)
     94 
     95 static const HB_OpenTypeFeature tibetan_features[] = {
     96     { HB_MAKE_TAG('c', 'c', 'm', 'p'), CcmpProperty },
     97     { HB_MAKE_TAG('a', 'b', 'v', 's'), AboveSubstProperty },
     98     { HB_MAKE_TAG('b', 'l', 'w', 's'), BelowSubstProperty },
     99     { HB_MAKE_TAG('c', 'a', 'l', 't'), CaltProperty },
    100     {0, 0}
    101 };
    102 
    103 static HB_Bool tibetan_shape_syllable(HB_Bool openType, HB_ShaperItem *item, HB_Bool invalid)
    104 {
    105     hb_uint32 i;
    106     const HB_UChar16 *str = item->string + item->item.pos;
    107     int len = item->item.length;
    108 #ifndef NO_OPENTYPE
    109     const int availableGlyphs = item->num_glyphs;
    110 #endif
    111     HB_Bool haveGlyphs;
    112     HB_STACKARRAY(HB_UChar16, reordered, len + 4);
    113 
    114     if (item->num_glyphs < item->item.length + 4) {
    115         item->num_glyphs = item->item.length + 4;
    116         HB_FREE_STACKARRAY(reordered);
    117         return FALSE;
    118     }
    119 
    120     if (invalid) {
    121         *reordered = 0x25cc;
    122         memcpy(reordered+1, str, len*sizeof(HB_UChar16));
    123         len++;
    124         str = reordered;
    125     }
    126 
    127     haveGlyphs = item->font->klass->convertStringToGlyphIndices(item->font,
    128                                                                 str, len,
    129                                                                 item->glyphs, &item->num_glyphs,
    130                                                                 item->item.bidiLevel % 2);
    131 
    132     HB_FREE_STACKARRAY(reordered);
    133 
    134     if (!haveGlyphs)
    135         return FALSE;
    136 
    137     for (i = 0; i < item->item.length; i++) {
    138         item->attributes[i].mark = FALSE;
    139         item->attributes[i].clusterStart = FALSE;
    140         item->attributes[i].justification = 0;
    141         item->attributes[i].zeroWidth = FALSE;
    142 /*        IDEBUG("    %d: %4x", i, str[i]); */
    143     }
    144 
    145     /* now we have the syllable in the right order, and can start running it through open type. */
    146 
    147 #ifndef NO_OPENTYPE
    148     if (openType) {
    149         HB_OpenTypeShape(item, /*properties*/0);
    150         if (!HB_OpenTypePosition(item, availableGlyphs, /*doLogClusters*/FALSE))
    151             return FALSE;
    152     } else {
    153         HB_HeuristicPosition(item);
    154     }
    155 #endif
    156 
    157     item->attributes[0].clusterStart = TRUE;
    158     return TRUE;
    159 }
    160 
    161 
    162 static int tibetan_nextSyllableBoundary(const HB_UChar16 *s, int start, int end, HB_Bool *invalid)
    163 {
    164     const HB_UChar16 *uc = s + start;
    165 
    166     int pos = 0;
    167     TibetanForm state = tibetan_form(*uc);
    168 
    169 /*     qDebug("state[%d]=%d (uc=%4x)", pos, state, uc[pos]);*/
    170     pos++;
    171 
    172     if (state != TibetanHeadConsonant) {
    173         if (state != TibetanOther)
    174             *invalid = TRUE;
    175         goto finish;
    176     }
    177 
    178     while (pos < end - start) {
    179         TibetanForm newState = tibetan_form(uc[pos]);
    180         switch(newState) {
    181         case TibetanSubjoinedConsonant:
    182         case TibetanSubjoinedVowel:
    183             if (state != TibetanHeadConsonant &&
    184                  state != TibetanSubjoinedConsonant)
    185                 goto finish;
    186             state = newState;
    187             break;
    188         case TibetanVowel:
    189             if (state != TibetanHeadConsonant &&
    190                  state != TibetanSubjoinedConsonant &&
    191                  state != TibetanSubjoinedVowel)
    192                 goto finish;
    193             break;
    194         case TibetanOther:
    195         case TibetanHeadConsonant:
    196             goto finish;
    197         }
    198         pos++;
    199     }
    200 
    201 finish:
    202     *invalid = FALSE;
    203     return start+pos;
    204 }
    205 
    206 HB_Bool HB_TibetanShape(HB_ShaperItem *item)
    207 {
    208 
    209     HB_Bool openType = FALSE;
    210     unsigned short *logClusters = item->log_clusters;
    211 
    212     HB_ShaperItem syllable = *item;
    213     int first_glyph = 0;
    214 
    215     int sstart = item->item.pos;
    216     int end = sstart + item->item.length;
    217 
    218     assert(item->item.script == HB_Script_Tibetan);
    219 
    220 #ifndef QT_NO_OPENTYPE
    221     openType = HB_SelectScript(item, tibetan_features);
    222 #endif
    223 
    224     while (sstart < end) {
    225         HB_Bool invalid;
    226         int i;
    227         int send = tibetan_nextSyllableBoundary(item->string, sstart, end, &invalid);
    228 /*        IDEBUG("syllable from %d, length %d, invalid=%s", sstart, send-sstart,
    229                  invalid ? "TRUE" : "FALSE"); */
    230         syllable.item.pos = sstart;
    231         syllable.item.length = send-sstart;
    232         syllable.glyphs = item->glyphs + first_glyph;
    233         syllable.attributes = item->attributes + first_glyph;
    234         syllable.offsets = item->offsets + first_glyph;
    235         syllable.advances = item->advances + first_glyph;
    236         syllable.num_glyphs = item->num_glyphs - first_glyph;
    237         if (!tibetan_shape_syllable(openType, &syllable, invalid)) {
    238             item->num_glyphs += syllable.num_glyphs;
    239             return FALSE;
    240         }
    241         /* fix logcluster array */
    242         for (i = sstart; i < send; ++i)
    243             logClusters[i-item->item.pos] = first_glyph;
    244         sstart = send;
    245         first_glyph += syllable.num_glyphs;
    246     }
    247     item->num_glyphs = first_glyph;
    248     return TRUE;
    249 }
    250 
    251 void HB_TibetanAttributes(HB_Script script, const HB_UChar16 *text, hb_uint32 from, hb_uint32 len, HB_CharAttributes *attributes)
    252 {
    253     int end = from + len;
    254     const HB_UChar16 *uc = text + from;
    255     hb_uint32 i = 0;
    256     HB_UNUSED(script);
    257     attributes += from;
    258     while (i < len) {
    259         HB_Bool invalid;
    260         hb_uint32 boundary = tibetan_nextSyllableBoundary(text, from+i, end, &invalid) - from;
    261 
    262         attributes[i].charStop = TRUE;
    263 
    264         if (boundary > len-1) boundary = len;
    265         i++;
    266         while (i < boundary) {
    267             attributes[i].charStop = FALSE;
    268             ++uc;
    269             ++i;
    270         }
    271         assert(i == boundary);
    272     }
    273 }
    274 
    275 
    276