2014-08-21 20:12:42 +00:00
|
|
|
/*
|
|
|
|
* Copyright 2014 Google Inc.
|
|
|
|
*
|
|
|
|
* Use of this source code is governed by a BSD-style license that can be
|
|
|
|
* found in the LICENSE file.
|
|
|
|
*/
|
|
|
|
|
2014-10-08 12:17:02 +00:00
|
|
|
#if SK_SUPPORT_GPU
|
|
|
|
#include "GrLayerHoister.h"
|
|
|
|
#include "GrRecordReplaceDraw.h"
|
|
|
|
#endif
|
|
|
|
|
2014-08-21 20:12:42 +00:00
|
|
|
#include "SkCanvas.h"
|
|
|
|
#include "SkMultiPictureDraw.h"
|
|
|
|
#include "SkPicture.h"
|
2014-10-29 19:36:45 +00:00
|
|
|
#include "SkTaskGroup.h"
|
|
|
|
|
|
|
|
void SkMultiPictureDraw::DrawData::draw() {
|
|
|
|
fCanvas->drawPicture(fPicture, &fMatrix, fPaint);
|
|
|
|
}
|
|
|
|
|
|
|
|
void SkMultiPictureDraw::DrawData::init(SkCanvas* canvas, const SkPicture* picture,
|
|
|
|
const SkMatrix* matrix, const SkPaint* paint) {
|
|
|
|
fPicture = SkRef(picture);
|
|
|
|
fCanvas = SkRef(canvas);
|
|
|
|
if (matrix) {
|
|
|
|
fMatrix = *matrix;
|
|
|
|
} else {
|
|
|
|
fMatrix.setIdentity();
|
|
|
|
}
|
|
|
|
if (paint) {
|
|
|
|
fPaint = SkNEW_ARGS(SkPaint, (*paint));
|
|
|
|
} else {
|
|
|
|
fPaint = NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void SkMultiPictureDraw::DrawData::Reset(SkTDArray<DrawData>& data) {
|
|
|
|
for (int i = 0; i < data.count(); ++i) {
|
|
|
|
data[i].fPicture->unref();
|
|
|
|
data[i].fCanvas->unref();
|
|
|
|
SkDELETE(data[i].fPaint);
|
|
|
|
}
|
|
|
|
data.rewind();
|
|
|
|
}
|
|
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////////////////////
|
2014-08-21 20:12:42 +00:00
|
|
|
|
|
|
|
SkMultiPictureDraw::SkMultiPictureDraw(int reserve) {
|
|
|
|
if (reserve > 0) {
|
2014-10-29 19:36:45 +00:00
|
|
|
fGPUDrawData.setReserve(reserve);
|
|
|
|
fThreadSafeDrawData.setReserve(reserve);
|
2014-08-21 20:12:42 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void SkMultiPictureDraw::reset() {
|
2014-10-29 19:36:45 +00:00
|
|
|
DrawData::Reset(fGPUDrawData);
|
|
|
|
DrawData::Reset(fThreadSafeDrawData);
|
2014-08-21 20:12:42 +00:00
|
|
|
}
|
|
|
|
|
2014-10-29 21:17:13 +00:00
|
|
|
void SkMultiPictureDraw::add(SkCanvas* canvas,
|
2014-08-21 20:12:42 +00:00
|
|
|
const SkPicture* picture,
|
2014-10-29 21:17:13 +00:00
|
|
|
const SkMatrix* matrix,
|
2014-08-21 20:12:42 +00:00
|
|
|
const SkPaint* paint) {
|
|
|
|
if (NULL == canvas || NULL == picture) {
|
|
|
|
SkDEBUGFAIL("parameters to SkMultiPictureDraw::add should be non-NULL");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2014-10-29 19:36:45 +00:00
|
|
|
SkTDArray<DrawData>& array = canvas->getGrContext() ? fGPUDrawData : fThreadSafeDrawData;
|
|
|
|
array.append()->init(canvas, picture, matrix, paint);
|
2014-08-21 20:12:42 +00:00
|
|
|
}
|
|
|
|
|
2014-10-08 12:17:02 +00:00
|
|
|
#undef SK_IGNORE_GPU_LAYER_HOISTING
|
|
|
|
#define SK_IGNORE_GPU_LAYER_HOISTING 1
|
|
|
|
|
2014-10-29 19:36:45 +00:00
|
|
|
class AutoMPDReset : SkNoncopyable {
|
|
|
|
SkMultiPictureDraw* fMPD;
|
|
|
|
public:
|
|
|
|
AutoMPDReset(SkMultiPictureDraw* mpd) : fMPD(mpd) {}
|
|
|
|
~AutoMPDReset() { fMPD->reset(); }
|
|
|
|
};
|
|
|
|
|
2014-08-21 20:12:42 +00:00
|
|
|
void SkMultiPictureDraw::draw() {
|
2014-10-29 19:36:45 +00:00
|
|
|
AutoMPDReset mpdreset(this);
|
|
|
|
// we place the taskgroup after the MPDReset, to ensure that we don't delete the DrawData
|
|
|
|
// objects until after we're finished the tasks (which have pointers to the data).
|
|
|
|
|
2014-10-30 14:19:11 +00:00
|
|
|
#if 0
|
2014-10-29 19:36:45 +00:00
|
|
|
SkTaskGroup group;
|
2014-10-29 22:44:25 +00:00
|
|
|
group.batch(DrawData::Draw, fThreadSafeDrawData.begin(), fThreadSafeDrawData.count());
|
2014-10-29 19:36:45 +00:00
|
|
|
// we deliberately don't call wait() here, since the destructor will do that, this allows us
|
|
|
|
// to continue processing gpu-data without having to wait on the cpu tasks.
|
2014-10-30 14:19:11 +00:00
|
|
|
#else
|
|
|
|
for (int i = 0; i < fThreadSafeDrawData.count(); i++) {
|
|
|
|
DrawData::Draw(&fThreadSafeDrawData[i]);
|
|
|
|
}
|
|
|
|
#endif
|
2014-10-29 19:36:45 +00:00
|
|
|
|
|
|
|
const int count = fGPUDrawData.count();
|
|
|
|
if (0 == count) {
|
|
|
|
return;
|
|
|
|
}
|
2014-10-08 12:17:02 +00:00
|
|
|
|
|
|
|
#ifndef SK_IGNORE_GPU_LAYER_HOISTING
|
2014-10-29 19:36:45 +00:00
|
|
|
GrContext* context = fGPUDrawData[0].fCanvas->getGrContext();
|
|
|
|
SkASSERT(context);
|
2014-10-08 12:17:02 +00:00
|
|
|
|
2014-10-29 19:36:45 +00:00
|
|
|
// Start by collecting all the layers that are going to be atlased and render
|
2014-10-28 14:21:44 +00:00
|
|
|
// them (if necessary). Hoisting the free floating layers is deferred until
|
|
|
|
// drawing the canvas that requires them.
|
|
|
|
SkTDArray<GrHoistedLayer> atlasedNeedRendering, atlasedRecycled;
|
2014-10-08 12:17:02 +00:00
|
|
|
|
2014-10-29 19:36:45 +00:00
|
|
|
for (int i = 0; i < count; ++i) {
|
|
|
|
const DrawData& data = fGPUDrawData[i];
|
|
|
|
// we only expect 1 context for all the canvases
|
|
|
|
SkASSERT(data.canvas->getGrContext() == context);
|
2014-10-08 12:17:02 +00:00
|
|
|
|
2014-10-29 19:36:45 +00:00
|
|
|
if (!data.fPaint && data.fMatrix.isIdentity()) {
|
2014-10-08 12:17:02 +00:00
|
|
|
// TODO: this path always tries to optimize pictures. Should we
|
|
|
|
// switch to this API approach (vs. SkCanvas::EXPERIMENTAL_optimize)?
|
2014-10-29 19:36:45 +00:00
|
|
|
data.fCanvas->EXPERIMENTAL_optimize(data.fPicture);
|
2014-10-08 12:17:02 +00:00
|
|
|
|
|
|
|
SkRect clipBounds;
|
2014-10-29 19:36:45 +00:00
|
|
|
if (!data.fCanvas->getClipBounds(&clipBounds)) {
|
2014-10-08 12:17:02 +00:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2014-10-28 14:21:44 +00:00
|
|
|
// TODO: sorting the cacheable layers from smallest to largest
|
|
|
|
// would improve the packing and reduce the number of swaps
|
|
|
|
// TODO: another optimization would be to make a first pass to
|
|
|
|
// lock any required layer that is already in the atlas
|
2014-10-29 19:36:45 +00:00
|
|
|
GrLayerHoister::FindLayersToAtlas(context, data.fPicture,
|
2014-10-29 21:17:13 +00:00
|
|
|
clipBounds,
|
2014-10-28 14:21:44 +00:00
|
|
|
&atlasedNeedRendering, &atlasedRecycled);
|
2014-10-08 12:17:02 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-10-29 19:36:45 +00:00
|
|
|
GrLayerHoister::DrawLayersToAtlas(context, atlasedNeedRendering);
|
2014-10-28 14:21:44 +00:00
|
|
|
|
|
|
|
SkTDArray<GrHoistedLayer> needRendering, recycled;
|
2014-10-08 12:17:02 +00:00
|
|
|
#endif
|
|
|
|
|
2014-10-29 19:36:45 +00:00
|
|
|
for (int i = 0; i < count; ++i) {
|
|
|
|
const DrawData& data = fGPUDrawData[i];
|
|
|
|
SkCanvas* canvas = data.fCanvas;
|
|
|
|
const SkPicture* picture = data.fPicture;
|
|
|
|
|
2014-10-08 12:17:02 +00:00
|
|
|
#ifndef SK_IGNORE_GPU_LAYER_HOISTING
|
2014-10-29 19:36:45 +00:00
|
|
|
if (!data.fPaint && data.fMatrix.isIdentity()) {
|
2014-10-28 14:21:44 +00:00
|
|
|
|
|
|
|
SkRect clipBounds;
|
2014-10-29 19:36:45 +00:00
|
|
|
if (!canvas->getClipBounds(&clipBounds)) {
|
2014-10-28 14:21:44 +00:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Find the layers required by this canvas. It will return atlased
|
|
|
|
// layers in the 'recycled' list since they have already been drawn.
|
2014-10-29 19:36:45 +00:00
|
|
|
GrLayerHoister::FindLayersToHoist(context, picture,
|
2014-10-28 14:21:44 +00:00
|
|
|
clipBounds, &needRendering, &recycled);
|
|
|
|
|
|
|
|
GrLayerHoister::DrawLayers(context, needRendering);
|
|
|
|
|
|
|
|
GrReplacements replacements;
|
|
|
|
|
|
|
|
GrLayerHoister::ConvertLayersToReplacements(needRendering, &replacements);
|
|
|
|
GrLayerHoister::ConvertLayersToReplacements(recycled, &replacements);
|
|
|
|
|
2014-10-29 19:36:45 +00:00
|
|
|
const SkMatrix initialMatrix = canvas->getTotalMatrix();
|
2014-10-08 12:17:02 +00:00
|
|
|
|
2014-10-28 14:21:44 +00:00
|
|
|
// Render the entire picture using new layers
|
2014-10-29 19:36:45 +00:00
|
|
|
GrRecordReplaceDraw(picture, canvas, &replacements, initialMatrix, NULL);
|
2014-10-28 14:21:44 +00:00
|
|
|
|
|
|
|
GrLayerHoister::UnlockLayers(context, needRendering);
|
|
|
|
GrLayerHoister::UnlockLayers(context, recycled);
|
|
|
|
|
|
|
|
needRendering.rewind();
|
|
|
|
recycled.rewind();
|
|
|
|
} else
|
2014-10-08 12:17:02 +00:00
|
|
|
#endif
|
|
|
|
{
|
2014-10-29 19:36:45 +00:00
|
|
|
canvas->drawPicture(picture, &data.fMatrix, data.fPaint);
|
2014-10-08 12:17:02 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifndef SK_IGNORE_GPU_LAYER_HOISTING
|
2014-10-29 19:36:45 +00:00
|
|
|
GrLayerHoister::UnlockLayers(context, atlasedNeedRendering);
|
|
|
|
GrLayerHoister::UnlockLayers(context, atlasedRecycled);
|
2014-10-29 20:56:02 +00:00
|
|
|
#if !GR_CACHE_HOISTED_LAYERS
|
|
|
|
GrLayerHoister::PurgeCache(context);
|
|
|
|
#endif
|
2014-10-08 12:17:02 +00:00
|
|
|
#endif
|
2014-08-21 20:12:42 +00:00
|
|
|
}
|
|
|
|
|