Home | History | Annotate | Download | only in ports
      1 /*
      2  * Copyright 2014 Google Inc.
      3  *
      4  * Use of this source code is governed by a BSD-style license that can be
      5  * found in the LICENSE file.
      6  */
      7 #include "SkDWriteNTDDI_VERSION.h"
      8 
      9 #include "SkTypes.h"
     10 #if defined(SK_BUILD_FOR_WIN)
     11 
     12 // SkTypes will include Windows.h, which will pull in all of the GDI defines.
     13 // GDI #defines GetGlyphIndices to GetGlyphIndicesA or GetGlyphIndicesW, but
     14 // IDWriteFontFace has a method called GetGlyphIndices. Since this file does
     15 // not use GDI, undefing GetGlyphIndices makes things less confusing.
     16 #undef GetGlyphIndices
     17 
     18 #include "SkDWrite.h"
     19 #include "SkDWriteFontFileStream.h"
     20 #include "SkFontDescriptor.h"
     21 #include "SkFontStream.h"
     22 #include "SkOTTable_OS_2.h"
     23 #include "SkOTTable_fvar.h"
     24 #include "SkOTTable_head.h"
     25 #include "SkOTTable_hhea.h"
     26 #include "SkOTTable_post.h"
     27 #include "SkOTUtils.h"
     28 #include "SkScalerContext.h"
     29 #include "SkScalerContext_win_dw.h"
     30 #include "SkTo.h"
     31 #include "SkTypeface_win_dw.h"
     32 #include "SkUtils.h"
     33 
     34 void DWriteFontTypeface::onGetFamilyName(SkString* familyName) const {
     35     SkTScopedComPtr<IDWriteLocalizedStrings> familyNames;
     36     HRV(fDWriteFontFamily->GetFamilyNames(&familyNames));
     37 
     38     sk_get_locale_string(familyNames.get(), nullptr/*fMgr->fLocaleName.get()*/, familyName);
     39 }
     40 
     41 void DWriteFontTypeface::onGetFontDescriptor(SkFontDescriptor* desc,
     42                                              bool* isLocalStream) const {
     43     // Get the family name.
     44     SkTScopedComPtr<IDWriteLocalizedStrings> familyNames;
     45     HRV(fDWriteFontFamily->GetFamilyNames(&familyNames));
     46 
     47     SkString utf8FamilyName;
     48     sk_get_locale_string(familyNames.get(), nullptr/*fMgr->fLocaleName.get()*/, &utf8FamilyName);
     49 
     50     desc->setFamilyName(utf8FamilyName.c_str());
     51     desc->setStyle(this->fontStyle());
     52     *isLocalStream = SkToBool(fDWriteFontFileLoader.get());
     53 }
     54 
     55 static SkUnichar next_utf8(const void** chars) {
     56     return SkUTF8_NextUnichar((const char**)chars);
     57 }
     58 
     59 static SkUnichar next_utf16(const void** chars) {
     60     return SkUTF16_NextUnichar((const uint16_t**)chars);
     61 }
     62 
     63 static SkUnichar next_utf32(const void** chars) {
     64     const SkUnichar** uniChars = (const SkUnichar**)chars;
     65     SkUnichar uni = **uniChars;
     66     *uniChars += 1;
     67     return uni;
     68 }
     69 
     70 typedef SkUnichar (*EncodingProc)(const void**);
     71 
     72 static EncodingProc find_encoding_proc(SkTypeface::Encoding enc) {
     73     static const EncodingProc gProcs[] = {
     74         next_utf8, next_utf16, next_utf32
     75     };
     76     SkASSERT((size_t)enc < SK_ARRAY_COUNT(gProcs));
     77     return gProcs[enc];
     78 }
     79 
     80 int DWriteFontTypeface::onCharsToGlyphs(const void* chars, Encoding encoding,
     81                                         uint16_t glyphs[], int glyphCount) const
     82 {
     83     if (nullptr == glyphs) {
     84         EncodingProc next_ucs4_proc = find_encoding_proc(encoding);
     85         for (int i = 0; i < glyphCount; ++i) {
     86             const SkUnichar c = next_ucs4_proc(&chars);
     87             BOOL exists;
     88             fDWriteFont->HasCharacter(c, &exists);
     89             if (!exists) {
     90                 return i;
     91             }
     92         }
     93         return glyphCount;
     94     }
     95 
     96     switch (encoding) {
     97     case SkTypeface::kUTF8_Encoding:
     98     case SkTypeface::kUTF16_Encoding: {
     99         static const int scratchCount = 256;
    100         UINT32 scratch[scratchCount];
    101         EncodingProc next_ucs4_proc = find_encoding_proc(encoding);
    102         for (int baseGlyph = 0; baseGlyph < glyphCount; baseGlyph += scratchCount) {
    103             int glyphsLeft = glyphCount - baseGlyph;
    104             int limit = SkTMin(glyphsLeft, scratchCount);
    105             for (int i = 0; i < limit; ++i) {
    106                 scratch[i] = next_ucs4_proc(&chars);
    107             }
    108             fDWriteFontFace->GetGlyphIndices(scratch, limit, &glyphs[baseGlyph]);
    109         }
    110         break;
    111     }
    112     case SkTypeface::kUTF32_Encoding: {
    113         const UINT32* utf32 = reinterpret_cast<const UINT32*>(chars);
    114         fDWriteFontFace->GetGlyphIndices(utf32, glyphCount, glyphs);
    115         break;
    116     }
    117     default:
    118         SK_ABORT("Invalid Text Encoding");
    119     }
    120 
    121     for (int i = 0; i < glyphCount; ++i) {
    122         if (0 == glyphs[i]) {
    123             return i;
    124         }
    125     }
    126     return glyphCount;
    127 }
    128 
    129 int DWriteFontTypeface::onCountGlyphs() const {
    130     return fDWriteFontFace->GetGlyphCount();
    131 }
    132 
    133 int DWriteFontTypeface::onGetUPEM() const {
    134     DWRITE_FONT_METRICS metrics;
    135     fDWriteFontFace->GetMetrics(&metrics);
    136     return metrics.designUnitsPerEm;
    137 }
    138 
    139 class LocalizedStrings_IDWriteLocalizedStrings : public SkTypeface::LocalizedStrings {
    140 public:
    141     /** Takes ownership of the IDWriteLocalizedStrings. */
    142     explicit LocalizedStrings_IDWriteLocalizedStrings(IDWriteLocalizedStrings* strings)
    143         : fIndex(0), fStrings(strings)
    144     { }
    145 
    146     bool next(SkTypeface::LocalizedString* localizedString) override {
    147         if (fIndex >= fStrings->GetCount()) {
    148             return false;
    149         }
    150 
    151         // String
    152         UINT32 stringLen;
    153         HRBM(fStrings->GetStringLength(fIndex, &stringLen), "Could not get string length.");
    154 
    155         SkSMallocWCHAR wString(stringLen+1);
    156         HRBM(fStrings->GetString(fIndex, wString.get(), stringLen+1), "Could not get string.");
    157 
    158         HRB(sk_wchar_to_skstring(wString.get(), stringLen, &localizedString->fString));
    159 
    160         // Locale
    161         UINT32 localeLen;
    162         HRBM(fStrings->GetLocaleNameLength(fIndex, &localeLen), "Could not get locale length.");
    163 
    164         SkSMallocWCHAR wLocale(localeLen+1);
    165         HRBM(fStrings->GetLocaleName(fIndex, wLocale.get(), localeLen+1), "Could not get locale.");
    166 
    167         HRB(sk_wchar_to_skstring(wLocale.get(), localeLen, &localizedString->fLanguage));
    168 
    169         ++fIndex;
    170         return true;
    171     }
    172 
    173 private:
    174     UINT32 fIndex;
    175     SkTScopedComPtr<IDWriteLocalizedStrings> fStrings;
    176 };
    177 
    178 SkTypeface::LocalizedStrings* DWriteFontTypeface::onCreateFamilyNameIterator() const {
    179     sk_sp<SkTypeface::LocalizedStrings> nameIter =
    180         SkOTUtils::LocalizedStrings_NameTable::MakeForFamilyNames(*this);
    181     if (!nameIter) {
    182         SkTScopedComPtr<IDWriteLocalizedStrings> familyNames;
    183         HRNM(fDWriteFontFamily->GetFamilyNames(&familyNames), "Could not obtain family names.");
    184         nameIter = sk_make_sp<LocalizedStrings_IDWriteLocalizedStrings>(familyNames.release());
    185     }
    186     return nameIter.release();
    187 }
    188 
    189 int DWriteFontTypeface::onGetVariationDesignPosition(
    190     SkFontArguments::VariationPosition::Coordinate coordinates[], int coordinateCount) const
    191 {
    192 
    193 #if defined(NTDDI_WIN10_RS3) && NTDDI_VERSION >= NTDDI_WIN10_RS3
    194 
    195     SkTScopedComPtr<IDWriteFontFace5> fontFace5;
    196     if (FAILED(fDWriteFontFace->QueryInterface(&fontFace5))) {
    197         return -1;
    198     }
    199 
    200     // Return 0 if the font is not variable font.
    201     if (!fontFace5->HasVariations()) {
    202         return 0;
    203     }
    204 
    205     UINT32 fontAxisCount = fontFace5->GetFontAxisValueCount();
    206     SkTScopedComPtr<IDWriteFontResource> fontResource;
    207     HR_GENERAL(fontFace5->GetFontResource(&fontResource), nullptr, -1);
    208     int variableAxisCount = 0;
    209     for (UINT32 i = 0; i < fontAxisCount; ++i) {
    210         if (fontResource->GetFontAxisAttributes(i) & DWRITE_FONT_AXIS_ATTRIBUTES_VARIABLE) {
    211             variableAxisCount++;
    212         }
    213     }
    214 
    215     if (!coordinates || coordinateCount < variableAxisCount) {
    216         return variableAxisCount;
    217     }
    218 
    219     SkAutoSTMalloc<8, DWRITE_FONT_AXIS_VALUE> fontAxisValue(fontAxisCount);
    220     HR_GENERAL(fontFace5->GetFontAxisValues(fontAxisValue.get(), fontAxisCount), nullptr, -1);
    221     UINT32 coordIndex = 0;
    222 
    223     for (UINT32 axisIndex = 0; axisIndex < fontAxisCount; ++axisIndex) {
    224         if (fontResource->GetFontAxisAttributes(axisIndex) & DWRITE_FONT_AXIS_ATTRIBUTES_VARIABLE) {
    225             coordinates[coordIndex].axis = SkEndian_SwapBE32(fontAxisValue[axisIndex].axisTag);
    226             coordinates[coordIndex].value = fontAxisValue[axisIndex].value;
    227         }
    228     }
    229 
    230     return variableAxisCount;
    231 
    232 #endif
    233 
    234     return -1;
    235 }
    236 
    237 int DWriteFontTypeface::onGetVariationDesignParameters(
    238     SkFontParameters::Variation::Axis parameters[], int parameterCount) const
    239 {
    240 
    241 #if defined(NTDDI_WIN10_RS3) && NTDDI_VERSION >= NTDDI_WIN10_RS3
    242 
    243     SkTScopedComPtr<IDWriteFontFace5> fontFace5;
    244     if (FAILED(fDWriteFontFace->QueryInterface(&fontFace5))) {
    245         return -1;
    246     }
    247 
    248     // Return 0 if the font is not variable font.
    249     if (!fontFace5->HasVariations()) {
    250         return 0;
    251     }
    252 
    253     UINT32 fontAxisCount = fontFace5->GetFontAxisValueCount();
    254     SkTScopedComPtr<IDWriteFontResource> fontResource;
    255     HR_GENERAL(fontFace5->GetFontResource(&fontResource), nullptr, -1);
    256     int variableAxisCount = 0;
    257     for (UINT32 i = 0; i < fontAxisCount; ++i) {
    258         if (fontResource->GetFontAxisAttributes(i) & DWRITE_FONT_AXIS_ATTRIBUTES_VARIABLE) {
    259             variableAxisCount++;
    260         }
    261     }
    262 
    263     if (!parameters || parameterCount < variableAxisCount) {
    264         return variableAxisCount;
    265     }
    266 
    267     SkAutoSTMalloc<8, DWRITE_FONT_AXIS_RANGE> fontAxisRange(fontAxisCount);
    268     HR_GENERAL(fontResource->GetFontAxisRanges(fontAxisRange.get(), fontAxisCount), nullptr, -1);
    269     SkAutoSTMalloc<8, DWRITE_FONT_AXIS_VALUE> fontAxisDefaultValue(fontAxisCount);
    270     HR_GENERAL(fontResource->GetDefaultFontAxisValues(fontAxisDefaultValue.get(), fontAxisCount),
    271                nullptr, -1);
    272     UINT32 coordIndex = 0;
    273 
    274     for (UINT32 axisIndex = 0; axisIndex < fontAxisCount; ++axisIndex) {
    275         if (fontResource->GetFontAxisAttributes(axisIndex) & DWRITE_FONT_AXIS_ATTRIBUTES_VARIABLE) {
    276             parameters[coordIndex].tag = SkEndian_SwapBE32(fontAxisDefaultValue[axisIndex].axisTag);
    277             parameters[coordIndex].min = fontAxisRange[axisIndex].minValue;
    278             parameters[coordIndex].def = fontAxisDefaultValue[axisIndex].value;
    279             parameters[coordIndex].max = fontAxisRange[axisIndex].maxValue;
    280             parameters[coordIndex].setHidden(fontResource->GetFontAxisAttributes(axisIndex) &
    281                                              DWRITE_FONT_AXIS_ATTRIBUTES_HIDDEN);
    282         }
    283     }
    284 
    285     return variableAxisCount;
    286 
    287 #endif
    288 
    289     return -1;
    290 }
    291 
    292 int DWriteFontTypeface::onGetTableTags(SkFontTableTag tags[]) const {
    293     DWRITE_FONT_FACE_TYPE type = fDWriteFontFace->GetType();
    294     if (type != DWRITE_FONT_FACE_TYPE_CFF &&
    295         type != DWRITE_FONT_FACE_TYPE_TRUETYPE &&
    296         type != DWRITE_FONT_FACE_TYPE_TRUETYPE_COLLECTION)
    297     {
    298         return 0;
    299     }
    300 
    301     int ttcIndex;
    302     std::unique_ptr<SkStream> stream(this->openStream(&ttcIndex));
    303     return stream.get() ? SkFontStream::GetTableTags(stream.get(), ttcIndex, tags) : 0;
    304 }
    305 
    306 size_t DWriteFontTypeface::onGetTableData(SkFontTableTag tag, size_t offset,
    307                                           size_t length, void* data) const
    308 {
    309     AutoDWriteTable table(fDWriteFontFace.get(), SkEndian_SwapBE32(tag));
    310     if (!table.fExists) {
    311         return 0;
    312     }
    313 
    314     if (offset > table.fSize) {
    315         return 0;
    316     }
    317     size_t size = SkTMin(length, table.fSize - offset);
    318     if (data) {
    319         memcpy(data, table.fData + offset, size);
    320     }
    321 
    322     return size;
    323 }
    324 
    325 sk_sp<SkTypeface> DWriteFontTypeface::onMakeClone(const SkFontArguments& args) const {
    326     // Skip if the current face index does not match the ttcIndex
    327     if (fDWriteFontFace->GetIndex() != SkTo<UINT32>(args.getCollectionIndex())) {
    328         return sk_ref_sp(this);
    329     }
    330 
    331 #if defined(NTDDI_WIN10_RS3) && NTDDI_VERSION >= NTDDI_WIN10_RS3
    332 
    333     SkTScopedComPtr<IDWriteFontFace5> fontFace5;
    334 
    335     if (SUCCEEDED(fDWriteFontFace->QueryInterface(&fontFace5)) && fontFace5->HasVariations()) {
    336         UINT32 fontAxisCount = fontFace5->GetFontAxisValueCount();
    337         UINT32 argsCoordCount = args.getVariationDesignPosition().coordinateCount;
    338         SkAutoSTMalloc<8, DWRITE_FONT_AXIS_VALUE> fontAxisValue(fontAxisCount);
    339         HRN(fontFace5->GetFontAxisValues(fontAxisValue.get(), fontAxisCount));
    340 
    341         for (UINT32 fontIndex = 0; fontIndex < fontAxisCount; ++fontIndex) {
    342             for (UINT32 argsIndex = 0; argsIndex < argsCoordCount; ++argsIndex) {
    343                 if (SkEndian_SwapBE32(fontAxisValue[fontIndex].axisTag) ==
    344                     args.getVariationDesignPosition().coordinates[argsIndex].axis) {
    345                     fontAxisValue[fontIndex].value =
    346                         args.getVariationDesignPosition().coordinates[argsIndex].value;
    347                 }
    348             }
    349         }
    350         SkTScopedComPtr<IDWriteFontResource> fontResource;
    351         HRN(fontFace5->GetFontResource(&fontResource));
    352         SkTScopedComPtr<IDWriteFontFace5> newFontFace5;
    353         HRN(fontResource->CreateFontFace(fDWriteFont->GetSimulations(),
    354                                          fontAxisValue.get(),
    355                                          fontAxisCount,
    356                                          &newFontFace5));
    357 
    358         SkTScopedComPtr<IDWriteFontFace> newFontFace;
    359         HRN(newFontFace5->QueryInterface(&newFontFace));
    360         return sk_sp<SkTypeface>(DWriteFontTypeface::Create(fFactory.get(),
    361                                                             newFontFace.get(),
    362                                                             fDWriteFont.get(),
    363                                                             fDWriteFontFamily.get(),
    364                                                             fDWriteFontFileLoader.get(),
    365                                                             fDWriteFontCollectionLoader.get()));
    366     }
    367 
    368 #endif
    369 
    370     return sk_ref_sp(this);
    371 }
    372 
    373 SkStreamAsset* DWriteFontTypeface::onOpenStream(int* ttcIndex) const {
    374     *ttcIndex = fDWriteFontFace->GetIndex();
    375 
    376     UINT32 numFiles;
    377     HRNM(fDWriteFontFace->GetFiles(&numFiles, nullptr),
    378          "Could not get number of font files.");
    379     if (numFiles != 1) {
    380         return nullptr;
    381     }
    382 
    383     SkTScopedComPtr<IDWriteFontFile> fontFile;
    384     HRNM(fDWriteFontFace->GetFiles(&numFiles, &fontFile), "Could not get font files.");
    385 
    386     const void* fontFileKey;
    387     UINT32 fontFileKeySize;
    388     HRNM(fontFile->GetReferenceKey(&fontFileKey, &fontFileKeySize),
    389          "Could not get font file reference key.");
    390 
    391     SkTScopedComPtr<IDWriteFontFileLoader> fontFileLoader;
    392     HRNM(fontFile->GetLoader(&fontFileLoader), "Could not get font file loader.");
    393 
    394     SkTScopedComPtr<IDWriteFontFileStream> fontFileStream;
    395     HRNM(fontFileLoader->CreateStreamFromKey(fontFileKey, fontFileKeySize,
    396                                              &fontFileStream),
    397          "Could not create font file stream.");
    398 
    399     return new SkDWriteFontFileStream(fontFileStream.get());
    400 }
    401 
    402 SkScalerContext* DWriteFontTypeface::onCreateScalerContext(const SkScalerContextEffects& effects,
    403                                                            const SkDescriptor* desc) const {
    404     return new SkScalerContext_DW(sk_ref_sp(const_cast<DWriteFontTypeface*>(this)), effects, desc);
    405 }
    406 
    407 void DWriteFontTypeface::onFilterRec(SkScalerContextRec* rec) const {
    408     if (rec->fFlags & SkScalerContext::kLCD_Vertical_Flag) {
    409         rec->fMaskFormat = SkMask::kA8_Format;
    410         rec->fFlags |= SkScalerContext::kGenA8FromLCD_Flag;
    411     }
    412 
    413     unsigned flagsWeDontSupport = SkScalerContext::kForceAutohinting_Flag |
    414                                   SkScalerContext::kEmbolden_Flag |
    415                                   SkScalerContext::kLCD_Vertical_Flag;
    416     rec->fFlags &= ~flagsWeDontSupport;
    417 
    418     SkFontHinting h = rec->getHinting();
    419     // DirectWrite2 allows for hinting to be turned off. Force everything else to normal.
    420     if (h != kNo_SkFontHinting || !fFactory2 || !fDWriteFontFace2) {
    421         h = kNormal_SkFontHinting;
    422     }
    423     rec->setHinting(h);
    424 
    425 #if defined(SK_FONT_HOST_USE_SYSTEM_SETTINGS)
    426     IDWriteFactory* factory = sk_get_dwrite_factory();
    427     if (factory != nullptr) {
    428         SkTScopedComPtr<IDWriteRenderingParams> defaultRenderingParams;
    429         if (SUCCEEDED(factory->CreateRenderingParams(&defaultRenderingParams))) {
    430             float gamma = defaultRenderingParams->GetGamma();
    431             rec->setDeviceGamma(gamma);
    432             rec->setPaintGamma(gamma);
    433 
    434             rec->setContrast(defaultRenderingParams->GetEnhancedContrast());
    435         }
    436     }
    437 #endif
    438 }
    439 
    440 ///////////////////////////////////////////////////////////////////////////////
    441 //PDF Support
    442 
    443 void DWriteFontTypeface::getGlyphToUnicodeMap(SkUnichar* glyphToUnicode) const {
    444     unsigned glyphCount = fDWriteFontFace->GetGlyphCount();
    445     sk_bzero(glyphToUnicode, sizeof(SkUnichar) * glyphCount);
    446     IDWriteFontFace* fontFace = fDWriteFontFace.get();
    447     int maxGlyph = -1;
    448     unsigned remainingGlyphCount = glyphCount;
    449     for (UINT32 c = 0; c < 0x10FFFF && remainingGlyphCount != 0; ++c) {
    450         UINT16 glyph = 0;
    451         HRVM(fontFace->GetGlyphIndices(&c, 1, &glyph), "Failed to get glyph index.");
    452         // Intermittent DW bug on Windows 10. See crbug.com/470146.
    453         if (glyph >= glyphCount) {
    454             return;
    455         }
    456         if (0 < glyph && glyphToUnicode[glyph] == 0) {
    457             maxGlyph = SkTMax(static_cast<int>(glyph), maxGlyph);
    458             glyphToUnicode[glyph] = c;  // Always use lowest-index unichar.
    459             --remainingGlyphCount;
    460         }
    461     }
    462 }
    463 
    464 std::unique_ptr<SkAdvancedTypefaceMetrics> DWriteFontTypeface::onGetAdvancedMetrics() const {
    465 
    466     std::unique_ptr<SkAdvancedTypefaceMetrics> info(nullptr);
    467 
    468     DWRITE_FONT_METRICS dwfm;
    469     fDWriteFontFace->GetMetrics(&dwfm);
    470 
    471     info.reset(new SkAdvancedTypefaceMetrics);
    472 
    473     info->fAscent = SkToS16(dwfm.ascent);
    474     info->fDescent = SkToS16(dwfm.descent);
    475     info->fCapHeight = SkToS16(dwfm.capHeight);
    476 
    477     {
    478         SkTScopedComPtr<IDWriteLocalizedStrings> postScriptNames;
    479         BOOL exists = FALSE;
    480         if (FAILED(fDWriteFont->GetInformationalStrings(
    481                         DWRITE_INFORMATIONAL_STRING_POSTSCRIPT_NAME,
    482                         &postScriptNames,
    483                         &exists)) ||
    484             !exists ||
    485             FAILED(sk_get_locale_string(postScriptNames.get(), nullptr, &info->fPostScriptName)))
    486         {
    487             SkDEBUGF("Unable to get postscript name for typeface %p\n", this);
    488         }
    489     }
    490 
    491     // SkAdvancedTypefaceMetrics::fFontName must actually be a family name.
    492     SkTScopedComPtr<IDWriteLocalizedStrings> familyNames;
    493     if (FAILED(fDWriteFontFamily->GetFamilyNames(&familyNames)) ||
    494         FAILED(sk_get_locale_string(familyNames.get(), nullptr, &info->fFontName)))
    495     {
    496         SkDEBUGF("Unable to get family name for typeface 0x%p\n", this);
    497     }
    498     if (info->fPostScriptName.isEmpty()) {
    499         info->fPostScriptName = info->fFontName;
    500     }
    501 
    502     DWRITE_FONT_FACE_TYPE fontType = fDWriteFontFace->GetType();
    503     if (fontType != DWRITE_FONT_FACE_TYPE_TRUETYPE &&
    504         fontType != DWRITE_FONT_FACE_TYPE_TRUETYPE_COLLECTION)
    505     {
    506         return info;
    507     }
    508 
    509     // Simulated fonts aren't really TrueType fonts.
    510     if (fDWriteFontFace->GetSimulations() == DWRITE_FONT_SIMULATIONS_NONE) {
    511         info->fType = SkAdvancedTypefaceMetrics::kTrueType_Font;
    512     }
    513 
    514     AutoTDWriteTable<SkOTTableHead> headTable(fDWriteFontFace.get());
    515     AutoTDWriteTable<SkOTTablePostScript> postTable(fDWriteFontFace.get());
    516     AutoTDWriteTable<SkOTTableHorizontalHeader> hheaTable(fDWriteFontFace.get());
    517     AutoTDWriteTable<SkOTTableOS2> os2Table(fDWriteFontFace.get());
    518     if (!headTable.fExists || !postTable.fExists || !hheaTable.fExists || !os2Table.fExists) {
    519         return info;
    520     }
    521 
    522     SkOTUtils::SetAdvancedTypefaceFlags(os2Table->version.v4.fsType, info.get());
    523 
    524     // There are versions of DirectWrite which support named instances for system variation fonts,
    525     // but no means to indicate that such a typeface is a variation.
    526     AutoTDWriteTable<SkOTTableFontVariations> fvarTable(fDWriteFontFace.get());
    527     if (fvarTable.fExists) {
    528         info->fFlags |= SkAdvancedTypefaceMetrics::kMultiMaster_FontFlag;
    529     }
    530 
    531     //There exist CJK fonts which set the IsFixedPitch and Monospace bits,
    532     //but have full width, latin half-width, and half-width kana.
    533     bool fixedWidth = (postTable->isFixedPitch &&
    534                       (1 == SkEndian_SwapBE16(hheaTable->numberOfHMetrics)));
    535     //Monospace
    536     if (fixedWidth) {
    537         info->fStyle |= SkAdvancedTypefaceMetrics::kFixedPitch_Style;
    538     }
    539     //Italic
    540     if (os2Table->version.v0.fsSelection.field.Italic) {
    541         info->fStyle |= SkAdvancedTypefaceMetrics::kItalic_Style;
    542     }
    543     //Serif
    544     using SerifStyle = SkPanose::Data::TextAndDisplay::SerifStyle;
    545     SerifStyle serifStyle = os2Table->version.v0.panose.data.textAndDisplay.bSerifStyle;
    546     if (SkPanose::FamilyType::TextAndDisplay == os2Table->version.v0.panose.bFamilyType) {
    547         if (SerifStyle::Cove == serifStyle ||
    548             SerifStyle::ObtuseCove == serifStyle ||
    549             SerifStyle::SquareCove == serifStyle ||
    550             SerifStyle::ObtuseSquareCove == serifStyle ||
    551             SerifStyle::Square == serifStyle ||
    552             SerifStyle::Thin == serifStyle ||
    553             SerifStyle::Bone == serifStyle ||
    554             SerifStyle::Exaggerated == serifStyle ||
    555             SerifStyle::Triangle == serifStyle)
    556         {
    557             info->fStyle |= SkAdvancedTypefaceMetrics::kSerif_Style;
    558         }
    559     //Script
    560     } else if (SkPanose::FamilyType::Script == os2Table->version.v0.panose.bFamilyType) {
    561         info->fStyle |= SkAdvancedTypefaceMetrics::kScript_Style;
    562     }
    563 
    564     info->fItalicAngle = SkEndian_SwapBE32(postTable->italicAngle) >> 16;
    565 
    566     info->fBBox = SkIRect::MakeLTRB((int32_t)SkEndian_SwapBE16((uint16_t)headTable->xMin),
    567                                     (int32_t)SkEndian_SwapBE16((uint16_t)headTable->yMax),
    568                                     (int32_t)SkEndian_SwapBE16((uint16_t)headTable->xMax),
    569                                     (int32_t)SkEndian_SwapBE16((uint16_t)headTable->yMin));
    570     return info;
    571 }
    572 #endif//defined(SK_BUILD_FOR_WIN)
    573