| /* | 
 |  * Copyright 2011 Google Inc. | 
 |  * | 
 |  * Use of this source code is governed by a BSD-style license that can be | 
 |  * found in the LICENSE file. | 
 |  */ | 
 | #include "src/utils/win/SkDWriteNTDDI_VERSION.h" | 
 |  | 
 | #include "include/core/SkTypes.h" | 
 | #if defined(SK_BUILD_FOR_WIN) | 
 |  | 
 | #undef GetGlyphIndices | 
 |  | 
 | #include "include/codec/SkCodec.h" | 
 | #include "include/codec/SkPngDecoder.h" | 
 | #include "include/core/SkBBHFactory.h" | 
 | #include "include/core/SkBitmap.h" | 
 | #include "include/core/SkData.h" | 
 | #include "include/core/SkDrawable.h" | 
 | #include "include/core/SkFontMetrics.h" | 
 | #include "include/core/SkGraphics.h" | 
 | #include "include/core/SkImage.h" | 
 | #include "include/core/SkOpenTypeSVGDecoder.h" | 
 | #include "include/core/SkPath.h" | 
 | #include "include/core/SkPictureRecorder.h" | 
 | #include "include/effects/SkGradientShader.h" | 
 | #include "include/private/base/SkMutex.h" | 
 | #include "include/private/base/SkTo.h" | 
 | #include "src/base/SkEndian.h" | 
 | #include "src/base/SkScopeExit.h" | 
 | #include "src/base/SkSharedMutex.h" | 
 | #include "src/core/SkDraw.h" | 
 | #include "src/core/SkGlyph.h" | 
 | #include "src/core/SkMaskGamma.h" | 
 | #include "src/core/SkRasterClip.h" | 
 | #include "src/core/SkScalerContext.h" | 
 | #include "src/ports/SkScalerContext_win_dw.h" | 
 | #include "src/ports/SkTypeface_win_dw.h" | 
 | #include "src/sfnt/SkOTTable_EBLC.h" | 
 | #include "src/sfnt/SkOTTable_EBSC.h" | 
 | #include "src/sfnt/SkOTTable_gasp.h" | 
 | #include "src/sfnt/SkOTTable_maxp.h" | 
 | #include "src/utils/SkMatrix22.h" | 
 | #include "src/utils/win/SkDWrite.h" | 
 | #include "src/utils/win/SkDWriteGeometrySink.h" | 
 | #include "src/utils/win/SkHRESULT.h" | 
 | #include "src/utils/win/SkTScopedComPtr.h" | 
 |  | 
 | #include <dwrite.h> | 
 | #include <dwrite_1.h> | 
 | #include <dwrite_3.h> | 
 |  | 
 | namespace { | 
 | static inline const constexpr bool kSkShowTextBlitCoverage = false; | 
 |  | 
 | /* Note: | 
 |  * In versions 8 and 8.1 of Windows, some calls in DWrite are not thread safe. | 
 |  * The mutex returned from maybe_dw_mutex protects the calls that are | 
 |  * problematic. | 
 |  */ | 
 | static SkSharedMutex* maybe_dw_mutex(DWriteFontTypeface& typeface) { | 
 |     static SkSharedMutex mutex; | 
 |     return typeface.fDWriteFontFace4 ? nullptr : &mutex; | 
 | } | 
 |  | 
 | class SK_SCOPED_CAPABILITY Exclusive { | 
 | public: | 
 |     explicit Exclusive(SkSharedMutex* maybe_lock) SK_ACQUIRE(*maybe_lock) | 
 |         : fLock(maybe_lock) { | 
 |         if (fLock) { | 
 |             fLock->acquire(); | 
 |         } | 
 |     } | 
 |     ~Exclusive() SK_RELEASE_CAPABILITY() { | 
 |         if (fLock) { | 
 |             fLock->release(); | 
 |         } | 
 |     } | 
 |  | 
 | private: | 
 |     SkSharedMutex* fLock; | 
 | }; | 
 | class SK_SCOPED_CAPABILITY Shared { | 
 | public: | 
 |     explicit Shared(SkSharedMutex* maybe_lock) SK_ACQUIRE_SHARED(*maybe_lock) | 
 |         : fLock(maybe_lock)  { | 
 |         if (fLock) { | 
 |             fLock->acquireShared(); | 
 |         } | 
 |     } | 
 |  | 
 |     // You would think this should be SK_RELEASE_SHARED_CAPABILITY, but SK_SCOPED_CAPABILITY | 
 |     // doesn't fully understand the difference between shared and exclusive. | 
 |     // Please review https://reviews.llvm.org/D52578 for more information. | 
 |     ~Shared() SK_RELEASE_CAPABILITY() { | 
 |         if (fLock) { | 
 |             fLock->releaseShared(); | 
 |         } | 
 |     } | 
 |  | 
 | private: | 
 |     SkSharedMutex* fLock; | 
 | }; | 
 |  | 
 | static bool isLCD(const SkScalerContextRec& rec) { | 
 |     return SkMask::kLCD16_Format == rec.fMaskFormat; | 
 | } | 
 |  | 
 | static bool is_hinted(DWriteFontTypeface* typeface) { | 
 |     Exclusive l(maybe_dw_mutex(*typeface)); | 
 |     AutoTDWriteTable<SkOTTableMaximumProfile> maxp(typeface->fDWriteFontFace.get()); | 
 |     if (!maxp.fExists) { | 
 |         return false; | 
 |     } | 
 |     if (maxp.fSize < sizeof(SkOTTableMaximumProfile::Version::TT)) { | 
 |         return false; | 
 |     } | 
 |     if (maxp->version.version != SkOTTableMaximumProfile::Version::TT::VERSION) { | 
 |         return false; | 
 |     } | 
 |     return (0 != maxp->version.tt.maxSizeOfInstructions); | 
 | } | 
 |  | 
 | /** A GaspRange is inclusive, [min, max]. */ | 
 | struct GaspRange { | 
 |     using Behavior = SkOTTableGridAndScanProcedure::GaspRange::behavior; | 
 |     GaspRange(int min, int max, int version, Behavior flags) | 
 |         : fMin(min), fMax(max), fVersion(version), fFlags(flags) { } | 
 |     int fMin; | 
 |     int fMax; | 
 |     int fVersion; | 
 |     Behavior fFlags; | 
 | }; | 
 |  | 
 | bool get_gasp_range(DWriteFontTypeface* typeface, int size, GaspRange* range) { | 
 |     AutoTDWriteTable<SkOTTableGridAndScanProcedure> gasp(typeface->fDWriteFontFace.get()); | 
 |     if (!gasp.fExists) { | 
 |         return false; | 
 |     } | 
 |     if (gasp.fSize < sizeof(SkOTTableGridAndScanProcedure)) { | 
 |         return false; | 
 |     } | 
 |     if (gasp->version != SkOTTableGridAndScanProcedure::version0 && | 
 |         gasp->version != SkOTTableGridAndScanProcedure::version1) | 
 |     { | 
 |         return false; | 
 |     } | 
 |  | 
 |     uint16_t numRanges = SkEndianSwap16(gasp->numRanges); | 
 |     if (numRanges > 1024 || | 
 |         gasp.fSize < sizeof(SkOTTableGridAndScanProcedure) + | 
 |         sizeof(SkOTTableGridAndScanProcedure::GaspRange) * numRanges) | 
 |     { | 
 |         return false; | 
 |     } | 
 |  | 
 |     const SkOTTableGridAndScanProcedure::GaspRange* rangeTable = | 
 |             SkTAfter<const SkOTTableGridAndScanProcedure::GaspRange>(gasp.get()); | 
 |     int minPPEM = -1; | 
 |     for (uint16_t i = 0; i < numRanges; ++i, ++rangeTable) { | 
 |         int maxPPEM = SkEndianSwap16(rangeTable->maxPPEM); | 
 |         if (minPPEM < size && size <= maxPPEM) { | 
 |             range->fMin = minPPEM + 1; | 
 |             range->fMax = maxPPEM; | 
 |             range->fVersion = SkEndian_SwapBE16(gasp->version); | 
 |             range->fFlags = rangeTable->flags; | 
 |             return true; | 
 |         } | 
 |         minPPEM = maxPPEM; | 
 |     } | 
 |     return false; | 
 | } | 
 | /** If the rendering mode for the specified 'size' is gridfit, then place | 
 |  *  the gridfit range into 'range'. Otherwise, leave 'range' alone. | 
 |  */ | 
 | static bool is_gridfit_only(GaspRange::Behavior flags) { | 
 |     return flags.raw.value == GaspRange::Behavior::Raw::GridfitMask; | 
 | } | 
 |  | 
 | static bool has_bitmap_strike(DWriteFontTypeface* typeface, GaspRange range) { | 
 |     Exclusive l(maybe_dw_mutex(*typeface)); | 
 |     { | 
 |         AutoTDWriteTable<SkOTTableEmbeddedBitmapLocation> eblc(typeface->fDWriteFontFace.get()); | 
 |         if (!eblc.fExists) { | 
 |             return false; | 
 |         } | 
 |         if (eblc.fSize < sizeof(SkOTTableEmbeddedBitmapLocation)) { | 
 |             return false; | 
 |         } | 
 |         if (eblc->version != SkOTTableEmbeddedBitmapLocation::version_initial) { | 
 |             return false; | 
 |         } | 
 |  | 
 |         uint32_t numSizes = SkEndianSwap32(eblc->numSizes); | 
 |         if (numSizes > 1024 || | 
 |             eblc.fSize < sizeof(SkOTTableEmbeddedBitmapLocation) + | 
 |                          sizeof(SkOTTableEmbeddedBitmapLocation::BitmapSizeTable) * numSizes) | 
 |         { | 
 |             return false; | 
 |         } | 
 |  | 
 |         const SkOTTableEmbeddedBitmapLocation::BitmapSizeTable* sizeTable = | 
 |                 SkTAfter<const SkOTTableEmbeddedBitmapLocation::BitmapSizeTable>(eblc.get()); | 
 |         for (uint32_t i = 0; i < numSizes; ++i, ++sizeTable) { | 
 |             if (sizeTable->ppemX == sizeTable->ppemY && | 
 |                 range.fMin <= sizeTable->ppemX && sizeTable->ppemX <= range.fMax) | 
 |             { | 
 |                 // TODO: determine if we should dig through IndexSubTableArray/IndexSubTable | 
 |                 // to determine the actual number of glyphs with bitmaps. | 
 |  | 
 |                 // TODO: Ensure that the bitmaps actually cover a significant portion of the strike. | 
 |  | 
 |                 // TODO: Ensure that the bitmaps are bi-level? | 
 |                 if (sizeTable->endGlyphIndex >= sizeTable->startGlyphIndex + 3) { | 
 |                     return true; | 
 |                 } | 
 |             } | 
 |         } | 
 |     } | 
 |  | 
 |     { | 
 |         AutoTDWriteTable<SkOTTableEmbeddedBitmapScaling> ebsc(typeface->fDWriteFontFace.get()); | 
 |         if (!ebsc.fExists) { | 
 |             return false; | 
 |         } | 
 |         if (ebsc.fSize < sizeof(SkOTTableEmbeddedBitmapScaling)) { | 
 |             return false; | 
 |         } | 
 |         if (ebsc->version != SkOTTableEmbeddedBitmapScaling::version_initial) { | 
 |             return false; | 
 |         } | 
 |  | 
 |         uint32_t numSizes = SkEndianSwap32(ebsc->numSizes); | 
 |         if (numSizes > 1024 || | 
 |             ebsc.fSize < sizeof(SkOTTableEmbeddedBitmapScaling) + | 
 |                          sizeof(SkOTTableEmbeddedBitmapScaling::BitmapScaleTable) * numSizes) | 
 |         { | 
 |             return false; | 
 |         } | 
 |  | 
 |         const SkOTTableEmbeddedBitmapScaling::BitmapScaleTable* scaleTable = | 
 |                 SkTAfter<const SkOTTableEmbeddedBitmapScaling::BitmapScaleTable>(ebsc.get()); | 
 |         for (uint32_t i = 0; i < numSizes; ++i, ++scaleTable) { | 
 |             if (scaleTable->ppemX == scaleTable->ppemY && | 
 |                 range.fMin <= scaleTable->ppemX && scaleTable->ppemX <= range.fMax) { | 
 |                 // EBSC tables are normally only found in bitmap only fonts. | 
 |                 return true; | 
 |             } | 
 |         } | 
 |     } | 
 |  | 
 |     return false; | 
 | } | 
 |  | 
 | static bool both_zero(SkScalar a, SkScalar b) { | 
 |     return 0 == a && 0 == b; | 
 | } | 
 |  | 
 | // returns false if there is any non-90-rotation or skew | 
 | static bool is_axis_aligned(const SkScalerContextRec& rec) { | 
 |     return 0 == rec.fPreSkewX && | 
 |            (both_zero(rec.fPost2x2[0][1], rec.fPost2x2[1][0]) || | 
 |             both_zero(rec.fPost2x2[0][0], rec.fPost2x2[1][1])); | 
 | } | 
 |  | 
 | }  //namespace | 
 |  | 
 | SkScalerContext_DW::SkScalerContext_DW(sk_sp<DWriteFontTypeface> typefaceRef, | 
 |                                        const SkScalerContextEffects& effects, | 
 |                                        const SkDescriptor* desc) | 
 |         : SkScalerContext(std::move(typefaceRef), effects, desc) | 
 | { | 
 |     DWriteFontTypeface* typeface = this->getDWriteTypeface(); | 
 |     fGlyphCount = typeface->fDWriteFontFace->GetGlyphCount(); | 
 |  | 
 |     // In general, all glyphs should use NATURAL_SYMMETRIC | 
 |     // except when bi-level rendering is requested or there are embedded | 
 |     // bi-level bitmaps (and the embedded bitmap flag is set and no rotation). | 
 |     // | 
 |     // DirectWrite's IDWriteFontFace::GetRecommendedRenderingMode does not do | 
 |     // this. As a result, determine the actual size of the text and then see if | 
 |     // there are any embedded bi-level bitmaps of that size. If there are, then | 
 |     // force bitmaps by requesting bi-level rendering. | 
 |     // | 
 |     // FreeType allows for separate ppemX and ppemY, but DirectWrite assumes | 
 |     // square pixels and only uses ppemY. Therefore the transform must track any | 
 |     // non-uniform x-scale. | 
 |     // | 
 |     // Also, rotated glyphs should have the same absolute advance widths as | 
 |     // horizontal glyphs and the subpixel flag should not affect glyph shapes. | 
 |  | 
 |     SkVector scale; | 
 |     fRec.computeMatrices(SkScalerContextRec::PreMatrixScale::kVertical, &scale, &fSkXform); | 
 |  | 
 |     fXform.m11 = SkScalarToFloat(fSkXform.getScaleX()); | 
 |     fXform.m12 = SkScalarToFloat(fSkXform.getSkewY()); | 
 |     fXform.m21 = SkScalarToFloat(fSkXform.getSkewX()); | 
 |     fXform.m22 = SkScalarToFloat(fSkXform.getScaleY()); | 
 |     fXform.dx = 0; | 
 |     fXform.dy = 0; | 
 |  | 
 |     // realTextSize is the actual device size we want (as opposed to the size the user requested). | 
 |     // gdiTextSize is the size we request when GDI compatible. | 
 |     // If the scale is negative, this means the matrix will do the flip anyway. | 
 |     const SkScalar realTextSize = scale.fY; | 
 |     // Due to floating point math, the lower bits are suspect. Round carefully. | 
 |     SkScalar gdiTextSize = SkScalarRoundToScalar(realTextSize * 64.0f) / 64.0f; | 
 |     if (gdiTextSize == 0) { | 
 |         gdiTextSize = SK_Scalar1; | 
 |     } | 
 |  | 
 |     bool bitmapRequested = SkToBool(fRec.fFlags & SkScalerContext::kEmbeddedBitmapText_Flag); | 
 |     bool treatLikeBitmap = false; | 
 |     bool axisAlignedBitmap = false; | 
 |     if (bitmapRequested) { | 
 |         // When embedded bitmaps are requested, treat the entire range like | 
 |         // a bitmap strike if the range is gridfit only and contains a bitmap. | 
 |         int bitmapPPEM = SkScalarTruncToInt(gdiTextSize); | 
 |         GaspRange range(bitmapPPEM, bitmapPPEM, 0, GaspRange::Behavior()); | 
 |         if (get_gasp_range(typeface, bitmapPPEM, &range)) { | 
 |             if (!is_gridfit_only(range.fFlags)) { | 
 |                 range = GaspRange(bitmapPPEM, bitmapPPEM, 0, GaspRange::Behavior()); | 
 |             } | 
 |         } | 
 |         treatLikeBitmap = has_bitmap_strike(typeface, range); | 
 |  | 
 |         axisAlignedBitmap = is_axis_aligned(fRec); | 
 |     } | 
 |  | 
 |     GaspRange range(0, 0xFFFF, 0, GaspRange::Behavior()); | 
 |  | 
 |     // If the user requested aliased, do so with aliased compatible metrics. | 
 |     if (SkMask::kBW_Format == fRec.fMaskFormat) { | 
 |         fTextSizeRender = gdiTextSize; | 
 |         fRenderingMode = DWRITE_RENDERING_MODE_ALIASED; | 
 |         fTextureType = DWRITE_TEXTURE_ALIASED_1x1; | 
 |         fTextSizeMeasure = gdiTextSize; | 
 |         fMeasuringMode = DWRITE_MEASURING_MODE_GDI_CLASSIC; | 
 |  | 
 |     // If we can use a bitmap, use gdi classic rendering and measurement. | 
 |     // This will not always provide a bitmap, but matches expected behavior. | 
 |     } else if (treatLikeBitmap && axisAlignedBitmap) { | 
 |         fTextSizeRender = gdiTextSize; | 
 |         fRenderingMode = DWRITE_RENDERING_MODE_GDI_CLASSIC; | 
 |         fTextureType = DWRITE_TEXTURE_CLEARTYPE_3x1; | 
 |         fTextSizeMeasure = gdiTextSize; | 
 |         fMeasuringMode = DWRITE_MEASURING_MODE_GDI_CLASSIC; | 
 |  | 
 |     // If rotated but the horizontal text could have used a bitmap, | 
 |     // render high quality rotated glyphs but measure using bitmap metrics. | 
 |     } else if (treatLikeBitmap) { | 
 |         fTextSizeRender = gdiTextSize; | 
 |         fRenderingMode = DWRITE_RENDERING_MODE_NATURAL_SYMMETRIC; | 
 |         fTextureType = DWRITE_TEXTURE_CLEARTYPE_3x1; | 
 |         fTextSizeMeasure = gdiTextSize; | 
 |         fMeasuringMode = DWRITE_MEASURING_MODE_GDI_CLASSIC; | 
 |  | 
 |     // If the font has a gasp table version 1, use it to determine symmetric rendering. | 
 |     } else if (get_gasp_range(typeface, SkScalarRoundToInt(gdiTextSize), &range) && | 
 |                range.fVersion >= 1) | 
 |     { | 
 |         fTextSizeRender = realTextSize; | 
 |         fRenderingMode = range.fFlags.field.SymmetricSmoothing | 
 |                        ? DWRITE_RENDERING_MODE_NATURAL_SYMMETRIC | 
 |                        : DWRITE_RENDERING_MODE_NATURAL; | 
 |         fTextureType = DWRITE_TEXTURE_CLEARTYPE_3x1; | 
 |         fTextSizeMeasure = realTextSize; | 
 |         fMeasuringMode = DWRITE_MEASURING_MODE_NATURAL; | 
 |  | 
 |     // If the requested size is above 20px or there are no bytecode hints, use symmetric rendering. | 
 |     } else if (realTextSize > SkIntToScalar(20) || !is_hinted(typeface)) { | 
 |         fTextSizeRender = realTextSize; | 
 |         fRenderingMode = DWRITE_RENDERING_MODE_NATURAL_SYMMETRIC; | 
 |         fTextureType = DWRITE_TEXTURE_CLEARTYPE_3x1; | 
 |         fTextSizeMeasure = realTextSize; | 
 |         fMeasuringMode = DWRITE_MEASURING_MODE_NATURAL; | 
 |  | 
 |     // Fonts with hints, no gasp or gasp version 0, and below 20px get non-symmetric rendering. | 
 |     // Often such fonts have hints which were only tested with GDI ClearType classic. | 
 |     // Some of these fonts rely on drop out control in the y direction in order to be legible. | 
 |     // Tenor Sans | 
 |     //    https://fonts.google.com/specimen/Tenor+Sans | 
 |     // Gill Sans W04 | 
 |     //    https://cdn.leagueoflegends.com/lolkit/1.1.9/resources/fonts/gill-sans-w04-book.woff | 
 |     //    https://na.leagueoflegends.com/en/news/game-updates/patch/patch-410-notes | 
 |     // See https://crbug.com/385897 | 
 |     } else { | 
 |         fTextSizeRender = gdiTextSize; | 
 |         fRenderingMode = DWRITE_RENDERING_MODE_NATURAL; | 
 |         fTextureType = DWRITE_TEXTURE_CLEARTYPE_3x1; | 
 |         fTextSizeMeasure = realTextSize; | 
 |         fMeasuringMode = DWRITE_MEASURING_MODE_NATURAL; | 
 |     } | 
 |  | 
 |     // DirectWrite2 allows for grayscale hinting. | 
 |     fAntiAliasMode = DWRITE_TEXT_ANTIALIAS_MODE_CLEARTYPE; | 
 |     if (typeface->fFactory2 && typeface->fDWriteFontFace2 && | 
 |         SkMask::kA8_Format == fRec.fMaskFormat && | 
 |         !(fRec.fFlags & SkScalerContext::kGenA8FromLCD_Flag)) | 
 |     { | 
 |         // DWRITE_TEXTURE_ALIASED_1x1 is now misnamed, it must also be used with grayscale. | 
 |         fTextureType = DWRITE_TEXTURE_ALIASED_1x1; | 
 |         fAntiAliasMode = DWRITE_TEXT_ANTIALIAS_MODE_GRAYSCALE; | 
 |     } | 
 |  | 
 |     // DirectWrite2 allows hinting to be disabled. | 
 |     fGridFitMode = DWRITE_GRID_FIT_MODE_ENABLED; | 
 |     if (fRec.getHinting() == SkFontHinting::kNone) { | 
 |         fGridFitMode = DWRITE_GRID_FIT_MODE_DISABLED; | 
 |         if (fRenderingMode != DWRITE_RENDERING_MODE_ALIASED) { | 
 |             fRenderingMode = DWRITE_RENDERING_MODE_NATURAL_SYMMETRIC; | 
 |         } | 
 |     } | 
 |  | 
 |     if (this->isLinearMetrics()) { | 
 |         fTextSizeMeasure = realTextSize; | 
 |         fMeasuringMode = DWRITE_MEASURING_MODE_NATURAL; | 
 |     } | 
 |  | 
 |     // The GDI measuring modes don't seem to work well with CBDT fonts (DWrite.dll 10.0.18362.836). | 
 |     if (fMeasuringMode != DWRITE_MEASURING_MODE_NATURAL) { | 
 |         constexpr UINT32 CBDTTag = DWRITE_MAKE_OPENTYPE_TAG('C','B','D','T'); | 
 |         AutoDWriteTable CBDT(typeface->fDWriteFontFace.get(), CBDTTag); | 
 |         if (CBDT.fExists) { | 
 |             fMeasuringMode = DWRITE_MEASURING_MODE_NATURAL; | 
 |         } | 
 |     } | 
 | } | 
 |  | 
 | SkScalerContext_DW::~SkScalerContext_DW() { | 
 | } | 
 |  | 
 | #if DWRITE_CORE || (defined(NTDDI_WIN11_ZN) && NTDDI_VERSION >= NTDDI_WIN11_ZN) | 
 |  | 
 | namespace { | 
 | SkColor4f sk_color_from(DWRITE_COLOR_F const& color) { | 
 |     // DWRITE_COLOR_F and SkColor4f are laid out the same and this should be a no-op. | 
 |     return SkColor4f{ color.r, color.g, color.b, color.a }; | 
 | } | 
 | DWRITE_COLOR_F dw_color_from(SkColor4f const& color) { | 
 |     // DWRITE_COLOR_F and SkColor4f are laid out the same and this should be a no-op. | 
 |     // Avoid brace initialization as DWRITE_COLOR_F can be defined as four floats (dxgitype.h, | 
 |     // d3d9types.h) or four unions of two floats (dwrite_2.h, d3dtypes.h). The type changed in | 
 |     // Direct3D 10, but the change does not appear to be documented. | 
 |     DWRITE_COLOR_F dwColor; | 
 |     dwColor.r = color.fR; | 
 |     dwColor.g = color.fG; | 
 |     dwColor.b = color.fB; | 
 |     dwColor.a = color.fA; | 
 |     return dwColor; | 
 | } | 
 |  | 
 | SkRect sk_rect_from(D2D_RECT_F const& rect) { | 
 |     // D2D_RECT_F and SkRect are both y-down and laid the same so this should be a no-op. | 
 |     return SkRect{ rect.left, rect.top, rect.right, rect.bottom }; | 
 | } | 
 | constexpr bool D2D_RECT_F_is_empty(const D2D_RECT_F& r) { | 
 |     return r.right <= r.left || r.bottom <= r.top; | 
 | } | 
 |  | 
 | SkMatrix sk_matrix_from(DWRITE_MATRIX const& m) { | 
 |     // DWRITE_MATRIX and SkMatrix are y-down. However DWRITE_MATRIX is affine only. | 
 |     return SkMatrix::MakeAll( | 
 |         m.m11, m.m21, m.dx, | 
 |         m.m12, m.m22, m.dy, | 
 |         0, 0, 1); | 
 | } | 
 |  | 
 | SkTileMode sk_tile_mode_from(D2D1_EXTEND_MODE extendMode) { | 
 |     switch (extendMode) { | 
 |     case D2D1_EXTEND_MODE_CLAMP: | 
 |         return SkTileMode::kClamp; | 
 |     case D2D1_EXTEND_MODE_WRAP: | 
 |         return SkTileMode::kRepeat; | 
 |     case D2D1_EXTEND_MODE_MIRROR: | 
 |         return SkTileMode::kMirror; | 
 |     default: | 
 |         return SkTileMode::kClamp; | 
 |     } | 
 | } | 
 |  | 
 | SkBlendMode sk_blend_mode_from(DWRITE_COLOR_COMPOSITE_MODE compositeMode) { | 
 |     switch (compositeMode) { | 
 |     case DWRITE_COLOR_COMPOSITE_CLEAR: | 
 |         return SkBlendMode::kClear; | 
 |     case DWRITE_COLOR_COMPOSITE_SRC: | 
 |         return SkBlendMode::kSrc; | 
 |     case DWRITE_COLOR_COMPOSITE_DEST: | 
 |         return SkBlendMode::kDst; | 
 |     case DWRITE_COLOR_COMPOSITE_SRC_OVER: | 
 |         return SkBlendMode::kSrcOver; | 
 |     case DWRITE_COLOR_COMPOSITE_DEST_OVER: | 
 |         return SkBlendMode::kDstOver; | 
 |     case DWRITE_COLOR_COMPOSITE_SRC_IN: | 
 |         return SkBlendMode::kSrcIn; | 
 |     case DWRITE_COLOR_COMPOSITE_DEST_IN: | 
 |         return SkBlendMode::kDstIn; | 
 |     case DWRITE_COLOR_COMPOSITE_SRC_OUT: | 
 |         return SkBlendMode::kSrcOut; | 
 |     case DWRITE_COLOR_COMPOSITE_DEST_OUT: | 
 |         return SkBlendMode::kDstOut; | 
 |     case DWRITE_COLOR_COMPOSITE_SRC_ATOP: | 
 |         return SkBlendMode::kSrcATop; | 
 |     case DWRITE_COLOR_COMPOSITE_DEST_ATOP: | 
 |         return SkBlendMode::kDstATop; | 
 |     case DWRITE_COLOR_COMPOSITE_XOR: | 
 |         return SkBlendMode::kXor; | 
 |     case DWRITE_COLOR_COMPOSITE_PLUS: | 
 |         return SkBlendMode::kPlus; | 
 |  | 
 |     case DWRITE_COLOR_COMPOSITE_SCREEN: | 
 |         return SkBlendMode::kScreen; | 
 |     case DWRITE_COLOR_COMPOSITE_OVERLAY: | 
 |         return SkBlendMode::kOverlay; | 
 |     case DWRITE_COLOR_COMPOSITE_DARKEN: | 
 |         return SkBlendMode::kDarken; | 
 |     case DWRITE_COLOR_COMPOSITE_LIGHTEN: | 
 |         return SkBlendMode::kLighten; | 
 |     case DWRITE_COLOR_COMPOSITE_COLOR_DODGE: | 
 |         return SkBlendMode::kColorDodge; | 
 |     case DWRITE_COLOR_COMPOSITE_COLOR_BURN: | 
 |         return SkBlendMode::kColorBurn; | 
 |     case DWRITE_COLOR_COMPOSITE_HARD_LIGHT: | 
 |         return SkBlendMode::kHardLight; | 
 |     case DWRITE_COLOR_COMPOSITE_SOFT_LIGHT: | 
 |         return SkBlendMode::kSoftLight; | 
 |     case DWRITE_COLOR_COMPOSITE_DIFFERENCE: | 
 |         return SkBlendMode::kDifference; | 
 |     case DWRITE_COLOR_COMPOSITE_EXCLUSION: | 
 |         return SkBlendMode::kExclusion; | 
 |     case DWRITE_COLOR_COMPOSITE_MULTIPLY: | 
 |         return SkBlendMode::kMultiply; | 
 |  | 
 |     case DWRITE_COLOR_COMPOSITE_HSL_HUE: | 
 |         return SkBlendMode::kHue; | 
 |     case DWRITE_COLOR_COMPOSITE_HSL_SATURATION: | 
 |         return SkBlendMode::kSaturation; | 
 |     case DWRITE_COLOR_COMPOSITE_HSL_COLOR: | 
 |         return SkBlendMode::kColor; | 
 |     case DWRITE_COLOR_COMPOSITE_HSL_LUMINOSITY: | 
 |         return SkBlendMode::kLuminosity; | 
 |     default: | 
 |         return SkBlendMode::kDst; | 
 |     } | 
 | } | 
 |  | 
 | inline SkPoint SkVectorProjection(SkPoint a, SkPoint b) { | 
 |     SkScalar length = b.length(); | 
 |     if (!length) { | 
 |         return SkPoint(); | 
 |     } | 
 |     SkPoint bNormalized = b; | 
 |     bNormalized.normalize(); | 
 |     bNormalized.scale(SkPoint::DotProduct(a, b) / length); | 
 |     return bNormalized; | 
 | } | 
 |  | 
 | // This linear interpolation is used for calculating a truncated color line in special edge cases. | 
 | // This interpolation needs to be kept in sync with what the gradient shader would normally do when | 
 | // truncating and drawing color lines. When drawing into N32 surfaces, this is expected to be true. | 
 | // If that changes, or if we support other color spaces in CPAL tables at some point, this needs to | 
 | // be looked at. | 
 | D2D1_COLOR_F lerpSkColor(D2D1_COLOR_F c0, D2D1_COLOR_F c1, float t) { | 
 |     // Due to the floating point calculation in the caller, when interpolating between very narrow | 
 |     // stops, we may get values outside the interpolation range, guard against these. | 
 |     if (t < 0) { | 
 |         return c0; | 
 |     } | 
 |     if (t > 1) { | 
 |         return c1; | 
 |     } | 
 |     const auto c0_4f = skvx::float4(c0.r, c0.g, c0.b, c0.a), | 
 |                c1_4f = skvx::float4(c1.r, c1.g, c1.b, c1.a), | 
 |                 c_4f = c0_4f + (c1_4f - c0_4f) * t; | 
 |     D2D1_COLOR_F r; | 
 |     c_4f.store(&r); | 
 |     return r; | 
 | } | 
 |  | 
 | enum TruncateStops { | 
 |     TruncateStart, | 
 |     TruncateEnd, | 
 | }; | 
 | // Truncate a vector of color stops at a previously computed stop position and insert at that | 
 | // position the color interpolated between the surrounding stops. | 
 | void truncateToStopInterpolating(SkScalar zeroRadiusStop, | 
 |                                  std::vector<D2D1_GRADIENT_STOP>& stops, | 
 |                                  TruncateStops truncateStops) { | 
 |     if (stops.size() <= 1u || | 
 |         zeroRadiusStop < stops.front().position || stops.back().position < zeroRadiusStop) { | 
 |         return; | 
 |     } | 
 |  | 
 |     auto lcmp = [](D2D1_GRADIENT_STOP const& stop, SkScalar position) { | 
 |         return stop.position < position; | 
 |     }; | 
 |     auto ucmp = [](SkScalar position, D2D1_GRADIENT_STOP const& stop) { | 
 |         return position < stop.position; | 
 |     }; | 
 |     size_t afterIndex = (truncateStops == TruncateStart) | 
 |         ? std::lower_bound(stops.begin(), stops.end(), zeroRadiusStop, lcmp) - stops.begin() | 
 |         : std::upper_bound(stops.begin(), stops.end(), zeroRadiusStop, ucmp) - stops.begin(); | 
 |  | 
 |     const float t = (zeroRadiusStop - stops[afterIndex - 1].position) / | 
 |         (stops[afterIndex].position - stops[afterIndex - 1].position); | 
 |     D2D1_COLOR_F lerpColor = lerpSkColor(stops[afterIndex - 1].color, stops[afterIndex].color, t); | 
 |  | 
 |     if (truncateStops == TruncateStart) { | 
 |         stops.erase(stops.begin(), stops.begin() + afterIndex); | 
 |         stops.insert(stops.begin(), { 0, lerpColor }); | 
 |     } else { | 
 |         stops.erase(stops.begin() + afterIndex, stops.end()); | 
 |         stops.insert(stops.end(), { 1, lerpColor }); | 
 |     } | 
 | } | 
 | } // namespace | 
 |  | 
 | bool SkScalerContext_DW::drawColorV1Paint(SkCanvas& canvas, | 
 |                                           IDWritePaintReader& reader, | 
 |                                           DWRITE_PAINT_ELEMENT const & element) | 
 | { | 
 |     // Helper to draw the specified number of children. | 
 |     auto drawChildren = [&](uint32_t childCount) -> bool { | 
 |         if (childCount != 0) { | 
 |             DWRITE_PAINT_ELEMENT childElement; | 
 |             HRB(reader.MoveToFirstChild(&childElement)); | 
 |             this->drawColorV1Paint(canvas, reader, childElement); | 
 |  | 
 |             for (uint32_t i = 1; i < childCount; i++) { | 
 |                 HRB(reader.MoveToNextSibling(&childElement)); | 
 |                 this->drawColorV1Paint(canvas, reader, childElement); | 
 |             } | 
 |  | 
 |             HRB(reader.MoveToParent()); | 
 |         } | 
 |         return true; | 
 |     }; | 
 |  | 
 |     SkAutoCanvasRestore restoreCanvas(&canvas, true); | 
 |     switch (element.paintType) { | 
 |     case DWRITE_PAINT_TYPE_NONE: | 
 |         return true; | 
 |  | 
 |     case DWRITE_PAINT_TYPE_LAYERS: { | 
 |         // A layers paint element has a variable number of children. | 
 |         return drawChildren(element.paint.layers.childCount); | 
 |     } | 
 |  | 
 |     case DWRITE_PAINT_TYPE_SOLID_GLYPH: { | 
 |         // A solid glyph paint element has no children. | 
 |         // glyphIndex, color.value, color.paletteEntryIndex, color.alpha, color.colorAttributes | 
 |         auto const& solidGlyph = element.paint.solidGlyph; | 
 |  | 
 |         SkPath path; | 
 |         SkTScopedComPtr<IDWriteGeometrySink> geometryToPath; | 
 |         HRBM(SkDWriteGeometrySink::Create(&path, &geometryToPath), | 
 |              "Could not create geometry to path converter."); | 
 |         UINT16 glyphId = SkTo<UINT16>(solidGlyph.glyphIndex); | 
 |         { | 
 |             Exclusive l(maybe_dw_mutex(*this->getDWriteTypeface())); | 
 |             HRBM(this->getDWriteTypeface()->fDWriteFontFace->GetGlyphRunOutline( | 
 |                      SkScalarToFloat(fTextSizeRender), | 
 |                      &glyphId, | 
 |                      nullptr, //advances | 
 |                      nullptr, //offsets | 
 |                      1, //num glyphs | 
 |                      FALSE, //sideways | 
 |                      FALSE, //rtl | 
 |                      geometryToPath.get()), | 
 |                  "Could not create glyph outline."); | 
 |         } | 
 |  | 
 |         path.transform(SkMatrix::Scale(1.0f / fTextSizeRender, 1.0f / fTextSizeRender)); | 
 |         SkPaint skPaint; | 
 |         skPaint.setColor4f(sk_color_from(solidGlyph.color.value)); | 
 |         skPaint.setAntiAlias(fRenderingMode != DWRITE_RENDERING_MODE_ALIASED); | 
 |         canvas.drawPath(path, skPaint); | 
 |         return true; | 
 |     } | 
 |  | 
 |     case DWRITE_PAINT_TYPE_SOLID: { | 
 |         // A solid paint element has no children. | 
 |         // value, paletteEntryIndex, alphaMultiplier, colorAttributes | 
 |         SkPaint skPaint; | 
 |         skPaint.setColor4f(sk_color_from(element.paint.solid.value)); | 
 |         canvas.drawPaint(skPaint); | 
 |         return true; | 
 |     } | 
 |  | 
 |     case DWRITE_PAINT_TYPE_LINEAR_GRADIENT: { | 
 |         auto const& linearGradient = element.paint.linearGradient; | 
 |         // A linear gradient paint element has no children. | 
 |         // x0, y0, x1, y1, x2, y2, extendMode, gradientStopCount, [colorStops] | 
 |  | 
 |         if (linearGradient.gradientStopCount == 0) { | 
 |             return true; | 
 |         } | 
 |         std::vector<D2D1_GRADIENT_STOP> stops; | 
 |         stops.resize(linearGradient.gradientStopCount); | 
 |  | 
 |         // If success stops will be ordered. | 
 |         HRBM(reader.GetGradientStops(0, stops.size(), stops.data()), | 
 |              "Could not get linear gradient stops."); | 
 |         SkPaint skPaint; | 
 |         if (stops.size() == 1) { | 
 |             skPaint.setColor4f(sk_color_from(stops[0].color)); | 
 |             canvas.drawPaint(skPaint); | 
 |             return true; | 
 |         } | 
 |         SkPoint linePositions[2] = { {linearGradient.x0, linearGradient.y0}, | 
 |                                      {linearGradient.x1, linearGradient.y1} }; | 
 |         SkPoint p0 = linePositions[0]; | 
 |         SkPoint p1 = linePositions[1]; | 
 |         SkPoint p2 = SkPoint::Make(linearGradient.x2, linearGradient.y2); | 
 |  | 
 |         // If p0p1 or p0p2 are degenerate probably nothing should be drawn. | 
 |         // If p0p1 and p0p2 are parallel then one side is the first color and the other side is | 
 |         // the last color, depending on the direction. | 
 |         // For now, just use the first color. | 
 |         if (p1 == p0 || p2 == p0 || !SkPoint::CrossProduct(p1 - p0, p2 - p0)) { | 
 |             skPaint.setColor4f(sk_color_from(stops[0].color)); | 
 |             canvas.drawPaint(skPaint); | 
 |             return true; | 
 |         } | 
 |  | 
 |         // Follow implementation note in nanoemoji: | 
 |         // https://github.com/googlefonts/nanoemoji/blob/0ac6e7bb4d8202db692574d8530a9b643f1b3b3c/src/nanoemoji/svg.py#L188 | 
 |         // to compute a new gradient end point P3 as the orthogonal | 
 |         // projection of the vector from p0 to p1 onto a line perpendicular | 
 |         // to line p0p2 and passing through p0. | 
 |         SkVector perpendicularToP2P0 = (p2 - p0); | 
 |         perpendicularToP2P0 = SkPoint::Make( perpendicularToP2P0.y(), | 
 |                                             -perpendicularToP2P0.x()); | 
 |         SkVector p3 = p0 + SkVectorProjection((p1 - p0), perpendicularToP2P0); | 
 |         linePositions[1] = p3; | 
 |  | 
 |         // Project/scale points according to stop extrema along p0p3 line, | 
 |         // p3 being the result of the projection above, then scale stops to | 
 |         // to [0, 1] range so that repeat modes work.  The Skia linear | 
 |         // gradient shader performs the repeat modes over the 0 to 1 range, | 
 |         // that's why we need to scale the stops to within that range. | 
 |         SkTileMode tileMode = sk_tile_mode_from(SkTo<D2D1_EXTEND_MODE>(linearGradient.extendMode)); | 
 |         SkScalar colorStopRange = stops.back().position - stops.front().position; | 
 |         // If the color stops are all at the same offset position, repeat and reflect modes | 
 |         // become meaningless. | 
 |         if (colorStopRange == 0.f) { | 
 |             if (tileMode != SkTileMode::kClamp) { | 
 |                 //skPaint.setColor(SK_ColorTRANSPARENT); | 
 |                 return true; | 
 |             } else { | 
 |                 // Insert duplicated fake color stop in pad case at +1.0f to enable the projection | 
 |                 // of circles for an originally 0-length color stop range. Adding this stop will | 
 |                 // paint the equivalent gradient, because: All font specified color stops are in the | 
 |                 // same spot, mode is pad, so everything before this spot is painted with the first | 
 |                 // color, everything after this spot is painted with the last color. Not adding this | 
 |                 // stop will skip the projection and result in specifying non-normalized color stops | 
 |                 // to the shader. | 
 |                 stops.push_back({ stops.back().position + 1.0f, stops.back().color }); | 
 |                 colorStopRange = 1.0f; | 
 |             } | 
 |         } | 
 |         SkASSERT(colorStopRange != 0.f); | 
 |  | 
 |         // If the colorStopRange is 0 at this point, the default behavior of the shader is to | 
 |         // clamp to 1 color stops that are above 1, clamp to 0 for color stops that are below 0, | 
 |         // and repeat the outer color stops at 0 and 1 if the color stops are inside the | 
 |         // range. That will result in the correct rendering. | 
 |         if ((colorStopRange != 1 || stops.front().position != 0.f)) { | 
 |             SkVector p0p3 = p3 - p0; | 
 |             SkVector p0Offset = p0p3; | 
 |             p0Offset.scale(stops.front().position); | 
 |             SkVector p1Offset = p0p3; | 
 |             p1Offset.scale(stops.back().position); | 
 |  | 
 |             linePositions[0] = p0 + p0Offset; | 
 |             linePositions[1] = p0 + p1Offset; | 
 |  | 
 |             SkScalar scaleFactor = 1 / colorStopRange; | 
 |             SkScalar startOffset = stops.front().position; | 
 |             for (D2D1_GRADIENT_STOP& stop : stops) { | 
 |                 stop.position = (stop.position - startOffset) * scaleFactor; | 
 |             } | 
 |         } | 
 |  | 
 |         std::unique_ptr<SkColor4f[]> skColors(new SkColor4f[stops.size()]); | 
 |         std::unique_ptr<SkScalar[]> skStops(new SkScalar[stops.size()]); | 
 |         for (size_t i = 0; i < stops.size(); ++i) { | 
 |             skColors[i] = sk_color_from(stops[i].color); | 
 |             skStops[i] = stops[i].position; | 
 |         } | 
 |  | 
 |         sk_sp<SkShader> shader(SkGradientShader::MakeLinear( | 
 |             linePositions, | 
 |             skColors.get(), SkColorSpace::MakeSRGB(), skStops.get(), stops.size(), | 
 |             tileMode, | 
 |             SkGradientShader::Interpolation{ | 
 |                 SkGradientShader::Interpolation::InPremul::kNo, | 
 |                 SkGradientShader::Interpolation::ColorSpace::kSRGB, | 
 |                 SkGradientShader::Interpolation::HueMethod::kShorter | 
 |             }, | 
 |             nullptr)); | 
 |  | 
 |         SkASSERT(shader); | 
 |         // An opaque color is needed to ensure the gradient is not modulated by alpha. | 
 |         skPaint.setColor(SK_ColorBLACK); | 
 |         skPaint.setShader(shader); | 
 |         canvas.drawPaint(skPaint); | 
 |         return true; | 
 |     } | 
 |  | 
 |     case DWRITE_PAINT_TYPE_RADIAL_GRADIENT: { | 
 |         auto const& radialGradient = element.paint.radialGradient; | 
 |         // A radial gradient paint element has no children. | 
 |         // x0, y0, radius0, x1, y1, radius1, extendMode, gradientStopCount, [colorsStops] | 
 |  | 
 |         SkPoint start = SkPoint::Make(radialGradient.x0, radialGradient.y0); | 
 |         SkScalar startRadius = radialGradient.radius0; | 
 |         SkPoint end = SkPoint::Make(radialGradient.x1, radialGradient.y1); | 
 |         SkScalar endRadius = radialGradient.radius1; | 
 |  | 
 |         if (radialGradient.gradientStopCount == 0) { | 
 |             return true; | 
 |         } | 
 |         std::vector<D2D1_GRADIENT_STOP> stops; | 
 |         stops.resize(radialGradient.gradientStopCount); | 
 |  | 
 |         // If success stops will be ordered. | 
 |         HRBM(reader.GetGradientStops(0, stops.size(), stops.data()), | 
 |              "Could not get radial gradient stops."); | 
 |         SkPaint skPaint; | 
 |         if (stops.size() == 1) { | 
 |             skPaint.setColor4f(sk_color_from(stops[0].color)); | 
 |             canvas.drawPaint(skPaint); | 
 |             return true; | 
 |         } | 
 |  | 
 |         SkScalar colorStopRange = stops.back().position - stops.front().position; | 
 |         SkTileMode tileMode = sk_tile_mode_from(SkTo<D2D1_EXTEND_MODE>(radialGradient.extendMode)); | 
 |  | 
 |         if (colorStopRange == 0.f) { | 
 |             if (tileMode != SkTileMode::kClamp) { | 
 |                 //skPaint.setColor(SK_ColorTRANSPARENT); | 
 |                 return true; | 
 |             } else { | 
 |                 // Insert duplicated fake color stop in pad case at +1.0f to enable the projection | 
 |                 // of circles for an originally 0-length color stop range. Adding this stop will | 
 |                 // paint the equivalent gradient, because: All font specified color stops are in the | 
 |                 // same spot, mode is pad, so everything before this spot is painted with the first | 
 |                 // color, everything after this spot is painted with the last color. Not adding this | 
 |                 // stop will skip the projection and result in specifying non-normalized color stops | 
 |                 // to the shader. | 
 |                 stops.push_back({ stops.back().position + 1.0f, stops.back().color }); | 
 |                 colorStopRange = 1.0f; | 
 |             } | 
 |         } | 
 |         SkASSERT(colorStopRange != 0.f); | 
 |  | 
 |         // If the colorStopRange is 0 at this point, the default behavior of the shader is to | 
 |         // clamp to 1 color stops that are above 1, clamp to 0 for color stops that are below 0, | 
 |         // and repeat the outer color stops at 0 and 1 if the color stops are inside the | 
 |         // range. That will result in the correct rendering. | 
 |         if (colorStopRange != 1 || stops.front().position != 0.f) { | 
 |             // For the Skia two-point caonical shader to understand the | 
 |             // COLRv1 color stops we need to scale stops to 0 to 1 range and | 
 |             // interpolate new centers and radii. Otherwise the shader | 
 |             // clamps stops outside the range to 0 and 1 (larger interval) | 
 |             // or repeats the outer stops at 0 and 1 if the (smaller | 
 |             // interval). | 
 |             SkVector startToEnd = end - start; | 
 |             SkScalar radiusDiff = endRadius - startRadius; | 
 |             SkScalar scaleFactor = 1 / colorStopRange; | 
 |             SkScalar stopsStartOffset = stops.front().position; | 
 |  | 
 |             SkVector startOffset = startToEnd; | 
 |             startOffset.scale(stops.front().position); | 
 |             SkVector endOffset = startToEnd; | 
 |             endOffset.scale(stops.back().position); | 
 |  | 
 |             // The order of the following computations is important in order to avoid | 
 |             // overwriting start or startRadius before the second reassignment. | 
 |             end = start + endOffset; | 
 |             start = start + startOffset; | 
 |             endRadius = startRadius + radiusDiff * stops.back().position; | 
 |             startRadius = startRadius + radiusDiff * stops.front().position; | 
 |  | 
 |             for (auto& stop : stops) { | 
 |                 stop.position = (stop.position - stopsStartOffset) * scaleFactor; | 
 |             } | 
 |         } | 
 |  | 
 |         // For negative radii, interpolation is needed to prepare parameters suitable | 
 |         // for invoking the shader. Implementation below as resolution discussed in | 
 |         // https://github.com/googlefonts/colr-gradients-spec/issues/367. | 
 |         // Truncate to manually interpolated color for tile mode clamp, otherwise | 
 |         // calculate positive projected circles. | 
 |         if (startRadius < 0 || endRadius < 0) { | 
 |             if (startRadius == endRadius && startRadius < 0) { | 
 |                 //skPaint.setColor(SK_ColorTRANSPARENT); | 
 |                 return true; | 
 |             } | 
 |  | 
 |             if (tileMode == SkTileMode::kClamp) { | 
 |                 SkVector startToEnd = end - start; | 
 |                 SkScalar radiusDiff = endRadius - startRadius; | 
 |                 SkScalar zeroRadiusStop = 0.f; | 
 |                 TruncateStops truncateSide = TruncateStart; | 
 |                 if (startRadius < 0) { | 
 |                     truncateSide = TruncateStart; | 
 |  | 
 |                     // Compute color stop position where radius is = 0.  After the scaling | 
 |                     // of stop positions to the normal 0,1 range that we have done above, | 
 |                     // the size of the radius as a function of the color stops is: r(x) = r0 | 
 |                     // + x*(r1-r0) Solving this function for r(x) = 0, we get: x = -r0 / | 
 |                     // (r1-r0) | 
 |                     zeroRadiusStop = -startRadius / (endRadius - startRadius); | 
 |                     startRadius = 0.f; | 
 |                     SkVector startEndDiff = end - start; | 
 |                     startEndDiff.scale(zeroRadiusStop); | 
 |                     start = start + startEndDiff; | 
 |                 } | 
 |  | 
 |                 if (endRadius < 0) { | 
 |                     truncateSide = TruncateEnd; | 
 |                     zeroRadiusStop = -startRadius / (endRadius - startRadius); | 
 |                     endRadius = 0.f; | 
 |                     SkVector startEndDiff = end - start; | 
 |                     startEndDiff.scale(1 - zeroRadiusStop); | 
 |                     end = end - startEndDiff; | 
 |                 } | 
 |  | 
 |                 if (!(startRadius == 0 && endRadius == 0)) { | 
 |                     truncateToStopInterpolating(zeroRadiusStop, stops, truncateSide); | 
 |                 } else { | 
 |                     // If both radii have become negative and where clamped to 0, we need to | 
 |                     // produce a single color cone, otherwise the shader colors the whole | 
 |                     // plane in a single color when two radii are specified as 0. | 
 |                     if (radiusDiff > 0) { | 
 |                         end = start + startToEnd; | 
 |                         endRadius = radiusDiff; | 
 |                         stops.erase(stops.begin(), stops.end() - 1); | 
 |                     } else { | 
 |                         start -= startToEnd; | 
 |                         startRadius = -radiusDiff; | 
 |                         stops.erase(stops.begin() + 1, stops.end()); | 
 |                     } | 
 |                 } | 
 |             } else { | 
 |                 if (startRadius < 0 || endRadius < 0) { | 
 |                     auto roundIntegerMultiple = [](SkScalar factorZeroCrossing, | 
 |                         SkTileMode tileMode) { | 
 |                             int roundedMultiple = factorZeroCrossing > 0 | 
 |                                 ? ceilf(factorZeroCrossing) | 
 |                                 : floorf(factorZeroCrossing) - 1; | 
 |                             if (tileMode == SkTileMode::kMirror && roundedMultiple % 2 != 0) { | 
 |                                 roundedMultiple += roundedMultiple < 0 ? -1 : 1; | 
 |                             } | 
 |                             return roundedMultiple; | 
 |                     }; | 
 |  | 
 |                     SkVector startToEnd = end - start; | 
 |                     SkScalar radiusDiff = endRadius - startRadius; | 
 |                     SkScalar factorZeroCrossing = (startRadius / (startRadius - endRadius)); | 
 |                     bool inRange = 0.f <= factorZeroCrossing && factorZeroCrossing <= 1.0f; | 
 |                     SkScalar direction = inRange && radiusDiff < 0 ? -1.0f : 1.0f; | 
 |                     SkScalar circleProjectionFactor = | 
 |                         roundIntegerMultiple(factorZeroCrossing * direction, tileMode); | 
 |                     startToEnd.scale(circleProjectionFactor); | 
 |                     startRadius += circleProjectionFactor * radiusDiff; | 
 |                     endRadius += circleProjectionFactor * radiusDiff; | 
 |                     start += startToEnd; | 
 |                     end += startToEnd; | 
 |                 } | 
 |             } | 
 |         } | 
 |  | 
 |         std::unique_ptr<SkColor4f[]> skColors(new SkColor4f[stops.size()]); | 
 |         std::unique_ptr<SkScalar[]> skStops(new SkScalar[stops.size()]); | 
 |         for (size_t i = 0; i < stops.size(); ++i) { | 
 |             skColors[i] = sk_color_from(stops[i].color); | 
 |             skStops[i] = stops[i].position; | 
 |         } | 
 |  | 
 |         // An opaque color is needed to ensure the gradient is not modulated by alpha. | 
 |         skPaint.setColor(SK_ColorBLACK); | 
 |         skPaint.setShader(SkGradientShader::MakeTwoPointConical( | 
 |             start, startRadius, end, endRadius, | 
 |             skColors.get(), SkColorSpace::MakeSRGB(), skStops.get(), stops.size(), | 
 |             tileMode, | 
 |             SkGradientShader::Interpolation{ | 
 |                 SkGradientShader::Interpolation::InPremul::kNo, | 
 |                 SkGradientShader::Interpolation::ColorSpace::kSRGB, | 
 |                 SkGradientShader::Interpolation::HueMethod::kShorter | 
 |             }, | 
 |             nullptr)); | 
 |         canvas.drawPaint(skPaint); | 
 |         return true; | 
 |     } | 
 |  | 
 |     case DWRITE_PAINT_TYPE_SWEEP_GRADIENT: { | 
 |         auto const& sweepGradient = element.paint.sweepGradient; | 
 |         // A sweep gradient paint element has no children. | 
 |         // centerX, centerY, startAngle, endAngle, extendMode, gradientStopCount, [colorStops] | 
 |  | 
 |         if (sweepGradient.gradientStopCount == 0) { | 
 |             return true; | 
 |         } | 
 |         std::vector<D2D1_GRADIENT_STOP> stops; | 
 |         stops.resize(sweepGradient.gradientStopCount); | 
 |  | 
 |         // If success stops will be ordered. | 
 |         HRBM(reader.GetGradientStops(0, stops.size(), stops.data()), | 
 |              "Could not get sweep gradient stops"); | 
 |         SkPaint skPaint; | 
 |         if (stops.size() == 1) { | 
 |             skPaint.setColor4f(sk_color_from(stops[0].color)); | 
 |             canvas.drawPaint(skPaint); | 
 |             return true; | 
 |         } | 
 |  | 
 |         SkPoint center = SkPoint::Make(sweepGradient.centerX, sweepGradient.centerY); | 
 |  | 
 |         SkScalar startAngle = sweepGradient.startAngle; | 
 |         SkScalar endAngle = sweepGradient.endAngle; | 
 |         // OpenType 1.9.1 adds a shift to the angle to ease specification of a 0 to 360 | 
 |         // degree sweep. This appears to already be applied by DW. | 
 |         //startAngle += 180.0f; | 
 |         //endAngle += 180.0f; | 
 |  | 
 |         // An opaque color is needed to ensure the gradient is not modulated by alpha. | 
 |         skPaint.setColor(SK_ColorBLACK); | 
 |  | 
 |         // New (Var)SweepGradient implementation compliant with OpenType 1.9.1 from here. | 
 |  | 
 |         // The shader expects stops from 0 to 1, so we need to account for | 
 |         // minimum and maximum stop positions being different from 0 and | 
 |         // 1. We do that by scaling minimum and maximum stop positions to | 
 |         // the 0 to 1 interval and scaling the angles inverse proportionally. | 
 |  | 
 |         // 1) Scale angles to their equivalent positions if stops were from 0 to 1. | 
 |  | 
 |         SkScalar sectorAngle = endAngle - startAngle; | 
 |         SkTileMode tileMode = sk_tile_mode_from(SkTo<D2D1_EXTEND_MODE>(sweepGradient.extendMode)); | 
 |         if (sectorAngle == 0 && tileMode != SkTileMode::kClamp) { | 
 |             // "If the ColorLine's extend mode is reflect or repeat and start and end angle | 
 |             // are equal, nothing is drawn.". | 
 |             //skPaint.setColor(SK_ColorTRANSPARENT); | 
 |             return true; | 
 |         } | 
 |  | 
 |         SkScalar startAngleScaled = startAngle + sectorAngle * stops.front().position; | 
 |         SkScalar endAngleScaled = startAngle + sectorAngle * stops.back().position; | 
 |  | 
 |         // 2) Scale stops accordingly to 0 to 1 range. | 
 |  | 
 |         float colorStopRange = stops.back().position - stops.front().position; | 
 |         if (colorStopRange == 0.f) { | 
 |             if (tileMode != SkTileMode::kClamp) { | 
 |                 //skPaint.setColor(SK_ColorTRANSPARENT); | 
 |                 return true; | 
 |             } else { | 
 |                 // Insert duplicated fake color stop in pad case at +1.0f to feed the shader correct | 
 |                 // values and enable painting a pad sweep gradient with two colors. Adding this stop | 
 |                 // will paint the equivalent gradient, because: All font specified color stops are | 
 |                 // in the same spot, mode is pad, so everything before this spot is painted with the | 
 |                 // first color, everything after this spot is painted with the last color. Not | 
 |                 // adding this stop will skip the projection and result in specifying non-normalized | 
 |                 // color stops to the shader. | 
 |                 stops.push_back({ stops.back().position + 1.0f, stops.back().color }); | 
 |                 colorStopRange = 1.0f; | 
 |             } | 
 |         } | 
 |  | 
 |         SkScalar scaleFactor = 1 / colorStopRange; | 
 |         SkScalar startOffset = stops.front().position; | 
 |  | 
 |         for (D2D1_GRADIENT_STOP& stop : stops) { | 
 |             stop.position = (stop.position - startOffset) * scaleFactor; | 
 |         } | 
 |  | 
 |         /* https://docs.microsoft.com/en-us/typography/opentype/spec/colr#sweep-gradients | 
 |         * "The angles are expressed in counter-clockwise degrees from | 
 |         * the direction of the positive x-axis on the design | 
 |         * grid. [...]  The color line progresses from the start angle | 
 |         * to the end angle in the counter-clockwise direction;" - | 
 |         * Convert angles and stops from counter-clockwise to clockwise | 
 |         * for the shader if the gradient is not already reversed due to | 
 |         * start angle being larger than end angle. */ | 
 |         startAngleScaled = 360.f - startAngleScaled; | 
 |         endAngleScaled = 360.f - endAngleScaled; | 
 |         if (startAngleScaled >= endAngleScaled) { | 
 |             std::swap(startAngleScaled, endAngleScaled); | 
 |             std::reverse(stops.begin(), stops.end()); | 
 |             for (auto& stop : stops) { | 
 |                 stop.position = 1.0f - stop.position; | 
 |             } | 
 |         } | 
 |  | 
 |         std::unique_ptr<SkColor4f[]> skColors(new SkColor4f[stops.size()]); | 
 |         std::unique_ptr<SkScalar[]> skStops(new SkScalar[stops.size()]); | 
 |         for (size_t i = 0; i < stops.size(); ++i) { | 
 |             skColors[i] = sk_color_from(stops[i].color); | 
 |             skStops[i] = stops[i].position; | 
 |         } | 
 |  | 
 |         skPaint.setShader(SkGradientShader::MakeSweep( | 
 |             center.x(), center.y(), | 
 |             skColors.get(), SkColorSpace::MakeSRGB(), skStops.get(), stops.size(), | 
 |             tileMode, | 
 |             startAngleScaled, endAngleScaled, | 
 |             SkGradientShader::Interpolation{ | 
 |                 SkGradientShader::Interpolation::InPremul::kNo, | 
 |                 SkGradientShader::Interpolation::ColorSpace::kSRGB, | 
 |                 SkGradientShader::Interpolation::HueMethod::kShorter | 
 |             }, | 
 |             nullptr)); | 
 |         canvas.drawPaint(skPaint); | 
 |         return true; | 
 |     } | 
 |  | 
 |     case DWRITE_PAINT_TYPE_GLYPH: { | 
 |         // A glyph paint element has one child, which is the fill for the glyph shape glyphIndex. | 
 |         SkPath path; | 
 |         SkTScopedComPtr<IDWriteGeometrySink> geometryToPath; | 
 |         HRBM(SkDWriteGeometrySink::Create(&path, &geometryToPath), | 
 |              "Could not create geometry to path converter."); | 
 |         UINT16 glyphId = SkTo<UINT16>(element.paint.glyph.glyphIndex); | 
 |         { | 
 |             Exclusive l(maybe_dw_mutex(*this->getDWriteTypeface())); | 
 |             HRBM(this->getDWriteTypeface()->fDWriteFontFace->GetGlyphRunOutline( | 
 |                      SkScalarToFloat(fTextSizeRender), | 
 |                      &glyphId, | 
 |                      nullptr, //advances | 
 |                      nullptr, //offsets | 
 |                      1, //num glyphs | 
 |                      FALSE, //sideways | 
 |                      FALSE, //rtl | 
 |                      geometryToPath.get()), | 
 |                  "Could not create glyph outline."); | 
 |         } | 
 |  | 
 |         path.transform(SkMatrix::Scale(1.0f / fTextSizeRender, 1.0f / fTextSizeRender)); | 
 |         canvas.clipPath(path, fRenderingMode != DWRITE_RENDERING_MODE_ALIASED); | 
 |  | 
 |         drawChildren(1); | 
 |         return true; | 
 |     } | 
 |  | 
 |     case DWRITE_PAINT_TYPE_COLOR_GLYPH: { | 
 |         auto const& colorGlyph = element.paint.colorGlyph; | 
 |         // A color glyph paint element has one child, the root of the paint tree for glyphIndex. | 
 |         // glyphIndex, clipBox | 
 |         if (D2D_RECT_F_is_empty(colorGlyph.clipBox)) { | 
 |             // Does not have a clip box | 
 |         } else { | 
 |             SkRect r = sk_rect_from(colorGlyph.clipBox); | 
 |             canvas.clipRect(r, fRenderingMode != DWRITE_RENDERING_MODE_ALIASED); | 
 |         } | 
 |  | 
 |         drawChildren(1); | 
 |         return true; | 
 |     } | 
 |  | 
 |     case DWRITE_PAINT_TYPE_TRANSFORM: { | 
 |         // A transform paint element always has one child, the transformed content. | 
 |         canvas.concat(sk_matrix_from(element.paint.transform)); | 
 |         drawChildren(1); | 
 |         return true; | 
 |     } | 
 |  | 
 |     case DWRITE_PAINT_TYPE_COMPOSITE: { | 
 |         // A composite paint element has two children, the source and destination of the operation. | 
 |  | 
 |         SkPaint blendModePaint; | 
 |         blendModePaint.setBlendMode(sk_blend_mode_from(element.paint.composite.mode)); | 
 |  | 
 |         SkAutoCanvasRestore acr(&canvas, false); | 
 |  | 
 |         // Need to visit the second child first and do savelayers, so manually handle children. | 
 |         DWRITE_PAINT_ELEMENT sourceElement; | 
 |         DWRITE_PAINT_ELEMENT backdropElement; | 
 |  | 
 |         HRBM(reader.MoveToFirstChild(&sourceElement), "Could not move to child."); | 
 |         HRBM(reader.MoveToNextSibling(&backdropElement), "Could not move to sibiling."); | 
 |         canvas.saveLayer(nullptr, nullptr); | 
 |         this->drawColorV1Paint(canvas, reader, backdropElement); | 
 |  | 
 |         HRBM(reader.MoveToParent(), "Could not move to parent."); | 
 |         HRBM(reader.MoveToFirstChild(&sourceElement), "Could not move to child."); | 
 |         canvas.saveLayer(nullptr, &blendModePaint); | 
 |         this->drawColorV1Paint(canvas, reader, sourceElement); | 
 |  | 
 |         HRBM(reader.MoveToParent(), "Could not move to parent."); | 
 |  | 
 |         return true; | 
 |     } | 
 |  | 
 |     default: | 
 |         return false; | 
 |     } | 
 | } | 
 |  | 
 | bool SkScalerContext_DW::drawColorV1Image(const SkGlyph& glyph, SkCanvas& canvas) { | 
 |     DWriteFontTypeface* typeface = this->getDWriteTypeface(); | 
 |     IDWriteFontFace7* fontFace = typeface->fDWriteFontFace7/*.get()*/; | 
 |     if (!fontFace) { | 
 |         return false; | 
 |     } | 
 |     UINT32 glyphIndex = glyph.getGlyphID(); | 
 |  | 
 |     SkTScopedComPtr<IDWritePaintReader> paintReader; | 
 |     HRBM(fontFace->CreatePaintReader(DWRITE_GLYPH_IMAGE_FORMATS_COLR_PAINT_TREE, | 
 |                                      DWRITE_PAINT_FEATURE_LEVEL_COLR_V1, | 
 |                                      &paintReader), | 
 |          "Could not create paint reader."); | 
 |  | 
 |     DWRITE_PAINT_ELEMENT paintElement; | 
 |     D2D_RECT_F clipBox; | 
 |     DWRITE_PAINT_ATTRIBUTES attributes; | 
 |     HRBM(paintReader->SetCurrentGlyph(glyphIndex, &paintElement, &clipBox, &attributes), | 
 |          "Could not set current glyph."); | 
 |  | 
 |     if (paintElement.paintType == DWRITE_PAINT_TYPE_NONE) { | 
 |         // Does not have paint layers, try another format. | 
 |         return false; | 
 |     } | 
 |  | 
 |     // All coordinates (including top level clip) are reported in "em"s (1 == em). | 
 |     // Size up all em units to the current size and transform. | 
 |     // Get glyph paths at render size, divide out the render size to get em units. | 
 |  | 
 |     SkMatrix matrix = fSkXform; | 
 |     SkScalar scale = fTextSizeRender; | 
 |     matrix.preScale(scale, scale); | 
 |     if (this->isSubpixel()) { | 
 |         matrix.postTranslate(SkFixedToScalar(glyph.getSubXFixed()), | 
 |                              SkFixedToScalar(glyph.getSubYFixed())); | 
 |     } | 
 |     canvas.concat(matrix); | 
 |  | 
 |     if (D2D_RECT_F_is_empty(clipBox)) { | 
 |         // Does not have a clip box | 
 |     } else { | 
 |         canvas.clipRect(sk_rect_from(clipBox)); | 
 |     } | 
 |  | 
 |     // The DirectWrite interface returns resolved colors if these are provided. | 
 |     // Indexes and alphas are reported but there is no reason to duplicate the color calculation. | 
 |     paintReader->SetTextColor(dw_color_from(SkColor4f::FromColor(fRec.fForegroundColor))); | 
 |     paintReader->SetCustomColorPalette(typeface->fDWPalette.get(), typeface->fPaletteEntryCount); | 
 |  | 
 |     return this->drawColorV1Paint(canvas, *paintReader, paintElement); | 
 | } | 
 |  | 
 | bool SkScalerContext_DW::generateColorV1Image(const SkGlyph& glyph, void* imageBuffer) { | 
 |     SkASSERT(glyph.maskFormat() == SkMask::Format::kARGB32_Format); | 
 |  | 
 |     SkBitmap dstBitmap; | 
 |     // TODO: mark this as sRGB when the blits will be sRGB. | 
 |     dstBitmap.setInfo(SkImageInfo::Make(glyph.width(), glyph.height(), | 
 |                       kN32_SkColorType, kPremul_SkAlphaType), | 
 |                       glyph.rowBytes()); | 
 |     dstBitmap.setPixels(imageBuffer); | 
 |  | 
 |     SkCanvas canvas(dstBitmap); | 
 |     if constexpr (kSkShowTextBlitCoverage) { | 
 |         canvas.clear(0x33FF0000); | 
 |     } else { | 
 |         canvas.clear(SK_ColorTRANSPARENT); | 
 |     } | 
 |     canvas.translate(-SkIntToScalar(glyph.left()), -SkIntToScalar(glyph.top())); | 
 |  | 
 |     return this->drawColorV1Image(glyph, canvas); | 
 | } | 
 |  | 
 | bool SkScalerContext_DW::generateColorV1PaintBounds( | 
 |     SkMatrix* ctm, SkRect* bounds, | 
 |     IDWritePaintReader& reader, DWRITE_PAINT_ELEMENT const & element) | 
 | { | 
 |     // Helper to iterate over the specified number of children. | 
 |     auto boundChildren = [&](UINT32 childCount) -> bool { | 
 |         if (childCount == 0) { | 
 |             return true; | 
 |         } | 
 |         DWRITE_PAINT_ELEMENT childElement; | 
 |         HRB(reader.MoveToFirstChild(&childElement)); | 
 |         this->generateColorV1PaintBounds(ctm, bounds, reader, childElement); | 
 |  | 
 |         for (uint32_t i = 1; i < childCount; ++i) { | 
 |             HRB(reader.MoveToNextSibling(&childElement)); | 
 |             this->generateColorV1PaintBounds(ctm, bounds, reader, childElement); | 
 |         } | 
 |  | 
 |         HRB(reader.MoveToParent()); | 
 |         return true; | 
 |     }; | 
 |  | 
 |     SkMatrix restoreMatrix = *ctm; | 
 |     SK_AT_SCOPE_EXIT(*ctm = restoreMatrix); | 
 |  | 
 |     switch (element.paintType) { | 
 |     case DWRITE_PAINT_TYPE_NONE: | 
 |         return false; | 
 |  | 
 |     case DWRITE_PAINT_TYPE_LAYERS: { | 
 |         // A layers paint element has a variable number of children. | 
 |         return boundChildren(element.paint.layers.childCount); | 
 |     } | 
 |  | 
 |     case DWRITE_PAINT_TYPE_SOLID_GLYPH: { | 
 |         // A solid glyph paint element has no children. | 
 |         // glyphIndex, color.value, color.paletteEntryIndex, color.alpha, color.colorAttributes | 
 |  | 
 |         SkPath path; | 
 |         SkTScopedComPtr<IDWriteGeometrySink> geometryToPath; | 
 |         HRBM(SkDWriteGeometrySink::Create(&path, &geometryToPath), | 
 |             "Could not create geometry to path converter."); | 
 |         UINT16 glyphId = SkTo<UINT16>(element.paint.solidGlyph.glyphIndex); | 
 |         { | 
 |             Exclusive l(maybe_dw_mutex(*this->getDWriteTypeface())); | 
 |             HRBM(this->getDWriteTypeface()->fDWriteFontFace->GetGlyphRunOutline( | 
 |                     SkScalarToFloat(fTextSizeRender), | 
 |                     &glyphId, | 
 |                     nullptr, //advances | 
 |                     nullptr, //offsets | 
 |                     1, //num glyphs | 
 |                     FALSE, //sideways | 
 |                     FALSE, //rtl | 
 |                     geometryToPath.get()), | 
 |                 "Could not create glyph outline."); | 
 |         } | 
 |  | 
 |         SkMatrix t = *ctm; | 
 |         t.preConcat(SkMatrix::Scale(1.0f / fTextSizeRender, 1.0f / fTextSizeRender)); | 
 |         path.transform(t); | 
 |         bounds->join(path.getBounds()); | 
 |         return true; | 
 |     } | 
 |  | 
 |     case DWRITE_PAINT_TYPE_SOLID: { | 
 |         return true; | 
 |     } | 
 |  | 
 |     case DWRITE_PAINT_TYPE_LINEAR_GRADIENT: { | 
 |         return true; | 
 |     } | 
 |  | 
 |     case DWRITE_PAINT_TYPE_RADIAL_GRADIENT: { | 
 |         return true; | 
 |     } | 
 |  | 
 |     case DWRITE_PAINT_TYPE_SWEEP_GRADIENT: { | 
 |         return true; | 
 |     } | 
 |  | 
 |     case DWRITE_PAINT_TYPE_GLYPH: { | 
 |         // A glyph paint element has one child, which is the fill for the glyph shape glyphIndex. | 
 |         SkPath path; | 
 |         SkTScopedComPtr<IDWriteGeometrySink> geometryToPath; | 
 |         HRBM(SkDWriteGeometrySink::Create(&path, &geometryToPath), | 
 |              "Could not create geometry to path converter."); | 
 |         UINT16 glyphId = SkTo<UINT16>(element.paint.glyph.glyphIndex); | 
 |         { | 
 |             Exclusive l(maybe_dw_mutex(*this->getDWriteTypeface())); | 
 |             HRBM(this->getDWriteTypeface()->fDWriteFontFace->GetGlyphRunOutline( | 
 |                      SkScalarToFloat(fTextSizeRender), | 
 |                      &glyphId, | 
 |                      nullptr, //advances | 
 |                      nullptr, //offsets | 
 |                      1, //num glyphs | 
 |                      FALSE, //sideways | 
 |                      FALSE, //rtl | 
 |                      geometryToPath.get()), | 
 |                  "Could not create glyph outline."); | 
 |         } | 
 |  | 
 |         SkMatrix t = *ctm; | 
 |         t.preConcat(SkMatrix::Scale(1.0f / fTextSizeRender, 1.0f / fTextSizeRender)); | 
 |         path.transform(t); | 
 |         bounds->join(path.getBounds()); | 
 |         return true; | 
 |     } | 
 |  | 
 |     case DWRITE_PAINT_TYPE_COLOR_GLYPH: { | 
 |         // A color glyph paint element has one child, which is the root | 
 |         // of the paint tree for the glyph specified by glyphIndex. | 
 |         auto const& colorGlyph = element.paint.colorGlyph; | 
 |         if (D2D_RECT_F_is_empty(colorGlyph.clipBox)) { | 
 |             // Does not have a clip box | 
 |             return boundChildren(1); | 
 |         } | 
 |         SkRect r = sk_rect_from(colorGlyph.clipBox); | 
 |         ctm->mapRect(r); | 
 |         bounds->join(r); | 
 |         return true; | 
 |     } | 
 |  | 
 |     case DWRITE_PAINT_TYPE_TRANSFORM: { | 
 |         // A transform paint element always has one child, which is the transformed content. | 
 |         ctm->preConcat(sk_matrix_from(element.paint.transform)); | 
 |         return boundChildren(1); | 
 |     } | 
 |  | 
 |     case DWRITE_PAINT_TYPE_COMPOSITE: { | 
 |         // A composite paint element has two children, the source and destination of the operation. | 
 |         return boundChildren(2); | 
 |     } | 
 |  | 
 |     default: | 
 |         return false; | 
 |     } | 
 | } | 
 |  | 
 | bool SkScalerContext_DW::generateColorV1Metrics(const SkGlyph& glyph, SkRect* bounds) { | 
 |     DWriteFontTypeface* typeface = this->getDWriteTypeface(); | 
 |     IDWriteFontFace7* fontFace = typeface->fDWriteFontFace7/*.get()*/; | 
 |     if (!fontFace) { | 
 |         return false; | 
 |     } | 
 |     UINT32 glyphIndex = glyph.getGlyphID(); | 
 |  | 
 |     SkTScopedComPtr<IDWritePaintReader> paintReader; | 
 |     HRESULT hr; | 
 |     // No message on failure here, since this will fail if the font has no color glyphs. | 
 |     hr = fontFace->CreatePaintReader(DWRITE_GLYPH_IMAGE_FORMATS_COLR_PAINT_TREE, | 
 |                                      DWRITE_PAINT_FEATURE_LEVEL_COLR_V1, | 
 |                                      &paintReader); | 
 |     if (FAILED(hr)) { | 
 |         return false; | 
 |     } | 
 |  | 
 |     DWRITE_PAINT_ELEMENT paintElement; | 
 |     D2D_RECT_F clipBox; | 
 |     DWRITE_PAINT_ATTRIBUTES attributes; | 
 |     // If the glyph is not color this will succeed but return paintType NONE. | 
 |     HRBM(paintReader->SetCurrentGlyph(glyphIndex, &paintElement, &clipBox, &attributes), | 
 |          "Could not set the current glyph."); | 
 |  | 
 |     if (paintElement.paintType == DWRITE_PAINT_TYPE_NONE) { | 
 |         // Does not have paint layers, try another format. | 
 |         return false; | 
 |     } | 
 |  | 
 |     // All coordinates (including top level clip) are reported in "em"s (1 == em). | 
 |     // Size up all em units to the current size and transform. | 
 |  | 
 |     SkMatrix matrix = fSkXform; | 
 |     SkScalar scale = fTextSizeRender; | 
 |     matrix.preScale(scale, scale); | 
 |     if (this->isSubpixel()) { | 
 |         matrix.postTranslate(SkFixedToScalar(glyph.getSubXFixed()), | 
 |                              SkFixedToScalar(glyph.getSubYFixed())); | 
 |     } | 
 |  | 
 |     SkRect r; | 
 |     if (D2D_RECT_F_is_empty(clipBox)) { | 
 |         // Does not have a clip box. | 
 |         r = SkRect::MakeEmpty(); | 
 |         if (!this->generateColorV1PaintBounds(&matrix, &r, *paintReader, paintElement)) { | 
 |             return false; | 
 |         } | 
 |         *bounds = r; | 
 |     } else { | 
 |         *bounds = sk_rect_from(clipBox); | 
 |         matrix.mapRect(bounds); | 
 |     } | 
 |     return true; | 
 | } | 
 |  | 
 | #else  // DWRITE_CORE || (defined(NTDDI_WIN11_ZN) && NTDDI_VERSION >= NTDDI_WIN11_ZN) | 
 |  | 
 | bool SkScalerContext_DW::generateColorV1Metrics(const SkGlyph&, SkRect*) { return false; } | 
 | bool SkScalerContext_DW::generateColorV1Image(const SkGlyph&, void*) { return false; } | 
 | bool SkScalerContext_DW::drawColorV1Image(const SkGlyph&, SkCanvas&) { return false; } | 
 |  | 
 | #endif  // DWRITE_CORE || (defined(NTDDI_WIN11_ZN) && NTDDI_VERSION >= NTDDI_WIN11_ZN) | 
 |  | 
 | bool SkScalerContext_DW::setAdvance(const SkGlyph& glyph, SkVector* advance) { | 
 |     *advance = {0, 0}; | 
 |     uint16_t glyphId = glyph.getGlyphID(); | 
 |     DWriteFontTypeface* typeface = this->getDWriteTypeface(); | 
 |  | 
 |     // DirectWrite treats all out of bounds glyph ids as having the same data as glyph 0. | 
 |     // For consistency with all other backends, treat out of range glyph ids as an error. | 
 |     if (fGlyphCount <= glyphId) { | 
 |         return false; | 
 |     } | 
 |  | 
 |     DWRITE_GLYPH_METRICS gm; | 
 |  | 
 |     if (DWRITE_MEASURING_MODE_GDI_CLASSIC == fMeasuringMode || | 
 |         DWRITE_MEASURING_MODE_GDI_NATURAL == fMeasuringMode) | 
 |     { | 
 |         Exclusive l(maybe_dw_mutex(*typeface)); | 
 |         HRBM(typeface->fDWriteFontFace->GetGdiCompatibleGlyphMetrics( | 
 |                  fTextSizeMeasure, | 
 |                  1.0f, // pixelsPerDip | 
 |                  // This parameter does not act like the lpmat2 parameter to GetGlyphOutlineW. | 
 |                  // If it did then GsA here and G_inv below to mapVectors. | 
 |                  nullptr, | 
 |                  DWRITE_MEASURING_MODE_GDI_NATURAL == fMeasuringMode, | 
 |                  &glyphId, 1, | 
 |                  &gm), | 
 |              "Could not get gdi compatible glyph metrics."); | 
 |     } else { | 
 |         Exclusive l(maybe_dw_mutex(*typeface)); | 
 |         HRBM(typeface->fDWriteFontFace->GetDesignGlyphMetrics(&glyphId, 1, &gm), | 
 |              "Could not get design metrics."); | 
 |     } | 
 |  | 
 |     DWRITE_FONT_METRICS dwfm; | 
 |     { | 
 |         Shared l(maybe_dw_mutex(*typeface)); | 
 |         typeface->fDWriteFontFace->GetMetrics(&dwfm); | 
 |     } | 
 |     SkScalar advanceX = fTextSizeMeasure * gm.advanceWidth / dwfm.designUnitsPerEm; | 
 |  | 
 |     *advance = { advanceX, 0 }; | 
 |     if (DWRITE_MEASURING_MODE_GDI_CLASSIC == fMeasuringMode || | 
 |         DWRITE_MEASURING_MODE_GDI_NATURAL == fMeasuringMode) | 
 |     { | 
 |         // DirectWrite produced 'compatible' metrics, but while close, | 
 |         // the end result is not always an integer as it would be with GDI. | 
 |         advance->fX = SkScalarRoundToScalar(advance->fX); | 
 |     } | 
 |     fSkXform.mapVectors(advance, 1); | 
 |     return true; | 
 | } | 
 |  | 
 | bool SkScalerContext_DW::generateDWMetrics(const SkGlyph& glyph, | 
 |                                            DWRITE_RENDERING_MODE renderingMode, | 
 |                                            DWRITE_TEXTURE_TYPE textureType, | 
 |                                            SkRect* bounds) | 
 | { | 
 |     DWriteFontTypeface* typeface = this->getDWriteTypeface(); | 
 |  | 
 |     //Measure raster size. | 
 |     fXform.dx = SkFixedToFloat(glyph.getSubXFixed()); | 
 |     fXform.dy = SkFixedToFloat(glyph.getSubYFixed()); | 
 |  | 
 |     FLOAT advance = 0; | 
 |  | 
 |     UINT16 glyphId = glyph.getGlyphID(); | 
 |  | 
 |     DWRITE_GLYPH_OFFSET offset; | 
 |     offset.advanceOffset = 0.0f; | 
 |     offset.ascenderOffset = 0.0f; | 
 |  | 
 |     DWRITE_GLYPH_RUN run; | 
 |     run.glyphCount = 1; | 
 |     run.glyphAdvances = &advance; | 
 |     run.fontFace = typeface->fDWriteFontFace.get(); | 
 |     run.fontEmSize = SkScalarToFloat(fTextSizeRender); | 
 |     run.bidiLevel = 0; | 
 |     run.glyphIndices = &glyphId; | 
 |     run.isSideways = FALSE; | 
 |     run.glyphOffsets = &offset; | 
 |  | 
 |     SkTScopedComPtr<IDWriteGlyphRunAnalysis> glyphRunAnalysis; | 
 |     { | 
 |         Exclusive l(maybe_dw_mutex(*typeface)); | 
 |         // IDWriteFactory2::CreateGlyphRunAnalysis is very bad at aliased glyphs. | 
 |         if (typeface->fFactory2 && | 
 |                 (fGridFitMode == DWRITE_GRID_FIT_MODE_DISABLED || | 
 |                  fAntiAliasMode == DWRITE_TEXT_ANTIALIAS_MODE_GRAYSCALE)) | 
 |         { | 
 |             HRBM(typeface->fFactory2->CreateGlyphRunAnalysis( | 
 |                     &run, | 
 |                     &fXform, | 
 |                     renderingMode, | 
 |                     fMeasuringMode, | 
 |                     fGridFitMode, | 
 |                     fAntiAliasMode, | 
 |                     0.0f, // baselineOriginX, | 
 |                     0.0f, // baselineOriginY, | 
 |                     &glyphRunAnalysis), | 
 |                  "Could not create DW2 glyph run analysis."); | 
 |         } else { | 
 |             HRBM(typeface->fFactory->CreateGlyphRunAnalysis(&run, | 
 |                     1.0f, // pixelsPerDip, | 
 |                     &fXform, | 
 |                     renderingMode, | 
 |                     fMeasuringMode, | 
 |                     0.0f, // baselineOriginX, | 
 |                     0.0f, // baselineOriginY, | 
 |                     &glyphRunAnalysis), | 
 |                  "Could not create glyph run analysis."); | 
 |         } | 
 |     } | 
 |     RECT bbox; | 
 |     { | 
 |         Shared l(maybe_dw_mutex(*typeface)); | 
 |         HRBM(glyphRunAnalysis->GetAlphaTextureBounds(textureType, &bbox), | 
 |              "Could not get texture bounds."); | 
 |     } | 
 |  | 
 |     // GetAlphaTextureBounds succeeds but sometimes returns empty bounds like | 
 |     // { 0x80000000, 0x80000000, 0x80000000, 0x80000000 } | 
 |     // for small but not quite zero and large (but not really large) glyphs, | 
 |     // Only set as non-empty if the returned bounds are non-empty. | 
 |     if (bbox.left >= bbox.right || bbox.top >= bbox.bottom) { | 
 |         return false; | 
 |     } | 
 |  | 
 |     *bounds = SkRect::MakeLTRB(bbox.left, bbox.top, bbox.right, bbox.bottom); | 
 |     return true; | 
 | } | 
 |  | 
 | bool SkScalerContext_DW::getColorGlyphRun(const SkGlyph& glyph, | 
 |                                           IDWriteColorGlyphRunEnumerator** colorGlyph) | 
 | { | 
 |     FLOAT advance = 0; | 
 |     UINT16 glyphId = glyph.getGlyphID(); | 
 |  | 
 |     DWRITE_GLYPH_OFFSET offset; | 
 |     offset.advanceOffset = 0.0f; | 
 |     offset.ascenderOffset = 0.0f; | 
 |  | 
 |     DWRITE_GLYPH_RUN run; | 
 |     run.glyphCount = 1; | 
 |     run.glyphAdvances = &advance; | 
 |     run.fontFace = this->getDWriteTypeface()->fDWriteFontFace.get(); | 
 |     run.fontEmSize = SkScalarToFloat(fTextSizeRender); | 
 |     run.bidiLevel = 0; | 
 |     run.glyphIndices = &glyphId; | 
 |     run.isSideways = FALSE; | 
 |     run.glyphOffsets = &offset; | 
 |  | 
 |     HRESULT hr = this->getDWriteTypeface()->fFactory2->TranslateColorGlyphRun( | 
 |         0, 0, &run, nullptr, fMeasuringMode, &fXform, 0, colorGlyph); | 
 |     if (hr == DWRITE_E_NOCOLOR) { | 
 |         return false; | 
 |     } | 
 |     HRBM(hr, "Failed to translate color glyph run"); | 
 |     return true; | 
 | } | 
 |  | 
 | bool SkScalerContext_DW::generateColorMetrics(const SkGlyph& glyph, SkRect* bounds) { | 
 |     SkTScopedComPtr<IDWriteColorGlyphRunEnumerator> colorLayers; | 
 |     if (!getColorGlyphRun(glyph, &colorLayers)) { | 
 |         return false; | 
 |     } | 
 |     SkASSERT(colorLayers.get()); | 
 |  | 
 |     *bounds = SkRect::MakeEmpty(); | 
 |     BOOL hasNextRun = FALSE; | 
 |     while (SUCCEEDED(colorLayers->MoveNext(&hasNextRun)) && hasNextRun) { | 
 |         const DWRITE_COLOR_GLYPH_RUN* colorGlyph; | 
 |         HRBM(colorLayers->GetCurrentRun(&colorGlyph), "Could not get current color glyph run"); | 
 |  | 
 |         SkPath path; | 
 |         SkTScopedComPtr<IDWriteGeometrySink> geometryToPath; | 
 |         HRBM(SkDWriteGeometrySink::Create(&path, &geometryToPath), | 
 |              "Could not create geometry to path converter."); | 
 |         { | 
 |             Exclusive l(maybe_dw_mutex(*this->getDWriteTypeface())); | 
 |             HRBM(colorGlyph->glyphRun.fontFace->GetGlyphRunOutline( | 
 |                     colorGlyph->glyphRun.fontEmSize, | 
 |                     colorGlyph->glyphRun.glyphIndices, | 
 |                     colorGlyph->glyphRun.glyphAdvances, | 
 |                     colorGlyph->glyphRun.glyphOffsets, | 
 |                     colorGlyph->glyphRun.glyphCount, | 
 |                     colorGlyph->glyphRun.isSideways, | 
 |                     colorGlyph->glyphRun.bidiLevel % 2, //rtl | 
 |                     geometryToPath.get()), | 
 |                  "Could not create glyph outline."); | 
 |         } | 
 |         bounds->join(path.getBounds()); | 
 |     } | 
 |     SkMatrix matrix = fSkXform; | 
 |     if (this->isSubpixel()) { | 
 |         matrix.postTranslate(SkFixedToScalar(glyph.getSubXFixed()), | 
 |                              SkFixedToScalar(glyph.getSubYFixed())); | 
 |     } | 
 |     matrix.mapRect(bounds); | 
 |     return true; | 
 | } | 
 |  | 
 | bool SkScalerContext_DW::generateSVGMetrics(const SkGlyph& glyph, SkRect* bounds) { | 
 |     SkPictureRecorder recorder; | 
 |     SkRect infiniteRect = SkRect::MakeLTRB(-SK_ScalarInfinity, -SK_ScalarInfinity, | 
 |                                             SK_ScalarInfinity,  SK_ScalarInfinity); | 
 |     sk_sp<SkBBoxHierarchy> bboxh = SkRTreeFactory()(); | 
 |     SkCanvas* recordingCanvas = recorder.beginRecording(infiniteRect, bboxh); | 
 |     if (!this->drawSVGImage(glyph, *recordingCanvas)) { | 
 |         return false; | 
 |     } | 
 |     sk_sp<SkPicture> pic = recorder.finishRecordingAsPicture(); | 
 |     *bounds = pic->cullRect(); | 
 |     SkASSERT(bounds->isFinite()); | 
 |     bounds->roundOut(bounds); | 
 |     return true; | 
 | } | 
 |  | 
 | namespace { | 
 | struct Context { | 
 |     SkTScopedComPtr<IDWriteFontFace4> fontFace4; | 
 |     void* glyphDataContext; | 
 |     Context(IDWriteFontFace4* face4, void* context) | 
 |         : fontFace4(SkRefComPtr(face4)) | 
 |         , glyphDataContext(context) | 
 |     {} | 
 | }; | 
 |  | 
 | static void ReleaseProc(const void* ptr, void* context) { | 
 |     Context* ctx = (Context*)context; | 
 |     ctx->fontFace4->ReleaseGlyphImageData(ctx->glyphDataContext); | 
 |     delete ctx; | 
 | } | 
 | } | 
 |  | 
 | bool SkScalerContext_DW::generatePngMetrics(const SkGlyph& glyph, SkRect* bounds) { | 
 |     IDWriteFontFace4* fontFace4 = this->getDWriteTypeface()->fDWriteFontFace4.get(); | 
 |     if (!fontFace4) { | 
 |         return false; | 
 |     } | 
 |  | 
 |     DWRITE_GLYPH_IMAGE_FORMATS imageFormats; | 
 |     HRBM(fontFace4->GetGlyphImageFormats(glyph.getGlyphID(), 0, UINT32_MAX, &imageFormats), | 
 |          "Cannot get glyph image formats."); | 
 |     if (!(imageFormats & DWRITE_GLYPH_IMAGE_FORMATS_PNG)) { | 
 |         return false; | 
 |     } | 
 |  | 
 |     DWRITE_GLYPH_IMAGE_DATA glyphData; | 
 |     void* glyphDataContext; | 
 |     HRBM(fontFace4->GetGlyphImageData(glyph.getGlyphID(), | 
 |                                       fTextSizeRender, | 
 |                                       DWRITE_GLYPH_IMAGE_FORMATS_PNG, | 
 |                                       &glyphData, | 
 |                                       &glyphDataContext), | 
 |          "Glyph image data could not be acquired."); | 
 |  | 
 |     Context* context = new Context(fontFace4, glyphDataContext); | 
 |     sk_sp<SkData> data = SkData::MakeWithProc(glyphData.imageData, | 
 |                                               glyphData.imageDataSize, | 
 |                                               &ReleaseProc, | 
 |                                               context); | 
 |  | 
 |     std::unique_ptr<SkCodec> codec = SkPngDecoder::Decode(std::move(data), nullptr); | 
 |     if (!codec) { | 
 |         return false; | 
 |     } | 
 |  | 
 |     SkImageInfo info = codec->getInfo(); | 
 |     *bounds = SkRect::MakeLTRB(SkIntToScalar(info.bounds().fLeft), | 
 |                                SkIntToScalar(info.bounds().fTop), | 
 |                                SkIntToScalar(info.bounds().fRight), | 
 |                                SkIntToScalar(info.bounds().fBottom)); | 
 |  | 
 |     SkMatrix matrix = fSkXform; | 
 |     SkScalar scale = fTextSizeRender / glyphData.pixelsPerEm; | 
 |     matrix.preScale(scale, scale); | 
 |     matrix.preTranslate(-glyphData.horizontalLeftOrigin.x, -glyphData.horizontalLeftOrigin.y); | 
 |     if (this->isSubpixel()) { | 
 |         matrix.postTranslate(SkFixedToScalar(glyph.getSubXFixed()), | 
 |                              SkFixedToScalar(glyph.getSubYFixed())); | 
 |     } | 
 |     matrix.mapRect(bounds); | 
 |     bounds->roundOut(bounds); | 
 |     return true; | 
 | } | 
 |  | 
 | SkScalerContext::GlyphMetrics SkScalerContext_DW::generateMetrics(const SkGlyph& glyph, | 
 |                                                                   SkArenaAlloc* alloc) { | 
 |     GlyphMetrics mx(glyph.maskFormat()); | 
 |  | 
 |     mx.extraBits = ScalerContextBits::NONE; | 
 |  | 
 |     if (!this->setAdvance(glyph, &mx.advance)) { | 
 |         return mx; | 
 |     } | 
 |  | 
 |     DWriteFontTypeface* typeface = this->getDWriteTypeface(); | 
 |     if (typeface->fIsColorFont) { | 
 |         if (generateColorV1Metrics(glyph, &mx.bounds)) { | 
 |             mx.maskFormat = SkMask::kARGB32_Format; | 
 |             mx.extraBits |= ScalerContextBits::COLRv1; | 
 |             mx.neverRequestPath = true; | 
 |             return mx; | 
 |         } | 
 |  | 
 |         if (generateColorMetrics(glyph, &mx.bounds)) { | 
 |             mx.maskFormat = SkMask::kARGB32_Format; | 
 |             mx.extraBits |= ScalerContextBits::COLR; | 
 |             mx.neverRequestPath = true; | 
 |             return mx; | 
 |         } | 
 |  | 
 |         if (generateSVGMetrics(glyph, &mx.bounds)) { | 
 |             mx.maskFormat = SkMask::kARGB32_Format; | 
 |             mx.extraBits |= ScalerContextBits::SVG; | 
 |             mx.neverRequestPath = true; | 
 |             return mx; | 
 |         } | 
 |  | 
 |         if (generatePngMetrics(glyph, &mx.bounds)) { | 
 |             mx.maskFormat = SkMask::kARGB32_Format; | 
 |             mx.extraBits |= ScalerContextBits::PNG; | 
 |             mx.neverRequestPath = true; | 
 |             return mx; | 
 |         } | 
 |     } | 
 |  | 
 |     if (this->generateDWMetrics(glyph, fRenderingMode, fTextureType, &mx.bounds)) { | 
 |         mx.extraBits = ScalerContextBits::DW; | 
 |         return mx; | 
 |     } | 
 |  | 
 |     // GetAlphaTextureBounds succeeds but returns an empty RECT if there are no | 
 |     // glyphs of the specified texture type or it is too big for smoothing. | 
 |     // When this happens, try with the alternate texture type. | 
 |     if (DWRITE_TEXTURE_ALIASED_1x1 != fTextureType || | 
 |         DWRITE_TEXT_ANTIALIAS_MODE_GRAYSCALE == fAntiAliasMode) | 
 |     { | 
 |         if (this->generateDWMetrics(glyph, | 
 |                                     DWRITE_RENDERING_MODE_ALIASED, | 
 |                                     DWRITE_TEXTURE_ALIASED_1x1, | 
 |                                     &mx.bounds)) | 
 |         { | 
 |             mx.maskFormat = SkMask::kBW_Format; | 
 |             mx.extraBits = ScalerContextBits::DW_1; | 
 |             return mx; | 
 |         } | 
 |     } | 
 |     // TODO: Try DWRITE_TEXTURE_CLEARTYPE_3x1 if DWRITE_TEXTURE_ALIASED_1x1 fails | 
 |  | 
 |     // GetAlphaTextureBounds can fail for various reasons. | 
 |     // As a fallback, attempt to generate the metrics and image from the path. | 
 |     mx.computeFromPath = true; | 
 |     mx.extraBits = ScalerContextBits::PATH; | 
 |     return mx; | 
 | } | 
 |  | 
 | void SkScalerContext_DW::generateFontMetrics(SkFontMetrics* metrics) { | 
 |     if (nullptr == metrics) { | 
 |         return; | 
 |     } | 
 |  | 
 |     sk_bzero(metrics, sizeof(*metrics)); | 
 |  | 
 |     DWRITE_FONT_METRICS dwfm; | 
 |     if (DWRITE_MEASURING_MODE_GDI_CLASSIC == fMeasuringMode || | 
 |         DWRITE_MEASURING_MODE_GDI_NATURAL == fMeasuringMode) | 
 |     { | 
 |         this->getDWriteTypeface()->fDWriteFontFace->GetGdiCompatibleMetrics( | 
 |              fTextSizeRender, | 
 |              1.0f, // pixelsPerDip | 
 |              &fXform, | 
 |              &dwfm); | 
 |     } else { | 
 |         this->getDWriteTypeface()->fDWriteFontFace->GetMetrics(&dwfm); | 
 |     } | 
 |  | 
 |     SkScalar upem = SkIntToScalar(dwfm.designUnitsPerEm); | 
 |  | 
 |     metrics->fAscent = -fTextSizeRender * SkIntToScalar(dwfm.ascent) / upem; | 
 |     metrics->fDescent = fTextSizeRender * SkIntToScalar(dwfm.descent) / upem; | 
 |     metrics->fLeading = fTextSizeRender * SkIntToScalar(dwfm.lineGap) / upem; | 
 |     metrics->fXHeight = fTextSizeRender * SkIntToScalar(dwfm.xHeight) / upem; | 
 |     metrics->fCapHeight = fTextSizeRender * SkIntToScalar(dwfm.capHeight) / upem; | 
 |     metrics->fUnderlineThickness = fTextSizeRender * SkIntToScalar(dwfm.underlineThickness) / upem; | 
 |     metrics->fUnderlinePosition = -(fTextSizeRender * SkIntToScalar(dwfm.underlinePosition) / upem); | 
 |     metrics->fStrikeoutThickness = fTextSizeRender * SkIntToScalar(dwfm.strikethroughThickness) / upem; | 
 |     metrics->fStrikeoutPosition = -(fTextSizeRender * SkIntToScalar(dwfm.strikethroughPosition) / upem); | 
 |  | 
 |     metrics->fFlags |= SkFontMetrics::kUnderlineThicknessIsValid_Flag; | 
 |     metrics->fFlags |= SkFontMetrics::kUnderlinePositionIsValid_Flag; | 
 |     metrics->fFlags |= SkFontMetrics::kStrikeoutThicknessIsValid_Flag; | 
 |     metrics->fFlags |= SkFontMetrics::kStrikeoutPositionIsValid_Flag; | 
 |  | 
 |     SkTScopedComPtr<IDWriteFontFace5> fontFace5; | 
 |     if (SUCCEEDED(this->getDWriteTypeface()->fDWriteFontFace->QueryInterface(&fontFace5))) { | 
 |         if (fontFace5->HasVariations()) { | 
 |             // The bounds are only valid for the default variation. | 
 |             metrics->fFlags |= SkFontMetrics::kBoundsInvalid_Flag; | 
 |         } | 
 |     } | 
 |  | 
 |     if (this->getDWriteTypeface()->fDWriteFontFace1.get()) { | 
 |         DWRITE_FONT_METRICS1 dwfm1; | 
 |         this->getDWriteTypeface()->fDWriteFontFace1->GetMetrics(&dwfm1); | 
 |         metrics->fTop = -fTextSizeRender * SkIntToScalar(dwfm1.glyphBoxTop) / upem; | 
 |         metrics->fBottom = -fTextSizeRender * SkIntToScalar(dwfm1.glyphBoxBottom) / upem; | 
 |         metrics->fXMin = fTextSizeRender * SkIntToScalar(dwfm1.glyphBoxLeft) / upem; | 
 |         metrics->fXMax = fTextSizeRender * SkIntToScalar(dwfm1.glyphBoxRight) / upem; | 
 |  | 
 |         metrics->fMaxCharWidth = metrics->fXMax - metrics->fXMin; | 
 |         return; | 
 |     } | 
 |  | 
 |     AutoTDWriteTable<SkOTTableHead> head(this->getDWriteTypeface()->fDWriteFontFace.get()); | 
 |     if (head.fExists && | 
 |         head.fSize >= sizeof(SkOTTableHead) && | 
 |         head->version == SkOTTableHead::version1) | 
 |     { | 
 |         metrics->fTop = -fTextSizeRender * (int16_t)SkEndian_SwapBE16(head->yMax) / upem; | 
 |         metrics->fBottom = -fTextSizeRender * (int16_t)SkEndian_SwapBE16(head->yMin) / upem; | 
 |         metrics->fXMin = fTextSizeRender * (int16_t)SkEndian_SwapBE16(head->xMin) / upem; | 
 |         metrics->fXMax = fTextSizeRender * (int16_t)SkEndian_SwapBE16(head->xMax) / upem; | 
 |  | 
 |         metrics->fMaxCharWidth = metrics->fXMax - metrics->fXMin; | 
 |         return; | 
 |     } | 
 |  | 
 |     // The real bounds weren't actually available. | 
 |     metrics->fFlags |= SkFontMetrics::kBoundsInvalid_Flag; | 
 |     metrics->fTop = metrics->fAscent; | 
 |     metrics->fBottom = metrics->fDescent; | 
 | } | 
 |  | 
 | /////////////////////////////////////////////////////////////////////////////// | 
 |  | 
 | #include "include/private/SkColorData.h" | 
 |  | 
 | void SkScalerContext_DW::BilevelToBW(const uint8_t* SK_RESTRICT src, | 
 |                                      const SkGlyph& glyph, void* imageBuffer) { | 
 |     const int width = glyph.width(); | 
 |     const size_t dstRB = (width + 7) >> 3; | 
 |     uint8_t* SK_RESTRICT dst = static_cast<uint8_t*>(imageBuffer); | 
 |  | 
 |     int byteCount = width >> 3; | 
 |     int bitCount = width & 7; | 
 |  | 
 |     for (int y = 0; y < glyph.height(); ++y) { | 
 |         if (byteCount > 0) { | 
 |             for (int i = 0; i < byteCount; ++i) { | 
 |                 unsigned byte = 0; | 
 |                 byte |= src[0] & (1 << 7); | 
 |                 byte |= src[1] & (1 << 6); | 
 |                 byte |= src[2] & (1 << 5); | 
 |                 byte |= src[3] & (1 << 4); | 
 |                 byte |= src[4] & (1 << 3); | 
 |                 byte |= src[5] & (1 << 2); | 
 |                 byte |= src[6] & (1 << 1); | 
 |                 byte |= src[7] & (1 << 0); | 
 |                 dst[i] = byte; | 
 |                 src += 8; | 
 |             } | 
 |         } | 
 |         if (bitCount > 0) { | 
 |             unsigned byte = 0; | 
 |             unsigned mask = 0x80; | 
 |             for (int i = 0; i < bitCount; i++) { | 
 |                 byte |= (src[i]) & mask; | 
 |                 mask >>= 1; | 
 |             } | 
 |             dst[byteCount] = byte; | 
 |         } | 
 |         src += bitCount; | 
 |         dst += dstRB; | 
 |     } | 
 |  | 
 |     if constexpr (kSkShowTextBlitCoverage) { | 
 |         dst = static_cast<uint8_t*>(imageBuffer); | 
 |         for (unsigned y = 0; y < (unsigned)glyph.height(); y += 2) { | 
 |             for (unsigned x = (y & 0x2); x < (unsigned)glyph.width(); x+=4) { | 
 |                 uint8_t& b = dst[(dstRB * y) + (x >> 3)]; | 
 |                 b = b ^ (1 << (0x7 - (x & 0x7))); | 
 |             } | 
 |         } | 
 |     } | 
 | } | 
 |  | 
 | template<bool APPLY_PREBLEND> | 
 | void SkScalerContext_DW::GrayscaleToA8(const uint8_t* SK_RESTRICT src, | 
 |                                        const SkGlyph& glyph, void* imageBuffer, | 
 |                                        const uint8_t* table8) { | 
 |     const size_t dstRB = glyph.rowBytes(); | 
 |     const int width = glyph.width(); | 
 |     uint8_t* SK_RESTRICT dst = static_cast<uint8_t*>(imageBuffer); | 
 |  | 
 |     for (int y = 0; y < glyph.height(); y++) { | 
 |         for (int i = 0; i < width; i++) { | 
 |             U8CPU a = *(src++); | 
 |             dst[i] = sk_apply_lut_if<APPLY_PREBLEND>(a, table8); | 
 |             if constexpr (kSkShowTextBlitCoverage) { | 
 |                 dst[i] = std::max<U8CPU>(0x30, dst[i]); | 
 |             } | 
 |         } | 
 |         dst = SkTAddOffset<uint8_t>(dst, dstRB); | 
 |     } | 
 | } | 
 |  | 
 | template<bool APPLY_PREBLEND> | 
 | void SkScalerContext_DW::RGBToA8(const uint8_t* SK_RESTRICT src, | 
 |                                  const SkGlyph& glyph, void* imageBuffer, | 
 |                                  const uint8_t* table8) { | 
 |     const size_t dstRB = glyph.rowBytes(); | 
 |     const int width = glyph.width(); | 
 |     uint8_t* SK_RESTRICT dst = static_cast<uint8_t*>(imageBuffer); | 
 |  | 
 |     for (int y = 0; y < glyph.height(); y++) { | 
 |         for (int i = 0; i < width; i++) { | 
 |             U8CPU r = *(src++); | 
 |             U8CPU g = *(src++); | 
 |             U8CPU b = *(src++); | 
 |             dst[i] = sk_apply_lut_if<APPLY_PREBLEND>((r + g + b) / 3, table8); | 
 |             if constexpr (kSkShowTextBlitCoverage) { | 
 |                 dst[i] = std::max<U8CPU>(0x30, dst[i]); | 
 |             } | 
 |         } | 
 |         dst = SkTAddOffset<uint8_t>(dst, dstRB); | 
 |     } | 
 | } | 
 |  | 
 | template<bool APPLY_PREBLEND, bool RGB> | 
 | void SkScalerContext_DW::RGBToLcd16(const uint8_t* SK_RESTRICT src, const SkGlyph& glyph, | 
 |                                     void* imageBuffer, | 
 |                                     const uint8_t* tableR, const uint8_t* tableG, | 
 |                                     const uint8_t* tableB) { | 
 |     const size_t dstRB = glyph.rowBytes(); | 
 |     const int width = glyph.width(); | 
 |     uint16_t* SK_RESTRICT dst = static_cast<uint16_t*>(imageBuffer); | 
 |  | 
 |     for (int y = 0; y < glyph.height(); y++) { | 
 |         for (int i = 0; i < width; i++) { | 
 |             U8CPU r, g, b; | 
 |             if (RGB) { | 
 |                 r = sk_apply_lut_if<APPLY_PREBLEND>(*(src++), tableR); | 
 |                 g = sk_apply_lut_if<APPLY_PREBLEND>(*(src++), tableG); | 
 |                 b = sk_apply_lut_if<APPLY_PREBLEND>(*(src++), tableB); | 
 |             } else { | 
 |                 b = sk_apply_lut_if<APPLY_PREBLEND>(*(src++), tableB); | 
 |                 g = sk_apply_lut_if<APPLY_PREBLEND>(*(src++), tableG); | 
 |                 r = sk_apply_lut_if<APPLY_PREBLEND>(*(src++), tableR); | 
 |             } | 
 |             if constexpr (kSkShowTextBlitCoverage) { | 
 |                 r = std::max<U8CPU>(0x30, r); | 
 |                 g = std::max<U8CPU>(0x30, g); | 
 |                 b = std::max<U8CPU>(0x30, b); | 
 |             } | 
 |             dst[i] = SkPack888ToRGB16(r, g, b); | 
 |         } | 
 |         dst = SkTAddOffset<uint16_t>(dst, dstRB); | 
 |     } | 
 | } | 
 |  | 
 | const void* SkScalerContext_DW::getDWMaskBits(const SkGlyph& glyph, | 
 |                                               DWRITE_RENDERING_MODE renderingMode, | 
 |                                               DWRITE_TEXTURE_TYPE textureType) | 
 | { | 
 |     DWriteFontTypeface* typeface = this->getDWriteTypeface(); | 
 |  | 
 |     int sizeNeeded = glyph.width() * glyph.height(); | 
 |     if (DWRITE_TEXTURE_CLEARTYPE_3x1 == textureType) { | 
 |         sizeNeeded *= 3; | 
 |     } | 
 |     if (sizeNeeded > fBits.size()) { | 
 |         fBits.resize(sizeNeeded); | 
 |     } | 
 |  | 
 |     // erase | 
 |     memset(fBits.begin(), 0, sizeNeeded); | 
 |  | 
 |     fXform.dx = SkFixedToFloat(glyph.getSubXFixed()); | 
 |     fXform.dy = SkFixedToFloat(glyph.getSubYFixed()); | 
 |  | 
 |     FLOAT advance = 0.0f; | 
 |  | 
 |     UINT16 index = glyph.getGlyphID(); | 
 |  | 
 |     DWRITE_GLYPH_OFFSET offset; | 
 |     offset.advanceOffset = 0.0f; | 
 |     offset.ascenderOffset = 0.0f; | 
 |  | 
 |     DWRITE_GLYPH_RUN run; | 
 |     run.glyphCount = 1; | 
 |     run.glyphAdvances = &advance; | 
 |     run.fontFace = typeface->fDWriteFontFace.get(); | 
 |     run.fontEmSize = SkScalarToFloat(fTextSizeRender); | 
 |     run.bidiLevel = 0; | 
 |     run.glyphIndices = &index; | 
 |     run.isSideways = FALSE; | 
 |     run.glyphOffsets = &offset; | 
 |     { | 
 |         SkTScopedComPtr<IDWriteGlyphRunAnalysis> glyphRunAnalysis; | 
 |         { | 
 |             Exclusive l(maybe_dw_mutex(*typeface)); | 
 |             // IDWriteFactory2::CreateGlyphRunAnalysis is very bad at aliased glyphs. | 
 |             if (typeface->fFactory2 && | 
 |                     (fGridFitMode == DWRITE_GRID_FIT_MODE_DISABLED || | 
 |                      fAntiAliasMode == DWRITE_TEXT_ANTIALIAS_MODE_GRAYSCALE)) | 
 |             { | 
 |                 HRNM(typeface->fFactory2->CreateGlyphRunAnalysis(&run, | 
 |                          &fXform, | 
 |                          renderingMode, | 
 |                          fMeasuringMode, | 
 |                          fGridFitMode, | 
 |                          fAntiAliasMode, | 
 |                          0.0f, // baselineOriginX, | 
 |                          0.0f, // baselineOriginY, | 
 |                          &glyphRunAnalysis), | 
 |                      "Could not create DW2 glyph run analysis."); | 
 |             } else { | 
 |                 HRNM(typeface->fFactory->CreateGlyphRunAnalysis(&run, | 
 |                          1.0f, // pixelsPerDip, | 
 |                          &fXform, | 
 |                          renderingMode, | 
 |                          fMeasuringMode, | 
 |                          0.0f, // baselineOriginX, | 
 |                          0.0f, // baselineOriginY, | 
 |                          &glyphRunAnalysis), | 
 |                      "Could not create glyph run analysis."); | 
 |             } | 
 |         } | 
 |         //NOTE: this assumes that the glyph has already been measured | 
 |         //with an exact same glyph run analysis. | 
 |         RECT bbox; | 
 |         bbox.left = glyph.left(); | 
 |         bbox.top = glyph.top(); | 
 |         bbox.right = glyph.left() + glyph.width(); | 
 |         bbox.bottom = glyph.top() + glyph.height(); | 
 |         { | 
 |             Shared l(maybe_dw_mutex(*typeface)); | 
 |             HRNM(glyphRunAnalysis->CreateAlphaTexture(textureType, | 
 |                     &bbox, | 
 |                     fBits.begin(), | 
 |                     sizeNeeded), | 
 |                  "Could not draw mask."); | 
 |         } | 
 |     } | 
 |     return fBits.begin(); | 
 | } | 
 |  | 
 | bool SkScalerContext_DW::generateDWImage(const SkGlyph& glyph, void* imageBuffer) { | 
 |     //Create the mask. | 
 |     ScalerContextBits::value_type format = glyph.extraBits(); | 
 |     DWRITE_RENDERING_MODE renderingMode = fRenderingMode; | 
 |     DWRITE_TEXTURE_TYPE textureType = fTextureType; | 
 |     if (format == ScalerContextBits::DW_1) { | 
 |         renderingMode = DWRITE_RENDERING_MODE_ALIASED; | 
 |         textureType = DWRITE_TEXTURE_ALIASED_1x1; | 
 |     } | 
 |     const void* bits = this->getDWMaskBits(glyph, renderingMode, textureType); | 
 |     if (!bits) { | 
 |         sk_bzero(imageBuffer, glyph.imageSize()); | 
 |         return false; | 
 |     } | 
 |  | 
 |     //Copy the mask into the glyph. | 
 |     const uint8_t* src = (const uint8_t*)bits; | 
 |     if (DWRITE_RENDERING_MODE_ALIASED == renderingMode) { | 
 |         SkASSERT(SkMask::kBW_Format == glyph.maskFormat()); | 
 |         SkASSERT(DWRITE_TEXTURE_ALIASED_1x1 == textureType); | 
 |         BilevelToBW(src, glyph, imageBuffer); | 
 |     } else if (!isLCD(fRec)) { | 
 |         if (textureType == DWRITE_TEXTURE_ALIASED_1x1) { | 
 |             if (fPreBlend.isApplicable()) { | 
 |                 GrayscaleToA8<true>(src, glyph, imageBuffer, fPreBlend.fG); | 
 |             } else { | 
 |                 GrayscaleToA8<false>(src, glyph, imageBuffer, fPreBlend.fG); | 
 |             } | 
 |         } else { | 
 |             if (fPreBlend.isApplicable()) { | 
 |                 RGBToA8<true>(src, glyph, imageBuffer, fPreBlend.fG); | 
 |             } else { | 
 |                 RGBToA8<false>(src, glyph, imageBuffer, fPreBlend.fG); | 
 |             } | 
 |         } | 
 |     } else { | 
 |         SkASSERT(SkMask::kLCD16_Format == glyph.maskFormat()); | 
 |         if (fPreBlend.isApplicable()) { | 
 |             if (fRec.fFlags & SkScalerContext::kLCD_BGROrder_Flag) { | 
 |                 RGBToLcd16<true, false>(src, glyph, imageBuffer, | 
 |                                         fPreBlend.fR, fPreBlend.fG, fPreBlend.fB); | 
 |             } else { | 
 |                 RGBToLcd16<true, true>(src, glyph, imageBuffer, | 
 |                                        fPreBlend.fR, fPreBlend.fG, fPreBlend.fB); | 
 |             } | 
 |         } else { | 
 |             if (fRec.fFlags & SkScalerContext::kLCD_BGROrder_Flag) { | 
 |                 RGBToLcd16<false, false>(src, glyph, imageBuffer, | 
 |                                          fPreBlend.fR, fPreBlend.fG, fPreBlend.fB); | 
 |             } else { | 
 |                 RGBToLcd16<false, true>(src, glyph, imageBuffer, | 
 |                                         fPreBlend.fR, fPreBlend.fG, fPreBlend.fB); | 
 |             } | 
 |         } | 
 |     } | 
 |     return true; | 
 | } | 
 |  | 
 | bool SkScalerContext_DW::drawColorImage(const SkGlyph& glyph, SkCanvas& canvas) { | 
 |     SkTScopedComPtr<IDWriteColorGlyphRunEnumerator> colorLayers; | 
 |     if (!getColorGlyphRun(glyph, &colorLayers)) { | 
 |         SkASSERTF(false, "Could not get color layers"); | 
 |         return false; | 
 |     } | 
 |  | 
 |     SkPaint paint; | 
 |     paint.setAntiAlias(fRenderingMode != DWRITE_RENDERING_MODE_ALIASED); | 
 |  | 
 |     if (this->isSubpixel()) { | 
 |         canvas.translate(SkFixedToScalar(glyph.getSubXFixed()), | 
 |                          SkFixedToScalar(glyph.getSubYFixed())); | 
 |     } | 
 |     canvas.concat(fSkXform); | 
 |  | 
 |     DWriteFontTypeface* typeface = this->getDWriteTypeface(); | 
 |     size_t paletteEntryCount = typeface->fPaletteEntryCount; | 
 |     SkColor* palette = typeface->fPalette.get(); | 
 |     BOOL hasNextRun = FALSE; | 
 |     while (SUCCEEDED(colorLayers->MoveNext(&hasNextRun)) && hasNextRun) { | 
 |         const DWRITE_COLOR_GLYPH_RUN* colorGlyph; | 
 |         HRBM(colorLayers->GetCurrentRun(&colorGlyph), "Could not get current color glyph run"); | 
 |  | 
 |         SkColor color; | 
 |         if (colorGlyph->paletteIndex == 0xffff) { | 
 |             color = fRec.fForegroundColor; | 
 |         } else if (colorGlyph->paletteIndex < paletteEntryCount) { | 
 |             color = palette[colorGlyph->paletteIndex]; | 
 |         } else { | 
 |             SK_TRACEHR(DWRITE_E_NOCOLOR, "Invalid palette index."); | 
 |             color = SK_ColorBLACK; | 
 |         } | 
 |         paint.setColor(color); | 
 |  | 
 |         SkPath path; | 
 |         SkTScopedComPtr<IDWriteGeometrySink> geometryToPath; | 
 |         HRBM(SkDWriteGeometrySink::Create(&path, &geometryToPath), | 
 |              "Could not create geometry to path converter."); | 
 |         { | 
 |             Exclusive l(maybe_dw_mutex(*this->getDWriteTypeface())); | 
 |             HRBM(colorGlyph->glyphRun.fontFace->GetGlyphRunOutline( | 
 |                      colorGlyph->glyphRun.fontEmSize, | 
 |                      colorGlyph->glyphRun.glyphIndices, | 
 |                      colorGlyph->glyphRun.glyphAdvances, | 
 |                      colorGlyph->glyphRun.glyphOffsets, | 
 |                      colorGlyph->glyphRun.glyphCount, | 
 |                      colorGlyph->glyphRun.isSideways, | 
 |                      colorGlyph->glyphRun.bidiLevel % 2, //rtl | 
 |                      geometryToPath.get()), | 
 |                  "Could not create glyph outline."); | 
 |         } | 
 |         canvas.drawPath(path, paint); | 
 |     } | 
 |     return true; | 
 | } | 
 |  | 
 | bool SkScalerContext_DW::generateColorImage(const SkGlyph& glyph, void* imageBuffer) { | 
 |     SkASSERT(glyph.maskFormat() == SkMask::Format::kARGB32_Format); | 
 |  | 
 |     SkBitmap dstBitmap; | 
 |     // TODO: mark this as sRGB when the blits will be sRGB. | 
 |     dstBitmap.setInfo(SkImageInfo::Make(glyph.width(), glyph.height(), | 
 |                                         kN32_SkColorType, kPremul_SkAlphaType), | 
 |                                         glyph.rowBytes()); | 
 |     dstBitmap.setPixels(imageBuffer); | 
 |  | 
 |     SkCanvas canvas(dstBitmap); | 
 |     if constexpr (kSkShowTextBlitCoverage) { | 
 |         canvas.clear(0x33FF0000); | 
 |     } else { | 
 |         canvas.clear(SK_ColorTRANSPARENT); | 
 |     } | 
 |     canvas.translate(-SkIntToScalar(glyph.left()), -SkIntToScalar(glyph.top())); | 
 |  | 
 |     return this->drawColorImage(glyph, canvas); | 
 | } | 
 |  | 
 | bool SkScalerContext_DW::drawSVGImage(const SkGlyph& glyph, SkCanvas& canvas) { | 
 |     DWriteFontTypeface* typeface = this->getDWriteTypeface(); | 
 |     IDWriteFontFace4* fontFace4 = typeface->fDWriteFontFace4.get(); | 
 |     if (!fontFace4) { | 
 |         return false; | 
 |     } | 
 |  | 
 |     DWRITE_GLYPH_IMAGE_FORMATS imageFormats; | 
 |     HRBM(fontFace4->GetGlyphImageFormats(glyph.getGlyphID(), 0, UINT32_MAX, &imageFormats), | 
 |          "Cannot get glyph image formats."); | 
 |     if (!(imageFormats & DWRITE_GLYPH_IMAGE_FORMATS_SVG)) { | 
 |         return false; | 
 |     } | 
 |  | 
 |     SkGraphics::OpenTypeSVGDecoderFactory svgFactory = SkGraphics::GetOpenTypeSVGDecoderFactory(); | 
 |     if (!svgFactory) { | 
 |         return false; | 
 |     } | 
 |  | 
 |     DWRITE_GLYPH_IMAGE_DATA glyphData; | 
 |     void* glyphDataContext; | 
 |     HRBM(fontFace4->GetGlyphImageData(glyph.getGlyphID(), | 
 |                                       fTextSizeRender, | 
 |                                       DWRITE_GLYPH_IMAGE_FORMATS_SVG, | 
 |                                       &glyphData, | 
 |                                       &glyphDataContext), | 
 |          "Glyph SVG data could not be acquired."); | 
 |     auto svgDecoder = svgFactory((const uint8_t*)glyphData.imageData, glyphData.imageDataSize); | 
 |     fontFace4->ReleaseGlyphImageData(glyphDataContext); | 
 |     if (!svgDecoder) { | 
 |         return false; | 
 |     } | 
 |  | 
 |     size_t paletteEntryCount = typeface->fPaletteEntryCount; | 
 |     SkColor* palette = typeface->fPalette.get(); | 
 |     int upem = typeface->getUnitsPerEm(); | 
 |  | 
 |     SkMatrix matrix = fSkXform; | 
 |     SkScalar scale = fTextSizeRender / upem; | 
 |     matrix.preScale(scale, scale); | 
 |     matrix.preTranslate(-glyphData.horizontalLeftOrigin.x, -glyphData.horizontalLeftOrigin.y); | 
 |     if (this->isSubpixel()) { | 
 |         matrix.postTranslate(SkFixedToScalar(glyph.getSubXFixed()), | 
 |                              SkFixedToScalar(glyph.getSubYFixed())); | 
 |     } | 
 |     canvas.concat(matrix); | 
 |  | 
 |     return svgDecoder->render(canvas, upem, glyph.getGlyphID(), | 
 |                               fRec.fForegroundColor, SkSpan(palette, paletteEntryCount)); | 
 | } | 
 |  | 
 | bool SkScalerContext_DW::generateSVGImage(const SkGlyph& glyph, void* imageBuffer) { | 
 |     SkASSERT(glyph.maskFormat() == SkMask::Format::kARGB32_Format); | 
 |  | 
 |     SkBitmap dstBitmap; | 
 |     // TODO: mark this as sRGB when the blits will be sRGB. | 
 |     dstBitmap.setInfo(SkImageInfo::Make(glyph.width(), glyph.height(), | 
 |                                         kN32_SkColorType, kPremul_SkAlphaType), | 
 |                       glyph.rowBytes()); | 
 |     dstBitmap.setPixels(imageBuffer); | 
 |  | 
 |     SkCanvas canvas(dstBitmap); | 
 |     if constexpr (kSkShowTextBlitCoverage) { | 
 |         canvas.clear(0x33FF0000); | 
 |     } else { | 
 |         canvas.clear(SK_ColorTRANSPARENT); | 
 |     } | 
 |     canvas.translate(-SkIntToScalar(glyph.left()), -SkIntToScalar(glyph.top())); | 
 |  | 
 |     return this->drawSVGImage(glyph, canvas); | 
 | } | 
 |  | 
 | bool SkScalerContext_DW::drawPngImage(const SkGlyph& glyph, SkCanvas& canvas) { | 
 |     IDWriteFontFace4* fontFace4 = this->getDWriteTypeface()->fDWriteFontFace4.get(); | 
 |     if (!fontFace4) { | 
 |         return false; | 
 |     } | 
 |  | 
 |     DWRITE_GLYPH_IMAGE_DATA glyphData; | 
 |     void* glyphDataContext; | 
 |     HRBM(fontFace4->GetGlyphImageData(glyph.getGlyphID(), | 
 |                                       fTextSizeRender, | 
 |                                       DWRITE_GLYPH_IMAGE_FORMATS_PNG, | 
 |                                       &glyphData, | 
 |                                       &glyphDataContext), | 
 |          "Glyph image data could not be acquired."); | 
 |     Context* context = new Context(fontFace4, glyphDataContext); | 
 |     sk_sp<SkData> data = SkData::MakeWithProc(glyphData.imageData, | 
 |                                               glyphData.imageDataSize, | 
 |                                               &ReleaseProc, | 
 |                                               context); | 
 |     sk_sp<SkImage> image = SkImages::DeferredFromEncodedData(std::move(data)); | 
 |     if (!image) { | 
 |         return false; | 
 |     } | 
 |  | 
 |     if (this->isSubpixel()) { | 
 |         canvas.translate(SkFixedToScalar(glyph.getSubXFixed()), | 
 |                          SkFixedToScalar(glyph.getSubYFixed())); | 
 |     } | 
 |     canvas.concat(fSkXform); | 
 |     SkScalar ratio = fTextSizeRender / glyphData.pixelsPerEm; | 
 |     canvas.scale(ratio, ratio); | 
 |     canvas.translate(-glyphData.horizontalLeftOrigin.x, -glyphData.horizontalLeftOrigin.y); | 
 |     canvas.drawImage(image, 0, 0); | 
 |     return true; | 
 | } | 
 |  | 
 | bool SkScalerContext_DW::generatePngImage(const SkGlyph& glyph, void* imageBuffer) { | 
 |     SkASSERT(glyph.maskFormat() == SkMask::Format::kARGB32_Format); | 
 |  | 
 |     SkBitmap dstBitmap; | 
 |     dstBitmap.setInfo(SkImageInfo::Make(glyph.width(), glyph.height(), | 
 |                                         kN32_SkColorType, kPremul_SkAlphaType), | 
 |                       glyph.rowBytes()); | 
 |     dstBitmap.setPixels(imageBuffer); | 
 |  | 
 |     SkCanvas canvas(dstBitmap); | 
 |     canvas.clear(SK_ColorTRANSPARENT); | 
 |     canvas.translate(-glyph.left(), -glyph.top()); | 
 |  | 
 |     return this->drawPngImage(glyph, canvas); | 
 | } | 
 |  | 
 | void SkScalerContext_DW::generateImage(const SkGlyph& glyph, void* imageBuffer) { | 
 |     ScalerContextBits::value_type format = glyph.extraBits(); | 
 |     if (format == ScalerContextBits::DW || | 
 |         format == ScalerContextBits::DW_1) | 
 |     { | 
 |         this->generateDWImage(glyph, imageBuffer); | 
 |     } else if (format == ScalerContextBits::COLRv1) { | 
 |         this->generateColorV1Image(glyph, imageBuffer); | 
 |     } else if (format == ScalerContextBits::COLR) { | 
 |         this->generateColorImage(glyph, imageBuffer); | 
 |     } else if (format == ScalerContextBits::SVG) { | 
 |         this->generateSVGImage(glyph, imageBuffer); | 
 |     } else if (format == ScalerContextBits::PNG) { | 
 |         this->generatePngImage(glyph, imageBuffer); | 
 |     } else if (format == ScalerContextBits::PATH) { | 
 |         const SkPath* devPath = glyph.path(); | 
 |         SkASSERT_RELEASE(devPath); | 
 |         SkMaskBuilder mask(static_cast<uint8_t*>(imageBuffer), | 
 |                            glyph.iRect(), glyph.rowBytes(), glyph.maskFormat()); | 
 |         SkASSERT(SkMask::kARGB32_Format != mask.fFormat); | 
 |         const bool doBGR = SkToBool(fRec.fFlags & SkScalerContext::kLCD_BGROrder_Flag); | 
 |         const bool doVert = SkToBool(fRec.fFlags & SkScalerContext::kLCD_Vertical_Flag); | 
 |         const bool a8LCD = SkToBool(fRec.fFlags & SkScalerContext::kGenA8FromLCD_Flag); | 
 |         const bool hairline = glyph.pathIsHairline(); | 
 |         GenerateImageFromPath(mask, *devPath, fPreBlend, doBGR, doVert, a8LCD, hairline); | 
 |     } else { | 
 |         SK_ABORT("Bad format"); | 
 |     } | 
 | } | 
 |  | 
 | bool SkScalerContext_DW::generatePath(const SkGlyph& glyph, SkPath* path) { | 
 |     SkASSERT(path); | 
 |     path->reset(); | 
 |  | 
 |     SkGlyphID glyphID = glyph.getGlyphID(); | 
 |  | 
 |     // DirectWrite treats all out of bounds glyph ids as having the same data as glyph 0. | 
 |     // For consistency with all other backends, treat out of range glyph ids as an error. | 
 |     if (fGlyphCount <= glyphID) { | 
 |         return false; | 
 |     } | 
 |  | 
 |     SkTScopedComPtr<IDWriteGeometrySink> geometryToPath; | 
 |     HRBM(SkDWriteGeometrySink::Create(path, &geometryToPath), | 
 |          "Could not create geometry to path converter."); | 
 |     UINT16 glyphId = SkTo<UINT16>(glyphID); | 
 |     { | 
 |         Exclusive l(maybe_dw_mutex(*this->getDWriteTypeface())); | 
 |         //TODO: convert to<->from DIUs? This would make a difference if hinting. | 
 |         //It may not be needed, it appears that DirectWrite only hints at em size. | 
 |         HRBM(this->getDWriteTypeface()->fDWriteFontFace->GetGlyphRunOutline( | 
 |              SkScalarToFloat(fTextSizeRender), | 
 |              &glyphId, | 
 |              nullptr, //advances | 
 |              nullptr, //offsets | 
 |              1, //num glyphs | 
 |              FALSE, //sideways | 
 |              FALSE, //rtl | 
 |              geometryToPath.get()), | 
 |              "Could not create glyph outline."); | 
 |     } | 
 |  | 
 |     path->transform(fSkXform); | 
 |     return true; | 
 | } | 
 |  | 
 | sk_sp<SkDrawable> SkScalerContext_DW::generateDrawable(const SkGlyph& glyph) { | 
 |     struct GlyphDrawable : public SkDrawable { | 
 |         SkScalerContext_DW* fSelf; | 
 |         SkGlyph fGlyph; | 
 |         GlyphDrawable(SkScalerContext_DW* self, const SkGlyph& glyph) : fSelf(self), fGlyph(glyph){} | 
 |         SkRect onGetBounds() override { return fGlyph.rect();  } | 
 |         size_t onApproximateBytesUsed() override { return sizeof(GlyphDrawable); } | 
 |         void maybeShowTextBlitCoverage(SkCanvas* canvas) { | 
 |             if constexpr (kSkShowTextBlitCoverage) { | 
 |                 SkPaint paint; | 
 |                 paint.setColor(0x3300FF00); | 
 |                 paint.setStyle(SkPaint::kFill_Style); | 
 |                 canvas->drawRect(this->onGetBounds(), paint); | 
 |             } | 
 |         } | 
 |     }; | 
 |     struct COLRv1GlyphDrawable : public GlyphDrawable { | 
 |         using GlyphDrawable::GlyphDrawable; | 
 |         void onDraw(SkCanvas* canvas) override { | 
 |             this->maybeShowTextBlitCoverage(canvas); | 
 |             fSelf->drawColorV1Image(fGlyph, *canvas); | 
 |         } | 
 |     }; | 
 |     struct COLRGlyphDrawable : public GlyphDrawable { | 
 |         using GlyphDrawable::GlyphDrawable; | 
 |         void onDraw(SkCanvas* canvas) override { | 
 |             this->maybeShowTextBlitCoverage(canvas); | 
 |             fSelf->drawColorImage(fGlyph, *canvas); | 
 |         } | 
 |     }; | 
 |     struct SVGGlyphDrawable : public GlyphDrawable { | 
 |         using GlyphDrawable::GlyphDrawable; | 
 |         void onDraw(SkCanvas* canvas) override { | 
 |             this->maybeShowTextBlitCoverage(canvas); | 
 |             fSelf->drawSVGImage(fGlyph, *canvas); | 
 |         } | 
 |     }; | 
 |     ScalerContextBits::value_type format = glyph.extraBits(); | 
 |     if (format == ScalerContextBits::COLRv1) { | 
 |         return sk_sp<SkDrawable>(new COLRv1GlyphDrawable(this, glyph)); | 
 |     } | 
 |     if (format == ScalerContextBits::COLR) { | 
 |         return sk_sp<SkDrawable>(new COLRGlyphDrawable(this, glyph)); | 
 |     } | 
 |     if (format == ScalerContextBits::SVG) { | 
 |         return sk_sp<SkDrawable>(new SVGGlyphDrawable(this, glyph)); | 
 |     } | 
 |     return nullptr; | 
 | } | 
 |  | 
 | #endif//defined(SK_BUILD_FOR_WIN) |