| 1 | /* | 
|---|
| 2 | * Copyright 2020 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 | #include "src/gpu/GrDynamicAtlas.h" | 
|---|
| 9 |  | 
|---|
| 10 | #include "src/core/SkIPoint16.h" | 
|---|
| 11 | #include "src/gpu/GrOnFlushResourceProvider.h" | 
|---|
| 12 | #include "src/gpu/GrProxyProvider.h" | 
|---|
| 13 | #include "src/gpu/GrRectanizerPow2.h" | 
|---|
| 14 | #include "src/gpu/GrRectanizerSkyline.h" | 
|---|
| 15 | #include "src/gpu/GrRenderTarget.h" | 
|---|
| 16 | #include "src/gpu/GrRenderTargetContextPriv.h" | 
|---|
| 17 |  | 
|---|
| 18 | // Each Node covers a sub-rectangle of the final atlas. When a GrDynamicAtlas runs out of room, we | 
|---|
| 19 | // create a new Node the same size as all combined nodes in the atlas as-is, and then place the new | 
|---|
| 20 | // Node immediately below or beside the others (thereby doubling the size of the GyDynamicAtlas). | 
|---|
| 21 | class GrDynamicAtlas::Node { | 
|---|
| 22 | public: | 
|---|
| 23 | Node(Node* previous, GrRectanizer* rectanizer, int x, int y) | 
|---|
| 24 | : fPrevious(previous), fRectanizer(rectanizer), fX(x), fY(y) {} | 
|---|
| 25 |  | 
|---|
| 26 | Node* previous() const { return fPrevious; } | 
|---|
| 27 |  | 
|---|
| 28 | bool addRect(int w, int h, SkIPoint16* loc) { | 
|---|
| 29 | // Pad all paths except those that are expected to take up an entire physical texture. | 
|---|
| 30 | if (w < fRectanizer->width()) { | 
|---|
| 31 | w = std::min(w + kPadding, fRectanizer->width()); | 
|---|
| 32 | } | 
|---|
| 33 | if (h < fRectanizer->height()) { | 
|---|
| 34 | h = std::min(h + kPadding, fRectanizer->height()); | 
|---|
| 35 | } | 
|---|
| 36 | if (!fRectanizer->addRect(w, h, loc)) { | 
|---|
| 37 | return false; | 
|---|
| 38 | } | 
|---|
| 39 | loc->fX += fX; | 
|---|
| 40 | loc->fY += fY; | 
|---|
| 41 | return true; | 
|---|
| 42 | } | 
|---|
| 43 |  | 
|---|
| 44 | private: | 
|---|
| 45 | Node* const fPrevious; | 
|---|
| 46 | GrRectanizer* const fRectanizer; | 
|---|
| 47 | const int fX, fY; | 
|---|
| 48 | }; | 
|---|
| 49 |  | 
|---|
| 50 | sk_sp<GrTextureProxy> GrDynamicAtlas::MakeLazyAtlasProxy( | 
|---|
| 51 | LazyInstantiateAtlasCallback&& callback, | 
|---|
| 52 | GrColorType colorType, | 
|---|
| 53 | InternalMultisample internalMultisample, | 
|---|
| 54 | const GrCaps& caps, | 
|---|
| 55 | GrSurfaceProxy::UseAllocator useAllocator) { | 
|---|
| 56 | GrBackendFormat format = caps.getDefaultBackendFormat(colorType, GrRenderable::kYes); | 
|---|
| 57 |  | 
|---|
| 58 | int sampleCount = 1; | 
|---|
| 59 | if (!caps.mixedSamplesSupport() && InternalMultisample::kYes == internalMultisample) { | 
|---|
| 60 | sampleCount = caps.internalMultisampleCount(format); | 
|---|
| 61 | } | 
|---|
| 62 |  | 
|---|
| 63 | sk_sp<GrTextureProxy> proxy = | 
|---|
| 64 | GrProxyProvider::MakeFullyLazyProxy(std::move(callback), format, GrRenderable::kYes, | 
|---|
| 65 | sampleCount, GrProtected::kNo, caps, useAllocator); | 
|---|
| 66 |  | 
|---|
| 67 | return proxy; | 
|---|
| 68 | } | 
|---|
| 69 |  | 
|---|
| 70 | GrDynamicAtlas::GrDynamicAtlas(GrColorType colorType, InternalMultisample internalMultisample, | 
|---|
| 71 | SkISize initialSize, int maxAtlasSize, const GrCaps& caps, | 
|---|
| 72 | RectanizerAlgorithm algorithm) | 
|---|
| 73 | : fColorType(colorType) | 
|---|
| 74 | , fInternalMultisample(internalMultisample) | 
|---|
| 75 | , fMaxAtlasSize(maxAtlasSize) | 
|---|
| 76 | , fRectanizerAlgorithm(algorithm) { | 
|---|
| 77 | SkASSERT(fMaxAtlasSize <= caps.maxTextureSize()); | 
|---|
| 78 | this->reset(initialSize, caps); | 
|---|
| 79 | } | 
|---|
| 80 |  | 
|---|
| 81 | GrDynamicAtlas::~GrDynamicAtlas() { | 
|---|
| 82 | } | 
|---|
| 83 |  | 
|---|
| 84 | void GrDynamicAtlas::reset(SkISize initialSize, const GrCaps& caps) { | 
|---|
| 85 | fNodeAllocator.reset(); | 
|---|
| 86 | fWidth = std::min(SkNextPow2(initialSize.width()), fMaxAtlasSize); | 
|---|
| 87 | fHeight = std::min(SkNextPow2(initialSize.height()), fMaxAtlasSize); | 
|---|
| 88 | fTopNode = nullptr; | 
|---|
| 89 | fDrawBounds.setEmpty(); | 
|---|
| 90 | fTextureProxy = MakeLazyAtlasProxy( | 
|---|
| 91 | [this](GrResourceProvider* resourceProvider, const LazyAtlasDesc& desc) { | 
|---|
| 92 | if (!fBackingTexture) { | 
|---|
| 93 | fBackingTexture = resourceProvider->createTexture( | 
|---|
| 94 | {fWidth, fHeight}, desc.fFormat, desc.fRenderable, desc.fSampleCnt, | 
|---|
| 95 | desc.fMipmapped, desc.fBudgeted, desc.fProtected); | 
|---|
| 96 | } | 
|---|
| 97 | return GrSurfaceProxy::LazyCallbackResult(fBackingTexture); | 
|---|
| 98 | }, | 
|---|
| 99 | fColorType, fInternalMultisample, caps, GrSurfaceProxy::UseAllocator::kNo); | 
|---|
| 100 | fBackingTexture = nullptr; | 
|---|
| 101 | } | 
|---|
| 102 |  | 
|---|
| 103 | GrDynamicAtlas::Node* GrDynamicAtlas::makeNode(Node* previous, int l, int t, int r, int b) { | 
|---|
| 104 | int width = r - l; | 
|---|
| 105 | int height = b - t; | 
|---|
| 106 | GrRectanizer* rectanizer = (fRectanizerAlgorithm == RectanizerAlgorithm::kSkyline) | 
|---|
| 107 | ? (GrRectanizer*)fNodeAllocator.make<GrRectanizerSkyline>(width, height) | 
|---|
| 108 | : fNodeAllocator.make<GrRectanizerPow2>(width, height); | 
|---|
| 109 | return fNodeAllocator.make<Node>(previous, rectanizer, l, t); | 
|---|
| 110 | } | 
|---|
| 111 |  | 
|---|
| 112 | bool GrDynamicAtlas::addRect(int width, int height, SkIPoint16* location) { | 
|---|
| 113 | // This can't be called anymore once instantiate() has been called. | 
|---|
| 114 | SkASSERT(!this->isInstantiated()); | 
|---|
| 115 |  | 
|---|
| 116 | if (!this->internalPlaceRect(width, height, location)) { | 
|---|
| 117 | return false; | 
|---|
| 118 | } | 
|---|
| 119 |  | 
|---|
| 120 | fDrawBounds.fWidth = std::max(fDrawBounds.width(), location->x() + width); | 
|---|
| 121 | fDrawBounds.fHeight = std::max(fDrawBounds.height(), location->y() + height); | 
|---|
| 122 | return true; | 
|---|
| 123 | } | 
|---|
| 124 |  | 
|---|
| 125 | bool GrDynamicAtlas::internalPlaceRect(int w, int h, SkIPoint16* loc) { | 
|---|
| 126 | if (std::max(h, w) > fMaxAtlasSize) { | 
|---|
| 127 | return false; | 
|---|
| 128 | } | 
|---|
| 129 | if (std::min(h, w) <= 0) { | 
|---|
| 130 | loc->set(0, 0); | 
|---|
| 131 | return true; | 
|---|
| 132 | } | 
|---|
| 133 |  | 
|---|
| 134 | if (!fTopNode) { | 
|---|
| 135 | if (w > fWidth) { | 
|---|
| 136 | fWidth = std::min(SkNextPow2(w), fMaxAtlasSize); | 
|---|
| 137 | } | 
|---|
| 138 | if (h > fHeight) { | 
|---|
| 139 | fHeight = std::min(SkNextPow2(h), fMaxAtlasSize); | 
|---|
| 140 | } | 
|---|
| 141 | fTopNode = this->makeNode(nullptr, 0, 0, fWidth, fHeight); | 
|---|
| 142 | } | 
|---|
| 143 |  | 
|---|
| 144 | for (Node* node = fTopNode; node; node = node->previous()) { | 
|---|
| 145 | if (node->addRect(w, h, loc)) { | 
|---|
| 146 | return true; | 
|---|
| 147 | } | 
|---|
| 148 | } | 
|---|
| 149 |  | 
|---|
| 150 | // The rect didn't fit. Grow the atlas and try again. | 
|---|
| 151 | do { | 
|---|
| 152 | if (fWidth >= fMaxAtlasSize && fHeight >= fMaxAtlasSize) { | 
|---|
| 153 | return false; | 
|---|
| 154 | } | 
|---|
| 155 | if (fHeight <= fWidth) { | 
|---|
| 156 | int top = fHeight; | 
|---|
| 157 | fHeight = std::min(fHeight * 2, fMaxAtlasSize); | 
|---|
| 158 | fTopNode = this->makeNode(fTopNode, 0, top, fWidth, fHeight); | 
|---|
| 159 | } else { | 
|---|
| 160 | int left = fWidth; | 
|---|
| 161 | fWidth = std::min(fWidth * 2, fMaxAtlasSize); | 
|---|
| 162 | fTopNode = this->makeNode(fTopNode, left, 0, fWidth, fHeight); | 
|---|
| 163 | } | 
|---|
| 164 | } while (!fTopNode->addRect(w, h, loc)); | 
|---|
| 165 |  | 
|---|
| 166 | return true; | 
|---|
| 167 | } | 
|---|
| 168 |  | 
|---|
| 169 | std::unique_ptr<GrRenderTargetContext> GrDynamicAtlas::instantiate( | 
|---|
| 170 | GrOnFlushResourceProvider* onFlushRP, sk_sp<GrTexture> backingTexture) { | 
|---|
| 171 | SkASSERT(!this->isInstantiated());  // This method should only be called once. | 
|---|
| 172 | // Caller should have cropped any paths to the destination render target instead of asking for | 
|---|
| 173 | // an atlas larger than maxRenderTargetSize. | 
|---|
| 174 | SkASSERT(std::max(fHeight, fWidth) <= fMaxAtlasSize); | 
|---|
| 175 | SkASSERT(fMaxAtlasSize <= onFlushRP->caps()->maxRenderTargetSize()); | 
|---|
| 176 |  | 
|---|
| 177 | // Finalize the content size of our proxy. The GPU can potentially make optimizations if it | 
|---|
| 178 | // knows we only intend to write out a smaller sub-rectangle of the backing texture. | 
|---|
| 179 | fTextureProxy->priv().setLazyDimensions(fDrawBounds); | 
|---|
| 180 |  | 
|---|
| 181 | if (backingTexture) { | 
|---|
| 182 | #ifdef SK_DEBUG | 
|---|
| 183 | auto backingRT = backingTexture->asRenderTarget(); | 
|---|
| 184 | SkASSERT(backingRT); | 
|---|
| 185 | SkASSERT(backingRT->backendFormat() == fTextureProxy->backendFormat()); | 
|---|
| 186 | SkASSERT(backingRT->numSamples() == fTextureProxy->asRenderTargetProxy()->numSamples()); | 
|---|
| 187 | SkASSERT(backingRT->width() == fWidth); | 
|---|
| 188 | SkASSERT(backingRT->height() == fHeight); | 
|---|
| 189 | #endif | 
|---|
| 190 | fBackingTexture = std::move(backingTexture); | 
|---|
| 191 | } | 
|---|
| 192 | auto rtc = onFlushRP->makeRenderTargetContext(fTextureProxy, kTextureOrigin, fColorType, | 
|---|
| 193 | nullptr, nullptr); | 
|---|
| 194 | if (!rtc) { | 
|---|
| 195 | onFlushRP->printWarningMessage(SkStringPrintf( | 
|---|
| 196 | "WARNING: failed to allocate a %ix%i atlas. Some masks will not be drawn.\n", | 
|---|
| 197 | fWidth, fHeight).c_str()); | 
|---|
| 198 | return nullptr; | 
|---|
| 199 | } | 
|---|
| 200 |  | 
|---|
| 201 | SkIRect clearRect = SkIRect::MakeSize(fDrawBounds); | 
|---|
| 202 | rtc->priv().clearAtLeast(clearRect, SK_PMColor4fTRANSPARENT); | 
|---|
| 203 | return rtc; | 
|---|
| 204 | } | 
|---|
| 205 |  | 
|---|