2018-03-01 15:24:02 +00:00
|
|
|
/*
|
|
|
|
* Copyright 2018 Google Inc.
|
|
|
|
*
|
|
|
|
* Use of this source code is governed by a BSD-style license that can be
|
|
|
|
* found in the LICENSE file.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "SkTypes.h"
|
|
|
|
|
|
|
|
#if SK_SUPPORT_GPU
|
|
|
|
|
|
|
|
#include "GrContextPriv.h"
|
|
|
|
#include "Test.h"
|
2018-03-01 16:36:45 +00:00
|
|
|
#include "text/GrGlyphCache.h"
|
2018-03-01 15:24:02 +00:00
|
|
|
|
|
|
|
static const int kNumPlots = 2;
|
|
|
|
static const int kPlotSize = 32;
|
|
|
|
static const int kAtlasSize = kNumPlots * kPlotSize;
|
|
|
|
|
|
|
|
int GrDrawOpAtlas::numAllocated_TestingOnly() const {
|
|
|
|
int count = 0;
|
|
|
|
for (uint32_t i = 0; i < this->maxPages(); ++i) {
|
|
|
|
if (fProxies[i]->priv().isInstantiated()) {
|
|
|
|
++count;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return count;
|
|
|
|
}
|
|
|
|
|
|
|
|
void EvictionFunc(GrDrawOpAtlas::AtlasID atlasID, void*) {
|
|
|
|
SkASSERT(0); // The unit test shouldn't exercise this code path
|
|
|
|
}
|
|
|
|
|
|
|
|
static void check(skiatest::Reporter* r, GrDrawOpAtlas* atlas,
|
|
|
|
uint32_t expectedActive, uint32_t expectedMax, int expectedAlloced) {
|
|
|
|
REPORTER_ASSERT(r, expectedActive == atlas->numActivePages());
|
|
|
|
REPORTER_ASSERT(r, expectedMax == atlas->maxPages());
|
|
|
|
REPORTER_ASSERT(r, expectedAlloced == atlas->numAllocated_TestingOnly());
|
|
|
|
}
|
|
|
|
|
|
|
|
class TestingUploadTarget : public GrDeferredUploadTarget {
|
|
|
|
public:
|
|
|
|
TestingUploadTarget() { }
|
|
|
|
|
|
|
|
const GrTokenTracker* tokenTracker() final {
|
|
|
|
return &fTokenTracker;
|
|
|
|
}
|
|
|
|
|
|
|
|
GrDeferredUploadToken addInlineUpload(GrDeferredTextureUploadFn&&) final {
|
|
|
|
SkASSERT(0); // this test shouldn't invoke this code path
|
|
|
|
return fTokenTracker.nextDrawToken();
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual GrDeferredUploadToken addASAPUpload(GrDeferredTextureUploadFn&& upload) final {
|
|
|
|
return fTokenTracker.nextTokenToFlush();
|
|
|
|
}
|
|
|
|
|
|
|
|
void issueDrawToken() { fTokenTracker.issueDrawToken(); }
|
|
|
|
void flushToken() { fTokenTracker.flushToken(); }
|
|
|
|
|
|
|
|
private:
|
|
|
|
GrTokenTracker fTokenTracker;
|
|
|
|
|
|
|
|
typedef GrDeferredUploadTarget INHERITED;
|
|
|
|
};
|
|
|
|
|
|
|
|
static bool fill_plot(GrDrawOpAtlas* atlas,
|
|
|
|
GrResourceProvider* resourceProvider,
|
|
|
|
GrDeferredUploadTarget* target,
|
|
|
|
GrDrawOpAtlas::AtlasID* atlasID,
|
|
|
|
int alpha) {
|
|
|
|
SkImageInfo ii = SkImageInfo::MakeA8(kPlotSize, kPlotSize);
|
|
|
|
|
|
|
|
SkBitmap data;
|
|
|
|
data.allocPixels(ii);
|
|
|
|
data.eraseARGB(alpha, 0, 0, 0);
|
|
|
|
|
|
|
|
SkIPoint16 loc;
|
|
|
|
bool result = atlas->addToAtlas(resourceProvider, atlasID, target, kPlotSize, kPlotSize,
|
|
|
|
data.getAddr(0, 0), &loc);
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
DEF_GPUTEST_FOR_RENDERING_CONTEXTS(DrawOpAtlas, reporter, ctxInfo) {
|
|
|
|
auto context = ctxInfo.grContext();
|
|
|
|
auto proxyProvider = context->contextPriv().proxyProvider();
|
|
|
|
auto resourceProvider = context->contextPriv().resourceProvider();
|
|
|
|
auto drawingManager = context->contextPriv().drawingManager();
|
|
|
|
|
|
|
|
GrOnFlushResourceProvider onFlushResourceProvider(drawingManager);
|
|
|
|
TestingUploadTarget uploadTarget;
|
|
|
|
|
|
|
|
std::unique_ptr<GrDrawOpAtlas> atlas = GrDrawOpAtlas::Make(
|
|
|
|
proxyProvider,
|
|
|
|
kAlpha_8_GrPixelConfig,
|
|
|
|
kAtlasSize, kAtlasSize,
|
|
|
|
kNumPlots, kNumPlots,
|
|
|
|
GrDrawOpAtlas::AllowMultitexturing::kYes,
|
|
|
|
EvictionFunc, nullptr);
|
|
|
|
check(reporter, atlas.get(), 0, 4, 0);
|
|
|
|
|
|
|
|
// Fill up the first level
|
|
|
|
GrDrawOpAtlas::AtlasID atlasIDs[kNumPlots * kNumPlots];
|
|
|
|
for (int i = 0; i < kNumPlots * kNumPlots; ++i) {
|
|
|
|
bool result = fill_plot(atlas.get(), resourceProvider, &uploadTarget, &atlasIDs[i], i*32);
|
|
|
|
REPORTER_ASSERT(reporter, result);
|
|
|
|
check(reporter, atlas.get(), 1, 4, 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
atlas->instantiate(&onFlushResourceProvider);
|
|
|
|
check(reporter, atlas.get(), 1, 4, 1);
|
|
|
|
|
|
|
|
// Force allocation of a second level
|
|
|
|
GrDrawOpAtlas::AtlasID atlasID;
|
|
|
|
bool result = fill_plot(atlas.get(), resourceProvider, &uploadTarget, &atlasID, 4*32);
|
|
|
|
REPORTER_ASSERT(reporter, result);
|
|
|
|
check(reporter, atlas.get(), 2, 4, 2);
|
|
|
|
|
|
|
|
// Simulate a lot of draws using only the first plot. The last texture should be compacted.
|
|
|
|
for (int i = 0; i < 512; ++i) {
|
|
|
|
atlas->setLastUseToken(atlasIDs[0], uploadTarget.tokenTracker()->nextDrawToken());
|
|
|
|
uploadTarget.issueDrawToken();
|
|
|
|
uploadTarget.flushToken();
|
|
|
|
atlas->compact(uploadTarget.tokenTracker()->nextTokenToFlush());
|
|
|
|
}
|
|
|
|
|
|
|
|
check(reporter, atlas.get(), 1, 4, 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif
|