| /* |
| * Copyright 2015 Google Inc. |
| * |
| * Use of this source code is governed by a BSD-style license that can be |
| * found in the LICENSE file. |
| */ |
| |
| #ifndef GrAtlasGlyphCache_DEFINED |
| #define GrAtlasGlyphCache_DEFINED |
| |
| #include "GrCaps.h" |
| #include "GrDrawOpAtlas.h" |
| #include "GrGlyph.h" |
| #include "SkGlyphCache.h" |
| #include "SkTDynamicHash.h" |
| #include "SkVarAlloc.h" |
| |
| class GrAtlasGlyphCache; |
| class GrGpu; |
| |
| /** |
| * The GrAtlasTextStrike manages a pool of CPU backing memory for GrGlyphs. This backing memory |
| * is indexed by a PackedID and SkGlyphCache. The SkGlyphCache is what actually creates the mask. |
| * The GrAtlasTextStrike may outlive the generating SkGlyphCache. However, it retains a copy |
| * of it's SkDescriptor as a key to access (or regenerate) the SkGlyphCache. GrAtlasTextStrike are |
| * created by and owned by a GrAtlasGlyphCache. |
| */ |
| class GrAtlasTextStrike : public SkNVRefCnt<GrAtlasTextStrike> { |
| public: |
| /** Owner is the cache that owns this strike. */ |
| GrAtlasTextStrike(GrAtlasGlyphCache* owner, const SkDescriptor& fontScalerKey); |
| ~GrAtlasTextStrike(); |
| |
| inline GrGlyph* getGlyph(const SkGlyph& skGlyph, GrGlyph::PackedID packed, |
| SkGlyphCache* cache) { |
| GrGlyph* glyph = fCache.find(packed); |
| if (nullptr == glyph) { |
| glyph = this->generateGlyph(skGlyph, packed, cache); |
| } |
| return glyph; |
| } |
| |
| // This variant of the above function is called by GrAtlasTextOp. At this point, it is possible |
| // that the maskformat of the glyph differs from what we expect. In these cases we will just |
| // draw a clear square. |
| // skbug:4143 crbug:510931 |
| inline GrGlyph* getGlyph(GrGlyph::PackedID packed, |
| GrMaskFormat expectedMaskFormat, |
| SkGlyphCache* cache) { |
| GrGlyph* glyph = fCache.find(packed); |
| if (nullptr == glyph) { |
| // We could return this to the caller, but in practice it adds code complexity for |
| // potentially little benefit(ie, if the glyph is not in our font cache, then its not |
| // in the atlas and we're going to be doing a texture upload anyways). |
| const SkGlyph& skGlyph = GrToSkGlyph(cache, packed); |
| glyph = this->generateGlyph(skGlyph, packed, cache); |
| glyph->fMaskFormat = expectedMaskFormat; |
| } |
| return glyph; |
| } |
| |
| // returns true if glyph successfully added to texture atlas, false otherwise. If the glyph's |
| // mask format has changed, then addGlyphToAtlas will draw a clear box. This will almost never |
| // happen. |
| // TODO we can handle some of these cases if we really want to, but the long term solution is to |
| // get the actual glyph image itself when we get the glyph metrics. |
| bool addGlyphToAtlas(GrDrawOp::Target*, GrGlyph*, SkGlyphCache*, |
| GrMaskFormat expectedMaskFormat); |
| |
| // testing |
| int countGlyphs() const { return fCache.count(); } |
| |
| // remove any references to this plot |
| void removeID(GrDrawOpAtlas::AtlasID); |
| |
| // If a TextStrike is abandoned by the cache, then the caller must get a new strike |
| bool isAbandoned() const { return fIsAbandoned; } |
| |
| static const SkDescriptor& GetKey(const GrAtlasTextStrike& ts) { |
| return *ts.fFontScalerKey.getDesc(); |
| } |
| |
| static uint32_t Hash(const SkDescriptor& desc) { return desc.getChecksum(); } |
| |
| private: |
| SkTDynamicHash<GrGlyph, GrGlyph::PackedID> fCache; |
| SkAutoDescriptor fFontScalerKey; |
| SkVarAlloc fPool; |
| |
| GrAtlasGlyphCache* fAtlasGlyphCache; |
| int fAtlasedGlyphs; |
| bool fIsAbandoned; |
| |
| static const SkGlyph& GrToSkGlyph(SkGlyphCache* cache, GrGlyph::PackedID id) { |
| return cache->getGlyphIDMetrics(GrGlyph::UnpackID(id), |
| GrGlyph::UnpackFixedX(id), |
| GrGlyph::UnpackFixedY(id)); |
| } |
| |
| GrGlyph* generateGlyph(const SkGlyph&, GrGlyph::PackedID, SkGlyphCache*); |
| |
| friend class GrAtlasGlyphCache; |
| }; |
| |
| /** |
| * GrAtlasGlyphCache manages strikes which are indexed by a SkGlyphCache. These strikes can then be |
| * used to generate individual Glyph Masks. The GrAtlasGlyphCache also manages GrDrawOpAtlases, |
| * though this is more or less transparent to the client(aside from atlasGeneration, described |
| * below). |
| */ |
| class GrAtlasGlyphCache { |
| public: |
| GrAtlasGlyphCache(GrContext*); |
| ~GrAtlasGlyphCache(); |
| // The user of the cache may hold a long-lived ref to the returned strike. However, actions by |
| // another client of the cache may cause the strike to be purged while it is still reffed. |
| // Therefore, the caller must check GrAtlasTextStrike::isAbandoned() if there are other |
| // interactions with the cache since the strike was received. |
| inline GrAtlasTextStrike* getStrike(const SkGlyphCache* cache) { |
| GrAtlasTextStrike* strike = fCache.find(cache->getDescriptor()); |
| if (nullptr == strike) { |
| strike = this->generateStrike(cache); |
| } |
| return strike; |
| } |
| |
| void freeAll(); |
| |
| // if texture returns nullptr, the client must not try to use other functions on the |
| // GrAtlasGlyphCache which use the atlas. This function *must* be called first, before other |
| // functions which use the atlas. |
| GrTexture* getTexture(GrMaskFormat format) { |
| if (this->initAtlas(format)) { |
| return this->getAtlas(format)->getTexture(); |
| } |
| return nullptr; |
| } |
| |
| bool hasGlyph(GrGlyph* glyph) { |
| SkASSERT(glyph); |
| return this->getAtlas(glyph->fMaskFormat)->hasID(glyph->fID); |
| } |
| |
| // To ensure the GrDrawOpAtlas does not evict the Glyph Mask from its texture backing store, |
| // the client must pass in the current op token along with the GrGlyph. |
| // A BulkUseTokenUpdater is used to manage bulk last use token updating in the Atlas. |
| // For convenience, this function will also set the use token for the current glyph if required |
| // NOTE: the bulk uploader is only valid if the subrun has a valid atlasGeneration |
| void addGlyphToBulkAndSetUseToken(GrDrawOpAtlas::BulkUseTokenUpdater* updater, GrGlyph* glyph, |
| GrDrawOpUploadToken token) { |
| SkASSERT(glyph); |
| updater->add(glyph->fID); |
| this->getAtlas(glyph->fMaskFormat)->setLastUseToken(glyph->fID, token); |
| } |
| |
| void setUseTokenBulk(const GrDrawOpAtlas::BulkUseTokenUpdater& updater, |
| GrDrawOpUploadToken token, |
| GrMaskFormat format) { |
| this->getAtlas(format)->setLastUseTokenBulk(updater, token); |
| } |
| |
| // add to texture atlas that matches this format |
| bool addToAtlas(GrAtlasTextStrike* strike, GrDrawOpAtlas::AtlasID* id, GrDrawOp::Target* target, |
| GrMaskFormat format, int width, int height, const void* image, |
| SkIPoint16* loc) { |
| fPreserveStrike = strike; |
| return this->getAtlas(format)->addToAtlas(id, target, width, height, image, loc); |
| } |
| |
| // Some clients may wish to verify the integrity of the texture backing store of the |
| // GrDrawOpAtlas. The atlasGeneration returned below is a monotonically increasing number which |
| // changes every time something is removed from the texture backing store. |
| uint64_t atlasGeneration(GrMaskFormat format) const { |
| return this->getAtlas(format)->atlasGeneration(); |
| } |
| |
| int log2Width(GrMaskFormat format) { return fAtlasConfigs[format].fLog2Width; } |
| int log2Height(GrMaskFormat format) { return fAtlasConfigs[format].fLog2Height; } |
| |
| /////////////////////////////////////////////////////////////////////////// |
| // Functions intended debug only |
| #ifdef SK_DEBUG |
| void dump() const; |
| #endif |
| |
| void setAtlasSizes_ForTesting(const GrDrawOpAtlasConfig configs[3]); |
| |
| private: |
| static GrPixelConfig MaskFormatToPixelConfig(GrMaskFormat format, const GrCaps& caps) { |
| switch (format) { |
| case kA8_GrMaskFormat: |
| return kAlpha_8_GrPixelConfig; |
| case kA565_GrMaskFormat: |
| return kRGB_565_GrPixelConfig; |
| case kARGB_GrMaskFormat: |
| return caps.srgbSupport() ? kSRGBA_8888_GrPixelConfig : kRGBA_8888_GrPixelConfig; |
| default: |
| SkDEBUGFAIL("unsupported GrMaskFormat"); |
| return kAlpha_8_GrPixelConfig; |
| } |
| } |
| |
| // There is a 1:1 mapping between GrMaskFormats and atlas indices |
| static int MaskFormatToAtlasIndex(GrMaskFormat format) { |
| static const int sAtlasIndices[] = { |
| kA8_GrMaskFormat, |
| kA565_GrMaskFormat, |
| kARGB_GrMaskFormat, |
| }; |
| static_assert(SK_ARRAY_COUNT(sAtlasIndices) == kMaskFormatCount, "array_size_mismatch"); |
| |
| SkASSERT(sAtlasIndices[format] < kMaskFormatCount); |
| return sAtlasIndices[format]; |
| } |
| |
| bool initAtlas(GrMaskFormat); |
| |
| GrAtlasTextStrike* generateStrike(const SkGlyphCache* cache) { |
| GrAtlasTextStrike* strike = new GrAtlasTextStrike(this, cache->getDescriptor()); |
| fCache.add(strike); |
| return strike; |
| } |
| |
| GrDrawOpAtlas* getAtlas(GrMaskFormat format) const { |
| int atlasIndex = MaskFormatToAtlasIndex(format); |
| SkASSERT(fAtlases[atlasIndex]); |
| return fAtlases[atlasIndex].get(); |
| } |
| |
| static void HandleEviction(GrDrawOpAtlas::AtlasID, void*); |
| |
| using StrikeHash = SkTDynamicHash<GrAtlasTextStrike, SkDescriptor>; |
| GrContext* fContext; |
| StrikeHash fCache; |
| std::unique_ptr<GrDrawOpAtlas> fAtlases[kMaskFormatCount]; |
| GrAtlasTextStrike* fPreserveStrike; |
| GrDrawOpAtlasConfig fAtlasConfigs[kMaskFormatCount]; |
| }; |
| |
| #endif |