1*c8dee2aaSAndroid Build Coastguard Worker /*
2*c8dee2aaSAndroid Build Coastguard Worker * Copyright 2011 Google Inc.
3*c8dee2aaSAndroid Build Coastguard Worker *
4*c8dee2aaSAndroid Build Coastguard Worker * Use of this source code is governed by a BSD-style license that can be
5*c8dee2aaSAndroid Build Coastguard Worker * found in the LICENSE file.
6*c8dee2aaSAndroid Build Coastguard Worker */
7*c8dee2aaSAndroid Build Coastguard Worker
8*c8dee2aaSAndroid Build Coastguard Worker #include "include/core/SkData.h"
9*c8dee2aaSAndroid Build Coastguard Worker #include "include/core/SkRect.h"
10*c8dee2aaSAndroid Build Coastguard Worker #include "include/core/SkRefCnt.h"
11*c8dee2aaSAndroid Build Coastguard Worker #include "include/core/SkScalar.h"
12*c8dee2aaSAndroid Build Coastguard Worker #include "include/core/SkTypes.h"
13*c8dee2aaSAndroid Build Coastguard Worker #include "include/private/base/SkAlign.h"
14*c8dee2aaSAndroid Build Coastguard Worker #include "include/private/base/SkTemplates.h"
15*c8dee2aaSAndroid Build Coastguard Worker #include "src/base/SkAutoMalloc.h"
16*c8dee2aaSAndroid Build Coastguard Worker #include "src/base/SkRandom.h"
17*c8dee2aaSAndroid Build Coastguard Worker #include "src/core/SkReadBuffer.h"
18*c8dee2aaSAndroid Build Coastguard Worker #include "src/core/SkWriter32.h"
19*c8dee2aaSAndroid Build Coastguard Worker #include "tests/Test.h"
20*c8dee2aaSAndroid Build Coastguard Worker
21*c8dee2aaSAndroid Build Coastguard Worker #include <array>
22*c8dee2aaSAndroid Build Coastguard Worker #include <cstdint>
23*c8dee2aaSAndroid Build Coastguard Worker #include <cstring>
24*c8dee2aaSAndroid Build Coastguard Worker
25*c8dee2aaSAndroid Build Coastguard Worker using namespace skia_private;
26*c8dee2aaSAndroid Build Coastguard Worker
check_contents(skiatest::Reporter * reporter,const SkWriter32 & writer,const void * expected,size_t size)27*c8dee2aaSAndroid Build Coastguard Worker static void check_contents(skiatest::Reporter* reporter, const SkWriter32& writer,
28*c8dee2aaSAndroid Build Coastguard Worker const void* expected, size_t size) {
29*c8dee2aaSAndroid Build Coastguard Worker SkAutoSMalloc<256> storage(size);
30*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(reporter, writer.bytesWritten() == size);
31*c8dee2aaSAndroid Build Coastguard Worker writer.flatten(storage.get());
32*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(reporter, !memcmp(storage.get(), expected, size));
33*c8dee2aaSAndroid Build Coastguard Worker }
34*c8dee2aaSAndroid Build Coastguard Worker
35*c8dee2aaSAndroid Build Coastguard Worker
test_reserve(skiatest::Reporter * reporter)36*c8dee2aaSAndroid Build Coastguard Worker static void test_reserve(skiatest::Reporter* reporter) {
37*c8dee2aaSAndroid Build Coastguard Worker // There used to be a bug where we'd assert your first reservation had to
38*c8dee2aaSAndroid Build Coastguard Worker // fit in external storage if you used it. This would crash in debug mode.
39*c8dee2aaSAndroid Build Coastguard Worker uint8_t storage[4];
40*c8dee2aaSAndroid Build Coastguard Worker SkWriter32 writer(storage, sizeof(storage));
41*c8dee2aaSAndroid Build Coastguard Worker writer.reserve(40);
42*c8dee2aaSAndroid Build Coastguard Worker }
43*c8dee2aaSAndroid Build Coastguard Worker
test_string_null(skiatest::Reporter * reporter)44*c8dee2aaSAndroid Build Coastguard Worker static void test_string_null(skiatest::Reporter* reporter) {
45*c8dee2aaSAndroid Build Coastguard Worker uint8_t storage[8];
46*c8dee2aaSAndroid Build Coastguard Worker SkWriter32 writer(storage, sizeof(storage));
47*c8dee2aaSAndroid Build Coastguard Worker
48*c8dee2aaSAndroid Build Coastguard Worker // Can we write nullptr?
49*c8dee2aaSAndroid Build Coastguard Worker writer.writeString(nullptr);
50*c8dee2aaSAndroid Build Coastguard Worker const int32_t expected[] = { 0x0, 0x0 };
51*c8dee2aaSAndroid Build Coastguard Worker check_contents(reporter, writer, expected, sizeof(expected));
52*c8dee2aaSAndroid Build Coastguard Worker }
53*c8dee2aaSAndroid Build Coastguard Worker
test_rewind(skiatest::Reporter * reporter)54*c8dee2aaSAndroid Build Coastguard Worker static void test_rewind(skiatest::Reporter* reporter) {
55*c8dee2aaSAndroid Build Coastguard Worker SkSWriter32<32> swriter;
56*c8dee2aaSAndroid Build Coastguard Worker int32_t array[3] = { 1, 2, 4 };
57*c8dee2aaSAndroid Build Coastguard Worker
58*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(reporter, 0 == swriter.bytesWritten());
59*c8dee2aaSAndroid Build Coastguard Worker for (size_t i = 0; i < std::size(array); ++i) {
60*c8dee2aaSAndroid Build Coastguard Worker swriter.writeInt(array[i]);
61*c8dee2aaSAndroid Build Coastguard Worker }
62*c8dee2aaSAndroid Build Coastguard Worker check_contents(reporter, swriter, array, sizeof(array));
63*c8dee2aaSAndroid Build Coastguard Worker
64*c8dee2aaSAndroid Build Coastguard Worker swriter.rewindToOffset(2*sizeof(int32_t));
65*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(reporter, sizeof(array) - 4 == swriter.bytesWritten());
66*c8dee2aaSAndroid Build Coastguard Worker swriter.writeInt(3);
67*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(reporter, sizeof(array) == swriter.bytesWritten());
68*c8dee2aaSAndroid Build Coastguard Worker array[2] = 3;
69*c8dee2aaSAndroid Build Coastguard Worker check_contents(reporter, swriter, array, sizeof(array));
70*c8dee2aaSAndroid Build Coastguard Worker
71*c8dee2aaSAndroid Build Coastguard Worker // test rewinding past allocated chunks. This used to crash because we
72*c8dee2aaSAndroid Build Coastguard Worker // didn't truncate our link-list after freeing trailing blocks
73*c8dee2aaSAndroid Build Coastguard Worker SkWriter32 writer;
74*c8dee2aaSAndroid Build Coastguard Worker for (int i = 0; i < 100; ++i) {
75*c8dee2aaSAndroid Build Coastguard Worker writer.writeInt(i);
76*c8dee2aaSAndroid Build Coastguard Worker }
77*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(reporter, 100*4 == writer.bytesWritten());
78*c8dee2aaSAndroid Build Coastguard Worker for (int j = 100*4; j >= 0; j -= 16) {
79*c8dee2aaSAndroid Build Coastguard Worker writer.rewindToOffset(j);
80*c8dee2aaSAndroid Build Coastguard Worker }
81*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(reporter, writer.bytesWritten() < 16);
82*c8dee2aaSAndroid Build Coastguard Worker }
83*c8dee2aaSAndroid Build Coastguard Worker
test1(skiatest::Reporter * reporter,SkWriter32 * writer)84*c8dee2aaSAndroid Build Coastguard Worker static void test1(skiatest::Reporter* reporter, SkWriter32* writer) {
85*c8dee2aaSAndroid Build Coastguard Worker const uint32_t data[] = { 0, 1, 2, 3, 4, 5, 6, 7, 8, 9 };
86*c8dee2aaSAndroid Build Coastguard Worker for (size_t i = 0; i < std::size(data); ++i) {
87*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(reporter, i*4 == writer->bytesWritten());
88*c8dee2aaSAndroid Build Coastguard Worker writer->write32(data[i]);
89*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(reporter, data[i] == writer->readTAt<uint32_t>(i * 4));
90*c8dee2aaSAndroid Build Coastguard Worker }
91*c8dee2aaSAndroid Build Coastguard Worker
92*c8dee2aaSAndroid Build Coastguard Worker char buffer[sizeof(data)];
93*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(reporter, sizeof(buffer) == writer->bytesWritten());
94*c8dee2aaSAndroid Build Coastguard Worker writer->flatten(buffer);
95*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(reporter, !memcmp(data, buffer, sizeof(buffer)));
96*c8dee2aaSAndroid Build Coastguard Worker }
97*c8dee2aaSAndroid Build Coastguard Worker
testWritePad(skiatest::Reporter * reporter,SkWriter32 * writer)98*c8dee2aaSAndroid Build Coastguard Worker static void testWritePad(skiatest::Reporter* reporter, SkWriter32* writer) {
99*c8dee2aaSAndroid Build Coastguard Worker // Create some random data to write.
100*c8dee2aaSAndroid Build Coastguard Worker const size_t dataSize = 10;
101*c8dee2aaSAndroid Build Coastguard Worker
102*c8dee2aaSAndroid Build Coastguard Worker AutoTMalloc<uint32_t> originalData(dataSize);
103*c8dee2aaSAndroid Build Coastguard Worker {
104*c8dee2aaSAndroid Build Coastguard Worker SkRandom rand(0);
105*c8dee2aaSAndroid Build Coastguard Worker for (size_t i = 0; i < dataSize; i++) {
106*c8dee2aaSAndroid Build Coastguard Worker originalData[(int) i] = rand.nextU();
107*c8dee2aaSAndroid Build Coastguard Worker }
108*c8dee2aaSAndroid Build Coastguard Worker
109*c8dee2aaSAndroid Build Coastguard Worker // Write the random data to the writer at different lengths for
110*c8dee2aaSAndroid Build Coastguard Worker // different alignments.
111*c8dee2aaSAndroid Build Coastguard Worker for (size_t len = 0; len < dataSize; len++) {
112*c8dee2aaSAndroid Build Coastguard Worker writer->writePad(originalData.get(), len);
113*c8dee2aaSAndroid Build Coastguard Worker }
114*c8dee2aaSAndroid Build Coastguard Worker }
115*c8dee2aaSAndroid Build Coastguard Worker
116*c8dee2aaSAndroid Build Coastguard Worker size_t totalBytes = writer->bytesWritten();
117*c8dee2aaSAndroid Build Coastguard Worker
118*c8dee2aaSAndroid Build Coastguard Worker SkAutoMalloc readStorage(totalBytes);
119*c8dee2aaSAndroid Build Coastguard Worker writer->flatten(readStorage.get());
120*c8dee2aaSAndroid Build Coastguard Worker
121*c8dee2aaSAndroid Build Coastguard Worker SkReadBuffer reader(readStorage.get(), totalBytes);
122*c8dee2aaSAndroid Build Coastguard Worker
123*c8dee2aaSAndroid Build Coastguard Worker for (size_t len = 0; len < dataSize; len++) {
124*c8dee2aaSAndroid Build Coastguard Worker const char* readPtr = static_cast<const char*>(reader.skip(len));
125*c8dee2aaSAndroid Build Coastguard Worker // Ensure that the data read is the same as what was written.
126*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(reporter, memcmp(readPtr, originalData.get(), len) == 0);
127*c8dee2aaSAndroid Build Coastguard Worker // Ensure that the rest is padded with zeroes.
128*c8dee2aaSAndroid Build Coastguard Worker const char* stop = readPtr + SkAlign4(len);
129*c8dee2aaSAndroid Build Coastguard Worker readPtr += len;
130*c8dee2aaSAndroid Build Coastguard Worker while (readPtr < stop) {
131*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(reporter, *readPtr++ == 0);
132*c8dee2aaSAndroid Build Coastguard Worker }
133*c8dee2aaSAndroid Build Coastguard Worker }
134*c8dee2aaSAndroid Build Coastguard Worker }
135*c8dee2aaSAndroid Build Coastguard Worker
testOverwriteT(skiatest::Reporter * reporter,SkWriter32 * writer)136*c8dee2aaSAndroid Build Coastguard Worker static void testOverwriteT(skiatest::Reporter* reporter, SkWriter32* writer) {
137*c8dee2aaSAndroid Build Coastguard Worker const size_t padding = 64;
138*c8dee2aaSAndroid Build Coastguard Worker
139*c8dee2aaSAndroid Build Coastguard Worker const uint32_t uint1 = 0x12345678;
140*c8dee2aaSAndroid Build Coastguard Worker const uint32_t uint2 = 0x98765432;
141*c8dee2aaSAndroid Build Coastguard Worker const SkScalar scalar1 = 1234.5678f;
142*c8dee2aaSAndroid Build Coastguard Worker const SkScalar scalar2 = 9876.5432f;
143*c8dee2aaSAndroid Build Coastguard Worker const SkRect rect1 = SkRect::MakeXYWH(1, 2, 3, 4);
144*c8dee2aaSAndroid Build Coastguard Worker const SkRect rect2 = SkRect::MakeXYWH(5, 6, 7, 8);
145*c8dee2aaSAndroid Build Coastguard Worker
146*c8dee2aaSAndroid Build Coastguard Worker for (size_t i = 0; i < (padding / 4); ++i) {
147*c8dee2aaSAndroid Build Coastguard Worker writer->write32(0);
148*c8dee2aaSAndroid Build Coastguard Worker }
149*c8dee2aaSAndroid Build Coastguard Worker
150*c8dee2aaSAndroid Build Coastguard Worker writer->write32(uint1);
151*c8dee2aaSAndroid Build Coastguard Worker writer->writeRect(rect1);
152*c8dee2aaSAndroid Build Coastguard Worker writer->writeScalar(scalar1);
153*c8dee2aaSAndroid Build Coastguard Worker
154*c8dee2aaSAndroid Build Coastguard Worker for (size_t i = 0; i < (padding / 4); ++i) {
155*c8dee2aaSAndroid Build Coastguard Worker writer->write32(0);
156*c8dee2aaSAndroid Build Coastguard Worker }
157*c8dee2aaSAndroid Build Coastguard Worker
158*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(reporter, writer->readTAt<uint32_t>(padding) == uint1);
159*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(reporter, writer->readTAt<SkRect>(padding + sizeof(uint32_t)) == rect1);
160*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(reporter, writer->readTAt<SkScalar>(
161*c8dee2aaSAndroid Build Coastguard Worker padding + sizeof(uint32_t) + sizeof(SkRect)) == scalar1);
162*c8dee2aaSAndroid Build Coastguard Worker
163*c8dee2aaSAndroid Build Coastguard Worker writer->overwriteTAt(padding, uint2);
164*c8dee2aaSAndroid Build Coastguard Worker writer->overwriteTAt(padding + sizeof(uint32_t), rect2);
165*c8dee2aaSAndroid Build Coastguard Worker writer->overwriteTAt(padding + sizeof(uint32_t) + sizeof(SkRect), scalar2);
166*c8dee2aaSAndroid Build Coastguard Worker
167*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(reporter, writer->readTAt<uint32_t>(padding) == uint2);
168*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(reporter, writer->readTAt<SkRect>(padding + sizeof(uint32_t)) == rect2);
169*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(reporter, writer->readTAt<SkScalar>(
170*c8dee2aaSAndroid Build Coastguard Worker padding + sizeof(uint32_t) + sizeof(SkRect)) == scalar2);
171*c8dee2aaSAndroid Build Coastguard Worker }
172*c8dee2aaSAndroid Build Coastguard Worker
DEF_TEST(Writer32_dynamic,reporter)173*c8dee2aaSAndroid Build Coastguard Worker DEF_TEST(Writer32_dynamic, reporter) {
174*c8dee2aaSAndroid Build Coastguard Worker SkWriter32 writer;
175*c8dee2aaSAndroid Build Coastguard Worker test1(reporter, &writer);
176*c8dee2aaSAndroid Build Coastguard Worker
177*c8dee2aaSAndroid Build Coastguard Worker writer.reset();
178*c8dee2aaSAndroid Build Coastguard Worker testWritePad(reporter, &writer);
179*c8dee2aaSAndroid Build Coastguard Worker
180*c8dee2aaSAndroid Build Coastguard Worker writer.reset();
181*c8dee2aaSAndroid Build Coastguard Worker testOverwriteT(reporter, &writer);
182*c8dee2aaSAndroid Build Coastguard Worker }
183*c8dee2aaSAndroid Build Coastguard Worker
DEF_TEST(Writer32_small,reporter)184*c8dee2aaSAndroid Build Coastguard Worker DEF_TEST(Writer32_small, reporter) {
185*c8dee2aaSAndroid Build Coastguard Worker SkSWriter32<8 * sizeof(intptr_t)> writer;
186*c8dee2aaSAndroid Build Coastguard Worker test1(reporter, &writer);
187*c8dee2aaSAndroid Build Coastguard Worker
188*c8dee2aaSAndroid Build Coastguard Worker writer.reset(); // should just rewind our storage
189*c8dee2aaSAndroid Build Coastguard Worker testWritePad(reporter, &writer);
190*c8dee2aaSAndroid Build Coastguard Worker
191*c8dee2aaSAndroid Build Coastguard Worker writer.reset();
192*c8dee2aaSAndroid Build Coastguard Worker testOverwriteT(reporter, &writer);
193*c8dee2aaSAndroid Build Coastguard Worker }
194*c8dee2aaSAndroid Build Coastguard Worker
DEF_TEST(Writer32_large,reporter)195*c8dee2aaSAndroid Build Coastguard Worker DEF_TEST(Writer32_large, reporter) {
196*c8dee2aaSAndroid Build Coastguard Worker SkSWriter32<1024 * sizeof(intptr_t)> writer;
197*c8dee2aaSAndroid Build Coastguard Worker test1(reporter, &writer);
198*c8dee2aaSAndroid Build Coastguard Worker
199*c8dee2aaSAndroid Build Coastguard Worker writer.reset(); // should just rewind our storage
200*c8dee2aaSAndroid Build Coastguard Worker testWritePad(reporter, &writer);
201*c8dee2aaSAndroid Build Coastguard Worker
202*c8dee2aaSAndroid Build Coastguard Worker writer.reset();
203*c8dee2aaSAndroid Build Coastguard Worker testOverwriteT(reporter, &writer);
204*c8dee2aaSAndroid Build Coastguard Worker }
205*c8dee2aaSAndroid Build Coastguard Worker
DEF_TEST(Writer32_misc,reporter)206*c8dee2aaSAndroid Build Coastguard Worker DEF_TEST(Writer32_misc, reporter) {
207*c8dee2aaSAndroid Build Coastguard Worker test_reserve(reporter);
208*c8dee2aaSAndroid Build Coastguard Worker test_string_null(reporter);
209*c8dee2aaSAndroid Build Coastguard Worker test_rewind(reporter);
210*c8dee2aaSAndroid Build Coastguard Worker }
211*c8dee2aaSAndroid Build Coastguard Worker
DEF_TEST(Writer32_data,reporter)212*c8dee2aaSAndroid Build Coastguard Worker DEF_TEST(Writer32_data, reporter) {
213*c8dee2aaSAndroid Build Coastguard Worker const char* str = "0123456789";
214*c8dee2aaSAndroid Build Coastguard Worker sk_sp<SkData> data0(SkData::MakeWithCString(str));
215*c8dee2aaSAndroid Build Coastguard Worker sk_sp<SkData> data1(SkData::MakeEmpty());
216*c8dee2aaSAndroid Build Coastguard Worker
217*c8dee2aaSAndroid Build Coastguard Worker const size_t sizes[] = {
218*c8dee2aaSAndroid Build Coastguard Worker SkWriter32::WriteDataSize(nullptr),
219*c8dee2aaSAndroid Build Coastguard Worker SkWriter32::WriteDataSize(data0.get()),
220*c8dee2aaSAndroid Build Coastguard Worker SkWriter32::WriteDataSize(data1.get()),
221*c8dee2aaSAndroid Build Coastguard Worker };
222*c8dee2aaSAndroid Build Coastguard Worker
223*c8dee2aaSAndroid Build Coastguard Worker SkSWriter32<1000> writer;
224*c8dee2aaSAndroid Build Coastguard Worker size_t sizeWritten = 0;
225*c8dee2aaSAndroid Build Coastguard Worker
226*c8dee2aaSAndroid Build Coastguard Worker writer.writeData(nullptr);
227*c8dee2aaSAndroid Build Coastguard Worker sizeWritten += sizes[0];
228*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(reporter, sizeWritten == writer.bytesWritten());
229*c8dee2aaSAndroid Build Coastguard Worker
230*c8dee2aaSAndroid Build Coastguard Worker writer.writeData(data0.get());
231*c8dee2aaSAndroid Build Coastguard Worker sizeWritten += sizes[1];
232*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(reporter, sizeWritten == writer.bytesWritten());
233*c8dee2aaSAndroid Build Coastguard Worker
234*c8dee2aaSAndroid Build Coastguard Worker writer.writeData(data1.get());
235*c8dee2aaSAndroid Build Coastguard Worker sizeWritten += sizes[2];
236*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(reporter, sizeWritten == writer.bytesWritten());
237*c8dee2aaSAndroid Build Coastguard Worker
238*c8dee2aaSAndroid Build Coastguard Worker auto result(writer.snapshotAsData());
239*c8dee2aaSAndroid Build Coastguard Worker
240*c8dee2aaSAndroid Build Coastguard Worker SkReadBuffer reader(result->data(), result->size());
241*c8dee2aaSAndroid Build Coastguard Worker auto d0(reader.readByteArrayAsData()),
242*c8dee2aaSAndroid Build Coastguard Worker d1(reader.readByteArrayAsData()),
243*c8dee2aaSAndroid Build Coastguard Worker d2(reader.readByteArrayAsData());
244*c8dee2aaSAndroid Build Coastguard Worker
245*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(reporter, 0 == d0->size());
246*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(reporter, strlen(str)+1 == d1->size());
247*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(reporter, !memcmp(str, d1->data(), strlen(str)+1));
248*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(reporter, 0 == d2->size());
249*c8dee2aaSAndroid Build Coastguard Worker
250*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(reporter, reader.offset() == sizeWritten);
251*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(reporter, reader.eof());
252*c8dee2aaSAndroid Build Coastguard Worker }
253