/* * Copyright 2010 Google Inc. * * Use of this source code is governed by a BSD-style license that can be * found in the LICENSE file. */ #include "GrFontScaler.h" #include "SkDescriptor.h" #include "SkDistanceFieldGen.h" #include "SkGlyphCache.h" /////////////////////////////////////////////////////////////////////////////// GrFontScaler::GrFontScaler(SkGlyphCache* strike) { fStrike = strike; fKey = nullptr; } GrFontScaler::~GrFontScaler() { SkSafeUnref(fKey); } GrMaskFormat GrFontScaler::getMaskFormat() const { SkMask::Format format = fStrike->getMaskFormat(); switch (format) { case SkMask::kBW_Format: // fall through to kA8 -- we store BW glyphs in our 8-bit cache case SkMask::kA8_Format: return kA8_GrMaskFormat; case SkMask::kLCD16_Format: return kA565_GrMaskFormat; case SkMask::kARGB32_Format: return kARGB_GrMaskFormat; default: SkDEBUGFAIL("unsupported SkMask::Format"); return kA8_GrMaskFormat; } } const GrFontDescKey* GrFontScaler::getKey() { if (nullptr == fKey) { fKey = new GrFontDescKey(fStrike->getDescriptor()); } return fKey; } GrMaskFormat GrFontScaler::getPackedGlyphMaskFormat(const SkGlyph& glyph) const { SkMask::Format format = static_cast<SkMask::Format>(glyph.fMaskFormat); switch (format) { case SkMask::kBW_Format: // fall through to kA8 -- we store BW glyphs in our 8-bit cache case SkMask::kA8_Format: return kA8_GrMaskFormat; case SkMask::kLCD16_Format: return kA565_GrMaskFormat; case SkMask::kARGB32_Format: return kARGB_GrMaskFormat; default: SkDEBUGFAIL("unsupported SkMask::Format"); return kA8_GrMaskFormat; } } bool GrFontScaler::getPackedGlyphBounds(const SkGlyph& glyph, SkIRect* bounds) { #if 1 // crbug:510931 // Retrieving the image from the cache can actually change the mask format. fStrike->findImage(glyph); #endif bounds->setXYWH(glyph.fLeft, glyph.fTop, glyph.fWidth, glyph.fHeight); return true; } bool GrFontScaler::getPackedGlyphDFBounds(const SkGlyph& glyph, SkIRect* bounds) { #if 1 // crbug:510931 // Retrieving the image from the cache can actually change the mask format. fStrike->findImage(glyph); #endif bounds->setXYWH(glyph.fLeft, glyph.fTop, glyph.fWidth, glyph.fHeight); bounds->outset(SK_DistanceFieldPad, SK_DistanceFieldPad); return true; } namespace { // expands each bit in a bitmask to 0 or ~0 of type INT_TYPE. Used to expand a BW glyph mask to // A8, RGB565, or RGBA8888. template <typename INT_TYPE> void expand_bits(INT_TYPE* dst, const uint8_t* src, int width, int height, int dstRowBytes, int srcRowBytes) { for (int i = 0; i < height; ++i) { int rowWritesLeft = width; const uint8_t* s = src; INT_TYPE* d = dst; while (rowWritesLeft > 0) { unsigned mask = *s++; for (int i = 7; i >= 0 && rowWritesLeft; --i, --rowWritesLeft) { *d++ = (mask & (1 << i)) ? (INT_TYPE)(~0UL) : 0; } } dst = reinterpret_cast<INT_TYPE*>(reinterpret_cast<intptr_t>(dst) + dstRowBytes); src += srcRowBytes; } } } bool GrFontScaler::getPackedGlyphImage(const SkGlyph& glyph, int width, int height, int dstRB, GrMaskFormat expectedMaskFormat, void* dst) { SkASSERT(glyph.fWidth == width); SkASSERT(glyph.fHeight == height); const void* src = fStrike->findImage(glyph); if (nullptr == src) { return false; } // crbug:510931 // Retrieving the image from the cache can actually change the mask format. This case is very // uncommon so for now we just draw a clear box for these glyphs. if (getPackedGlyphMaskFormat(glyph) != expectedMaskFormat) { const int bpp = GrMaskFormatBytesPerPixel(expectedMaskFormat); for (int y = 0; y < height; y++) { sk_bzero(dst, width * bpp); dst = (char*)dst + dstRB; } return true; } int srcRB = glyph.rowBytes(); // The windows font host sometimes has BW glyphs in a non-BW strike. So it is important here to // check the glyph's format, not the strike's format, and to be able to convert to any of the // GrMaskFormats. if (SkMask::kBW_Format == glyph.fMaskFormat) { // expand bits to our mask type const uint8_t* bits = reinterpret_cast<const uint8_t*>(src); switch (expectedMaskFormat) { case kA8_GrMaskFormat:{ uint8_t* bytes = reinterpret_cast<uint8_t*>(dst); expand_bits(bytes, bits, width, height, dstRB, srcRB); break; } case kA565_GrMaskFormat: { uint16_t* rgb565 = reinterpret_cast<uint16_t*>(dst); expand_bits(rgb565, bits, width, height, dstRB, srcRB); break; } default: SkFAIL("Invalid GrMaskFormat"); } } else if (srcRB == dstRB) { memcpy(dst, src, dstRB * height); } else { const int bbp = GrMaskFormatBytesPerPixel(expectedMaskFormat); for (int y = 0; y < height; y++) { memcpy(dst, src, width * bbp); src = (const char*)src + srcRB; dst = (char*)dst + dstRB; } } return true; } bool GrFontScaler::getPackedGlyphDFImage(const SkGlyph& glyph, int width, int height, void* dst) { SkASSERT(glyph.fWidth + 2*SK_DistanceFieldPad == width); SkASSERT(glyph.fHeight + 2*SK_DistanceFieldPad == height); const void* image = fStrike->findImage(glyph); if (nullptr == image) { return false; } // now generate the distance field SkASSERT(dst); SkMask::Format maskFormat = static_cast<SkMask::Format>(glyph.fMaskFormat); if (SkMask::kA8_Format == maskFormat) { // make the distance field from the image SkGenerateDistanceFieldFromA8Image((unsigned char*)dst, (unsigned char*)image, glyph.fWidth, glyph.fHeight, glyph.rowBytes()); } else if (SkMask::kBW_Format == maskFormat) { // make the distance field from the image SkGenerateDistanceFieldFromBWImage((unsigned char*)dst, (unsigned char*)image, glyph.fWidth, glyph.fHeight, glyph.rowBytes()); } else { return false; } return true; } const SkPath* GrFontScaler::getGlyphPath(const SkGlyph& glyph) { return fStrike->findPath(glyph); } const SkGlyph& GrFontScaler::grToSkGlyph(GrGlyph::PackedID id) { return fStrike->getGlyphIDMetrics(GrGlyph::UnpackID(id), GrGlyph::UnpackFixedX(id), GrGlyph::UnpackFixedY(id)); }