• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
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 {
FamilyRecFamilyRec63     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 
getSingletonInterface()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 
GetSingletonDirectInterface()158 SkFontConfigInterface* SkFontConfigInterface::GetSingletonDirectInterface() {
159     return getSingletonInterface();
160 }
161 
162 ///////////////////////////////////////////////////////////////////////////////
163 
has_font(const SkTArray<FontRec> & array,const SkString & filename)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 
get_path_for_sys_fonts(SkString * full,const char name[])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 
insert_into_name_dict(SkTDict<FamilyRecID> & familyNameDict,const char * name,FamilyRecID familyRecID)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 
SkFontConfigInterfaceAndroid(SkTDArray<FontFamily * > & fontFamilies)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 
~SkFontConfigInterfaceAndroid()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 
addFallbackFamily(FamilyRecID familyRecID)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 
find_best_style(const FamilyRec & family,SkTypeface::Style style)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 
matchFamilyName(const char familyName[],SkTypeface::Style style,FontIdentity * outFontIdentifier,SkString * outFamilyName,SkTypeface::Style * outStyle)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 
openStream(const FontIdentity & identity)457 SkStream* SkFontConfigInterfaceAndroid::openStream(const FontIdentity& identity) {
458     return SkStream::NewFromFile(identity.fString.c_str());
459 }
460 
getFamilyNames()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 
matchFamilySet(const char inFamilyName[],SkString * outFamilyName,SkTArray<FontIdentity> *)479 bool SkFontConfigInterfaceAndroid::matchFamilySet(const char inFamilyName[],
480                                                   SkString* outFamilyName,
481                                                   SkTArray<FontIdentity>*) {
482     return false;
483 }
484 
find_proc(SkTypeface * face,SkTypeface::Style style,void * ctx)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 
getTypefaceForFontRec(FontRecID fontRecID)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 
getFallbackFamilyNameForChar(SkUnichar uni,const char * lang,SkString * name)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 
getTypefaceForChar(SkUnichar uni,SkTypeface::Style style,SkPaintOptionsAndroid::FontVariant fontVariant)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 
getCurrentLocaleFallbackFontList()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 
findFallbackFontList(const SkLanguage & lang,bool isOriginal)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 
nextLogicalTypeface(SkFontID currFontID,SkFontID origFontID,const SkPaintOptionsAndroid & opts)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 
getTypefaceForGlyphID(uint16_t glyphID,const SkTypeface * origTypeface,const SkPaintOptionsAndroid & opts,int * lBounds,int * uBounds)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 
SkGetFallbackFamilyNameForChar(SkUnichar uni,SkString * name)742 bool SkGetFallbackFamilyNameForChar(SkUnichar uni, SkString* name) {
743     SkFontConfigInterfaceAndroid* fontConfig = getSingletonInterface();
744     return fontConfig->getFallbackFamilyNameForChar(uni, NULL, name);
745 }
746 
SkGetFallbackFamilyNameForChar(SkUnichar uni,const char * lang,SkString * name)747 bool SkGetFallbackFamilyNameForChar(SkUnichar uni, const char* lang, SkString* name) {
748     SkFontConfigInterfaceAndroid* fontConfig = getSingletonInterface();
749     return fontConfig->getFallbackFamilyNameForChar(uni, lang, name);
750 }
751 
SkUseTestFontConfigFile(const char * mainconf,const char * fallbackconf,const char * fontsdir)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 
SkAndroidNextLogicalTypeface(SkFontID currFontID,SkFontID origFontID,const SkPaintOptionsAndroid & options)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 
SkGetTypefaceForGlyphID(uint16_t glyphID,const SkTypeface * origTypeface,const SkPaintOptionsAndroid & options,int * lowerBounds,int * upperBounds)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 
getHBScriptFromHBScriptOld(HB_Script script_old)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"
getUnicodeFromHBScript(hb_script_t script)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 
typefaceLookupCompare(const TypefaceLookupStruct & first,const TypefaceLookupStruct & second)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 
SkCreateTypefaceForScriptNG(hb_script_t script,SkTypeface::Style style,SkPaintOptionsAndroid::FontVariant fontVariant)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 
SkCreateTypefaceForScript(HB_Script script,SkTypeface::Style style,SkPaintOptionsAndroid::FontVariant fontVariant)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 
Factory()932 SkFontMgr* SkFontMgr::Factory() {
933     return NULL;
934 }
935