'<(skia_src_path)/core/SkMessageBus.h',
'<(skia_src_path)/core/SkMetaData.cpp',
'<(skia_src_path)/core/SkMipMap.cpp',
- '<(skia_src_path)/core/SkReadBuffer.cpp',
- '<(skia_src_path)/core/SkWriteBuffer.cpp',
+ '<(skia_src_path)/core/SkOffsetTable.h',
'<(skia_src_path)/core/SkPackBits.cpp',
'<(skia_src_path)/core/SkPaint.cpp',
'<(skia_src_path)/core/SkPaintOptionsAndroid.cpp',
'<(skia_src_path)/core/SkQuadTreePicture.h',
'<(skia_src_path)/core/SkRasterClip.cpp',
'<(skia_src_path)/core/SkRasterizer.cpp',
+ '<(skia_src_path)/core/SkReadBuffer.cpp',
'<(skia_src_path)/core/SkRect.cpp',
'<(skia_src_path)/core/SkRefDict.cpp',
'<(skia_src_path)/core/SkRegion.cpp',
'<(skia_src_path)/core/SkUnPreMultiply.cpp',
'<(skia_src_path)/core/SkUtils.cpp',
'<(skia_src_path)/core/SkValidatingReadBuffer.cpp',
+ '<(skia_src_path)/core/SkWriteBuffer.cpp',
'<(skia_src_path)/core/SkWriter32.cpp',
'<(skia_src_path)/core/SkXfermode.cpp',
--- /dev/null
+/*
+ * Copyright 2014 Google Inc.
+ *
+ * Use of this source code is governed by a BSD-style license that can be
+ * found in the LICENSE file.
+ */
+
+#ifndef SkOffsetTable_DEFINED
+#define SkOffsetTable_DEFINED
+
+#include "SkRefCnt.h"
+#include "SkTDArray.h"
+
+// A 2D table of skp offsets. Each row is indexed by an int. This is used
+// to store the command offsets that reference a particular bitmap using
+// the bitmap's index in the bitmap heap as the 'id' here. It has to be
+// ref-countable so SkPicturePlayback can take ownership of it.
+// Note that this class assumes that the ids are densely packed.
+
+// TODO: This needs to be sped up. We could replace the offset table with
+// a hash table.
+class SkOffsetTable : public SkRefCnt {
+public:
+ SkOffsetTable() {}
+ ~SkOffsetTable() {
+ fOffsetArrays.deleteAll();
+ }
+
+ // Record that this 'id' is used by the command starting at this 'offset'.
+ // Offsets for a given 'id' should always be added in increasing order.
+ void add(int id, size_t offset) {
+ if (id >= fOffsetArrays.count()) {
+ int oldCount = fOffsetArrays.count();
+ fOffsetArrays.setCount(id+1);
+ for (int i = oldCount; i <= id; ++i) {
+ fOffsetArrays[i] = NULL;
+ }
+ }
+
+ if (NULL == fOffsetArrays[id]) {
+ fOffsetArrays[id] = SkNEW(OffsetArray);
+ }
+ fOffsetArrays[id]->add(offset);
+ }
+
+ int numIDs() const {
+ return fOffsetArrays.count();
+ }
+
+ // Do the offsets of any commands referencing this ID fall in the
+ // range [min, max] (both inclusive)
+ bool overlap(int id, size_t min, size_t max) {
+ SkASSERT(id < fOffsetArrays.count());
+
+ if (NULL == fOffsetArrays[id]) {
+ return false;
+ }
+
+ // If this id has an offset array it should have at least one use
+ SkASSERT(fOffsetArrays[id]->count() > 0);
+ if (max < fOffsetArrays[id]->min() || min > fOffsetArrays[id]->max()) {
+ return false;
+ }
+
+ return true;
+ }
+
+ bool includes(int id, size_t offset) {
+ SkASSERT(id < fOffsetArrays.count());
+
+ OffsetArray* array = fOffsetArrays[id];
+
+ for (int i = 0; i < array->fOffsets.count(); ++i) {
+ if (array->fOffsets[i] == offset) {
+ return true;
+ } else if (array->fOffsets[i] > offset) {
+ return false;
+ }
+ }
+
+ // Calls to 'includes' should be gaurded by an overlap() call, so we
+ // should always find something.
+ SkASSERT(0);
+ return false;
+ }
+
+protected:
+ class OffsetArray {
+ public:
+ void add(size_t offset) {
+ SkASSERT(fOffsets.count() == 0 || offset > this->max());
+ *fOffsets.append() = offset;
+ }
+ size_t min() const {
+ SkASSERT(fOffsets.count() > 0);
+ return fOffsets[0];
+ }
+ size_t max() const {
+ SkASSERT(fOffsets.count() > 0);
+ return fOffsets[fOffsets.count()-1];
+ }
+ int count() const {
+ return fOffsets.count();
+ }
+
+ SkTDArray<size_t> fOffsets;
+ };
+
+ SkTDArray<OffsetArray*> fOffsetArrays;
+
+private:
+ typedef SkRefCnt INHERITED;
+};
+
+#endif
* Use of this source code is governed by a BSD-style license that can be
* found in the LICENSE file.
*/
-#include "SkPicturePlayback.h"
-#include "SkPictureRecord.h"
-#include "SkTypeface.h"
-#include "SkReadBuffer.h"
-#include "SkWriteBuffer.h"
#include <new>
#include "SkBBoxHierarchy.h"
+#include "SkOffsetTable.h"
+#include "SkPicturePlayback.h"
+#include "SkPictureRecord.h"
#include "SkPictureStateTree.h"
+#include "SkReadBuffer.h"
+#include "SkTypeface.h"
#include "SkTSort.h"
+#include "SkWriteBuffer.h"
template <typename T> int SafeCount(const T* obj) {
return obj ? obj->count() : 0;
fBitmapHeap.reset(SkSafeRef(record.fBitmapHeap));
fPathHeap.reset(SkSafeRef(record.fPathHeap));
+ fBitmapUseOffsets.reset(SkSafeRef(record.fBitmapUseOffsets.get()));
+
// ensure that the paths bounds are pre-computed
if (fPathHeap.get()) {
for (int i = 0; i < fPathHeap->count(); i++) {
return (DrawType) op;
}
+// The activeOps parameter is actually "const SkTDArray<SkPictureStateTree::Draw*>&".
+// It represents the operations about to be drawn, as generated by some spatial
+// subdivision helper class. It should already be in 'fOffset' sorted order.
+void SkPicturePlayback::preLoadBitmaps(const SkTDArray<void*>& activeOps) {
+ if (0 == activeOps.count() || NULL == fBitmapUseOffsets) {
+ return;
+ }
+
+ SkTDArray<int> active;
+
+ SkAutoTDeleteArray<bool> needToCheck(new bool[fBitmapUseOffsets->numIDs()]);
+ for (int i = 0; i < fBitmapUseOffsets->numIDs(); ++i) {
+ needToCheck.get()[i] = true;
+ }
+
+ uint32_t max = ((SkPictureStateTree::Draw*)activeOps[activeOps.count()-1])->fOffset;
+
+ for (int i = 0; i < activeOps.count(); ++i) {
+ SkPictureStateTree::Draw* draw = (SkPictureStateTree::Draw*) activeOps[i];
+
+ for (int j = 0; j < fBitmapUseOffsets->numIDs(); ++j) {
+ if (!needToCheck.get()[j]) {
+ continue;
+ }
+
+ if (!fBitmapUseOffsets->overlap(j, draw->fOffset, max)) {
+ needToCheck.get()[j] = false;
+ continue;
+ }
+
+ if (!fBitmapUseOffsets->includes(j, draw->fOffset)) {
+ continue;
+ }
+
+ *active.append() = j;
+ needToCheck.get()[j] = false;
+ }
+ }
+
+ for (int i = 0; i < active.count(); ++i) {
+ SkDebugf("preload texture %d\n", active[i]);
+ }
+}
+
void SkPicturePlayback::draw(SkCanvas& canvas, SkDrawPictureCallback* callback) {
#ifdef ENABLE_TIME_DRAW
SkAutoTime at("SkPicture::draw", 50);
SkReader32 reader(fOpData->bytes(), fOpData->size());
TextContainer text;
- SkTDArray<void*> results;
+ SkTDArray<void*> activeOps;
if (NULL != fStateTree && NULL != fBoundingHierarchy) {
SkRect clipBounds;
if (canvas.getClipBounds(&clipBounds)) {
SkIRect query;
clipBounds.roundOut(&query);
- fBoundingHierarchy->search(query, &results);
- if (results.count() == 0) {
+ fBoundingHierarchy->search(query, &activeOps);
+ if (activeOps.count() == 0) {
return;
}
SkTQSort<SkPictureStateTree::Draw>(
- reinterpret_cast<SkPictureStateTree::Draw**>(results.begin()),
- reinterpret_cast<SkPictureStateTree::Draw**>(results.end()-1));
+ reinterpret_cast<SkPictureStateTree::Draw**>(activeOps.begin()),
+ reinterpret_cast<SkPictureStateTree::Draw**>(activeOps.end()-1));
}
}
SkPictureStateTree::Iterator it = (NULL == fStateTree) ?
SkPictureStateTree::Iterator() :
- fStateTree->getIterator(results, &canvas);
+ fStateTree->getIterator(activeOps, &canvas);
if (it.isValid()) {
uint32_t skipTo = it.draw();
reader.setOffset(skipTo);
}
+ this->preLoadBitmaps(activeOps);
+
// Record this, so we can concat w/ it if we encounter a setMatrix()
SkMatrix initialMatrix = canvas.getTotalMatrix();
int originalSaveCount = canvas.getSaveCount();
class SkWStream;
class SkBBoxHierarchy;
class SkPictureStateTree;
+class SkOffsetTable;
struct SkPictInfo {
enum Flags {
virtual void postDraw(int opIndex);
#endif
+ void preLoadBitmaps(const SkTDArray<void*>& results);
+
private:
class TextContainer {
public:
SkTRefArray<SkPaint>* fPaints;
SkData* fOpData; // opcodes and parameters
+ SkAutoTUnref<SkOffsetTable> fBitmapUseOffsets;
SkPicture** fPictureRefs;
int fPictureCount;
#include "SkRRect.h"
#include "SkBBoxHierarchy.h"
#include "SkDevice.h"
+#include "SkOffsetTable.h"
#include "SkPictureStateTree.h"
#include "SkSurface.h"
fBitmapHeap = SkNEW(SkBitmapHeap);
fFlattenableHeap.setBitmapStorage(fBitmapHeap);
fPathHeap = NULL; // lazy allocate
+
#ifndef SK_COLLAPSE_MATRIX_CLIP_STATE
fFirstSavedLayerIndex = kNoSavedLayerIndex;
#endif
size_t initialOffset = this->addDraw(DRAW_BITMAP, &size);
SkASSERT(initialOffset+getPaintOffset(DRAW_BITMAP, size) == fWriter.bytesWritten());
this->addPaintPtr(paint);
- this->addBitmap(bitmap);
+ int bitmapID = this->addBitmap(bitmap);
this->addScalar(left);
this->addScalar(top);
this->validate(initialOffset, size);
+ this->trackBitmapUse(bitmapID, initialOffset);
}
void SkPictureRecord::drawBitmapRectToRect(const SkBitmap& bitmap, const SkRect* src,
SkASSERT(initialOffset+getPaintOffset(DRAW_BITMAP_RECT_TO_RECT, size)
== fWriter.bytesWritten());
this->addPaintPtr(paint);
- this->addBitmap(bitmap);
+ int bitmapID = this->addBitmap(bitmap);
this->addRectPtr(src); // may be null
this->addRect(dst);
this->addInt(flags);
this->validate(initialOffset, size);
+ this->trackBitmapUse(bitmapID, initialOffset);
}
void SkPictureRecord::drawBitmapMatrix(const SkBitmap& bitmap, const SkMatrix& matrix,
size_t initialOffset = this->addDraw(DRAW_BITMAP_MATRIX, &size);
SkASSERT(initialOffset+getPaintOffset(DRAW_BITMAP_MATRIX, size) == fWriter.bytesWritten());
this->addPaintPtr(paint);
- this->addBitmap(bitmap);
+ int bitmapID = this->addBitmap(bitmap);
this->addMatrix(matrix);
this->validate(initialOffset, size);
+ this->trackBitmapUse(bitmapID, initialOffset);
}
void SkPictureRecord::drawBitmapNine(const SkBitmap& bitmap, const SkIRect& center,
size_t initialOffset = this->addDraw(DRAW_BITMAP_NINE, &size);
SkASSERT(initialOffset+getPaintOffset(DRAW_BITMAP_NINE, size) == fWriter.bytesWritten());
this->addPaintPtr(paint);
- this->addBitmap(bitmap);
+ int bitmapID = this->addBitmap(bitmap);
this->addIRect(center);
this->addRect(dst);
this->validate(initialOffset, size);
+ this->trackBitmapUse(bitmapID, initialOffset);
}
void SkPictureRecord::drawSprite(const SkBitmap& bitmap, int left, int top,
size_t initialOffset = this->addDraw(DRAW_SPRITE, &size);
SkASSERT(initialOffset+getPaintOffset(DRAW_SPRITE, size) == fWriter.bytesWritten());
this->addPaintPtr(paint);
- this->addBitmap(bitmap);
+ int bitmapID = this->addBitmap(bitmap);
this->addInt(left);
this->addInt(top);
this->validate(initialOffset, size);
+ this->trackBitmapUse(bitmapID, initialOffset);
}
void SkPictureRecord::ComputeFontMetricsTopBottom(const SkPaint& paint, SkScalar topbot[2]) {
return SkSurface::NewPicture(info.fWidth, info.fHeight);
}
-void SkPictureRecord::addBitmap(const SkBitmap& bitmap) {
+void SkPictureRecord::trackBitmapUse(int bitmapID, size_t offset) {
+#ifndef SK_ALLOW_BITMAP_TRACKING
+ return;
+#endif
+
+ if (!(fRecordFlags & SkPicture::kOptimizeForClippedPlayback_RecordingFlag)) {
+ return;
+ }
+
+ if (SkBitmapHeap::INVALID_SLOT == bitmapID) {
+ return;
+ }
+
+ if (NULL == fBitmapUseOffsets) {
+ fBitmapUseOffsets.reset(SkNEW(SkOffsetTable));
+ }
+
+ fBitmapUseOffsets->add(bitmapID, offset);
+}
+
+int SkPictureRecord::addBitmap(const SkBitmap& bitmap) {
const int index = fBitmapHeap->insert(bitmap);
// In debug builds, a bad return value from insert() will crash, allowing for debugging. In
// release builds, the invalid value will be recorded so that the reader will know that there
// was a problem.
SkASSERT(index != SkBitmapHeap::INVALID_SLOT);
this->addInt(index);
+ return index;
}
void SkPictureRecord::addMatrix(const SkMatrix& matrix) {
#include "SkTemplates.h"
#include "SkWriter32.h"
-class SkPictureStateTree;
class SkBBoxHierarchy;
+class SkOffsetTable;
+class SkPictureStateTree;
// These macros help with packing and unpacking a single byte value and
// a 3 byte value into/out of a uint32_t
fWriter.writeScalar(scalar);
}
- void addBitmap(const SkBitmap& bitmap);
+ // The command at 'offset' in the skp uses the specified bitmap
+ void trackBitmapUse(int bitmapID, size_t offset);
+ int addBitmap(const SkBitmap& bitmap);
void addMatrix(const SkMatrix& matrix);
const SkFlatData* addPaint(const SkPaint& paint) { return this->addPaintPtr(&paint); }
const SkFlatData* addPaintPtr(const SkPaint* paint);
bool fOptsEnabled;
int fInitialSaveCount;
+ SkAutoTUnref<SkOffsetTable> fBitmapUseOffsets;
+
friend class SkPicturePlayback;
friend class SkPictureTester; // for unit testing