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 GrBatchAtlas_DEFINED
9 #define GrBatchAtlas_DEFINED
10 
11 #include "GrTexture.h"
12 #include "SkPoint.h"
13 #include "SkTDArray.h"
14 #include "SkTInternalLList.h"
15 
16 #include "batches/GrDrawBatch.h"
17 
18 class GrRectanizer;
19 
20 struct GrBatchAtlasConfig {
numPlotsXGrBatchAtlasConfig21     int numPlotsX() const { return fWidth / fPlotWidth; }
numPlotsYGrBatchAtlasConfig22     int numPlotsY() const { return fHeight / fPlotWidth; }
23     int fWidth;
24     int fHeight;
25     int fLog2Width;
26     int fLog2Height;
27     int fPlotWidth;
28     int fPlotHeight;
29 };
30 
31 class GrBatchAtlas {
32 public:
33     // An AtlasID is an opaque handle which callers can use to determine if the atlas contains
34     // a specific piece of data
35     typedef uint64_t AtlasID;
36     static const uint32_t kInvalidAtlasID = 0;
37     static const uint64_t kInvalidAtlasGeneration = 0;
38 
39     // A function pointer for use as a callback during eviction.  Whenever GrBatchAtlas evicts a
40     // specific AtlasID, it will call all of the registered listeners so they can optionally process
41     // the eviction
42     typedef void (*EvictionFunc)(GrBatchAtlas::AtlasID, void*);
43 
44     GrBatchAtlas(GrTexture*, int numPlotsX, int numPlotsY);
45     ~GrBatchAtlas();
46 
47     // Adds a width x height subimage to the atlas. Upon success it returns
48     // the containing GrPlot and absolute location in the backing texture.
49     // nullptr is returned if the subimage cannot fit in the atlas.
50     // If provided, the image data will be written to the CPU-side backing bitmap.
51     // NOTE: If the client intends to refer to the atlas, they should immediately call 'setUseToken'
52     // with the currentToken from the batch target, otherwise the next call to addToAtlas might
53     // cause an eviction
54     bool addToAtlas(AtlasID*, GrDrawBatch::Target*, int width, int height, const void* image,
55                     SkIPoint16* loc);
56 
getTexture()57     GrTexture* getTexture() const { return fTexture; }
58 
atlasGeneration()59     uint64_t atlasGeneration() const { return fAtlasGeneration; }
60 
hasID(AtlasID id)61     inline bool hasID(AtlasID id) {
62         uint32_t index = GetIndexFromID(id);
63         SkASSERT(index < fNumPlots);
64         return fPlotArray[index]->genID() == GetGenerationFromID(id);
65     }
66 
67     // To ensure the atlas does not evict a given entry, the client must set the last use token
setLastUseToken(AtlasID id,GrBatchToken batchToken)68     inline void setLastUseToken(AtlasID id, GrBatchToken batchToken) {
69         SkASSERT(this->hasID(id));
70         uint32_t index = GetIndexFromID(id);
71         SkASSERT(index < fNumPlots);
72         this->makeMRU(fPlotArray[index]);
73         fPlotArray[index]->setLastUseToken(batchToken);
74     }
75 
registerEvictionCallback(EvictionFunc func,void * userData)76     inline void registerEvictionCallback(EvictionFunc func, void* userData) {
77         EvictionData* data = fEvictionCallbacks.append();
78         data->fFunc = func;
79         data->fData = userData;
80     }
81 
82     /*
83      * A class which can be handed back to GrBatchAtlas for updating in bulk last use tokens.  The
84      * current max number of plots the GrBatchAtlas can handle is 32, if in the future this is
85      * insufficient then we can move to a 64 bit int
86      */
87     class BulkUseTokenUpdater {
88     public:
BulkUseTokenUpdater()89         BulkUseTokenUpdater() : fPlotAlreadyUpdated(0) {}
BulkUseTokenUpdater(const BulkUseTokenUpdater & that)90         BulkUseTokenUpdater(const BulkUseTokenUpdater& that)
91             : fPlotsToUpdate(that.fPlotsToUpdate)
92             , fPlotAlreadyUpdated(that.fPlotAlreadyUpdated) {
93         }
94 
add(AtlasID id)95         void add(AtlasID id) {
96             int index = GrBatchAtlas::GetIndexFromID(id);
97             if (!this->find(index)) {
98                 this->set(index);
99             }
100         }
101 
reset()102         void reset() {
103             fPlotsToUpdate.reset();
104             fPlotAlreadyUpdated = 0;
105         }
106 
107     private:
find(int index)108         bool find(int index) const {
109             SkASSERT(index < kMaxPlots);
110             return (fPlotAlreadyUpdated >> index) & 1;
111         }
112 
set(int index)113         void set(int index) {
114             SkASSERT(!this->find(index));
115             fPlotAlreadyUpdated = fPlotAlreadyUpdated | (1 << index);
116             fPlotsToUpdate.push_back(index);
117         }
118 
119         static const int kMinItems = 4;
120         static const int kMaxPlots = 32;
121         SkSTArray<kMinItems, int, true> fPlotsToUpdate;
122         uint32_t fPlotAlreadyUpdated;
123 
124         friend class GrBatchAtlas;
125     };
126 
setLastUseTokenBulk(const BulkUseTokenUpdater & updater,GrBatchToken batchToken)127     void setLastUseTokenBulk(const BulkUseTokenUpdater& updater, GrBatchToken batchToken) {
128         int count = updater.fPlotsToUpdate.count();
129         for (int i = 0; i < count; i++) {
130             BatchPlot* plot = fPlotArray[updater.fPlotsToUpdate[i]];
131             this->makeMRU(plot);
132             plot->setLastUseToken(batchToken);
133         }
134     }
135 
136     static const int kGlyphMaxDim = 256;
GlyphTooLargeForAtlas(int width,int height)137     static bool GlyphTooLargeForAtlas(int width, int height) {
138         return width > kGlyphMaxDim || height > kGlyphMaxDim;
139     }
140 
141 private:
142     // The backing GrTexture for a GrBatchAtlas is broken into a spatial grid of BatchPlots.
143     // The BatchPlots keep track of subimage placement via their GrRectanizer. A BatchPlot
144     // manages the lifetime of its data using two tokens, a last use token and a last upload token.
145     // Once a BatchPlot is "full" (i.e. there is no room for the new subimage according to the
146     // GrRectanizer), it can no longer be used unless the last use of the GrPlot has already been
147     // flushed through to the gpu.
148     class BatchPlot : public SkRefCnt {
149         SK_DECLARE_INTERNAL_LLIST_INTERFACE(BatchPlot);
150 
151     public:
152         // index() is a unique id for the plot relative to the owning GrAtlas.  genID() is a
153         // monotonically incremented number which is bumped every time this plot is
154         // evicted from the cache (i.e., there is continuity in genID() across atlas spills).
index()155         uint32_t index() const { return fIndex; }
genID()156         uint64_t genID() const { return fGenID; }
id()157         GrBatchAtlas::AtlasID id() const {
158             SkASSERT(GrBatchAtlas::kInvalidAtlasID != fID);
159             return fID;
160         }
161         SkDEBUGCODE(size_t bpp() const { return fBytesPerPixel; })
162 
163         bool addSubImage(int width, int height, const void* image, SkIPoint16* loc);
164 
165         // To manage the lifetime of a plot, we use two tokens.  We use the last upload token to
166         // know when we can 'piggy back' uploads, ie if the last upload hasn't been flushed to gpu,
167         // we don't need to issue a new upload even if we update the cpu backing store.  We use
168         // lastUse to determine when we can evict a plot from the cache, ie if the last use has
169         // already flushed through the gpu then we can reuse the plot.
lastUploadToken()170         GrBatchToken lastUploadToken() const { return fLastUpload; }
lastUseToken()171         GrBatchToken lastUseToken() const { return fLastUse; }
setLastUploadToken(GrBatchToken batchToken)172         void setLastUploadToken(GrBatchToken batchToken) {
173             SkASSERT(batchToken >= fLastUpload);
174             fLastUpload = batchToken;
175         }
setLastUseToken(GrBatchToken batchToken)176         void setLastUseToken(GrBatchToken batchToken) {
177             SkASSERT(batchToken >= fLastUse);
178             fLastUse = batchToken;
179         }
180 
181         void uploadToTexture(GrBatchUploader::TextureUploader* uploader, GrTexture* texture);
182         void resetRects();
183 
184     private:
185         BatchPlot(int index, uint64_t genID, int offX, int offY, int width, int height,
186                   GrPixelConfig config);
187 
188         ~BatchPlot() override;
189 
190         // Create a clone of this plot. The cloned plot will take the place of the
191         // current plot in the atlas.
clone()192         BatchPlot* clone() const {
193             return new BatchPlot(fIndex, fGenID+1, fX, fY, fWidth, fHeight, fConfig);
194         }
195 
CreateId(uint32_t index,uint64_t generation)196         static GrBatchAtlas::AtlasID CreateId(uint32_t index, uint64_t generation) {
197             SkASSERT(index < (1 << 16));
198             SkASSERT(generation < ((uint64_t)1 << 48));
199             return generation << 16 | index;
200         }
201 
202         GrBatchToken          fLastUpload;
203         GrBatchToken          fLastUse;
204 
205         const uint32_t        fIndex;
206         uint64_t              fGenID;
207         GrBatchAtlas::AtlasID fID;
208         unsigned char*        fData;
209         const int             fWidth;
210         const int             fHeight;
211         const int             fX;
212         const int             fY;
213         GrRectanizer*         fRects;
214         const SkIPoint16      fOffset;        // the offset of the plot in the backing texture
215         const GrPixelConfig   fConfig;
216         const size_t          fBytesPerPixel;
217         SkIRect               fDirtyRect;
218         SkDEBUGCODE(bool      fDirty;)
219 
220         friend class GrBatchAtlas;
221 
222         typedef SkRefCnt INHERITED;
223     };
224 
225     typedef SkTInternalLList<BatchPlot> GrBatchPlotList;
226 
GetIndexFromID(AtlasID id)227     static uint32_t GetIndexFromID(AtlasID id) {
228         return id & 0xffff;
229     }
230 
231     // top 48 bits are reserved for the generation ID
GetGenerationFromID(AtlasID id)232     static uint64_t GetGenerationFromID(AtlasID id) {
233         return (id >> 16) & 0xffffffffffff;
234     }
235 
236     inline void updatePlot(GrDrawBatch::Target*, AtlasID*, BatchPlot*);
237 
makeMRU(BatchPlot * plot)238     inline void makeMRU(BatchPlot* plot) {
239         if (fPlotList.head() == plot) {
240             return;
241         }
242 
243         fPlotList.remove(plot);
244         fPlotList.addToHead(plot);
245     }
246 
247     inline void processEviction(AtlasID);
248 
249     friend class GrPlotUploader; // to access GrBatchPlot
250 
251     GrTexture* fTexture;
252     SkDEBUGCODE(uint32_t fNumPlots;)
253 
254     uint64_t fAtlasGeneration;
255 
256     struct EvictionData {
257         EvictionFunc fFunc;
258         void* fData;
259     };
260 
261     SkTDArray<EvictionData> fEvictionCallbacks;
262     // allocated array of GrBatchPlots
263     SkAutoTUnref<BatchPlot>* fPlotArray;
264     // LRU list of GrPlots (MRU at head - LRU at tail)
265     GrBatchPlotList fPlotList;
266 };
267 
268 #endif
269