blob: 11270f3df465a9f3b3e5f328fbb79e1a00944e5f [file] [log] [blame]
joshualitt5bf99f12015-03-13 11:47:42 -07001/*
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
Brian Salomon89527432016-12-16 09:52:16 -050016#include "ops/GrDrawOp.h"
joshualittddd22d82016-02-16 06:47:52 -080017
joshualitt5bf99f12015-03-13 11:47:42 -070018class GrRectanizer;
19
joshualittda04e0e2015-08-19 08:16:43 -070020struct GrBatchAtlasConfig {
21 int numPlotsX() const { return fWidth / fPlotWidth; }
22 int numPlotsY() const { return fHeight / fPlotWidth; }
23 int fWidth;
24 int fHeight;
jvanverth7023a002016-02-22 11:25:32 -080025 int fLog2Width;
26 int fLog2Height;
joshualittda04e0e2015-08-19 08:16:43 -070027 int fPlotWidth;
28 int fPlotHeight;
29};
30
joshualitt5bf99f12015-03-13 11:47:42 -070031class GrBatchAtlas {
32public:
joshualitt5bf99f12015-03-13 11:47:42 -070033 // An AtlasID is an opaque handle which callers can use to determine if the atlas contains
34 // a specific piece of data
joshualitt8db6fdc2015-07-31 08:25:07 -070035 typedef uint64_t AtlasID;
joshualitt7c3a2f82015-03-31 13:32:05 -070036 static const uint32_t kInvalidAtlasID = 0;
37 static const uint64_t kInvalidAtlasGeneration = 0;
joshualitt5bf99f12015-03-13 11:47:42 -070038
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
Ben Wagner594f9ed2016-11-08 14:13:39 -050044 GrBatchAtlas(sk_sp<GrTexture>, int numPlotsX, int numPlotsY);
joshualitt5bf99f12015-03-13 11:47:42 -070045
46 // Adds a width x height subimage to the atlas. Upon success it returns
47 // the containing GrPlot and absolute location in the backing texture.
halcanary96fcdcc2015-08-27 07:41:13 -070048 // nullptr is returned if the subimage cannot fit in the atlas.
joshualitt5bf99f12015-03-13 11:47:42 -070049 // If provided, the image data will be written to the CPU-side backing bitmap.
joshualittb4c507e2015-04-08 08:07:59 -070050 // NOTE: If the client intends to refer to the atlas, they should immediately call 'setUseToken'
51 // with the currentToken from the batch target, otherwise the next call to addToAtlas might
52 // cause an eviction
Brian Salomon9afd3712016-12-01 10:59:09 -050053 bool addToAtlas(AtlasID*, GrDrawOp::Target*, int width, int height, const void* image,
joshualitt5bf99f12015-03-13 11:47:42 -070054 SkIPoint16* loc);
55
Ben Wagner594f9ed2016-11-08 14:13:39 -050056 GrTexture* getTexture() const { return fTexture.get(); }
joshualitt5bf99f12015-03-13 11:47:42 -070057
joshualitt7c3a2f82015-03-31 13:32:05 -070058 uint64_t atlasGeneration() const { return fAtlasGeneration; }
joshualitt5df175e2015-11-18 13:37:54 -080059
60 inline bool hasID(AtlasID id) {
61 uint32_t index = GetIndexFromID(id);
62 SkASSERT(index < fNumPlots);
63 return fPlotArray[index]->genID() == GetGenerationFromID(id);
64 }
joshualittb4c507e2015-04-08 08:07:59 -070065
66 // To ensure the atlas does not evict a given entry, the client must set the last use token
Brian Salomon9afd3712016-12-01 10:59:09 -050067 inline void setLastUseToken(AtlasID id, GrDrawOpUploadToken batchToken) {
joshualitt5df175e2015-11-18 13:37:54 -080068 SkASSERT(this->hasID(id));
69 uint32_t index = GetIndexFromID(id);
70 SkASSERT(index < fNumPlots);
Hal Canary144caf52016-11-07 17:57:18 -050071 this->makeMRU(fPlotArray[index].get());
joshualitt5df175e2015-11-18 13:37:54 -080072 fPlotArray[index]->setLastUseToken(batchToken);
73 }
74
75 inline void registerEvictionCallback(EvictionFunc func, void* userData) {
joshualitt5bf99f12015-03-13 11:47:42 -070076 EvictionData* data = fEvictionCallbacks.append();
77 data->fFunc = func;
78 data->fData = userData;
79 }
80
joshualittb4c507e2015-04-08 08:07:59 -070081 /*
82 * A class which can be handed back to GrBatchAtlas for updating in bulk last use tokens. The
83 * current max number of plots the GrBatchAtlas can handle is 32, if in the future this is
84 * insufficient then we can move to a 64 bit int
85 */
86 class BulkUseTokenUpdater {
87 public:
joshualitt4314e082015-04-23 08:03:35 -070088 BulkUseTokenUpdater() : fPlotAlreadyUpdated(0) {}
joshualitt7e97b0b2015-07-31 15:18:08 -070089 BulkUseTokenUpdater(const BulkUseTokenUpdater& that)
90 : fPlotsToUpdate(that.fPlotsToUpdate)
91 , fPlotAlreadyUpdated(that.fPlotAlreadyUpdated) {
92 }
93
joshualittb4c507e2015-04-08 08:07:59 -070094 void add(AtlasID id) {
95 int index = GrBatchAtlas::GetIndexFromID(id);
96 if (!this->find(index)) {
97 this->set(index);
98 }
99 }
100
101 void reset() {
joshualitt4314e082015-04-23 08:03:35 -0700102 fPlotsToUpdate.reset();
joshualittb4c507e2015-04-08 08:07:59 -0700103 fPlotAlreadyUpdated = 0;
104 }
105
106 private:
107 bool find(int index) const {
108 SkASSERT(index < kMaxPlots);
109 return (fPlotAlreadyUpdated >> index) & 1;
110 }
111
112 void set(int index) {
113 SkASSERT(!this->find(index));
114 fPlotAlreadyUpdated = fPlotAlreadyUpdated | (1 << index);
joshualitt97202d22015-04-22 13:47:02 -0700115 fPlotsToUpdate.push_back(index);
joshualittb4c507e2015-04-08 08:07:59 -0700116 }
117
118 static const int kMinItems = 4;
119 static const int kMaxPlots = 32;
joshualitt97202d22015-04-22 13:47:02 -0700120 SkSTArray<kMinItems, int, true> fPlotsToUpdate;
joshualitt8672f4d2015-04-21 08:03:04 -0700121 uint32_t fPlotAlreadyUpdated;
joshualittb4c507e2015-04-08 08:07:59 -0700122
123 friend class GrBatchAtlas;
124 };
125
Brian Salomon9afd3712016-12-01 10:59:09 -0500126 void setLastUseTokenBulk(const BulkUseTokenUpdater& updater, GrDrawOpUploadToken batchToken) {
joshualitt5df175e2015-11-18 13:37:54 -0800127 int count = updater.fPlotsToUpdate.count();
128 for (int i = 0; i < count; i++) {
Hal Canary144caf52016-11-07 17:57:18 -0500129 BatchPlot* plot = fPlotArray[updater.fPlotsToUpdate[i]].get();
joshualitt5df175e2015-11-18 13:37:54 -0800130 this->makeMRU(plot);
131 plot->setLastUseToken(batchToken);
132 }
133 }
joshualittb4c507e2015-04-08 08:07:59 -0700134
joshualitt010db532015-04-21 10:07:26 -0700135 static const int kGlyphMaxDim = 256;
136 static bool GlyphTooLargeForAtlas(int width, int height) {
137 return width > kGlyphMaxDim || height > kGlyphMaxDim;
138 }
139
joshualitt5bf99f12015-03-13 11:47:42 -0700140private:
joshualitt5df175e2015-11-18 13:37:54 -0800141 // The backing GrTexture for a GrBatchAtlas is broken into a spatial grid of BatchPlots.
142 // The BatchPlots keep track of subimage placement via their GrRectanizer. A BatchPlot
143 // manages the lifetime of its data using two tokens, a last use token and a last upload token.
144 // Once a BatchPlot is "full" (i.e. there is no room for the new subimage according to the
145 // GrRectanizer), it can no longer be used unless the last use of the GrPlot has already been
146 // flushed through to the gpu.
147 class BatchPlot : public SkRefCnt {
148 SK_DECLARE_INTERNAL_LLIST_INTERFACE(BatchPlot);
149
150 public:
151 // index() is a unique id for the plot relative to the owning GrAtlas. genID() is a
152 // monotonically incremented number which is bumped every time this plot is
153 // evicted from the cache (i.e., there is continuity in genID() across atlas spills).
154 uint32_t index() const { return fIndex; }
155 uint64_t genID() const { return fGenID; }
156 GrBatchAtlas::AtlasID id() const {
157 SkASSERT(GrBatchAtlas::kInvalidAtlasID != fID);
158 return fID;
159 }
160 SkDEBUGCODE(size_t bpp() const { return fBytesPerPixel; })
161
162 bool addSubImage(int width, int height, const void* image, SkIPoint16* loc);
163
164 // To manage the lifetime of a plot, we use two tokens. We use the last upload token to
165 // know when we can 'piggy back' uploads, ie if the last upload hasn't been flushed to gpu,
166 // we don't need to issue a new upload even if we update the cpu backing store. We use
167 // lastUse to determine when we can evict a plot from the cache, ie if the last use has
168 // already flushed through the gpu then we can reuse the plot.
Brian Salomon9afd3712016-12-01 10:59:09 -0500169 GrDrawOpUploadToken lastUploadToken() const { return fLastUpload; }
170 GrDrawOpUploadToken lastUseToken() const { return fLastUse; }
171 void setLastUploadToken(GrDrawOpUploadToken batchToken) { fLastUpload = batchToken; }
172 void setLastUseToken(GrDrawOpUploadToken batchToken) { fLastUse = batchToken; }
joshualitt5df175e2015-11-18 13:37:54 -0800173
Brian Salomon9afd3712016-12-01 10:59:09 -0500174 void uploadToTexture(GrDrawOp::WritePixelsFn&, GrTexture* texture);
joshualitt5df175e2015-11-18 13:37:54 -0800175 void resetRects();
176
177 private:
178 BatchPlot(int index, uint64_t genID, int offX, int offY, int width, int height,
jvanverthc3d706f2016-04-20 10:33:27 -0700179 GrPixelConfig config);
joshualitt5df175e2015-11-18 13:37:54 -0800180
181 ~BatchPlot() override;
182
183 // Create a clone of this plot. The cloned plot will take the place of the
184 // current plot in the atlas.
185 BatchPlot* clone() const {
jvanverthc3d706f2016-04-20 10:33:27 -0700186 return new BatchPlot(fIndex, fGenID+1, fX, fY, fWidth, fHeight, fConfig);
joshualitt5df175e2015-11-18 13:37:54 -0800187 }
188
189 static GrBatchAtlas::AtlasID CreateId(uint32_t index, uint64_t generation) {
190 SkASSERT(index < (1 << 16));
191 SkASSERT(generation < ((uint64_t)1 << 48));
192 return generation << 16 | index;
193 }
194
Brian Salomon9afd3712016-12-01 10:59:09 -0500195 GrDrawOpUploadToken fLastUpload;
196 GrDrawOpUploadToken fLastUse;
joshualitt5df175e2015-11-18 13:37:54 -0800197
198 const uint32_t fIndex;
199 uint64_t fGenID;
200 GrBatchAtlas::AtlasID fID;
jvanverthc3d706f2016-04-20 10:33:27 -0700201 unsigned char* fData;
joshualitt5df175e2015-11-18 13:37:54 -0800202 const int fWidth;
203 const int fHeight;
204 const int fX;
205 const int fY;
206 GrRectanizer* fRects;
207 const SkIPoint16 fOffset; // the offset of the plot in the backing texture
208 const GrPixelConfig fConfig;
209 const size_t fBytesPerPixel;
210 SkIRect fDirtyRect;
211 SkDEBUGCODE(bool fDirty;)
212
213 friend class GrBatchAtlas;
214
215 typedef SkRefCnt INHERITED;
216 };
217
robertphillips2b0536f2015-11-06 14:10:42 -0800218 typedef SkTInternalLList<BatchPlot> GrBatchPlotList;
219
joshualitt8db6fdc2015-07-31 08:25:07 -0700220 static uint32_t GetIndexFromID(AtlasID id) {
joshualitt5bf99f12015-03-13 11:47:42 -0700221 return id & 0xffff;
222 }
223
joshualitt8db6fdc2015-07-31 08:25:07 -0700224 // top 48 bits are reserved for the generation ID
225 static uint64_t GetGenerationFromID(AtlasID id) {
226 return (id >> 16) & 0xffffffffffff;
joshualitt5bf99f12015-03-13 11:47:42 -0700227 }
228
Brian Salomon9afd3712016-12-01 10:59:09 -0500229 inline void updatePlot(GrDrawOp::Target*, AtlasID*, BatchPlot*);
joshualitt5bf99f12015-03-13 11:47:42 -0700230
joshualitt5df175e2015-11-18 13:37:54 -0800231 inline void makeMRU(BatchPlot* plot) {
232 if (fPlotList.head() == plot) {
233 return;
234 }
235
236 fPlotList.remove(plot);
237 fPlotList.addToHead(plot);
238 }
joshualitt5bf99f12015-03-13 11:47:42 -0700239
240 inline void processEviction(AtlasID);
241
Ben Wagner594f9ed2016-11-08 14:13:39 -0500242 sk_sp<GrTexture> fTexture;
243 int fPlotWidth;
244 int fPlotHeight;
robertphillips2b0536f2015-11-06 14:10:42 -0800245 SkDEBUGCODE(uint32_t fNumPlots;)
246
joshualitt7c3a2f82015-03-31 13:32:05 -0700247 uint64_t fAtlasGeneration;
joshualitt5bf99f12015-03-13 11:47:42 -0700248
249 struct EvictionData {
250 EvictionFunc fFunc;
251 void* fData;
252 };
253
254 SkTDArray<EvictionData> fEvictionCallbacks;
255 // allocated array of GrBatchPlots
Ben Wagner594f9ed2016-11-08 14:13:39 -0500256 std::unique_ptr<sk_sp<BatchPlot>[]> fPlotArray;
joshualitt5bf99f12015-03-13 11:47:42 -0700257 // LRU list of GrPlots (MRU at head - LRU at tail)
258 GrBatchPlotList fPlotList;
259};
260
261#endif