Home | History | Annotate | Download | only in ports
      1 
      2 /*
      3  * Copyright 2013 The Android Open Source Project
      4  *
      5  * Use of this source code is governed by a BSD-style license that can be
      6  * found in the LICENSE file.
      7  */
      8 
      9 #include "SkFontConfigInterface.h"
     10 #include "SkTypeface_android.h"
     11 
     12 #include "SkFontConfigParser_android.h"
     13 #include "SkFontConfigTypeface.h"
     14 #include "SkFontMgr.h"
     15 #include "SkGlyphCache.h"
     16 #include "SkPaint.h"
     17 #include "SkString.h"
     18 #include "SkStream.h"
     19 #include "SkThread.h"
     20 #include "SkTypefaceCache.h"
     21 #include "SkTArray.h"
     22 #include "SkTDict.h"
     23 #include "SkTSearch.h"
     24 
     25 #include <stdio.h>
     26 #include <string.h>
     27 
     28 #ifndef SK_DEBUG_FONTS
     29     #define SK_DEBUG_FONTS 0
     30 #endif
     31 
     32 #if SK_DEBUG_FONTS
     33     #define DEBUG_FONT(args) SkDebugf args
     34 #else
     35     #define DEBUG_FONT(args)
     36 #endif
     37 
     38 ///////////////////////////////////////////////////////////////////////////////
     39 
     40 // For test only.
     41 static const char* gTestMainConfigFile = NULL;
     42 static const char* gTestFallbackConfigFile = NULL;
     43 static const char* gTestFontFilePrefix = NULL;
     44 
     45 ///////////////////////////////////////////////////////////////////////////////
     46 
     47 typedef int32_t FontRecID;
     48 #define INVALID_FONT_REC_ID -1
     49 
     50 typedef int32_t FamilyRecID;
     51 #define INVALID_FAMILY_REC_ID -1
     52 
     53 // used to record our notion of the pre-existing fonts
     54 struct FontRec {
     55     SkRefPtr<SkTypeface> fTypeface;
     56     SkString fFileName;
     57     SkTypeface::Style fStyle;
     58     bool fIsValid;
     59     FamilyRecID fFamilyRecID;
     60 };
     61 
     62 struct FamilyRec {
     63     FamilyRec() {
     64         memset(fFontRecID, INVALID_FONT_REC_ID, sizeof(fFontRecID));
     65     }
     66 
     67     static const int FONT_STYLE_COUNT = 4;
     68     FontRecID fFontRecID[FONT_STYLE_COUNT];
     69     bool fIsFallbackFont;
     70     SkString fFallbackName;
     71     SkPaintOptionsAndroid fPaintOptions;
     72 };
     73 
     74 
     75 typedef SkTDArray<FamilyRecID> FallbackFontList;
     76 
     77 class SkFontConfigInterfaceAndroid : public SkFontConfigInterface {
     78 public:
     79     SkFontConfigInterfaceAndroid(SkTDArray<FontFamily*>& fontFamilies);
     80     virtual ~SkFontConfigInterfaceAndroid();
     81 
     82     virtual bool matchFamilyName(const char familyName[],
     83                                  SkTypeface::Style requested,
     84                                  FontIdentity* outFontIdentifier,
     85                                  SkString* outFamilyName,
     86                                  SkTypeface::Style* outStyle) SK_OVERRIDE;
     87     virtual SkStream* openStream(const FontIdentity&) SK_OVERRIDE;
     88 
     89     // new APIs
     90     virtual SkDataTable* getFamilyNames() SK_OVERRIDE;
     91     virtual bool matchFamilySet(const char inFamilyName[],
     92                                 SkString* outFamilyName,
     93                                 SkTArray<FontIdentity>*) SK_OVERRIDE;
     94 
     95     /**
     96      *  Get the family name of the font in the default fallback font list that
     97      *  contains the specified chararacter. if no font is found, returns false.
     98      */
     99     bool getFallbackFamilyNameForChar(SkUnichar uni, const char* lang, SkString* name);
    100     /**
    101      *
    102      */
    103     SkTypeface* getTypefaceForChar(SkUnichar uni, SkTypeface::Style style,
    104                                    SkPaintOptionsAndroid::FontVariant fontVariant);
    105     SkTypeface* nextLogicalTypeface(SkFontID currFontID, SkFontID origFontID,
    106                                     const SkPaintOptionsAndroid& options);
    107     SkTypeface* getTypefaceForGlyphID(uint16_t glyphID, const SkTypeface* origTypeface,
    108                                       const SkPaintOptionsAndroid& options,
    109                                       int* lowerBounds, int* upperBounds);
    110 
    111 private:
    112     void addFallbackFamily(FamilyRecID fontRecID);
    113     SkTypeface* getTypefaceForFontRec(FontRecID fontRecID);
    114     FallbackFontList* getCurrentLocaleFallbackFontList();
    115     FallbackFontList* findFallbackFontList(const SkLanguage& lang, bool isOriginal = true);
    116 
    117     SkTArray<FontRec> fFonts;
    118     SkTArray<FamilyRec> fFontFamilies;
    119     SkTDict<FamilyRecID> fFamilyNameDict;
    120     FamilyRecID fDefaultFamilyRecID;
    121 
    122     // (SkLanguage)<->(fallback chain index) translation
    123     SkTDict<FallbackFontList*> fFallbackFontDict;
    124     SkTDict<FallbackFontList*> fFallbackFontAliasDict;
    125     FallbackFontList fDefaultFallbackList;
    126 
    127     // fallback info for current locale
    128     SkString fCachedLocale;
    129     FallbackFontList* fLocaleFallbackFontList;
    130 };
    131 
    132 ///////////////////////////////////////////////////////////////////////////////
    133 
    134 static SkFontConfigInterfaceAndroid* getSingletonInterface() {
    135     SK_DECLARE_STATIC_MUTEX(gMutex);
    136     static SkFontConfigInterfaceAndroid* gFontConfigInterface;
    137 
    138     SkAutoMutexAcquire ac(gMutex);
    139     if (NULL == gFontConfigInterface) {
    140         // load info from a configuration file that we can use to populate the
    141         // system/fallback font structures
    142         SkTDArray<FontFamily*> fontFamilies;
    143         if (!gTestMainConfigFile) {
    144             SkFontConfigParser::GetFontFamilies(fontFamilies);
    145         } else {
    146             SkFontConfigParser::GetTestFontFamilies(fontFamilies, gTestMainConfigFile,
    147                                                     gTestFallbackConfigFile);
    148         }
    149 
    150         gFontConfigInterface = new SkFontConfigInterfaceAndroid(fontFamilies);
    151 
    152         // cleanup the data we received from the parser
    153         fontFamilies.deleteAll();
    154     }
    155     return gFontConfigInterface;
    156 }
    157 
    158 SkFontConfigInterface* SkFontConfigInterface::GetSingletonDirectInterface() {
    159     return getSingletonInterface();
    160 }
    161 
    162 ///////////////////////////////////////////////////////////////////////////////
    163 
    164 static bool has_font(const SkTArray<FontRec>& array, const SkString& filename) {
    165     for (int i = 0; i < array.count(); i++) {
    166         if (array[i].fFileName == filename) {
    167             return true;
    168         }
    169     }
    170     return false;
    171 }
    172 
    173 #ifndef SK_FONT_FILE_PREFIX
    174     #define SK_FONT_FILE_PREFIX          "/fonts/"
    175 #endif
    176 
    177 static void get_path_for_sys_fonts(SkString* full, const char name[]) {
    178     if (gTestFontFilePrefix) {
    179         full->set(gTestFontFilePrefix);
    180     } else {
    181         full->set(getenv("ANDROID_ROOT"));
    182         full->append(SK_FONT_FILE_PREFIX);
    183     }
    184     full->append(name);
    185 }
    186 
    187 static void insert_into_name_dict(SkTDict<FamilyRecID>& familyNameDict,
    188                                   const char* name, FamilyRecID familyRecID) {
    189     SkAutoAsciiToLC tolc(name);
    190     if (familyNameDict.find(tolc.lc())) {
    191         SkDebugf("---- system font attempting to use a the same name [%s] for"
    192                  "multiple families. skipping subsequent occurrences", tolc.lc());
    193     } else {
    194         familyNameDict.set(tolc.lc(), familyRecID);
    195     }
    196 }
    197 
    198 // Defined in SkFontHost_FreeType.cpp
    199 bool find_name_and_attributes(SkStream* stream, SkString* name,
    200                               SkTypeface::Style* style, bool* isFixedWidth);
    201 
    202 ///////////////////////////////////////////////////////////////////////////////
    203 
    204 SkFontConfigInterfaceAndroid::SkFontConfigInterfaceAndroid(SkTDArray<FontFamily*>& fontFamilies) :
    205         fFonts(fontFamilies.count()),
    206         fFontFamilies(fontFamilies.count() / FamilyRec::FONT_STYLE_COUNT),
    207         fFamilyNameDict(1024),
    208         fDefaultFamilyRecID(INVALID_FAMILY_REC_ID),
    209         fFallbackFontDict(128),
    210         fFallbackFontAliasDict(128),
    211         fLocaleFallbackFontList(NULL) {
    212 
    213     for (int i = 0; i < fontFamilies.count(); ++i) {
    214         FontFamily* family = fontFamilies[i];
    215 
    216         // defer initializing the familyRec until we can be sure that at least
    217         // one of it's children contains a valid font file
    218         FamilyRec* familyRec = NULL;
    219         FamilyRecID familyRecID = INVALID_FAMILY_REC_ID;
    220 
    221         for (int j = 0; j < family->fFontFiles.count(); ++j) {
    222             SkString filename;
    223             get_path_for_sys_fonts(&filename, family->fFontFiles[j]->fFileName);
    224 
    225             if (has_font(fFonts, filename)) {
    226                 SkDebugf("---- system font and fallback font files specify a duplicate "
    227                         "font %s, skipping the second occurrence", filename.c_str());
    228                 continue;
    229             }
    230 
    231             FontRec& fontRec = fFonts.push_back();
    232             fontRec.fFileName = filename;
    233             fontRec.fStyle = SkTypeface::kNormal;
    234             fontRec.fIsValid = false;
    235             fontRec.fFamilyRecID = familyRecID;
    236 
    237             const FontRecID fontRecID = fFonts.count() - 1;
    238 
    239             SkAutoTUnref<SkStream> stream(SkStream::NewFromFile(filename.c_str()));
    240             if (stream.get() != NULL) {
    241                 bool isFixedWidth;
    242                 SkString name;
    243                 fontRec.fIsValid = find_name_and_attributes(stream.get(), &name,
    244                                                             &fontRec.fStyle, &isFixedWidth);
    245             } else {
    246                 if (!family->fIsFallbackFont) {
    247                     SkDebugf("---- failed to open <%s> as a font\n", filename.c_str());
    248                 }
    249             }
    250 
    251             if (fontRec.fIsValid) {
    252                 DEBUG_FONT(("---- SystemFonts[%d][%d] fallback=%d file=%s",
    253                            i, fFonts.count() - 1, family->fIsFallbackFont, filename.c_str()));
    254             } else {
    255                 DEBUG_FONT(("---- SystemFonts[%d][%d] fallback=%d file=%s (INVALID)",
    256                            i, fFonts.count() - 1, family->fIsFallbackFont, filename.c_str()));
    257                 continue;
    258             }
    259 
    260             // create a familyRec now that we know that at least one font in
    261             // the family is valid
    262             if (familyRec == NULL) {
    263                 familyRec = &fFontFamilies.push_back();
    264                 familyRecID = fFontFamilies.count() - 1;
    265                 fontRec.fFamilyRecID = familyRecID;
    266 
    267                 familyRec->fIsFallbackFont = family->fIsFallbackFont;
    268                 familyRec->fPaintOptions = family->fFontFiles[j]->fPaintOptions;
    269 
    270             } else if (familyRec->fPaintOptions != family->fFontFiles[j]->fPaintOptions) {
    271                 SkDebugf("Every font file within a family must have identical"
    272                          "language and variant attributes");
    273                 sk_throw();
    274             }
    275 
    276             // add this font to the current familyRec
    277             if (INVALID_FONT_REC_ID != familyRec->fFontRecID[fontRec.fStyle]) {
    278                 DEBUG_FONT(("Overwriting familyRec for style[%d] old,new:(%d,%d)",
    279                             fontRec.fStyle, familyRec->fFontRecID[fontRec.fStyle],
    280                             fontRecID));
    281             }
    282             familyRec->fFontRecID[fontRec.fStyle] = fontRecID;
    283         }
    284 
    285         if (familyRec) {
    286             if (familyRec->fIsFallbackFont) {
    287                 // add the font to the appropriate fallback chains and also insert a
    288                 // unique name into the familyNameDict for internal usage
    289                 addFallbackFamily(familyRecID);
    290             } else {
    291                 // add the names that map to this family to the dictionary for easy lookup
    292                 const SkTDArray<const char*>& names = family->fNames;
    293                 if (names.isEmpty()) {
    294                     SkDEBUGFAIL("ERROR: non-fallback font with no name");
    295                     continue;
    296                 }
    297 
    298                 for (int i = 0; i < names.count(); i++) {
    299                     insert_into_name_dict(fFamilyNameDict, names[i], familyRecID);
    300                 }
    301             }
    302         }
    303     }
    304 
    305     DEBUG_FONT(("---- We have %d system fonts", fFonts.count()));
    306 
    307     if (fFontFamilies.count() > 0) {
    308         fDefaultFamilyRecID = 0;
    309     }
    310 
    311     // scans the default fallback font chain, adding every entry to every other
    312     // fallback font chain to which it does not belong. this results in every
    313     // language-specific fallback font chain having all of its fallback fonts at
    314     // the front of the chain, and everything else at the end.
    315     FallbackFontList* fallbackList;
    316     SkTDict<FallbackFontList*>::Iter iter(fFallbackFontDict);
    317     const char* fallbackLang = iter.next(&fallbackList);
    318     while(fallbackLang != NULL) {
    319         for (int i = 0; i < fDefaultFallbackList.count(); i++) {
    320             FamilyRecID familyRecID = fDefaultFallbackList[i];
    321             const SkString& fontLang = fFontFamilies[familyRecID].fPaintOptions.getLanguage().getTag();
    322             if (strcmp(fallbackLang, fontLang.c_str()) != 0) {
    323                 fallbackList->push(familyRecID);
    324             }
    325         }
    326         // move to the next fallback list in the dictionary
    327         fallbackLang = iter.next(&fallbackList);
    328     }
    329 }
    330 
    331 SkFontConfigInterfaceAndroid::~SkFontConfigInterfaceAndroid() {
    332     // iterate through and cleanup fFallbackFontDict
    333     SkTDict<FallbackFontList*>::Iter iter(fFallbackFontDict);
    334     FallbackFontList* fallbackList;
    335     while(iter.next(&fallbackList) != NULL) {
    336         SkDELETE(fallbackList);
    337     }
    338 }
    339 
    340 void SkFontConfigInterfaceAndroid::addFallbackFamily(FamilyRecID familyRecID) {
    341     SkASSERT(familyRecID < fFontFamilies.count());
    342     FamilyRec& familyRec = fFontFamilies[familyRecID];
    343     SkASSERT(familyRec.fIsFallbackFont);
    344 
    345     // add the fallback family to the name dictionary.  This is
    346     // needed by getFallbackFamilyNameForChar() so that fallback
    347     // families can be identified by a unique name. The unique
    348     // identifier that we've chosen is the familyID in hex (e.g. '0F##fallback').
    349     familyRec.fFallbackName.printf("%.2x##fallback", familyRecID);
    350     insert_into_name_dict(fFamilyNameDict, familyRec.fFallbackName.c_str(), familyRecID);
    351 
    352     // add to the default fallback list
    353     fDefaultFallbackList.push(familyRecID);
    354 
    355     // stop here if it is the default language tag
    356     const SkString& languageTag = familyRec.fPaintOptions.getLanguage().getTag();
    357     if (languageTag.isEmpty()) {
    358         return;
    359     }
    360 
    361     // add to the appropriate language's custom fallback list
    362     FallbackFontList* customList = NULL;
    363     if (!fFallbackFontDict.find(languageTag.c_str(), &customList)) {
    364         DEBUG_FONT(("----  Created fallback list for \"%s\"", languageTag.c_str()));
    365         customList = SkNEW(FallbackFontList);
    366         fFallbackFontDict.set(languageTag.c_str(), customList);
    367     }
    368     SkASSERT(customList != NULL);
    369     customList->push(familyRecID);
    370 }
    371 
    372 
    373 static FontRecID find_best_style(const FamilyRec& family, SkTypeface::Style style) {
    374 
    375     const FontRecID* fontRecIDs = family.fFontRecID;
    376 
    377     if (fontRecIDs[style] != INVALID_FONT_REC_ID) { // exact match
    378         return fontRecIDs[style];
    379     }
    380     // look for a matching bold
    381     style = (SkTypeface::Style)(style ^ SkTypeface::kItalic);
    382     if (fontRecIDs[style] != INVALID_FONT_REC_ID) {
    383         return fontRecIDs[style];
    384     }
    385     // look for the plain
    386     if (fontRecIDs[SkTypeface::kNormal] != INVALID_FONT_REC_ID) {
    387         return fontRecIDs[SkTypeface::kNormal];
    388     }
    389     // look for anything
    390     for (int i = 0; i < FamilyRec::FONT_STYLE_COUNT; i++) {
    391         if (fontRecIDs[i] != INVALID_FONT_REC_ID) {
    392             return fontRecIDs[i];
    393         }
    394     }
    395     // should never get here, since the fontRecID list should not be empty
    396     SkDEBUGFAIL("No valid fonts exist for this family");
    397     return -1;
    398 }
    399 
    400 bool SkFontConfigInterfaceAndroid::matchFamilyName(const char familyName[],
    401                                                    SkTypeface::Style style,
    402                                                    FontIdentity* outFontIdentifier,
    403                                                    SkString* outFamilyName,
    404                                                    SkTypeface::Style* outStyle) {
    405     // clip to legal style bits
    406     style = (SkTypeface::Style)(style & SkTypeface::kBoldItalic);
    407 
    408     bool exactNameMatch = false;
    409 
    410     FamilyRecID familyRecID = INVALID_FAMILY_REC_ID;
    411     if (NULL != familyName) {
    412         SkAutoAsciiToLC tolc(familyName);
    413         if (fFamilyNameDict.find(tolc.lc(), &familyRecID)) {
    414             exactNameMatch = true;
    415         }
    416     } else {
    417         familyRecID = fDefaultFamilyRecID;
    418 
    419     }
    420 
    421     // If no matching family name is found then return false. This allows clients
    422     // to be able to search for other fonts instead of forcing them to use the
    423     // default font.
    424     if (INVALID_FAMILY_REC_ID == familyRecID) {
    425         return false;
    426     }
    427 
    428     FontRecID fontRecID = find_best_style(fFontFamilies[familyRecID], style);
    429     FontRec& fontRec = fFonts[fontRecID];
    430 
    431     if (NULL != outFontIdentifier) {
    432         outFontIdentifier->fID = fontRecID;
    433         outFontIdentifier->fTTCIndex = 0;
    434         outFontIdentifier->fString.set(fontRec.fFileName);
    435 //        outFontIdentifier->fStyle = fontRec.fStyle;
    436     }
    437 
    438     if (NULL != outFamilyName) {
    439         if (exactNameMatch) {
    440             outFamilyName->set(familyName);
    441         } else {
    442             // find familyName from list of names
    443             const char* familyName = NULL;
    444             SkAssertResult(fFamilyNameDict.findKey(familyRecID, &familyName));
    445             SkASSERT(familyName);
    446             outFamilyName->set(familyName);
    447         }
    448     }
    449 
    450     if (NULL != outStyle) {
    451         *outStyle = fontRec.fStyle;
    452     }
    453 
    454     return true;
    455 }
    456 
    457 SkStream* SkFontConfigInterfaceAndroid::openStream(const FontIdentity& identity) {
    458     return SkStream::NewFromFile(identity.fString.c_str());
    459 }
    460 
    461 SkDataTable* SkFontConfigInterfaceAndroid::getFamilyNames() {
    462     SkTDArray<const char*> names;
    463     SkTDArray<size_t> sizes;
    464 
    465     SkTDict<FamilyRecID>::Iter iter(fFamilyNameDict);
    466     const char* familyName = iter.next(NULL);
    467     while(familyName != NULL) {
    468         *names.append() = familyName;
    469         *sizes.append() = strlen(familyName) + 1;
    470 
    471         // move to the next familyName in the dictionary
    472         familyName = iter.next(NULL);
    473     }
    474 
    475     return SkDataTable::NewCopyArrays((const void*const*)names.begin(),
    476                                       sizes.begin(), names.count());
    477 }
    478 
    479 bool SkFontConfigInterfaceAndroid::matchFamilySet(const char inFamilyName[],
    480                                                   SkString* outFamilyName,
    481                                                   SkTArray<FontIdentity>*) {
    482     return false;
    483 }
    484 
    485 static bool find_proc(SkTypeface* face, SkTypeface::Style style, void* ctx) {
    486     const FontRecID* fontRecID = (const FontRecID*)ctx;
    487     FontRecID currFontRecID = ((FontConfigTypeface*)face)->getIdentity().fID;
    488     return currFontRecID == *fontRecID;
    489 }
    490 
    491 SkTypeface* SkFontConfigInterfaceAndroid::getTypefaceForFontRec(FontRecID fontRecID) {
    492     FontRec& fontRec = fFonts[fontRecID];
    493     SkTypeface* face = fontRec.fTypeface.get();
    494     if (!face) {
    495         // look for it in the typeface cache
    496         face = SkTypefaceCache::FindByProcAndRef(find_proc, &fontRecID);
    497 
    498         // if it is not in the cache then create it
    499         if (!face) {
    500             const char* familyName = NULL;
    501             SkAssertResult(fFamilyNameDict.findKey(fontRec.fFamilyRecID, &familyName));
    502             SkASSERT(familyName);
    503             face = SkTypeface::CreateFromName(familyName, fontRec.fStyle);
    504         }
    505 
    506         // store the result for subsequent lookups
    507         fontRec.fTypeface = face;
    508     }
    509     SkASSERT(face);
    510     return face;
    511 }
    512 
    513 bool SkFontConfigInterfaceAndroid::getFallbackFamilyNameForChar(SkUnichar uni,
    514                                                                 const char* lang,
    515                                                                 SkString* name) {
    516     FallbackFontList* fallbackFontList = NULL;
    517     const SkString langTag(lang);
    518     if (langTag.isEmpty()) {
    519         fallbackFontList = this->getCurrentLocaleFallbackFontList();
    520     } else {
    521         fallbackFontList = this->findFallbackFontList(langTag);
    522     }
    523 
    524     for (int i = 0; i < fallbackFontList->count(); i++) {
    525         FamilyRecID familyRecID = fallbackFontList->getAt(i);
    526 
    527         // if it is not one of the accepted variants then move to the next family
    528         int32_t acceptedVariants = SkPaintOptionsAndroid::kDefault_Variant |
    529                                    SkPaintOptionsAndroid::kElegant_Variant;
    530         if (!(fFontFamilies[familyRecID].fPaintOptions.getFontVariant() & acceptedVariants)) {
    531             continue;
    532         }
    533 
    534         FontRecID fontRecID = find_best_style(fFontFamilies[familyRecID], SkTypeface::kNormal);
    535         SkTypeface* face = this->getTypefaceForFontRec(fontRecID);
    536 
    537         SkPaint paint;
    538         paint.setTypeface(face);
    539         paint.setTextEncoding(SkPaint::kUTF32_TextEncoding);
    540 
    541         uint16_t glyphID;
    542         paint.textToGlyphs(&uni, sizeof(uni), &glyphID);
    543         if (glyphID != 0) {
    544             name->set(fFontFamilies[familyRecID].fFallbackName);
    545             return true;
    546         }
    547     }
    548     return false;
    549 }
    550 
    551 SkTypeface* SkFontConfigInterfaceAndroid::getTypefaceForChar(SkUnichar uni,
    552                                                              SkTypeface::Style style,
    553                                                              SkPaintOptionsAndroid::FontVariant fontVariant) {
    554     FontRecID fontRecID = find_best_style(fFontFamilies[fDefaultFamilyRecID], style);
    555     SkTypeface* face = this->getTypefaceForFontRec(fontRecID);
    556 
    557     SkPaintOptionsAndroid paintOptions;
    558     paintOptions.setFontVariant(fontVariant);
    559     paintOptions.setUseFontFallbacks(true);
    560 
    561     SkPaint paint;
    562     paint.setTypeface(face);
    563     paint.setTextEncoding(SkPaint::kUTF16_TextEncoding);
    564     paint.setPaintOptionsAndroid(paintOptions);
    565 
    566     SkAutoGlyphCache autoCache(paint, NULL, NULL);
    567     SkGlyphCache*    cache = autoCache.getCache();
    568 
    569     SkScalerContext* ctx = cache->getScalerContext();
    570     if (ctx) {
    571         SkFontID fontID = ctx->findTypefaceIdForChar(uni);
    572         return SkTypefaceCache::FindByID(fontID);
    573     }
    574     return NULL;
    575 }
    576 
    577 FallbackFontList* SkFontConfigInterfaceAndroid::getCurrentLocaleFallbackFontList() {
    578     SkString locale = SkFontConfigParser::GetLocale();
    579     if (NULL == fLocaleFallbackFontList || locale != fCachedLocale) {
    580         fCachedLocale = locale;
    581         fLocaleFallbackFontList = this->findFallbackFontList(locale);
    582     }
    583     return fLocaleFallbackFontList;
    584 }
    585 
    586 FallbackFontList* SkFontConfigInterfaceAndroid::findFallbackFontList(const SkLanguage& lang,
    587                                                                      bool isOriginal) {
    588     const SkString& langTag = lang.getTag();
    589     if (langTag.isEmpty()) {
    590         return &fDefaultFallbackList;
    591     }
    592 
    593     FallbackFontList* fallbackFontList;
    594     if (fFallbackFontDict.find(langTag.c_str(), langTag.size(), &fallbackFontList) ||
    595         fFallbackFontAliasDict.find(langTag.c_str(), langTag.size(), &fallbackFontList)) {
    596         return fallbackFontList;
    597     }
    598 
    599     // attempt a recursive fuzzy match
    600     SkLanguage parent = lang.getParent();
    601     fallbackFontList = findFallbackFontList(parent, false);
    602 
    603     // cache the original lang so we don't have to do the recursion again.
    604     if (isOriginal) {
    605         DEBUG_FONT(("----  Created fallback list alias for \"%s\"", langTag.c_str()));
    606         fFallbackFontAliasDict.set(langTag.c_str(), fallbackFontList);
    607     }
    608     return fallbackFontList;
    609 }
    610 
    611 SkTypeface* SkFontConfigInterfaceAndroid::nextLogicalTypeface(SkFontID currFontID,
    612                                                               SkFontID origFontID,
    613                                                               const SkPaintOptionsAndroid& opts) {
    614     // Skia does not support font fallback by default. This enables clients such
    615     // as WebKit to customize their font selection. In any case, clients can use
    616     // GetFallbackFamilyNameForChar() to get the fallback font for individual
    617     // characters.
    618     if (!opts.isUsingFontFallbacks()) {
    619         return NULL;
    620     }
    621 
    622     FallbackFontList* currentFallbackList = findFallbackFontList(opts.getLanguage());
    623     SkASSERT(currentFallbackList);
    624 
    625     SkTypeface::Style origStyle = SkTypeface::kNormal;
    626     const SkTypeface* origTypeface = SkTypefaceCache::FindByID(origFontID);
    627     if (NULL != origTypeface) {
    628         origStyle = origTypeface->style();
    629     }
    630 
    631     // we must convert currTypeface into a FontRecID
    632     FontRecID currFontRecID = INVALID_FONT_REC_ID;
    633     const SkTypeface* currTypeface = SkTypefaceCache::FindByID(currFontID);
    634     // non-system fonts are not in the font cache so if we are asked to fallback
    635     // for a non-system font we will start at the front of the chain.
    636     if (NULL != currTypeface) {
    637         currFontRecID = ((FontConfigTypeface*)currTypeface)->getIdentity().fID;
    638         SkASSERT(INVALID_FONT_REC_ID != currFontRecID);
    639     }
    640 
    641     FamilyRecID currFamilyRecID = INVALID_FAMILY_REC_ID;
    642     if (INVALID_FONT_REC_ID != currFontRecID) {
    643         currFamilyRecID = fFonts[currFontRecID].fFamilyRecID;
    644     }
    645 
    646     // lookup the index next font in the chain
    647     int currFallbackFontIndex = currentFallbackList->find(currFamilyRecID);
    648     // We add 1 to the returned index for 2 reasons: (1) if find succeeds it moves
    649     // our index to the next entry in the list; (2) if find() fails it returns
    650     // -1 and incrementing it will set our starting index to 0 (the head of the list)
    651     int nextFallbackFontIndex = currFallbackFontIndex + 1;
    652 
    653     if(nextFallbackFontIndex >= currentFallbackList->count()) {
    654         return NULL;
    655     }
    656 
    657     // If a rec object is set to prefer "kDefault_Variant" it means they have no preference
    658     // In this case, we set the value to "kCompact_Variant"
    659     SkPaintOptionsAndroid::FontVariant variant = opts.getFontVariant();
    660     if (variant == SkPaintOptionsAndroid::kDefault_Variant) {
    661         variant = SkPaintOptionsAndroid::kCompact_Variant;
    662     }
    663 
    664     int32_t acceptedVariants = SkPaintOptionsAndroid::kDefault_Variant | variant;
    665 
    666     SkTypeface* nextLogicalTypeface = 0;
    667     while (nextFallbackFontIndex < currentFallbackList->count()) {
    668         FamilyRecID familyRecID = currentFallbackList->getAt(nextFallbackFontIndex);
    669         if ((fFontFamilies[familyRecID].fPaintOptions.getFontVariant() & acceptedVariants) != 0) {
    670             FontRecID matchedFont = find_best_style(fFontFamilies[familyRecID], origStyle);
    671             nextLogicalTypeface = this->getTypefaceForFontRec(matchedFont);
    672             break;
    673         }
    674         nextFallbackFontIndex++;
    675     }
    676 
    677     DEBUG_FONT(("---- nextLogicalFont: currFontID=%d, origFontID=%d, currRecID=%d, "
    678                 "lang=%s, variant=%d, nextFallbackIndex[%d,%d] => nextLogicalTypeface=%d",
    679                 currFontID, origFontID, currFontRecID, opts.getLanguage().getTag().c_str(),
    680                 variant, nextFallbackFontIndex, currentFallbackList->getAt(nextFallbackFontIndex),
    681                 (nextLogicalTypeface) ? nextLogicalTypeface->uniqueID() : 0));
    682     return SkSafeRef(nextLogicalTypeface);
    683 }
    684 
    685 SkTypeface* SkFontConfigInterfaceAndroid::getTypefaceForGlyphID(uint16_t glyphID,
    686                                                                 const SkTypeface* origTypeface,
    687                                                                 const SkPaintOptionsAndroid& opts,
    688                                                                 int* lBounds, int* uBounds) {
    689     // If we aren't using fallbacks then we shouldn't be calling this
    690     SkASSERT(opts.isUsingFontFallbacks());
    691     SkASSERT(origTypeface);
    692 
    693     SkTypeface* currentTypeface = NULL;
    694     int lowerBounds = 0; //inclusive
    695     int upperBounds = origTypeface->countGlyphs(); //exclusive
    696 
    697     // check to see if the glyph is in the bounds of the origTypeface
    698     if (glyphID < upperBounds) {
    699         currentTypeface = const_cast<SkTypeface*>(origTypeface);
    700     } else {
    701         FallbackFontList* currentFallbackList = findFallbackFontList(opts.getLanguage());
    702         SkASSERT(currentFallbackList);
    703 
    704         // If an object is set to prefer "kDefault_Variant" it means they have no preference
    705         // In this case, we set the value to "kCompact_Variant"
    706         SkPaintOptionsAndroid::FontVariant variant = opts.getFontVariant();
    707         if (variant == SkPaintOptionsAndroid::kDefault_Variant) {
    708             variant = SkPaintOptionsAndroid::kCompact_Variant;
    709         }
    710 
    711         int32_t acceptedVariants = SkPaintOptionsAndroid::kDefault_Variant | variant;
    712         SkTypeface::Style origStyle = origTypeface->style();
    713 
    714         for (int x = 0; x < currentFallbackList->count(); ++x) {
    715             const FamilyRecID familyRecID = currentFallbackList->getAt(x);
    716             const SkPaintOptionsAndroid& familyOptions = fFontFamilies[familyRecID].fPaintOptions;
    717             if ((familyOptions.getFontVariant() & acceptedVariants) != 0) {
    718                 FontRecID matchedFont = find_best_style(fFontFamilies[familyRecID], origStyle);
    719                 currentTypeface = this->getTypefaceForFontRec(matchedFont);
    720                 lowerBounds = upperBounds;
    721                 upperBounds += currentTypeface->countGlyphs();
    722                 if (glyphID < upperBounds) {
    723                     break;
    724                 }
    725             }
    726         }
    727     }
    728 
    729     if (NULL != currentTypeface) {
    730         if (lBounds) {
    731             *lBounds = lowerBounds;
    732         }
    733         if (uBounds) {
    734             *uBounds = upperBounds;
    735         }
    736     }
    737     return currentTypeface;
    738 }
    739 
    740 ///////////////////////////////////////////////////////////////////////////////
    741 
    742 bool SkGetFallbackFamilyNameForChar(SkUnichar uni, SkString* name) {
    743     SkFontConfigInterfaceAndroid* fontConfig = getSingletonInterface();
    744     return fontConfig->getFallbackFamilyNameForChar(uni, NULL, name);
    745 }
    746 
    747 bool SkGetFallbackFamilyNameForChar(SkUnichar uni, const char* lang, SkString* name) {
    748     SkFontConfigInterfaceAndroid* fontConfig = getSingletonInterface();
    749     return fontConfig->getFallbackFamilyNameForChar(uni, lang, name);
    750 }
    751 
    752 void SkUseTestFontConfigFile(const char* mainconf, const char* fallbackconf,
    753                              const char* fontsdir) {
    754     gTestMainConfigFile = mainconf;
    755     gTestFallbackConfigFile = fallbackconf;
    756     gTestFontFilePrefix = fontsdir;
    757     SkASSERT(gTestMainConfigFile);
    758     SkASSERT(gTestFallbackConfigFile);
    759     SkASSERT(gTestFontFilePrefix);
    760     SkDEBUGF(("Use Test Config File Main %s, Fallback %s, Font Dir %s",
    761               gTestMainConfigFile, gTestFallbackConfigFile, gTestFontFilePrefix));
    762 }
    763 
    764 SkTypeface* SkAndroidNextLogicalTypeface(SkFontID currFontID, SkFontID origFontID,
    765                                          const SkPaintOptionsAndroid& options) {
    766     SkFontConfigInterfaceAndroid* fontConfig = getSingletonInterface();
    767     return fontConfig->nextLogicalTypeface(currFontID, origFontID, options);
    768 
    769 }
    770 
    771 SkTypeface* SkGetTypefaceForGlyphID(uint16_t glyphID, const SkTypeface* origTypeface,
    772                                     const SkPaintOptionsAndroid& options,
    773                                     int* lowerBounds, int* upperBounds) {
    774     SkFontConfigInterfaceAndroid* fontConfig = getSingletonInterface();
    775     return fontConfig->getTypefaceForGlyphID(glyphID, origTypeface, options,
    776                                              lowerBounds, upperBounds);
    777 }
    778 
    779 ///////////////////////////////////////////////////////////////////////////////
    780 
    781 #ifdef SK_BUILD_FOR_ANDROID_FRAMEWORK
    782 
    783 struct HB_UnicodeMapping {
    784     // TODO: when the WebView no longer needs harfbuzz_old, remove
    785     HB_Script script_old;
    786     hb_script_t script;
    787     const SkUnichar unicode;
    788 };
    789 
    790 /*
    791  * The following scripts are not complex fonts and we do not expect them to be parsed by this table
    792  * HB_SCRIPT_COMMON,
    793  * HB_SCRIPT_GREEK,
    794  * HB_SCRIPT_CYRILLIC,
    795  * HB_SCRIPT_HANGUL
    796  * HB_SCRIPT_INHERITED
    797  */
    798 
    799 /* Harfbuzz (old) is missing a number of scripts in its table. For these,
    800  * we include a value which can never happen. We won't get complex script
    801  * shaping in these cases, but the library wouldn't know how to shape
    802  * them anyway. */
    803 #define HB_Script_Unknown HB_ScriptCount
    804 
    805 static HB_UnicodeMapping HB_UnicodeMappingArray[] = {
    806     {HB_Script_Armenian,   HB_SCRIPT_ARMENIAN,    0x0531},
    807     {HB_Script_Hebrew,     HB_SCRIPT_HEBREW,      0x0591},
    808     {HB_Script_Arabic,     HB_SCRIPT_ARABIC,      0x0600},
    809     {HB_Script_Syriac,     HB_SCRIPT_SYRIAC,      0x0710},
    810     {HB_Script_Thaana,     HB_SCRIPT_THAANA,      0x0780},
    811     {HB_Script_Nko,        HB_SCRIPT_NKO,         0x07C0},
    812     {HB_Script_Devanagari, HB_SCRIPT_DEVANAGARI,  0x0901},
    813     {HB_Script_Bengali,    HB_SCRIPT_BENGALI,     0x0981},
    814     {HB_Script_Gurmukhi,   HB_SCRIPT_GURMUKHI,    0x0A10},
    815     {HB_Script_Gujarati,   HB_SCRIPT_GUJARATI,    0x0A90},
    816     {HB_Script_Oriya,      HB_SCRIPT_ORIYA,       0x0B10},
    817     {HB_Script_Tamil,      HB_SCRIPT_TAMIL,       0x0B82},
    818     {HB_Script_Telugu,     HB_SCRIPT_TELUGU,      0x0C10},
    819     {HB_Script_Kannada,    HB_SCRIPT_KANNADA,     0x0C90},
    820     {HB_Script_Malayalam,  HB_SCRIPT_MALAYALAM,   0x0D10},
    821     {HB_Script_Sinhala,    HB_SCRIPT_SINHALA,     0x0D90},
    822     {HB_Script_Thai,       HB_SCRIPT_THAI,        0x0E01},
    823     {HB_Script_Lao,        HB_SCRIPT_LAO,         0x0E81},
    824     {HB_Script_Tibetan,    HB_SCRIPT_TIBETAN,     0x0F00},
    825     {HB_Script_Myanmar,    HB_SCRIPT_MYANMAR,     0x1000},
    826     {HB_Script_Georgian,   HB_SCRIPT_GEORGIAN,    0x10A0},
    827     {HB_Script_Unknown,    HB_SCRIPT_ETHIOPIC,    0x1200},
    828     {HB_Script_Unknown,    HB_SCRIPT_CHEROKEE,    0x13A0},
    829     {HB_Script_Ogham,      HB_SCRIPT_OGHAM,       0x1680},
    830     {HB_Script_Runic,      HB_SCRIPT_RUNIC,       0x16A0},
    831     {HB_Script_Khmer,      HB_SCRIPT_KHMER,       0x1780},
    832     {HB_Script_Unknown,    HB_SCRIPT_TAI_LE,      0x1950},
    833     {HB_Script_Unknown,    HB_SCRIPT_NEW_TAI_LUE, 0x1980},
    834     {HB_Script_Unknown,    HB_SCRIPT_TAI_THAM,    0x1A20},
    835     {HB_Script_Unknown,    HB_SCRIPT_CHAM,        0xAA00},
    836 };
    837 
    838 static hb_script_t getHBScriptFromHBScriptOld(HB_Script script_old) {
    839     hb_script_t script = HB_SCRIPT_INVALID;
    840     int numSupportedFonts = sizeof(HB_UnicodeMappingArray) / sizeof(HB_UnicodeMapping);
    841     for (int i = 0; i < numSupportedFonts; i++) {
    842         if (script_old == HB_UnicodeMappingArray[i].script_old) {
    843             script = HB_UnicodeMappingArray[i].script;
    844             break;
    845         }
    846     }
    847     return script;
    848 }
    849 
    850 // returns 0 for "Not Found"
    851 static SkUnichar getUnicodeFromHBScript(hb_script_t script) {
    852     SkUnichar unichar = 0;
    853     int numSupportedFonts = sizeof(HB_UnicodeMappingArray) / sizeof(HB_UnicodeMapping);
    854     for (int i = 0; i < numSupportedFonts; i++) {
    855         if (script == HB_UnicodeMappingArray[i].script) {
    856             unichar = HB_UnicodeMappingArray[i].unicode;
    857             break;
    858         }
    859     }
    860     return unichar;
    861 }
    862 
    863 struct TypefaceLookupStruct {
    864     hb_script_t script;
    865     SkTypeface::Style style;
    866     SkPaintOptionsAndroid::FontVariant fontVariant;
    867     SkTypeface* typeface;
    868 };
    869 
    870 SK_DECLARE_STATIC_MUTEX(gTypefaceTableMutex);  // This is the mutex for gTypefaceTable
    871 static SkTDArray<TypefaceLookupStruct> gTypefaceTable;  // This is protected by gTypefaceTableMutex
    872 
    873 static int typefaceLookupCompare(const TypefaceLookupStruct& first,
    874                                  const TypefaceLookupStruct& second) {
    875     if (first.script != second.script) {
    876         return (first.script > second.script) ? 1 : -1;
    877     }
    878     if (first.style != second.style) {
    879         return (first.style > second.style) ? 1 : -1;
    880     }
    881     if (first.fontVariant != second.fontVariant) {
    882         return (first.fontVariant > second.fontVariant) ? 1 : -1;
    883     }
    884     return 0;
    885 }
    886 
    887 SkTypeface* SkCreateTypefaceForScriptNG(hb_script_t script, SkTypeface::Style style,
    888                                         SkPaintOptionsAndroid::FontVariant fontVariant) {
    889     SkAutoMutexAcquire ac(gTypefaceTableMutex);
    890 
    891     TypefaceLookupStruct key;
    892     key.script = script;
    893     key.style = style;
    894     key.fontVariant = fontVariant;
    895 
    896     int index = SkTSearch<TypefaceLookupStruct>(
    897             (const TypefaceLookupStruct*) gTypefaceTable.begin(),
    898             gTypefaceTable.count(), key, sizeof(TypefaceLookupStruct),
    899             typefaceLookupCompare);
    900 
    901     SkTypeface* retTypeface = NULL;
    902     if (index >= 0) {
    903         retTypeface = gTypefaceTable[index].typeface;
    904     }
    905     else {
    906         SkUnichar unichar = getUnicodeFromHBScript(script);
    907         if (!unichar) {
    908             return NULL;
    909         }
    910 
    911         SkFontConfigInterfaceAndroid* fontConfig = getSingletonInterface();
    912         retTypeface = fontConfig->getTypefaceForChar(unichar, style, fontVariant);
    913 
    914         // add to the lookup table
    915         key.typeface = retTypeface;
    916         *gTypefaceTable.insert(~index) = key;
    917     }
    918 
    919     // we ref(), the caller is expected to unref when they are done
    920     return SkSafeRef(retTypeface);
    921 }
    922 
    923 SkTypeface* SkCreateTypefaceForScript(HB_Script script, SkTypeface::Style style,
    924                                       SkPaintOptionsAndroid::FontVariant fontVariant) {
    925     return SkCreateTypefaceForScriptNG(getHBScriptFromHBScriptOld(script), style, fontVariant);
    926 }
    927 
    928 #endif
    929 
    930 ///////////////////////////////////////////////////////////////////////////////
    931 
    932 SkFontMgr* SkFontMgr::Factory() {
    933     return NULL;
    934 }
    935