/* * 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 "src/gpu/text/GrAtlasManager.h" #include "src/gpu/GrGlyph.h" #include "src/gpu/GrImageInfo.h" #include "src/gpu/text/GrStrikeCache.h" GrAtlasManager::GrAtlasManager(GrProxyProvider* proxyProvider, GrStrikeCache* glyphCache, size_t maxTextureBytes, GrDrawOpAtlas::AllowMultitexturing allowMultitexturing) : fAllowMultitexturing{allowMultitexturing} , fProxyProvider{proxyProvider} , fCaps{fProxyProvider->refCaps()} , fGlyphCache{glyphCache} , fAtlasConfig{fCaps->maxTextureSize(), maxTextureBytes} { } GrAtlasManager::~GrAtlasManager() = default; void GrAtlasManager::freeAll() { for (int i = 0; i < kMaskFormatCount; ++i) { fAtlases[i] = nullptr; } } bool GrAtlasManager::hasGlyph(GrGlyph* glyph) { SkASSERT(glyph); return this->getAtlas(glyph->fMaskFormat)->hasID(glyph->fPlotLocator); } // add to texture atlas that matches this format GrDrawOpAtlas::ErrorCode GrAtlasManager::addToAtlas( GrResourceProvider* resourceProvider, GrDrawOpAtlas::PlotLocator* plotLocator, GrDeferredUploadTarget* target, GrMaskFormat format, int width, int height, const void* image, SkIPoint16* loc) { return this->getAtlas(format)->addToAtlas( resourceProvider, plotLocator, target, width, height, image, loc); } void GrAtlasManager::addGlyphToBulkAndSetUseToken(GrDrawOpAtlas::BulkUseTokenUpdater* updater, GrGlyph* glyph, GrDeferredUploadToken token) { SkASSERT(glyph); if (updater->add(glyph->fPlotLocator)) { this->getAtlas(glyph->fMaskFormat)->setLastUseToken(glyph->fPlotLocator, token); } } #ifdef SK_DEBUG #include "src/gpu/GrContextPriv.h" #include "src/gpu/GrSurfaceContext.h" #include "src/gpu/GrSurfaceProxy.h" #include "src/gpu/GrTextureProxy.h" #include "include/core/SkBitmap.h" #include "include/core/SkImageEncoder.h" #include "include/core/SkStream.h" #include /** * Write the contents of the surface proxy to a PNG. Returns true if successful. * @param filename Full path to desired file */ static bool save_pixels(GrContext* context, GrSurfaceProxyView view, GrColorType colorType, const char* filename) { if (!view.proxy()) { return false; } SkImageInfo ii = SkImageInfo::Make(view.proxy()->dimensions(), kRGBA_8888_SkColorType, kPremul_SkAlphaType); SkBitmap bm; if (!bm.tryAllocPixels(ii)) { return false; } auto sContext = GrSurfaceContext::Make(context, std::move(view), colorType, kUnknown_SkAlphaType, nullptr); if (!sContext || !sContext->asTextureProxy()) { return false; } bool result = sContext->readPixels(ii, bm.getPixels(), bm.rowBytes(), {0, 0}); if (!result) { SkDebugf("------ failed to read pixels for %s\n", filename); return false; } // remove any previous version of this file remove(filename); SkFILEWStream file(filename); if (!file.isValid()) { SkDebugf("------ failed to create file: %s\n", filename); remove(filename); // remove any partial file return false; } if (!SkEncodeImage(&file, bm, SkEncodedImageFormat::kPNG, 100)) { SkDebugf("------ failed to encode %s\n", filename); remove(filename); // remove any partial file return false; } return true; } void GrAtlasManager::dump(GrContext* context) const { static int gDumpCount = 0; for (int i = 0; i < kMaskFormatCount; ++i) { if (fAtlases[i]) { const GrSurfaceProxyView* views = fAtlases[i]->getViews(); for (uint32_t pageIdx = 0; pageIdx < fAtlases[i]->numActivePages(); ++pageIdx) { SkASSERT(views[pageIdx].proxy()); SkString filename; #ifdef SK_BUILD_FOR_ANDROID filename.printf("/sdcard/fontcache_%d%d%d.png", gDumpCount, i, pageIdx); #else filename.printf("fontcache_%d%d%d.png", gDumpCount, i, pageIdx); #endif auto ct = GrMaskFormatToColorType(AtlasIndexToMaskFormat(i)); save_pixels(context, views[pageIdx], ct, filename.c_str()); } } } ++gDumpCount; } #endif void GrAtlasManager::setAtlasDimensionsToMinimum_ForTesting() { // Delete any old atlases. // This should be safe to do as long as we are not in the middle of a flush. for (int i = 0; i < kMaskFormatCount; i++) { fAtlases[i] = nullptr; } // Set all the atlas sizes to 1x1 plot each. new (&fAtlasConfig) GrDrawOpAtlasConfig{}; } bool GrAtlasManager::initAtlas(GrMaskFormat format) { int index = MaskFormatToAtlasIndex(format); if (fAtlases[index] == nullptr) { GrColorType grColorType = GrMaskFormatToColorType(format); SkISize atlasDimensions = fAtlasConfig.atlasDimensions(format); SkISize plotDimensions = fAtlasConfig.plotDimensions(format); const GrBackendFormat format = fCaps->getDefaultBackendFormat(grColorType, GrRenderable::kNo); fAtlases[index] = GrDrawOpAtlas::Make( fProxyProvider, format, grColorType, atlasDimensions.width(), atlasDimensions.height(), plotDimensions.width(), plotDimensions.height(), this, fAllowMultitexturing, nullptr); if (!fAtlases[index]) { return false; } } return true; }