aboutsummaryrefslogtreecommitdiffhomepage
path: root/src/core/SkPictureFlat.cpp
blob: ec04495db9cdf1e429ec9241073721aabe05c1f2 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126

/*
 * Copyright 2011 Google Inc.
 *
 * Use of this source code is governed by a BSD-style license that can be
 * found in the LICENSE file.
 */
#include "SkPictureFlat.h"

#include "SkChecksum.h"
#include "SkColorFilter.h"
#include "SkDrawLooper.h"
#include "SkMaskFilter.h"
#include "SkRasterizer.h"
#include "SkShader.h"
#include "SkTypeface.h"
#include "SkXfermode.h"

///////////////////////////////////////////////////////////////////////////////

SkRefCntPlayback::SkRefCntPlayback() : fCount(0), fArray(NULL) {}

SkRefCntPlayback::~SkRefCntPlayback() {
    this->reset(NULL);
}

void SkRefCntPlayback::reset(const SkRefCntSet* rec) {
    for (int i = 0; i < fCount; i++) {
        SkASSERT(fArray[i]);
        fArray[i]->unref();
    }
    SkDELETE_ARRAY(fArray);
    
    if (rec) {
        fCount = rec->count();
        fArray = SkNEW_ARRAY(SkRefCnt*, fCount);
        rec->copyToArray(fArray);
        for (int i = 0; i < fCount; i++) {
            fArray[i]->ref();
        }
    } else {
        fCount = 0;
        fArray = NULL;
    }
}

void SkRefCntPlayback::setCount(int count) {
    this->reset(NULL);
    
    fCount = count;
    fArray = SkNEW_ARRAY(SkRefCnt*, count);
    sk_bzero(fArray, count * sizeof(SkRefCnt*));
}

SkRefCnt* SkRefCntPlayback::set(int index, SkRefCnt* obj) {
    SkASSERT((unsigned)index < (unsigned)fCount);
    SkRefCnt_SafeAssign(fArray[index], obj);
    return obj;
}

///////////////////////////////////////////////////////////////////////////////

SkFlatData* SkFlatData::Create(SkChunkAlloc* heap, const void* obj,
        int index, void (*flattenProc)(SkOrderedWriteBuffer&, const void*),
        SkRefCntSet* refCntRecorder, SkRefCntSet* faceRecorder,
        uint32_t writeBufferflags) {

    // a buffer of 256 bytes should be sufficient for most paints, regions,
    // and matrices.
    intptr_t storage[256];
    SkOrderedWriteBuffer buffer(256, storage, sizeof(storage));
    if (refCntRecorder) {
        buffer.setRefCntRecorder(refCntRecorder);
    }
    if (faceRecorder) {
        buffer.setTypefaceRecorder(faceRecorder);
    }
    buffer.setFlags(writeBufferflags);

    flattenProc(buffer, obj);
    uint32_t size = buffer.size();


#if !SK_PREFER_32BIT_CHECKSUM
    uint32_t unpaddedSize = size;
    size = SkAlign8(size);
#endif

    // allocate enough memory to hold both SkFlatData and the serialized
    // contents
    SkFlatData* result = (SkFlatData*) heap->allocThrow(size + sizeof(SkFlatData));
    result->fIndex = index;
    result->fAllocSize = size;

    // put the serialized contents into the data section of the new allocation
    buffer.flatten(result->data());
#if SK_PREFER_32BIT_CHECKSUM
    result->fChecksum =
        SkComputeChecksum32(reinterpret_cast<uint32_t*>(result->data()), size);
#else
    if (size != unpaddedSize) {
        // Flat data is padded: put zeros in the last 32 bits.
        SkASSERT(size - 4 == unpaddedSize);
        *((uint32_t*)((char*)result->data() + unpaddedSize)) = 0;
    }
    result->fChecksum =
        SkComputeChecksum64(reinterpret_cast<uint64_t*>(result->data()), size);
#endif
    return result;
}

void SkFlatData::unflatten(void* result,
        void (*unflattenProc)(SkOrderedReadBuffer&, void*),
        SkRefCntPlayback* refCntPlayback,
        SkTypefacePlayback* facePlayback) const {

    SkOrderedReadBuffer buffer(this->data(), fAllocSize);
    if (refCntPlayback) {
        refCntPlayback->setupBuffer(buffer);
    }
    if (facePlayback) {
        facePlayback->setupBuffer(buffer);
    }
    unflattenProc(buffer, result);
    SkASSERT(fAllocSize == SkAlign8((int32_t)buffer.offset()));
}