blob: ce3e40aec4830a3a1deb3827ef3b17ac231d2249 [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
16class BatchPlot;
17class GrBatchTarget;
18class GrRectanizer;
19
20typedef SkTInternalLList<BatchPlot> GrBatchPlotList;
21
22class GrBatchAtlas {
23public:
24 typedef uint64_t BatchToken;
25 // An AtlasID is an opaque handle which callers can use to determine if the atlas contains
26 // a specific piece of data
joshualitt8db6fdc2015-07-31 08:25:07 -070027 typedef uint64_t AtlasID;
joshualitt7c3a2f82015-03-31 13:32:05 -070028 static const uint32_t kInvalidAtlasID = 0;
29 static const uint64_t kInvalidAtlasGeneration = 0;
joshualitt5bf99f12015-03-13 11:47:42 -070030
31 // A function pointer for use as a callback during eviction. Whenever GrBatchAtlas evicts a
32 // specific AtlasID, it will call all of the registered listeners so they can optionally process
33 // the eviction
34 typedef void (*EvictionFunc)(GrBatchAtlas::AtlasID, void*);
35
36 GrBatchAtlas(GrTexture*, int numPlotsX, int numPlotsY);
37 ~GrBatchAtlas();
38
39 // Adds a width x height subimage to the atlas. Upon success it returns
40 // the containing GrPlot and absolute location in the backing texture.
41 // NULL is returned if the subimage cannot fit in the atlas.
42 // If provided, the image data will be written to the CPU-side backing bitmap.
joshualittb4c507e2015-04-08 08:07:59 -070043 // NOTE: If the client intends to refer to the atlas, they should immediately call 'setUseToken'
44 // with the currentToken from the batch target, otherwise the next call to addToAtlas might
45 // cause an eviction
joshualitt5bf99f12015-03-13 11:47:42 -070046 bool addToAtlas(AtlasID*, GrBatchTarget*, int width, int height, const void* image,
47 SkIPoint16* loc);
48
49 GrTexture* getTexture() const { return fTexture; }
50
joshualitt7c3a2f82015-03-31 13:32:05 -070051 uint64_t atlasGeneration() const { return fAtlasGeneration; }
joshualitt5bf99f12015-03-13 11:47:42 -070052 bool hasID(AtlasID id);
joshualittb4c507e2015-04-08 08:07:59 -070053
54 // To ensure the atlas does not evict a given entry, the client must set the last use token
55 void setLastUseToken(AtlasID id, BatchToken batchToken);
joshualitt5bf99f12015-03-13 11:47:42 -070056 void registerEvictionCallback(EvictionFunc func, void* userData) {
57 EvictionData* data = fEvictionCallbacks.append();
58 data->fFunc = func;
59 data->fData = userData;
60 }
61
joshualittb4c507e2015-04-08 08:07:59 -070062 /*
63 * A class which can be handed back to GrBatchAtlas for updating in bulk last use tokens. The
64 * current max number of plots the GrBatchAtlas can handle is 32, if in the future this is
65 * insufficient then we can move to a 64 bit int
66 */
67 class BulkUseTokenUpdater {
68 public:
joshualitt4314e082015-04-23 08:03:35 -070069 BulkUseTokenUpdater() : fPlotAlreadyUpdated(0) {}
joshualittb4c507e2015-04-08 08:07:59 -070070 void add(AtlasID id) {
71 int index = GrBatchAtlas::GetIndexFromID(id);
72 if (!this->find(index)) {
73 this->set(index);
74 }
75 }
76
77 void reset() {
joshualitt4314e082015-04-23 08:03:35 -070078 fPlotsToUpdate.reset();
joshualittb4c507e2015-04-08 08:07:59 -070079 fPlotAlreadyUpdated = 0;
80 }
81
82 private:
83 bool find(int index) const {
84 SkASSERT(index < kMaxPlots);
85 return (fPlotAlreadyUpdated >> index) & 1;
86 }
87
88 void set(int index) {
89 SkASSERT(!this->find(index));
90 fPlotAlreadyUpdated = fPlotAlreadyUpdated | (1 << index);
joshualitt97202d22015-04-22 13:47:02 -070091 fPlotsToUpdate.push_back(index);
joshualittb4c507e2015-04-08 08:07:59 -070092 }
93
94 static const int kMinItems = 4;
95 static const int kMaxPlots = 32;
joshualitt97202d22015-04-22 13:47:02 -070096 SkSTArray<kMinItems, int, true> fPlotsToUpdate;
joshualitt8672f4d2015-04-21 08:03:04 -070097 uint32_t fPlotAlreadyUpdated;
joshualittb4c507e2015-04-08 08:07:59 -070098
99 friend class GrBatchAtlas;
100 };
101
102 void setLastUseTokenBulk(const BulkUseTokenUpdater& reffer, BatchToken);
103
joshualitt010db532015-04-21 10:07:26 -0700104 static const int kGlyphMaxDim = 256;
105 static bool GlyphTooLargeForAtlas(int width, int height) {
106 return width > kGlyphMaxDim || height > kGlyphMaxDim;
107 }
108
joshualitt5bf99f12015-03-13 11:47:42 -0700109private:
joshualitt8db6fdc2015-07-31 08:25:07 -0700110 static uint32_t GetIndexFromID(AtlasID id) {
joshualitt5bf99f12015-03-13 11:47:42 -0700111 return id & 0xffff;
112 }
113
joshualitt8db6fdc2015-07-31 08:25:07 -0700114 // top 48 bits are reserved for the generation ID
115 static uint64_t GetGenerationFromID(AtlasID id) {
116 return (id >> 16) & 0xffffffffffff;
joshualitt5bf99f12015-03-13 11:47:42 -0700117 }
118
119 inline void updatePlot(GrBatchTarget*, AtlasID*, BatchPlot*);
120
121 inline void makeMRU(BatchPlot* plot);
122
123 inline void processEviction(AtlasID);
124
125 GrTexture* fTexture;
joshualitt8db6fdc2015-07-31 08:25:07 -0700126 uint32_t fNumPlotsX;
127 uint32_t fNumPlotsY;
128 uint32_t fPlotWidth;
129 uint32_t fPlotHeight;
joshualitt5bf99f12015-03-13 11:47:42 -0700130 size_t fBPP;
joshualitt7c3a2f82015-03-31 13:32:05 -0700131 uint64_t fAtlasGeneration;
joshualitt5bf99f12015-03-13 11:47:42 -0700132
133 struct EvictionData {
134 EvictionFunc fFunc;
135 void* fData;
136 };
137
138 SkTDArray<EvictionData> fEvictionCallbacks;
139 // allocated array of GrBatchPlots
140 SkAutoTUnref<BatchPlot>* fPlotArray;
141 // LRU list of GrPlots (MRU at head - LRU at tail)
142 GrBatchPlotList fPlotList;
143};
144
145#endif