/* * 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 "GrTemplates.h" #include "SkGr.h" #include "SkDescriptor.h" #include "SkGlyphCache.h" class SkGrDescKey : public GrKey { public: explicit SkGrDescKey(const SkDescriptor& desc); virtual ~SkGrDescKey(); protected: // overrides virtual bool lt(const GrKey& rh) const; virtual bool eq(const GrKey& rh) const; private: SkDescriptor* fDesc; enum { kMaxStorageInts = 16 }; uint32_t fStorage[kMaxStorageInts]; }; /////////////////////////////////////////////////////////////////////////////// SkGrDescKey::SkGrDescKey(const SkDescriptor& desc) : GrKey(desc.getChecksum()) { size_t size = desc.getLength(); if (size <= sizeof(fStorage)) { fDesc = GrTCast(fStorage); } else { fDesc = SkDescriptor::Alloc(size); } memcpy(fDesc, &desc, size); } SkGrDescKey::~SkGrDescKey() { if (fDesc != GrTCast(fStorage)) { SkDescriptor::Free(fDesc); } } bool SkGrDescKey::lt(const GrKey& rh) const { const SkDescriptor* srcDesc = ((const SkGrDescKey*)&rh)->fDesc; size_t lenLH = fDesc->getLength(); size_t lenRH = srcDesc->getLength(); int cmp = memcmp(fDesc, srcDesc, SkMin32(lenLH, lenRH)); if (0 == cmp) { return lenLH < lenRH; } else { return cmp < 0; } } bool SkGrDescKey::eq(const GrKey& rh) const { const SkDescriptor* srcDesc = ((const SkGrDescKey*)&rh)->fDesc; return fDesc->equals(*srcDesc); } /////////////////////////////////////////////////////////////////////////////// SkGrFontScaler::SkGrFontScaler(SkGlyphCache* strike) { fStrike = strike; fKey = NULL; } SkGrFontScaler::~SkGrFontScaler() { GrSafeUnref(fKey); } GrMaskFormat SkGrFontScaler::getMaskFormat() { 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::kLCD32_Format: return kA888_GrMaskFormat; default: GrAssert(!"unsupported SkMask::Format"); return kA8_GrMaskFormat; } } const GrKey* SkGrFontScaler::getKey() { if (NULL == fKey) { fKey = new SkGrDescKey(fStrike->getDescriptor()); } return fKey; } bool SkGrFontScaler::getPackedGlyphBounds(GrGlyph::PackedID packed, GrIRect* bounds) { const SkGlyph& glyph = fStrike->getGlyphIDMetrics(GrGlyph::UnpackID(packed), GrGlyph::UnpackFixedX(packed), GrGlyph::UnpackFixedY(packed)); bounds->setXYWH(glyph.fLeft, glyph.fTop, glyph.fWidth, glyph.fHeight); return true; } static void bits_to_bytes(const uint8_t bits[], uint8_t bytes[], int count) { while (count > 0) { unsigned mask = *bits++; for (int i = 7; i >= 0; --i) { *bytes++ = (mask & (1 << i)) ? 0xFF : 0; if (--count == 0) { return; } } } } bool SkGrFontScaler::getPackedGlyphImage(GrGlyph::PackedID packed, int width, int height, int dstRB, void* dst) { const SkGlyph& glyph = fStrike->getGlyphIDMetrics(GrGlyph::UnpackID(packed), GrGlyph::UnpackFixedX(packed), GrGlyph::UnpackFixedY(packed)); GrAssert(glyph.fWidth == width); GrAssert(glyph.fHeight == height); const void* src = fStrike->findImage(glyph); if (NULL == src) { return false; } int srcRB = glyph.rowBytes(); if (SkMask::kBW_Format == fStrike->getMaskFormat()) { // expand bits to bytes const uint8_t* bits = reinterpret_cast(src); uint8_t* bytes = reinterpret_cast(dst); for (int y = 0; y < height; y++) { bits_to_bytes(bits, bytes, width); bits += srcRB; bytes += dstRB; } } else if (srcRB == dstRB) { memcpy(dst, src, dstRB * height); } else { const int bbp = GrMaskFormatBytesPerPixel(this->getMaskFormat()); for (int y = 0; y < height; y++) { memcpy(dst, src, width * bbp); src = (const char*)src + srcRB; dst = (char*)dst + dstRB; } } return true; } // we should just return const SkPath* (NULL means false) bool SkGrFontScaler::getGlyphPath(uint16_t glyphID, SkPath* path) { const SkGlyph& glyph = fStrike->getGlyphIDMetrics(glyphID); const SkPath* skPath = fStrike->findPath(glyph); if (skPath) { *path = *skPath; return true; } return false; }