blob: 95e4e7ac98d5a6f418220f959ce628c1841d6c17 [file] [log] [blame]
/*
* Copyright 2014 Google Inc.
*
* Use of this source code is governed by a BSD-style license that can be
* found in the LICENSE file.
*/
#include "GrAADistanceFieldPathRenderer.h"
#include "GrAtlas.h"
#include "GrContext.h"
#include "GrDrawState.h"
#include "GrSurfacePriv.h"
#include "GrSWMaskHelper.h"
#include "GrTexturePriv.h"
#include "effects/GrDistanceFieldTextureEffect.h"
#include "SkDistanceFieldGen.h"
#include "SkRTConf.h"
#define ATLAS_TEXTURE_WIDTH 1024
#define ATLAS_TEXTURE_HEIGHT 1024
#define PLOT_WIDTH 256
#define PLOT_HEIGHT 256
#define NUM_PLOTS_X (ATLAS_TEXTURE_WIDTH / PLOT_WIDTH)
#define NUM_PLOTS_Y (ATLAS_TEXTURE_HEIGHT / PLOT_HEIGHT)
SK_CONF_DECLARE(bool, c_DumpPathCache, "gpu.dumpPathCache", false,
"Dump the contents of the path cache before every purge.");
////////////////////////////////////////////////////////////////////////////////
GrAADistanceFieldPathRenderer::~GrAADistanceFieldPathRenderer() {
PathDataList::Iter iter;
iter.init(fPathList, PathDataList::Iter::kHead_IterStart);
PathData* pathData;
while ((pathData = iter.get())) {
iter.next();
fPathList.remove(pathData);
SkDELETE(pathData);
}
SkDELETE(fAtlas);
}
////////////////////////////////////////////////////////////////////////////////
bool GrAADistanceFieldPathRenderer::canDrawPath(const SkPath& path,
const SkStrokeRec& stroke,
const GrDrawTarget* target,
bool antiAlias) const {
// TODO: Support inverse fill
// TODO: Support strokes
if (!target->caps()->shaderDerivativeSupport() || !antiAlias || path.isInverseFillType()
|| SkStrokeRec::kFill_Style != stroke.getStyle()) {
return false;
}
// currently don't support perspective or scaling more than 3x
const GrDrawState& drawState = target->getDrawState();
const SkMatrix& vm = drawState.getViewMatrix();
if (vm.hasPerspective() || vm.getMaxScale() > 3.0f) {
return false;
}
// only support paths smaller than 64 x 64
const SkRect& bounds = path.getBounds();
return bounds.width() < 64.f && bounds.height() < 64.f;
}
GrPathRenderer::StencilSupport GrAADistanceFieldPathRenderer::onGetStencilSupport(
const SkPath&,
const SkStrokeRec&,
const GrDrawTarget*) const {
return GrPathRenderer::kNoSupport_StencilSupport;
}
////////////////////////////////////////////////////////////////////////////////
// position + texture coord
extern const GrVertexAttrib gSDFPathVertexAttribs[] = {
{ kVec2f_GrVertexAttribType, 0, kPosition_GrVertexAttribBinding },
{ kVec2f_GrVertexAttribType, sizeof(SkPoint), kGeometryProcessor_GrVertexAttribBinding }
};
static const size_t kSDFPathVASize = 2 * sizeof(SkPoint);
bool GrAADistanceFieldPathRenderer::onDrawPath(const SkPath& path,
const SkStrokeRec& stroke,
GrDrawTarget* target,
bool antiAlias) {
// we've already bailed on inverse filled paths, so this is safe
if (path.isEmpty()) {
return true;
}
SkASSERT(fContext);
// check to see if path is cached
// TODO: handle stroked vs. filled version of same path
PathData* pathData = fPathCache.find(path.getGenerationID());
if (NULL == pathData) {
pathData = this->addPathToAtlas(path, stroke, antiAlias);
if (NULL == pathData) {
return false;
}
}
// use signed distance field to render
return this->internalDrawPath(path, pathData, target);
}
// factor used to scale the path prior to building distance field
const SkScalar kScaleFactor = 2.0f;
// padding around path bounds to allow for antialiased pixels
const SkScalar kAntiAliasPad = 1.0f;
GrAADistanceFieldPathRenderer::PathData* GrAADistanceFieldPathRenderer::addPathToAtlas(
const SkPath& path,
const SkStrokeRec& stroke,
bool antiAlias) {
// generate distance field and add to atlas
if (NULL == fAtlas) {
SkISize textureSize = SkISize::Make(ATLAS_TEXTURE_WIDTH, ATLAS_TEXTURE_HEIGHT);
fAtlas = SkNEW_ARGS(GrAtlas, (fContext->getGpu(), kAlpha_8_GrPixelConfig,
kNone_GrTextureFlags, textureSize,
NUM_PLOTS_X, NUM_PLOTS_Y, false));
if (NULL == fAtlas) {
return NULL;
}
}
const SkRect& bounds = path.getBounds();
// generate bounding rect for bitmap draw
SkRect scaledBounds = bounds;
// scale up to improve maxification range
scaledBounds.fLeft *= kScaleFactor;
scaledBounds.fTop *= kScaleFactor;
scaledBounds.fRight *= kScaleFactor;
scaledBounds.fBottom *= kScaleFactor;
// move the origin to an integer boundary (gives better results)
SkScalar dx = SkScalarFraction(scaledBounds.fLeft);
SkScalar dy = SkScalarFraction(scaledBounds.fTop);
scaledBounds.offset(-dx, -dy);
// get integer boundary
SkIRect devPathBounds;
scaledBounds.roundOut(&devPathBounds);
// pad to allow room for antialiasing
devPathBounds.outset(SkScalarCeilToInt(kAntiAliasPad), SkScalarCeilToInt(kAntiAliasPad));
// move origin to upper left corner
devPathBounds.offsetTo(0,0);
// draw path to bitmap
SkMatrix drawMatrix;
drawMatrix.setTranslate(-bounds.left(), -bounds.top());
drawMatrix.postScale(kScaleFactor, kScaleFactor);
drawMatrix.postTranslate(kAntiAliasPad, kAntiAliasPad);
GrSWMaskHelper helper(fContext);
if (!helper.init(devPathBounds, &drawMatrix)) {
return NULL;
}
helper.draw(path, stroke, SkRegion::kReplace_Op, antiAlias, 0xFF);
// generate signed distance field
devPathBounds.outset(SK_DistanceFieldPad, SK_DistanceFieldPad);
int width = devPathBounds.width();
int height = devPathBounds.height();
SkAutoSMalloc<1024> dfStorage(width*height*sizeof(unsigned char));
helper.toSDF((unsigned char*) dfStorage.get());
// add to atlas
SkIPoint16 atlasLocation;
GrPlot* plot = fAtlas->addToAtlas(&fPlotUsage, width, height, dfStorage.get(),
&atlasLocation);
// if atlas full
if (NULL == plot) {
if (this->freeUnusedPlot()) {
plot = fAtlas->addToAtlas(&fPlotUsage, width, height, dfStorage.get(),
&atlasLocation);
if (plot) {
goto HAS_ATLAS;
}
}
if (c_DumpPathCache) {
#ifdef SK_DEVELOPER
GrTexture* texture = fAtlas->getTexture();
texture->surfacePriv().savePixels("pathcache.png");
#endif
}
// before we purge the cache, we must flush any accumulated draws
fContext->flush();
if (this->freeUnusedPlot()) {
plot = fAtlas->addToAtlas(&fPlotUsage, width, height, dfStorage.get(),
&atlasLocation);
if (plot) {
goto HAS_ATLAS;
}
}
return NULL;
}
HAS_ATLAS:
// add to cache
PathData* pathData = SkNEW(PathData);
pathData->fGenID = path.getGenerationID();
pathData->fPlot = plot;
// change the scaled rect to match the size of the inset distance field
scaledBounds.fRight = scaledBounds.fLeft +
SkIntToScalar(devPathBounds.width() - 2*SK_DistanceFieldInset);
scaledBounds.fBottom = scaledBounds.fTop +
SkIntToScalar(devPathBounds.height() - 2*SK_DistanceFieldInset);
// shift the origin to the correct place relative to the distance field
// need to also restore the fractional translation
scaledBounds.offset(-SkIntToScalar(SK_DistanceFieldInset) - kAntiAliasPad + dx,
-SkIntToScalar(SK_DistanceFieldInset) - kAntiAliasPad + dy);
pathData->fBounds = scaledBounds;
// origin we render from is inset from distance field edge
atlasLocation.fX += SK_DistanceFieldInset;
atlasLocation.fY += SK_DistanceFieldInset;
pathData->fAtlasLocation = atlasLocation;
fPathCache.add(pathData);
fPathList.addToTail(pathData);
return pathData;
}
bool GrAADistanceFieldPathRenderer::freeUnusedPlot() {
// find an unused plot
GrPlot* plot = fAtlas->getUnusedPlot();
if (NULL == plot) {
return false;
}
plot->resetRects();
// remove any paths that use this plot
PathDataList::Iter iter;
iter.init(fPathList, PathDataList::Iter::kHead_IterStart);
PathData* pathData;
while ((pathData = iter.get())) {
iter.next();
if (plot == pathData->fPlot) {
fPathCache.remove(pathData->fGenID);
fPathList.remove(pathData);
SkDELETE(pathData);
}
}
// tell the atlas to free the plot
GrAtlas::RemovePlot(&fPlotUsage, plot);
return true;
}
bool GrAADistanceFieldPathRenderer::internalDrawPath(const SkPath& path,
const PathData* pathData,
GrDrawTarget* target) {
GrTexture* texture = fAtlas->getTexture();
GrDrawState* drawState = target->drawState();
GrDrawState::AutoRestoreEffects are(drawState);
SkASSERT(pathData->fPlot);
GrDrawTarget::DrawToken drawToken = target->getCurrentDrawToken();
pathData->fPlot->setDrawToken(drawToken);
// make me some vertices
drawState->setVertexAttribs<gSDFPathVertexAttribs>(SK_ARRAY_COUNT(gSDFPathVertexAttribs),
kSDFPathVASize);
void* vertices = NULL;
void* indices = NULL;
bool success = target->reserveVertexAndIndexSpace(4, 6, &vertices, &indices);
GrAlwaysAssert(success);
SkScalar dx = pathData->fBounds.fLeft;
SkScalar dy = pathData->fBounds.fTop;
SkScalar width = pathData->fBounds.width();
SkScalar height = pathData->fBounds.height();
SkScalar invScale = 1.0f/kScaleFactor;
dx *= invScale;
dy *= invScale;
width *= invScale;
height *= invScale;
SkFixed tx = SkIntToFixed(pathData->fAtlasLocation.fX);
SkFixed ty = SkIntToFixed(pathData->fAtlasLocation.fY);
SkFixed tw = SkScalarToFixed(pathData->fBounds.width());
SkFixed th = SkScalarToFixed(pathData->fBounds.height());
// vertex positions
SkRect r = SkRect::MakeXYWH(dx, dy, width, height);
size_t vertSize = 2 * sizeof(SkPoint);
SkPoint* positions = reinterpret_cast<SkPoint*>(vertices);
positions->setRectFan(r.left(), r.top(), r.right(), r.bottom(), vertSize);
// vertex texture coords
intptr_t intPtr = reinterpret_cast<intptr_t>(positions);
SkPoint* textureCoords = reinterpret_cast<SkPoint*>(intPtr + vertSize - sizeof(SkPoint));
textureCoords->setRectFan(SkFixedToFloat(texture->texturePriv().normalizeFixedX(tx)),
SkFixedToFloat(texture->texturePriv().normalizeFixedY(ty)),
SkFixedToFloat(texture->texturePriv().normalizeFixedX(tx + tw)),
SkFixedToFloat(texture->texturePriv().normalizeFixedY(ty + th)),
vertSize);
uint16_t* indexPtr = reinterpret_cast<uint16_t*>(indices);
*indexPtr++ = 0;
*indexPtr++ = 1;
*indexPtr++ = 2;
*indexPtr++ = 0;
*indexPtr++ = 2;
*indexPtr++ = 3;
// set up any flags
uint32_t flags = 0;
const SkMatrix& vm = drawState->getViewMatrix();
flags |= vm.isSimilarity() ? kSimilarity_DistanceFieldEffectFlag : 0;
GrTextureParams params(SkShader::kRepeat_TileMode, GrTextureParams::kBilerp_FilterMode);
drawState->setGeometryProcessor(GrDistanceFieldNoGammaTextureEffect::Create(texture,
params,
flags))->unref();
vm.mapRect(&r);
target->drawIndexedInstances(kTriangles_GrPrimitiveType, 1, 4, 6, &r);
target->resetVertexSource();
target->resetIndexSource();
return true;
}