Home | History | Annotate | Download | only in text
      1 /*
      2  * Copyright 2015 Google Inc.
      3  *
      4  * Use of this source code is governed by a BSD-style license that can be
      5  * found in the LICENSE file.
      6  */
      7 
      8 #ifndef GrAtlasGlyphCache_DEFINED
      9 #define GrAtlasGlyphCache_DEFINED
     10 
     11 #include "GrDrawOpAtlas.h"
     12 #include "GrGlyph.h"
     13 #include "SkArenaAlloc.h"
     14 #include "SkGlyphCache.h"
     15 #include "SkTDynamicHash.h"
     16 
     17 class GrGlyphCache;
     18 class GrAtlasManager;
     19 class GrGpu;
     20 
     21 /**
     22  *  The GrTextStrike manages a pool of CPU backing memory for GrGlyphs. This backing memory
     23  *  is indexed by a PackedID and SkGlyphCache. The SkGlyphCache is what actually creates the mask.
     24  *  The GrTextStrike may outlive the generating SkGlyphCache. However, it retains a copy
     25  *  of it's SkDescriptor as a key to access (or regenerate) the SkGlyphCache. GrTextStrikes are
     26  *  created by and owned by a GrGlyphCache.
     27  */
     28 class GrTextStrike : public SkNVRefCnt<GrTextStrike> {
     29 public:
     30     GrTextStrike(const SkDescriptor& fontScalerKey);
     31     ~GrTextStrike();
     32 
     33     inline GrGlyph* getGlyph(const SkGlyph& skGlyph, GrGlyph::PackedID packed,
     34                              SkGlyphCache* cache) {
     35         GrGlyph* glyph = fCache.find(packed);
     36         if (!glyph) {
     37             glyph = this->generateGlyph(skGlyph, packed, cache);
     38         }
     39         return glyph;
     40     }
     41 
     42     // This variant of the above function is called by GrAtlasTextOp. At this point, it is possible
     43     // that the maskformat of the glyph differs from what we expect.  In these cases we will just
     44     // draw a clear square.
     45     // skbug:4143 crbug:510931
     46     inline GrGlyph* getGlyph(GrGlyph::PackedID packed,
     47                              GrMaskFormat expectedMaskFormat,
     48                              SkGlyphCache* cache) {
     49         GrGlyph* glyph = fCache.find(packed);
     50         if (!glyph) {
     51             // We could return this to the caller, but in practice it adds code complexity for
     52             // potentially little benefit(ie, if the glyph is not in our font cache, then its not
     53             // in the atlas and we're going to be doing a texture upload anyways).
     54             const SkGlyph& skGlyph = GrToSkGlyph(cache, packed);
     55             glyph = this->generateGlyph(skGlyph, packed, cache);
     56             glyph->fMaskFormat = expectedMaskFormat;
     57         }
     58         return glyph;
     59     }
     60 
     61     // returns true if glyph successfully added to texture atlas, false otherwise.  If the glyph's
     62     // mask format has changed, then addGlyphToAtlas will draw a clear box.  This will almost never
     63     // happen.
     64     // TODO we can handle some of these cases if we really want to, but the long term solution is to
     65     // get the actual glyph image itself when we get the glyph metrics.
     66     bool addGlyphToAtlas(GrResourceProvider*, GrDeferredUploadTarget*, GrGlyphCache*,
     67                          GrAtlasManager*, GrGlyph*,
     68                          SkGlyphCache*, GrMaskFormat expectedMaskFormat);
     69 
     70     // testing
     71     int countGlyphs() const { return fCache.count(); }
     72 
     73     // remove any references to this plot
     74     void removeID(GrDrawOpAtlas::AtlasID);
     75 
     76     // If a TextStrike is abandoned by the cache, then the caller must get a new strike
     77     bool isAbandoned() const { return fIsAbandoned; }
     78 
     79     static const SkDescriptor& GetKey(const GrTextStrike& strike) {
     80         return *strike.fFontScalerKey.getDesc();
     81     }
     82 
     83     static uint32_t Hash(const SkDescriptor& desc) { return desc.getChecksum(); }
     84 
     85 private:
     86     SkTDynamicHash<GrGlyph, GrGlyph::PackedID> fCache;
     87     SkAutoDescriptor fFontScalerKey;
     88     SkArenaAlloc fPool{512};
     89 
     90     int fAtlasedGlyphs;
     91     bool fIsAbandoned;
     92 
     93     static const SkGlyph& GrToSkGlyph(SkGlyphCache* cache, GrGlyph::PackedID id) {
     94         return cache->getGlyphIDMetrics(GrGlyph::UnpackID(id),
     95                                         GrGlyph::UnpackFixedX(id),
     96                                         GrGlyph::UnpackFixedY(id));
     97     }
     98 
     99     GrGlyph* generateGlyph(const SkGlyph&, GrGlyph::PackedID, SkGlyphCache*);
    100 
    101     friend class GrGlyphCache;
    102 };
    103 
    104 /**
    105  * GrGlyphCache manages strikes which are indexed by a SkGlyphCache. These strikes can then be
    106  * used to generate individual Glyph Masks.
    107  */
    108 class GrGlyphCache {
    109 public:
    110     GrGlyphCache();
    111     ~GrGlyphCache();
    112 
    113     void setGlyphSizeLimit(SkScalar sizeLimit) { fGlyphSizeLimit = sizeLimit; }
    114     SkScalar getGlyphSizeLimit() const { return fGlyphSizeLimit; }
    115 
    116     void setStrikeToPreserve(GrTextStrike* strike) { fPreserveStrike = strike; }
    117 
    118     // The user of the cache may hold a long-lived ref to the returned strike. However, actions by
    119     // another client of the cache may cause the strike to be purged while it is still reffed.
    120     // Therefore, the caller must check GrTextStrike::isAbandoned() if there are other
    121     // interactions with the cache since the strike was received.
    122     inline sk_sp<GrTextStrike> getStrike(const SkGlyphCache* cache) {
    123         sk_sp<GrTextStrike> strike = sk_ref_sp(fCache.find(cache->getDescriptor()));
    124         if (!strike) {
    125             strike = this->generateStrike(cache);
    126         }
    127         return strike;
    128     }
    129 
    130     void freeAll();
    131 
    132     static void HandleEviction(GrDrawOpAtlas::AtlasID, void*);
    133 
    134 private:
    135     sk_sp<GrTextStrike> generateStrike(const SkGlyphCache* cache) {
    136         // 'fCache' get the construction ref
    137         sk_sp<GrTextStrike> strike = sk_ref_sp(new GrTextStrike(cache->getDescriptor()));
    138         fCache.add(strike.get());
    139         return strike;
    140     }
    141 
    142     using StrikeHash = SkTDynamicHash<GrTextStrike, SkDescriptor>;
    143 
    144     StrikeHash fCache;
    145     GrTextStrike* fPreserveStrike;
    146     SkScalar fGlyphSizeLimit;
    147 };
    148 
    149 #endif
    150