xref: /aosp_15_r20/external/skia/tests/StreamTest.cpp (revision c8dee2aa9b3f27cf6c858bd81872bdeb2c07ed17)
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/SkRefCnt.h"
10*c8dee2aaSAndroid Build Coastguard Worker #include "include/core/SkStream.h"
11*c8dee2aaSAndroid Build Coastguard Worker #include "include/core/SkString.h"
12*c8dee2aaSAndroid Build Coastguard Worker #include "include/core/SkTypes.h"
13*c8dee2aaSAndroid Build Coastguard Worker #include "include/private/base/SkTemplates.h"
14*c8dee2aaSAndroid Build Coastguard Worker #include "include/private/base/SkTo.h"
15*c8dee2aaSAndroid Build Coastguard Worker #include "src/base/SkAutoMalloc.h"
16*c8dee2aaSAndroid Build Coastguard Worker #include "src/base/SkBuffer.h"
17*c8dee2aaSAndroid Build Coastguard Worker #include "src/base/SkRandom.h"
18*c8dee2aaSAndroid Build Coastguard Worker #include "src/core/SkOSFile.h"
19*c8dee2aaSAndroid Build Coastguard Worker #include "src/core/SkStreamPriv.h"
20*c8dee2aaSAndroid Build Coastguard Worker #include "src/utils/SkOSPath.h"
21*c8dee2aaSAndroid Build Coastguard Worker #include "tests/Test.h"
22*c8dee2aaSAndroid Build Coastguard Worker #include "tools/Resources.h"
23*c8dee2aaSAndroid Build Coastguard Worker 
24*c8dee2aaSAndroid Build Coastguard Worker #include <algorithm>
25*c8dee2aaSAndroid Build Coastguard Worker #include <climits>
26*c8dee2aaSAndroid Build Coastguard Worker #include <cstdint>
27*c8dee2aaSAndroid Build Coastguard Worker #include <cstdio>
28*c8dee2aaSAndroid Build Coastguard Worker #include <cstring>
29*c8dee2aaSAndroid Build Coastguard Worker #include <functional>
30*c8dee2aaSAndroid Build Coastguard Worker #include <limits>
31*c8dee2aaSAndroid Build Coastguard Worker #include <memory>
32*c8dee2aaSAndroid Build Coastguard Worker #include <string>
33*c8dee2aaSAndroid Build Coastguard Worker 
34*c8dee2aaSAndroid Build Coastguard Worker using namespace skia_private;
35*c8dee2aaSAndroid Build Coastguard Worker 
36*c8dee2aaSAndroid Build Coastguard Worker #ifdef SK_ENABLE_ANDROID_UTILS
37*c8dee2aaSAndroid Build Coastguard Worker #include "client_utils/android/FrontBufferedStream.h"
38*c8dee2aaSAndroid Build Coastguard Worker #endif
39*c8dee2aaSAndroid Build Coastguard Worker 
40*c8dee2aaSAndroid Build Coastguard Worker #ifndef SK_BUILD_FOR_WIN
41*c8dee2aaSAndroid Build Coastguard Worker #include <fcntl.h>
42*c8dee2aaSAndroid Build Coastguard Worker #endif
43*c8dee2aaSAndroid Build Coastguard Worker 
44*c8dee2aaSAndroid Build Coastguard Worker #define MAX_SIZE    (256 * 1024)
45*c8dee2aaSAndroid Build Coastguard Worker 
test_loop_stream(skiatest::Reporter * reporter,SkStream * stream,const void * src,size_t len,int repeat)46*c8dee2aaSAndroid Build Coastguard Worker static void test_loop_stream(skiatest::Reporter* reporter, SkStream* stream,
47*c8dee2aaSAndroid Build Coastguard Worker                              const void* src, size_t len, int repeat) {
48*c8dee2aaSAndroid Build Coastguard Worker     SkAutoSMalloc<256> storage(len);
49*c8dee2aaSAndroid Build Coastguard Worker     void* tmp = storage.get();
50*c8dee2aaSAndroid Build Coastguard Worker 
51*c8dee2aaSAndroid Build Coastguard Worker     for (int i = 0; i < repeat; ++i) {
52*c8dee2aaSAndroid Build Coastguard Worker         size_t bytes = stream->read(tmp, len);
53*c8dee2aaSAndroid Build Coastguard Worker         REPORTER_ASSERT(reporter, bytes == len);
54*c8dee2aaSAndroid Build Coastguard Worker         REPORTER_ASSERT(reporter, !memcmp(tmp, src, len));
55*c8dee2aaSAndroid Build Coastguard Worker     }
56*c8dee2aaSAndroid Build Coastguard Worker 
57*c8dee2aaSAndroid Build Coastguard Worker     // expect EOF
58*c8dee2aaSAndroid Build Coastguard Worker     size_t bytes = stream->read(tmp, 1);
59*c8dee2aaSAndroid Build Coastguard Worker     REPORTER_ASSERT(reporter, 0 == bytes);
60*c8dee2aaSAndroid Build Coastguard Worker     // isAtEnd might not return true until after the first failing read.
61*c8dee2aaSAndroid Build Coastguard Worker     REPORTER_ASSERT(reporter, stream->isAtEnd());
62*c8dee2aaSAndroid Build Coastguard Worker }
63*c8dee2aaSAndroid Build Coastguard Worker 
test_filestreams(skiatest::Reporter * reporter,const char * tmpDir)64*c8dee2aaSAndroid Build Coastguard Worker static void test_filestreams(skiatest::Reporter* reporter, const char* tmpDir) {
65*c8dee2aaSAndroid Build Coastguard Worker     SkString path = SkOSPath::Join(tmpDir, "wstream_test");
66*c8dee2aaSAndroid Build Coastguard Worker 
67*c8dee2aaSAndroid Build Coastguard Worker     const char s[] = "abcdefghijklmnopqrstuvwxyz";
68*c8dee2aaSAndroid Build Coastguard Worker 
69*c8dee2aaSAndroid Build Coastguard Worker     {
70*c8dee2aaSAndroid Build Coastguard Worker         SkFILEWStream writer(path.c_str());
71*c8dee2aaSAndroid Build Coastguard Worker         if (!writer.isValid()) {
72*c8dee2aaSAndroid Build Coastguard Worker             ERRORF(reporter, "Failed to create tmp file %s\n", path.c_str());
73*c8dee2aaSAndroid Build Coastguard Worker             return;
74*c8dee2aaSAndroid Build Coastguard Worker         }
75*c8dee2aaSAndroid Build Coastguard Worker 
76*c8dee2aaSAndroid Build Coastguard Worker         for (int i = 0; i < 100; ++i) {
77*c8dee2aaSAndroid Build Coastguard Worker             writer.write(s, 26);
78*c8dee2aaSAndroid Build Coastguard Worker         }
79*c8dee2aaSAndroid Build Coastguard Worker     }
80*c8dee2aaSAndroid Build Coastguard Worker 
81*c8dee2aaSAndroid Build Coastguard Worker     {
82*c8dee2aaSAndroid Build Coastguard Worker         SkFILEStream stream(path.c_str());
83*c8dee2aaSAndroid Build Coastguard Worker         REPORTER_ASSERT(reporter, stream.isValid());
84*c8dee2aaSAndroid Build Coastguard Worker         test_loop_stream(reporter, &stream, s, 26, 100);
85*c8dee2aaSAndroid Build Coastguard Worker 
86*c8dee2aaSAndroid Build Coastguard Worker         std::unique_ptr<SkStreamAsset> stream2(stream.duplicate());
87*c8dee2aaSAndroid Build Coastguard Worker         test_loop_stream(reporter, stream2.get(), s, 26, 100);
88*c8dee2aaSAndroid Build Coastguard Worker     }
89*c8dee2aaSAndroid Build Coastguard Worker 
90*c8dee2aaSAndroid Build Coastguard Worker     {
91*c8dee2aaSAndroid Build Coastguard Worker         FILE* file = ::fopen(path.c_str(), "rb");
92*c8dee2aaSAndroid Build Coastguard Worker         SkFILEStream stream(file);
93*c8dee2aaSAndroid Build Coastguard Worker         REPORTER_ASSERT(reporter, stream.isValid());
94*c8dee2aaSAndroid Build Coastguard Worker         test_loop_stream(reporter, &stream, s, 26, 100);
95*c8dee2aaSAndroid Build Coastguard Worker 
96*c8dee2aaSAndroid Build Coastguard Worker         std::unique_ptr<SkStreamAsset> stream2(stream.duplicate());
97*c8dee2aaSAndroid Build Coastguard Worker         test_loop_stream(reporter, stream2.get(), s, 26, 100);
98*c8dee2aaSAndroid Build Coastguard Worker     }
99*c8dee2aaSAndroid Build Coastguard Worker }
100*c8dee2aaSAndroid Build Coastguard Worker 
TestWStream(skiatest::Reporter * reporter)101*c8dee2aaSAndroid Build Coastguard Worker static void TestWStream(skiatest::Reporter* reporter) {
102*c8dee2aaSAndroid Build Coastguard Worker     SkDynamicMemoryWStream  ds;
103*c8dee2aaSAndroid Build Coastguard Worker     const char s[] = "abcdefghijklmnopqrstuvwxyz";
104*c8dee2aaSAndroid Build Coastguard Worker     int i;
105*c8dee2aaSAndroid Build Coastguard Worker     for (i = 0; i < 100; i++) {
106*c8dee2aaSAndroid Build Coastguard Worker         REPORTER_ASSERT(reporter, ds.write(s, 26));
107*c8dee2aaSAndroid Build Coastguard Worker     }
108*c8dee2aaSAndroid Build Coastguard Worker     REPORTER_ASSERT(reporter, ds.bytesWritten() == 100 * 26);
109*c8dee2aaSAndroid Build Coastguard Worker 
110*c8dee2aaSAndroid Build Coastguard Worker     char* dst = new char[100 * 26 + 1];
111*c8dee2aaSAndroid Build Coastguard Worker     dst[100*26] = '*';
112*c8dee2aaSAndroid Build Coastguard Worker     ds.copyTo(dst);
113*c8dee2aaSAndroid Build Coastguard Worker     REPORTER_ASSERT(reporter, dst[100*26] == '*');
114*c8dee2aaSAndroid Build Coastguard Worker     for (i = 0; i < 100; i++) {
115*c8dee2aaSAndroid Build Coastguard Worker         REPORTER_ASSERT(reporter, memcmp(&dst[i * 26], s, 26) == 0);
116*c8dee2aaSAndroid Build Coastguard Worker     }
117*c8dee2aaSAndroid Build Coastguard Worker 
118*c8dee2aaSAndroid Build Coastguard Worker     {
119*c8dee2aaSAndroid Build Coastguard Worker         std::unique_ptr<SkStreamAsset> stream(ds.detachAsStream());
120*c8dee2aaSAndroid Build Coastguard Worker         REPORTER_ASSERT(reporter, 100 * 26 == stream->getLength());
121*c8dee2aaSAndroid Build Coastguard Worker         REPORTER_ASSERT(reporter, ds.bytesWritten() == 0);
122*c8dee2aaSAndroid Build Coastguard Worker         test_loop_stream(reporter, stream.get(), s, 26, 100);
123*c8dee2aaSAndroid Build Coastguard Worker 
124*c8dee2aaSAndroid Build Coastguard Worker         std::unique_ptr<SkStreamAsset> stream2(stream->duplicate());
125*c8dee2aaSAndroid Build Coastguard Worker         test_loop_stream(reporter, stream2.get(), s, 26, 100);
126*c8dee2aaSAndroid Build Coastguard Worker 
127*c8dee2aaSAndroid Build Coastguard Worker         std::unique_ptr<SkStreamAsset> stream3(stream->fork());
128*c8dee2aaSAndroid Build Coastguard Worker         REPORTER_ASSERT(reporter, stream3->isAtEnd());
129*c8dee2aaSAndroid Build Coastguard Worker         char tmp;
130*c8dee2aaSAndroid Build Coastguard Worker         size_t bytes = stream->read(&tmp, 1);
131*c8dee2aaSAndroid Build Coastguard Worker         REPORTER_ASSERT(reporter, 0 == bytes);
132*c8dee2aaSAndroid Build Coastguard Worker         stream3->rewind();
133*c8dee2aaSAndroid Build Coastguard Worker         test_loop_stream(reporter, stream3.get(), s, 26, 100);
134*c8dee2aaSAndroid Build Coastguard Worker     }
135*c8dee2aaSAndroid Build Coastguard Worker 
136*c8dee2aaSAndroid Build Coastguard Worker     for (i = 0; i < 100; i++) {
137*c8dee2aaSAndroid Build Coastguard Worker         REPORTER_ASSERT(reporter, ds.write(s, 26));
138*c8dee2aaSAndroid Build Coastguard Worker     }
139*c8dee2aaSAndroid Build Coastguard Worker     REPORTER_ASSERT(reporter, ds.bytesWritten() == 100 * 26);
140*c8dee2aaSAndroid Build Coastguard Worker 
141*c8dee2aaSAndroid Build Coastguard Worker     {
142*c8dee2aaSAndroid Build Coastguard Worker         // Test that this works after a snapshot.
143*c8dee2aaSAndroid Build Coastguard Worker         std::unique_ptr<SkStreamAsset> stream(ds.detachAsStream());
144*c8dee2aaSAndroid Build Coastguard Worker         REPORTER_ASSERT(reporter, ds.bytesWritten() == 0);
145*c8dee2aaSAndroid Build Coastguard Worker         test_loop_stream(reporter, stream.get(), s, 26, 100);
146*c8dee2aaSAndroid Build Coastguard Worker 
147*c8dee2aaSAndroid Build Coastguard Worker         std::unique_ptr<SkStreamAsset> stream2(stream->duplicate());
148*c8dee2aaSAndroid Build Coastguard Worker         test_loop_stream(reporter, stream2.get(), s, 26, 100);
149*c8dee2aaSAndroid Build Coastguard Worker     }
150*c8dee2aaSAndroid Build Coastguard Worker     delete[] dst;
151*c8dee2aaSAndroid Build Coastguard Worker 
152*c8dee2aaSAndroid Build Coastguard Worker     SkString tmpDir = skiatest::GetTmpDir();
153*c8dee2aaSAndroid Build Coastguard Worker     if (!tmpDir.isEmpty()) {
154*c8dee2aaSAndroid Build Coastguard Worker         test_filestreams(reporter, tmpDir.c_str());
155*c8dee2aaSAndroid Build Coastguard Worker     }
156*c8dee2aaSAndroid Build Coastguard Worker }
157*c8dee2aaSAndroid Build Coastguard Worker 
TestPackedUInt(skiatest::Reporter * reporter)158*c8dee2aaSAndroid Build Coastguard Worker static void TestPackedUInt(skiatest::Reporter* reporter) {
159*c8dee2aaSAndroid Build Coastguard Worker     // we know that packeduint tries to write 1, 2 or 4 bytes for the length,
160*c8dee2aaSAndroid Build Coastguard Worker     // so we test values around each of those transitions (and a few others)
161*c8dee2aaSAndroid Build Coastguard Worker     const size_t sizes[] = {
162*c8dee2aaSAndroid Build Coastguard Worker         0, 1, 2, 0xFC, 0xFD, 0xFE, 0xFF, 0x100, 0x101, 32767, 32768, 32769,
163*c8dee2aaSAndroid Build Coastguard Worker         0xFFFD, 0xFFFE, 0xFFFF, 0x10000, 0x10001,
164*c8dee2aaSAndroid Build Coastguard Worker         0xFFFFFD, 0xFFFFFE, 0xFFFFFF, 0x1000000, 0x1000001,
165*c8dee2aaSAndroid Build Coastguard Worker         0x7FFFFFFE, 0x7FFFFFFF, 0x80000000, 0x80000001, 0xFFFFFFFE, 0xFFFFFFFF
166*c8dee2aaSAndroid Build Coastguard Worker     };
167*c8dee2aaSAndroid Build Coastguard Worker 
168*c8dee2aaSAndroid Build Coastguard Worker 
169*c8dee2aaSAndroid Build Coastguard Worker     size_t i;
170*c8dee2aaSAndroid Build Coastguard Worker     SkDynamicMemoryWStream wstream;
171*c8dee2aaSAndroid Build Coastguard Worker 
172*c8dee2aaSAndroid Build Coastguard Worker     for (i = 0; i < std::size(sizes); ++i) {
173*c8dee2aaSAndroid Build Coastguard Worker         bool success = wstream.writePackedUInt(sizes[i]);
174*c8dee2aaSAndroid Build Coastguard Worker         REPORTER_ASSERT(reporter, success);
175*c8dee2aaSAndroid Build Coastguard Worker     }
176*c8dee2aaSAndroid Build Coastguard Worker 
177*c8dee2aaSAndroid Build Coastguard Worker     std::unique_ptr<SkStreamAsset> rstream(wstream.detachAsStream());
178*c8dee2aaSAndroid Build Coastguard Worker     for (i = 0; i < std::size(sizes); ++i) {
179*c8dee2aaSAndroid Build Coastguard Worker         size_t n;
180*c8dee2aaSAndroid Build Coastguard Worker         if (!rstream->readPackedUInt(&n)) {
181*c8dee2aaSAndroid Build Coastguard Worker             ERRORF(reporter, "[%zu] sizes:%zx could not be read\n", i, sizes[i]);
182*c8dee2aaSAndroid Build Coastguard Worker         }
183*c8dee2aaSAndroid Build Coastguard Worker         if (sizes[i] != n) {
184*c8dee2aaSAndroid Build Coastguard Worker             ERRORF(reporter, "[%zu] sizes:%zx != n:%zx\n", i, sizes[i], n);
185*c8dee2aaSAndroid Build Coastguard Worker         }
186*c8dee2aaSAndroid Build Coastguard Worker     }
187*c8dee2aaSAndroid Build Coastguard Worker }
188*c8dee2aaSAndroid Build Coastguard Worker 
189*c8dee2aaSAndroid Build Coastguard Worker // Test that setting an SkMemoryStream to a nullptr data does not result in a crash when calling
190*c8dee2aaSAndroid Build Coastguard Worker // methods that access fData.
TestDereferencingData(skiatest::Reporter * reporter,SkMemoryStream * memStream)191*c8dee2aaSAndroid Build Coastguard Worker static void TestDereferencingData(skiatest::Reporter* reporter, SkMemoryStream* memStream) {
192*c8dee2aaSAndroid Build Coastguard Worker     REPORTER_ASSERT(reporter, memStream->read(nullptr, 0) == 0);
193*c8dee2aaSAndroid Build Coastguard Worker     // Reading non-zero bytes from an empty stream should cleanly read zero bytes.
194*c8dee2aaSAndroid Build Coastguard Worker     char buf[1];
195*c8dee2aaSAndroid Build Coastguard Worker     REPORTER_ASSERT(reporter, memStream->read(buf, sizeof(buf)) == 0);
196*c8dee2aaSAndroid Build Coastguard Worker     memStream->getMemoryBase();
197*c8dee2aaSAndroid Build Coastguard Worker     (void)memStream->getData();
198*c8dee2aaSAndroid Build Coastguard Worker }
199*c8dee2aaSAndroid Build Coastguard Worker 
TestNullData(skiatest::Reporter * reporter)200*c8dee2aaSAndroid Build Coastguard Worker static void TestNullData(skiatest::Reporter* reporter) {
201*c8dee2aaSAndroid Build Coastguard Worker     SkMemoryStream memStream(nullptr);
202*c8dee2aaSAndroid Build Coastguard Worker     TestDereferencingData(reporter, &memStream);
203*c8dee2aaSAndroid Build Coastguard Worker 
204*c8dee2aaSAndroid Build Coastguard Worker     memStream.setData(nullptr);
205*c8dee2aaSAndroid Build Coastguard Worker     TestDereferencingData(reporter, &memStream);
206*c8dee2aaSAndroid Build Coastguard Worker 
207*c8dee2aaSAndroid Build Coastguard Worker }
208*c8dee2aaSAndroid Build Coastguard Worker 
DEF_TEST(Stream,reporter)209*c8dee2aaSAndroid Build Coastguard Worker DEF_TEST(Stream, reporter) {
210*c8dee2aaSAndroid Build Coastguard Worker     TestWStream(reporter);
211*c8dee2aaSAndroid Build Coastguard Worker     TestPackedUInt(reporter);
212*c8dee2aaSAndroid Build Coastguard Worker     TestNullData(reporter);
213*c8dee2aaSAndroid Build Coastguard Worker }
214*c8dee2aaSAndroid Build Coastguard Worker 
215*c8dee2aaSAndroid Build Coastguard Worker #ifndef SK_BUILD_FOR_IOS
216*c8dee2aaSAndroid Build Coastguard Worker /**
217*c8dee2aaSAndroid Build Coastguard Worker  *  Tests peeking and then reading the same amount. The two should provide the
218*c8dee2aaSAndroid Build Coastguard Worker  *  same results.
219*c8dee2aaSAndroid Build Coastguard Worker  *  Returns the amount successfully read minus the amount successfully peeked.
220*c8dee2aaSAndroid Build Coastguard Worker  */
compare_peek_to_read(skiatest::Reporter * reporter,SkStream * stream,size_t bytesToPeek)221*c8dee2aaSAndroid Build Coastguard Worker static size_t compare_peek_to_read(skiatest::Reporter* reporter,
222*c8dee2aaSAndroid Build Coastguard Worker                                    SkStream* stream, size_t bytesToPeek) {
223*c8dee2aaSAndroid Build Coastguard Worker     // The rest of our tests won't be very interesting if bytesToPeek is zero.
224*c8dee2aaSAndroid Build Coastguard Worker     REPORTER_ASSERT(reporter, bytesToPeek > 0);
225*c8dee2aaSAndroid Build Coastguard Worker     SkAutoMalloc peekStorage(bytesToPeek);
226*c8dee2aaSAndroid Build Coastguard Worker     SkAutoMalloc readStorage(bytesToPeek);
227*c8dee2aaSAndroid Build Coastguard Worker     void* peekPtr = peekStorage.get();
228*c8dee2aaSAndroid Build Coastguard Worker     void* readPtr = peekStorage.get();
229*c8dee2aaSAndroid Build Coastguard Worker 
230*c8dee2aaSAndroid Build Coastguard Worker     const size_t bytesPeeked = stream->peek(peekPtr, bytesToPeek);
231*c8dee2aaSAndroid Build Coastguard Worker     const size_t bytesRead = stream->read(readPtr, bytesToPeek);
232*c8dee2aaSAndroid Build Coastguard Worker 
233*c8dee2aaSAndroid Build Coastguard Worker     // bytesRead should only be less than attempted if the stream is at the
234*c8dee2aaSAndroid Build Coastguard Worker     // end.
235*c8dee2aaSAndroid Build Coastguard Worker     REPORTER_ASSERT(reporter, bytesRead == bytesToPeek || stream->isAtEnd());
236*c8dee2aaSAndroid Build Coastguard Worker 
237*c8dee2aaSAndroid Build Coastguard Worker     // peek and read should behave the same, except peek returned to the
238*c8dee2aaSAndroid Build Coastguard Worker     // original position, so they read the same data.
239*c8dee2aaSAndroid Build Coastguard Worker     REPORTER_ASSERT(reporter, !memcmp(peekPtr, readPtr, bytesPeeked));
240*c8dee2aaSAndroid Build Coastguard Worker 
241*c8dee2aaSAndroid Build Coastguard Worker     // A stream should never be able to peek more than it can read.
242*c8dee2aaSAndroid Build Coastguard Worker     REPORTER_ASSERT(reporter, bytesRead >= bytesPeeked);
243*c8dee2aaSAndroid Build Coastguard Worker 
244*c8dee2aaSAndroid Build Coastguard Worker     return bytesRead - bytesPeeked;
245*c8dee2aaSAndroid Build Coastguard Worker }
246*c8dee2aaSAndroid Build Coastguard Worker 
test_fully_peekable_stream(skiatest::Reporter * r,SkStream * stream,size_t limit)247*c8dee2aaSAndroid Build Coastguard Worker static void test_fully_peekable_stream(skiatest::Reporter* r, SkStream* stream, size_t limit) {
248*c8dee2aaSAndroid Build Coastguard Worker     for (size_t i = 1; !stream->isAtEnd(); i++) {
249*c8dee2aaSAndroid Build Coastguard Worker         REPORTER_ASSERT(r, compare_peek_to_read(r, stream, i) == 0);
250*c8dee2aaSAndroid Build Coastguard Worker     }
251*c8dee2aaSAndroid Build Coastguard Worker }
252*c8dee2aaSAndroid Build Coastguard Worker 
253*c8dee2aaSAndroid Build Coastguard Worker #ifdef SK_ENABLE_ANDROID_UTILS
test_peeking_front_buffered_stream(skiatest::Reporter * r,const SkStream & original,size_t bufferSize)254*c8dee2aaSAndroid Build Coastguard Worker static void test_peeking_front_buffered_stream(skiatest::Reporter* r,
255*c8dee2aaSAndroid Build Coastguard Worker                                                const SkStream& original,
256*c8dee2aaSAndroid Build Coastguard Worker                                                size_t bufferSize) {
257*c8dee2aaSAndroid Build Coastguard Worker     std::unique_ptr<SkStream> dupe(original.duplicate());
258*c8dee2aaSAndroid Build Coastguard Worker     REPORTER_ASSERT(r, dupe != nullptr);
259*c8dee2aaSAndroid Build Coastguard Worker     auto bufferedStream = android::skia::FrontBufferedStream::Make(
260*c8dee2aaSAndroid Build Coastguard Worker             std::move(dupe), bufferSize);
261*c8dee2aaSAndroid Build Coastguard Worker     REPORTER_ASSERT(r, bufferedStream != nullptr);
262*c8dee2aaSAndroid Build Coastguard Worker 
263*c8dee2aaSAndroid Build Coastguard Worker     size_t peeked = 0;
264*c8dee2aaSAndroid Build Coastguard Worker     for (size_t i = 1; !bufferedStream->isAtEnd(); i++) {
265*c8dee2aaSAndroid Build Coastguard Worker         const size_t unpeekableBytes = compare_peek_to_read(r, bufferedStream.get(), i);
266*c8dee2aaSAndroid Build Coastguard Worker         if (unpeekableBytes > 0) {
267*c8dee2aaSAndroid Build Coastguard Worker             // This could not have returned a number greater than i.
268*c8dee2aaSAndroid Build Coastguard Worker             REPORTER_ASSERT(r, unpeekableBytes <= i);
269*c8dee2aaSAndroid Build Coastguard Worker 
270*c8dee2aaSAndroid Build Coastguard Worker             // We have reached the end of the buffer. Verify that it was at least
271*c8dee2aaSAndroid Build Coastguard Worker             // bufferSize.
272*c8dee2aaSAndroid Build Coastguard Worker             REPORTER_ASSERT(r, peeked + i - unpeekableBytes >= bufferSize);
273*c8dee2aaSAndroid Build Coastguard Worker             // No more peeking is supported.
274*c8dee2aaSAndroid Build Coastguard Worker             break;
275*c8dee2aaSAndroid Build Coastguard Worker         }
276*c8dee2aaSAndroid Build Coastguard Worker         peeked += i;
277*c8dee2aaSAndroid Build Coastguard Worker     }
278*c8dee2aaSAndroid Build Coastguard Worker 
279*c8dee2aaSAndroid Build Coastguard Worker     // Test that attempting to peek beyond the length of the buffer does not prevent rewinding.
280*c8dee2aaSAndroid Build Coastguard Worker     bufferedStream = android::skia::FrontBufferedStream::Make(original.duplicate(), bufferSize);
281*c8dee2aaSAndroid Build Coastguard Worker     REPORTER_ASSERT(r, bufferedStream != nullptr);
282*c8dee2aaSAndroid Build Coastguard Worker 
283*c8dee2aaSAndroid Build Coastguard Worker     const size_t bytesToPeek = bufferSize + 1;
284*c8dee2aaSAndroid Build Coastguard Worker     SkAutoMalloc peekStorage(bytesToPeek);
285*c8dee2aaSAndroid Build Coastguard Worker     SkAutoMalloc readStorage(bytesToPeek);
286*c8dee2aaSAndroid Build Coastguard Worker 
287*c8dee2aaSAndroid Build Coastguard Worker     for (size_t start = 0; start <= bufferSize; start++) {
288*c8dee2aaSAndroid Build Coastguard Worker         // Skip to the starting point
289*c8dee2aaSAndroid Build Coastguard Worker         REPORTER_ASSERT(r, bufferedStream->skip(start) == start);
290*c8dee2aaSAndroid Build Coastguard Worker 
291*c8dee2aaSAndroid Build Coastguard Worker         const size_t bytesPeeked = bufferedStream->peek(peekStorage.get(), bytesToPeek);
292*c8dee2aaSAndroid Build Coastguard Worker         if (0 == bytesPeeked) {
293*c8dee2aaSAndroid Build Coastguard Worker             // Peeking should only fail completely if we have read/skipped beyond the buffer.
294*c8dee2aaSAndroid Build Coastguard Worker             REPORTER_ASSERT(r, start >= bufferSize);
295*c8dee2aaSAndroid Build Coastguard Worker             break;
296*c8dee2aaSAndroid Build Coastguard Worker         }
297*c8dee2aaSAndroid Build Coastguard Worker 
298*c8dee2aaSAndroid Build Coastguard Worker         // Only read the amount that was successfully peeked.
299*c8dee2aaSAndroid Build Coastguard Worker         const size_t bytesRead = bufferedStream->read(readStorage.get(), bytesPeeked);
300*c8dee2aaSAndroid Build Coastguard Worker         REPORTER_ASSERT(r, bytesRead == bytesPeeked);
301*c8dee2aaSAndroid Build Coastguard Worker         REPORTER_ASSERT(r, !memcmp(peekStorage.get(), readStorage.get(), bytesPeeked));
302*c8dee2aaSAndroid Build Coastguard Worker 
303*c8dee2aaSAndroid Build Coastguard Worker         // This should be safe to rewind.
304*c8dee2aaSAndroid Build Coastguard Worker         REPORTER_ASSERT(r, bufferedStream->rewind());
305*c8dee2aaSAndroid Build Coastguard Worker     }
306*c8dee2aaSAndroid Build Coastguard Worker }
307*c8dee2aaSAndroid Build Coastguard Worker #endif
308*c8dee2aaSAndroid Build Coastguard Worker 
309*c8dee2aaSAndroid Build Coastguard Worker // This test uses file system operations that don't work out of the
310*c8dee2aaSAndroid Build Coastguard Worker // box on iOS. It's likely that we don't need them on iOS. Ignoring for now.
311*c8dee2aaSAndroid Build Coastguard Worker // TODO(stephana): Re-evaluate if we need this in the future.
DEF_TEST(StreamPeek,reporter)312*c8dee2aaSAndroid Build Coastguard Worker DEF_TEST(StreamPeek, reporter) {
313*c8dee2aaSAndroid Build Coastguard Worker     // Test a memory stream.
314*c8dee2aaSAndroid Build Coastguard Worker     const char gAbcs[] = "abcdefghijklmnopqrstuvwxyz";
315*c8dee2aaSAndroid Build Coastguard Worker     SkMemoryStream memStream(gAbcs, strlen(gAbcs), false);
316*c8dee2aaSAndroid Build Coastguard Worker     test_fully_peekable_stream(reporter, &memStream, memStream.getLength());
317*c8dee2aaSAndroid Build Coastguard Worker 
318*c8dee2aaSAndroid Build Coastguard Worker     // Test an arbitrary file stream. file streams do not support peeking.
319*c8dee2aaSAndroid Build Coastguard Worker     auto tmpdir = skiatest::GetTmpDir();
320*c8dee2aaSAndroid Build Coastguard Worker     if (tmpdir.isEmpty()) {
321*c8dee2aaSAndroid Build Coastguard Worker         ERRORF(reporter, "no tmp dir!");
322*c8dee2aaSAndroid Build Coastguard Worker         return;
323*c8dee2aaSAndroid Build Coastguard Worker     }
324*c8dee2aaSAndroid Build Coastguard Worker     auto path = SkOSPath::Join(tmpdir.c_str(), "file");
325*c8dee2aaSAndroid Build Coastguard Worker     {
326*c8dee2aaSAndroid Build Coastguard Worker         SkFILEWStream wStream(path.c_str());
327*c8dee2aaSAndroid Build Coastguard Worker         constexpr char filename[] = "images/baby_tux.webp";
328*c8dee2aaSAndroid Build Coastguard Worker         auto data = GetResourceAsData(filename);
329*c8dee2aaSAndroid Build Coastguard Worker         if (!data || data->size() == 0) {
330*c8dee2aaSAndroid Build Coastguard Worker             ERRORF(reporter, "resource missing: %s\n", filename);
331*c8dee2aaSAndroid Build Coastguard Worker             return;
332*c8dee2aaSAndroid Build Coastguard Worker         }
333*c8dee2aaSAndroid Build Coastguard Worker         if (!wStream.isValid() || !wStream.write(data->data(), data->size())) {
334*c8dee2aaSAndroid Build Coastguard Worker             ERRORF(reporter, "error wrtiting to file %s", path.c_str());
335*c8dee2aaSAndroid Build Coastguard Worker             return;
336*c8dee2aaSAndroid Build Coastguard Worker         }
337*c8dee2aaSAndroid Build Coastguard Worker     }
338*c8dee2aaSAndroid Build Coastguard Worker     SkFILEStream fileStream(path.c_str());
339*c8dee2aaSAndroid Build Coastguard Worker     REPORTER_ASSERT(reporter, fileStream.isValid());
340*c8dee2aaSAndroid Build Coastguard Worker     if (!fileStream.isValid()) {
341*c8dee2aaSAndroid Build Coastguard Worker         return;
342*c8dee2aaSAndroid Build Coastguard Worker     }
343*c8dee2aaSAndroid Build Coastguard Worker     SkAutoMalloc storage(fileStream.getLength());
344*c8dee2aaSAndroid Build Coastguard Worker     for (size_t i = 1; i < fileStream.getLength(); i++) {
345*c8dee2aaSAndroid Build Coastguard Worker         REPORTER_ASSERT(reporter, fileStream.peek(storage.get(), i) == 0);
346*c8dee2aaSAndroid Build Coastguard Worker     }
347*c8dee2aaSAndroid Build Coastguard Worker 
348*c8dee2aaSAndroid Build Coastguard Worker #ifdef SK_ENABLE_ANDROID_UTILS
349*c8dee2aaSAndroid Build Coastguard Worker     // Now test some FrontBufferedStreams
350*c8dee2aaSAndroid Build Coastguard Worker     for (size_t i = 1; i < memStream.getLength(); i++) {
351*c8dee2aaSAndroid Build Coastguard Worker         test_peeking_front_buffered_stream(reporter, memStream, i);
352*c8dee2aaSAndroid Build Coastguard Worker     }
353*c8dee2aaSAndroid Build Coastguard Worker #endif
354*c8dee2aaSAndroid Build Coastguard Worker }
355*c8dee2aaSAndroid Build Coastguard Worker #endif
356*c8dee2aaSAndroid Build Coastguard Worker 
357*c8dee2aaSAndroid Build Coastguard Worker // Asserts that asset == expected and is peekable.
stream_peek_test(skiatest::Reporter * rep,SkStreamAsset * asset,const SkData * expected)358*c8dee2aaSAndroid Build Coastguard Worker static void stream_peek_test(skiatest::Reporter* rep,
359*c8dee2aaSAndroid Build Coastguard Worker                              SkStreamAsset* asset,
360*c8dee2aaSAndroid Build Coastguard Worker                              const SkData* expected) {
361*c8dee2aaSAndroid Build Coastguard Worker     if (asset->getLength() != expected->size()) {
362*c8dee2aaSAndroid Build Coastguard Worker         ERRORF(rep, "Unexpected length.");
363*c8dee2aaSAndroid Build Coastguard Worker         return;
364*c8dee2aaSAndroid Build Coastguard Worker     }
365*c8dee2aaSAndroid Build Coastguard Worker     SkRandom rand;
366*c8dee2aaSAndroid Build Coastguard Worker     uint8_t buffer[4096];
367*c8dee2aaSAndroid Build Coastguard Worker     const uint8_t* expect = expected->bytes();
368*c8dee2aaSAndroid Build Coastguard Worker     for (size_t i = 0; i < asset->getLength(); ++i) {
369*c8dee2aaSAndroid Build Coastguard Worker         uint32_t maxSize =
370*c8dee2aaSAndroid Build Coastguard Worker                 SkToU32(std::min(sizeof(buffer), asset->getLength() - i));
371*c8dee2aaSAndroid Build Coastguard Worker         size_t size = rand.nextRangeU(1, maxSize);
372*c8dee2aaSAndroid Build Coastguard Worker         SkASSERT(size >= 1);
373*c8dee2aaSAndroid Build Coastguard Worker         SkASSERT(size <= sizeof(buffer));
374*c8dee2aaSAndroid Build Coastguard Worker         SkASSERT(size + i <= asset->getLength());
375*c8dee2aaSAndroid Build Coastguard Worker         if (asset->peek(buffer, size) < size) {
376*c8dee2aaSAndroid Build Coastguard Worker             ERRORF(rep, "Peek Failed!");
377*c8dee2aaSAndroid Build Coastguard Worker             return;
378*c8dee2aaSAndroid Build Coastguard Worker         }
379*c8dee2aaSAndroid Build Coastguard Worker         if (0 != memcmp(buffer, &expect[i], size)) {
380*c8dee2aaSAndroid Build Coastguard Worker             ERRORF(rep, "Peek returned wrong bytes!");
381*c8dee2aaSAndroid Build Coastguard Worker             return;
382*c8dee2aaSAndroid Build Coastguard Worker         }
383*c8dee2aaSAndroid Build Coastguard Worker         uint8_t value;
384*c8dee2aaSAndroid Build Coastguard Worker         REPORTER_ASSERT(rep, 1 == asset->read(&value, 1));
385*c8dee2aaSAndroid Build Coastguard Worker         if (value != expect[i]) {
386*c8dee2aaSAndroid Build Coastguard Worker             ERRORF(rep, "Read Failed!");
387*c8dee2aaSAndroid Build Coastguard Worker             return;
388*c8dee2aaSAndroid Build Coastguard Worker         }
389*c8dee2aaSAndroid Build Coastguard Worker     }
390*c8dee2aaSAndroid Build Coastguard Worker }
391*c8dee2aaSAndroid Build Coastguard Worker 
DEF_TEST(StreamPeek_BlockMemoryStream,rep)392*c8dee2aaSAndroid Build Coastguard Worker DEF_TEST(StreamPeek_BlockMemoryStream, rep) {
393*c8dee2aaSAndroid Build Coastguard Worker     const static int kSeed = 1234;
394*c8dee2aaSAndroid Build Coastguard Worker     SkRandom valueSource(kSeed);
395*c8dee2aaSAndroid Build Coastguard Worker     SkRandom rand(kSeed << 1);
396*c8dee2aaSAndroid Build Coastguard Worker     uint8_t buffer[4096];
397*c8dee2aaSAndroid Build Coastguard Worker     SkDynamicMemoryWStream dynamicMemoryWStream;
398*c8dee2aaSAndroid Build Coastguard Worker     size_t totalWritten = 0;
399*c8dee2aaSAndroid Build Coastguard Worker     for (int i = 0; i < 32; ++i) {
400*c8dee2aaSAndroid Build Coastguard Worker         // Randomize the length of the blocks.
401*c8dee2aaSAndroid Build Coastguard Worker         size_t size = rand.nextRangeU(1, sizeof(buffer));
402*c8dee2aaSAndroid Build Coastguard Worker         for (size_t j = 0; j < size; ++j) {
403*c8dee2aaSAndroid Build Coastguard Worker             buffer[j] = valueSource.nextU() & 0xFF;
404*c8dee2aaSAndroid Build Coastguard Worker         }
405*c8dee2aaSAndroid Build Coastguard Worker         dynamicMemoryWStream.write(buffer, size);
406*c8dee2aaSAndroid Build Coastguard Worker         totalWritten += size;
407*c8dee2aaSAndroid Build Coastguard Worker         REPORTER_ASSERT(rep, totalWritten == dynamicMemoryWStream.bytesWritten());
408*c8dee2aaSAndroid Build Coastguard Worker     }
409*c8dee2aaSAndroid Build Coastguard Worker     std::unique_ptr<SkStreamAsset> asset(dynamicMemoryWStream.detachAsStream());
410*c8dee2aaSAndroid Build Coastguard Worker     sk_sp<SkData> expected(SkData::MakeUninitialized(asset->getLength()));
411*c8dee2aaSAndroid Build Coastguard Worker     uint8_t* expectedPtr = static_cast<uint8_t*>(expected->writable_data());
412*c8dee2aaSAndroid Build Coastguard Worker     valueSource.setSeed(kSeed);  // reseed.
413*c8dee2aaSAndroid Build Coastguard Worker     // We want the exact same same "random" string of numbers to put
414*c8dee2aaSAndroid Build Coastguard Worker     // in expected. i.e.: don't rely on SkDynamicMemoryStream to work
415*c8dee2aaSAndroid Build Coastguard Worker     // correctly while we are testing SkDynamicMemoryStream.
416*c8dee2aaSAndroid Build Coastguard Worker     for (size_t i = 0; i < asset->getLength(); ++i) {
417*c8dee2aaSAndroid Build Coastguard Worker         expectedPtr[i] = valueSource.nextU() & 0xFF;
418*c8dee2aaSAndroid Build Coastguard Worker     }
419*c8dee2aaSAndroid Build Coastguard Worker     stream_peek_test(rep, asset.get(), expected.get());
420*c8dee2aaSAndroid Build Coastguard Worker }
421*c8dee2aaSAndroid Build Coastguard Worker 
DEF_TEST(StreamRemainingLengthIsBelow_MemoryStream,rep)422*c8dee2aaSAndroid Build Coastguard Worker DEF_TEST(StreamRemainingLengthIsBelow_MemoryStream, rep) {
423*c8dee2aaSAndroid Build Coastguard Worker     SkMemoryStream stream(100);
424*c8dee2aaSAndroid Build Coastguard Worker     REPORTER_ASSERT(rep, !StreamRemainingLengthIsBelow(&stream, 0));
425*c8dee2aaSAndroid Build Coastguard Worker     REPORTER_ASSERT(rep, !StreamRemainingLengthIsBelow(&stream, 90));
426*c8dee2aaSAndroid Build Coastguard Worker     REPORTER_ASSERT(rep, !StreamRemainingLengthIsBelow(&stream, 100));
427*c8dee2aaSAndroid Build Coastguard Worker 
428*c8dee2aaSAndroid Build Coastguard Worker     REPORTER_ASSERT(rep, StreamRemainingLengthIsBelow(&stream, 101));
429*c8dee2aaSAndroid Build Coastguard Worker     REPORTER_ASSERT(rep, StreamRemainingLengthIsBelow(&stream, ULONG_MAX));
430*c8dee2aaSAndroid Build Coastguard Worker 
431*c8dee2aaSAndroid Build Coastguard Worker     uint8_t buff[75];
432*c8dee2aaSAndroid Build Coastguard Worker     REPORTER_ASSERT(rep, stream.read(buff, 75) == 75);
433*c8dee2aaSAndroid Build Coastguard Worker 
434*c8dee2aaSAndroid Build Coastguard Worker     REPORTER_ASSERT(rep, !StreamRemainingLengthIsBelow(&stream, 0));
435*c8dee2aaSAndroid Build Coastguard Worker     REPORTER_ASSERT(rep, !StreamRemainingLengthIsBelow(&stream, 24));
436*c8dee2aaSAndroid Build Coastguard Worker     REPORTER_ASSERT(rep, !StreamRemainingLengthIsBelow(&stream, 25));
437*c8dee2aaSAndroid Build Coastguard Worker 
438*c8dee2aaSAndroid Build Coastguard Worker     REPORTER_ASSERT(rep, StreamRemainingLengthIsBelow(&stream, 26));
439*c8dee2aaSAndroid Build Coastguard Worker     REPORTER_ASSERT(rep, StreamRemainingLengthIsBelow(&stream, 100));
440*c8dee2aaSAndroid Build Coastguard Worker     REPORTER_ASSERT(rep, StreamRemainingLengthIsBelow(&stream, ULONG_MAX));
441*c8dee2aaSAndroid Build Coastguard Worker }
442*c8dee2aaSAndroid Build Coastguard Worker 
443*c8dee2aaSAndroid Build Coastguard Worker namespace {
444*c8dee2aaSAndroid Build Coastguard Worker class DumbStream : public SkStream {
445*c8dee2aaSAndroid Build Coastguard Worker public:
DumbStream(const uint8_t * data,size_t n)446*c8dee2aaSAndroid Build Coastguard Worker     DumbStream(const uint8_t* data, size_t n)
447*c8dee2aaSAndroid Build Coastguard Worker         : fData(data), fCount(n), fIdx(0) {}
read(void * buffer,size_t size)448*c8dee2aaSAndroid Build Coastguard Worker     size_t read(void* buffer, size_t size) override {
449*c8dee2aaSAndroid Build Coastguard Worker         size_t copyCount = std::min(fCount - fIdx, size);
450*c8dee2aaSAndroid Build Coastguard Worker         if (copyCount) {
451*c8dee2aaSAndroid Build Coastguard Worker             memcpy(buffer, &fData[fIdx], copyCount);
452*c8dee2aaSAndroid Build Coastguard Worker             fIdx += copyCount;
453*c8dee2aaSAndroid Build Coastguard Worker         }
454*c8dee2aaSAndroid Build Coastguard Worker         return copyCount;
455*c8dee2aaSAndroid Build Coastguard Worker     }
isAtEnd() const456*c8dee2aaSAndroid Build Coastguard Worker     bool isAtEnd() const override {
457*c8dee2aaSAndroid Build Coastguard Worker         return fCount == fIdx;
458*c8dee2aaSAndroid Build Coastguard Worker     }
459*c8dee2aaSAndroid Build Coastguard Worker  private:
460*c8dee2aaSAndroid Build Coastguard Worker     const uint8_t* fData;
461*c8dee2aaSAndroid Build Coastguard Worker     size_t fCount, fIdx;
462*c8dee2aaSAndroid Build Coastguard Worker };
463*c8dee2aaSAndroid Build Coastguard Worker }  // namespace
464*c8dee2aaSAndroid Build Coastguard Worker 
stream_copy_test(skiatest::Reporter * reporter,const void * srcData,size_t N,SkStream * stream)465*c8dee2aaSAndroid Build Coastguard Worker static void stream_copy_test(skiatest::Reporter* reporter,
466*c8dee2aaSAndroid Build Coastguard Worker                              const void* srcData,
467*c8dee2aaSAndroid Build Coastguard Worker                              size_t N,
468*c8dee2aaSAndroid Build Coastguard Worker                              SkStream* stream) {
469*c8dee2aaSAndroid Build Coastguard Worker     SkDynamicMemoryWStream tgt;
470*c8dee2aaSAndroid Build Coastguard Worker     if (!SkStreamCopy(&tgt, stream)) {
471*c8dee2aaSAndroid Build Coastguard Worker         ERRORF(reporter, "SkStreamCopy failed");
472*c8dee2aaSAndroid Build Coastguard Worker         return;
473*c8dee2aaSAndroid Build Coastguard Worker     }
474*c8dee2aaSAndroid Build Coastguard Worker     sk_sp<SkData> data(tgt.detachAsData());
475*c8dee2aaSAndroid Build Coastguard Worker     if (data->size() != N) {
476*c8dee2aaSAndroid Build Coastguard Worker         ERRORF(reporter, "SkStreamCopy incorrect size");
477*c8dee2aaSAndroid Build Coastguard Worker         return;
478*c8dee2aaSAndroid Build Coastguard Worker     }
479*c8dee2aaSAndroid Build Coastguard Worker     if (0 != memcmp(data->data(), srcData, N)) {
480*c8dee2aaSAndroid Build Coastguard Worker         ERRORF(reporter, "SkStreamCopy bad copy");
481*c8dee2aaSAndroid Build Coastguard Worker     }
482*c8dee2aaSAndroid Build Coastguard Worker }
483*c8dee2aaSAndroid Build Coastguard Worker 
DEF_TEST(DynamicMemoryWStream_detachAsData,r)484*c8dee2aaSAndroid Build Coastguard Worker DEF_TEST(DynamicMemoryWStream_detachAsData, r) {
485*c8dee2aaSAndroid Build Coastguard Worker     const char az[] = "abcdefghijklmnopqrstuvwxyz";
486*c8dee2aaSAndroid Build Coastguard Worker     const unsigned N = 40000;
487*c8dee2aaSAndroid Build Coastguard Worker     SkDynamicMemoryWStream dmws;
488*c8dee2aaSAndroid Build Coastguard Worker     for (unsigned i = 0; i < N; ++i) {
489*c8dee2aaSAndroid Build Coastguard Worker         dmws.writeText(az);
490*c8dee2aaSAndroid Build Coastguard Worker     }
491*c8dee2aaSAndroid Build Coastguard Worker     REPORTER_ASSERT(r, dmws.bytesWritten() == N * strlen(az));
492*c8dee2aaSAndroid Build Coastguard Worker     auto data = dmws.detachAsData();
493*c8dee2aaSAndroid Build Coastguard Worker     REPORTER_ASSERT(r, data->size() == N * strlen(az));
494*c8dee2aaSAndroid Build Coastguard Worker     const uint8_t* ptr = data->bytes();
495*c8dee2aaSAndroid Build Coastguard Worker     for (unsigned i = 0; i < N; ++i) {
496*c8dee2aaSAndroid Build Coastguard Worker         if (0 != memcmp(ptr, az, strlen(az))) {
497*c8dee2aaSAndroid Build Coastguard Worker             ERRORF(r, "detachAsData() memcmp failed");
498*c8dee2aaSAndroid Build Coastguard Worker             return;
499*c8dee2aaSAndroid Build Coastguard Worker         }
500*c8dee2aaSAndroid Build Coastguard Worker         ptr += strlen(az);
501*c8dee2aaSAndroid Build Coastguard Worker     }
502*c8dee2aaSAndroid Build Coastguard Worker }
503*c8dee2aaSAndroid Build Coastguard Worker 
DEF_TEST(StreamCopy,reporter)504*c8dee2aaSAndroid Build Coastguard Worker DEF_TEST(StreamCopy, reporter) {
505*c8dee2aaSAndroid Build Coastguard Worker     SkRandom random(123456);
506*c8dee2aaSAndroid Build Coastguard Worker     static const int N = 10000;
507*c8dee2aaSAndroid Build Coastguard Worker     AutoTMalloc<uint8_t> src((size_t)N);
508*c8dee2aaSAndroid Build Coastguard Worker     for (int j = 0; j < N; ++j) {
509*c8dee2aaSAndroid Build Coastguard Worker         src[j] = random.nextU() & 0xff;
510*c8dee2aaSAndroid Build Coastguard Worker     }
511*c8dee2aaSAndroid Build Coastguard Worker     // SkStreamCopy had two code paths; this test both.
512*c8dee2aaSAndroid Build Coastguard Worker     DumbStream dumbStream(src.get(), (size_t)N);
513*c8dee2aaSAndroid Build Coastguard Worker     stream_copy_test(reporter, src, N, &dumbStream);
514*c8dee2aaSAndroid Build Coastguard Worker     SkMemoryStream smartStream(src.get(), (size_t)N);
515*c8dee2aaSAndroid Build Coastguard Worker     stream_copy_test(reporter, src, N, &smartStream);
516*c8dee2aaSAndroid Build Coastguard Worker }
517*c8dee2aaSAndroid Build Coastguard Worker 
DEF_TEST(StreamEmptyStreamMemoryBase,r)518*c8dee2aaSAndroid Build Coastguard Worker DEF_TEST(StreamEmptyStreamMemoryBase, r) {
519*c8dee2aaSAndroid Build Coastguard Worker     SkDynamicMemoryWStream tmp;
520*c8dee2aaSAndroid Build Coastguard Worker     std::unique_ptr<SkStreamAsset> asset(tmp.detachAsStream());
521*c8dee2aaSAndroid Build Coastguard Worker     REPORTER_ASSERT(r, nullptr == asset->getMemoryBase());
522*c8dee2aaSAndroid Build Coastguard Worker }
523*c8dee2aaSAndroid Build Coastguard Worker 
DEF_TEST(FILEStreamWithOffset,r)524*c8dee2aaSAndroid Build Coastguard Worker DEF_TEST(FILEStreamWithOffset, r) {
525*c8dee2aaSAndroid Build Coastguard Worker     if (GetResourcePath().isEmpty()) {
526*c8dee2aaSAndroid Build Coastguard Worker         return;
527*c8dee2aaSAndroid Build Coastguard Worker     }
528*c8dee2aaSAndroid Build Coastguard Worker 
529*c8dee2aaSAndroid Build Coastguard Worker     SkString filename = GetResourcePath("images/baby_tux.png");
530*c8dee2aaSAndroid Build Coastguard Worker     SkFILEStream stream1(filename.c_str());
531*c8dee2aaSAndroid Build Coastguard Worker     if (!stream1.isValid()) {
532*c8dee2aaSAndroid Build Coastguard Worker         ERRORF(r, "Could not create SkFILEStream from %s", filename.c_str());
533*c8dee2aaSAndroid Build Coastguard Worker         return;
534*c8dee2aaSAndroid Build Coastguard Worker     }
535*c8dee2aaSAndroid Build Coastguard Worker     REPORTER_ASSERT(r, stream1.hasLength());
536*c8dee2aaSAndroid Build Coastguard Worker     REPORTER_ASSERT(r, stream1.hasPosition());
537*c8dee2aaSAndroid Build Coastguard Worker 
538*c8dee2aaSAndroid Build Coastguard Worker     // Seek halfway through the file. The second SkFILEStream will be created
539*c8dee2aaSAndroid Build Coastguard Worker     // with the same filename and offset and therefore will treat that offset as
540*c8dee2aaSAndroid Build Coastguard Worker     // the beginning.
541*c8dee2aaSAndroid Build Coastguard Worker     const size_t size = stream1.getLength();
542*c8dee2aaSAndroid Build Coastguard Worker     const size_t middle = size / 2;
543*c8dee2aaSAndroid Build Coastguard Worker     if (!stream1.seek(middle)) {
544*c8dee2aaSAndroid Build Coastguard Worker         ERRORF(r, "Could not seek SkFILEStream to %zu out of %zu", middle, size);
545*c8dee2aaSAndroid Build Coastguard Worker         return;
546*c8dee2aaSAndroid Build Coastguard Worker     }
547*c8dee2aaSAndroid Build Coastguard Worker     REPORTER_ASSERT(r, stream1.getPosition() == middle);
548*c8dee2aaSAndroid Build Coastguard Worker 
549*c8dee2aaSAndroid Build Coastguard Worker     FILE* file = sk_fopen(filename.c_str(), kRead_SkFILE_Flag);
550*c8dee2aaSAndroid Build Coastguard Worker     if (!file) {
551*c8dee2aaSAndroid Build Coastguard Worker         ERRORF(r, "Could not open %s as a FILE", filename.c_str());
552*c8dee2aaSAndroid Build Coastguard Worker         return;
553*c8dee2aaSAndroid Build Coastguard Worker     }
554*c8dee2aaSAndroid Build Coastguard Worker 
555*c8dee2aaSAndroid Build Coastguard Worker     if (fseek(file, (long) middle, SEEK_SET) != 0) {
556*c8dee2aaSAndroid Build Coastguard Worker         ERRORF(r, "Could not fseek FILE to %zu out of %zu", middle, size);
557*c8dee2aaSAndroid Build Coastguard Worker         return;
558*c8dee2aaSAndroid Build Coastguard Worker     }
559*c8dee2aaSAndroid Build Coastguard Worker     SkFILEStream stream2(file);
560*c8dee2aaSAndroid Build Coastguard Worker 
561*c8dee2aaSAndroid Build Coastguard Worker     const size_t remaining = size - middle;
562*c8dee2aaSAndroid Build Coastguard Worker     AutoTMalloc<uint8_t> expected(remaining);
563*c8dee2aaSAndroid Build Coastguard Worker     REPORTER_ASSERT(r, stream1.read(expected.get(), remaining) == remaining);
564*c8dee2aaSAndroid Build Coastguard Worker 
565*c8dee2aaSAndroid Build Coastguard Worker     auto test_full_read = [&r, &expected, remaining](SkStream* stream) {
566*c8dee2aaSAndroid Build Coastguard Worker         AutoTMalloc<uint8_t> actual(remaining);
567*c8dee2aaSAndroid Build Coastguard Worker         REPORTER_ASSERT(r, stream->read(actual.get(), remaining) == remaining);
568*c8dee2aaSAndroid Build Coastguard Worker         REPORTER_ASSERT(r, !memcmp(expected.get(), actual.get(), remaining));
569*c8dee2aaSAndroid Build Coastguard Worker 
570*c8dee2aaSAndroid Build Coastguard Worker         REPORTER_ASSERT(r, stream->getPosition() == stream->getLength());
571*c8dee2aaSAndroid Build Coastguard Worker         REPORTER_ASSERT(r, stream->isAtEnd());
572*c8dee2aaSAndroid Build Coastguard Worker     };
573*c8dee2aaSAndroid Build Coastguard Worker 
574*c8dee2aaSAndroid Build Coastguard Worker     auto test_rewind = [&r, &expected, remaining](SkStream* stream) {
575*c8dee2aaSAndroid Build Coastguard Worker         // Rewind goes back to original offset.
576*c8dee2aaSAndroid Build Coastguard Worker         REPORTER_ASSERT(r, stream->rewind());
577*c8dee2aaSAndroid Build Coastguard Worker         REPORTER_ASSERT(r, stream->getPosition() == 0);
578*c8dee2aaSAndroid Build Coastguard Worker         AutoTMalloc<uint8_t> actual(remaining);
579*c8dee2aaSAndroid Build Coastguard Worker         REPORTER_ASSERT(r, stream->read(actual.get(), remaining) == remaining);
580*c8dee2aaSAndroid Build Coastguard Worker         REPORTER_ASSERT(r, !memcmp(expected.get(), actual.get(), remaining));
581*c8dee2aaSAndroid Build Coastguard Worker     };
582*c8dee2aaSAndroid Build Coastguard Worker 
583*c8dee2aaSAndroid Build Coastguard Worker     auto test_move = [&r, &expected, size, remaining](SkStream* stream) {
584*c8dee2aaSAndroid Build Coastguard Worker         // Cannot move to before the original offset.
585*c8dee2aaSAndroid Build Coastguard Worker         REPORTER_ASSERT(r, stream->move(- (long) size));
586*c8dee2aaSAndroid Build Coastguard Worker         REPORTER_ASSERT(r, stream->getPosition() == 0);
587*c8dee2aaSAndroid Build Coastguard Worker 
588*c8dee2aaSAndroid Build Coastguard Worker         REPORTER_ASSERT(r, stream->move(std::numeric_limits<long>::min()));
589*c8dee2aaSAndroid Build Coastguard Worker         REPORTER_ASSERT(r, stream->getPosition() == 0);
590*c8dee2aaSAndroid Build Coastguard Worker 
591*c8dee2aaSAndroid Build Coastguard Worker         AutoTMalloc<uint8_t> actual(remaining);
592*c8dee2aaSAndroid Build Coastguard Worker         REPORTER_ASSERT(r, stream->read(actual.get(), remaining) == remaining);
593*c8dee2aaSAndroid Build Coastguard Worker         REPORTER_ASSERT(r, !memcmp(expected.get(), actual.get(), remaining));
594*c8dee2aaSAndroid Build Coastguard Worker 
595*c8dee2aaSAndroid Build Coastguard Worker         REPORTER_ASSERT(r, stream->isAtEnd());
596*c8dee2aaSAndroid Build Coastguard Worker         REPORTER_ASSERT(r, stream->getPosition() == remaining);
597*c8dee2aaSAndroid Build Coastguard Worker 
598*c8dee2aaSAndroid Build Coastguard Worker         // Cannot move beyond the end.
599*c8dee2aaSAndroid Build Coastguard Worker         REPORTER_ASSERT(r, stream->move(1));
600*c8dee2aaSAndroid Build Coastguard Worker         REPORTER_ASSERT(r, stream->isAtEnd());
601*c8dee2aaSAndroid Build Coastguard Worker         REPORTER_ASSERT(r, stream->getPosition() == remaining);
602*c8dee2aaSAndroid Build Coastguard Worker     };
603*c8dee2aaSAndroid Build Coastguard Worker 
604*c8dee2aaSAndroid Build Coastguard Worker     auto test_seek = [&r, &expected, middle, remaining](SkStream* stream) {
605*c8dee2aaSAndroid Build Coastguard Worker         // Seek to an arbitrary position.
606*c8dee2aaSAndroid Build Coastguard Worker         const size_t arbitrary = middle / 2;
607*c8dee2aaSAndroid Build Coastguard Worker         REPORTER_ASSERT(r, stream->seek(arbitrary));
608*c8dee2aaSAndroid Build Coastguard Worker         REPORTER_ASSERT(r, stream->getPosition() == arbitrary);
609*c8dee2aaSAndroid Build Coastguard Worker         const size_t miniRemaining = remaining - arbitrary;
610*c8dee2aaSAndroid Build Coastguard Worker         AutoTMalloc<uint8_t> actual(miniRemaining);
611*c8dee2aaSAndroid Build Coastguard Worker         REPORTER_ASSERT(r, stream->read(actual.get(), miniRemaining) == miniRemaining);
612*c8dee2aaSAndroid Build Coastguard Worker         REPORTER_ASSERT(r, !memcmp(expected.get() + arbitrary, actual.get(), miniRemaining));
613*c8dee2aaSAndroid Build Coastguard Worker     };
614*c8dee2aaSAndroid Build Coastguard Worker 
615*c8dee2aaSAndroid Build Coastguard Worker     auto test_seek_beginning = [&r, &expected, remaining](SkStream* stream) {
616*c8dee2aaSAndroid Build Coastguard Worker         // Seek to the beginning.
617*c8dee2aaSAndroid Build Coastguard Worker         REPORTER_ASSERT(r, stream->seek(0));
618*c8dee2aaSAndroid Build Coastguard Worker         REPORTER_ASSERT(r, stream->getPosition() == 0);
619*c8dee2aaSAndroid Build Coastguard Worker         AutoTMalloc<uint8_t> actual(remaining);
620*c8dee2aaSAndroid Build Coastguard Worker         REPORTER_ASSERT(r, stream->read(actual.get(), remaining) == remaining);
621*c8dee2aaSAndroid Build Coastguard Worker         REPORTER_ASSERT(r, !memcmp(expected.get(), actual.get(), remaining));
622*c8dee2aaSAndroid Build Coastguard Worker     };
623*c8dee2aaSAndroid Build Coastguard Worker 
624*c8dee2aaSAndroid Build Coastguard Worker     auto test_seek_end = [&r, remaining](SkStream* stream) {
625*c8dee2aaSAndroid Build Coastguard Worker         // Cannot seek past the end.
626*c8dee2aaSAndroid Build Coastguard Worker         REPORTER_ASSERT(r, stream->isAtEnd());
627*c8dee2aaSAndroid Build Coastguard Worker 
628*c8dee2aaSAndroid Build Coastguard Worker         REPORTER_ASSERT(r, stream->seek(remaining + 1));
629*c8dee2aaSAndroid Build Coastguard Worker         REPORTER_ASSERT(r, stream->isAtEnd());
630*c8dee2aaSAndroid Build Coastguard Worker         REPORTER_ASSERT(r, stream->getPosition() == remaining);
631*c8dee2aaSAndroid Build Coastguard Worker 
632*c8dee2aaSAndroid Build Coastguard Worker         const size_t middle = remaining / 2;
633*c8dee2aaSAndroid Build Coastguard Worker         REPORTER_ASSERT(r, stream->seek(middle));
634*c8dee2aaSAndroid Build Coastguard Worker         REPORTER_ASSERT(r, !stream->isAtEnd());
635*c8dee2aaSAndroid Build Coastguard Worker         REPORTER_ASSERT(r, stream->getPosition() == middle);
636*c8dee2aaSAndroid Build Coastguard Worker 
637*c8dee2aaSAndroid Build Coastguard Worker         REPORTER_ASSERT(r, stream->seek(remaining * 2));
638*c8dee2aaSAndroid Build Coastguard Worker         REPORTER_ASSERT(r, stream->isAtEnd());
639*c8dee2aaSAndroid Build Coastguard Worker         REPORTER_ASSERT(r, stream->getPosition() == remaining);
640*c8dee2aaSAndroid Build Coastguard Worker 
641*c8dee2aaSAndroid Build Coastguard Worker         REPORTER_ASSERT(r, stream->seek(std::numeric_limits<long>::max()));
642*c8dee2aaSAndroid Build Coastguard Worker         REPORTER_ASSERT(r, stream->isAtEnd());
643*c8dee2aaSAndroid Build Coastguard Worker         REPORTER_ASSERT(r, stream->getPosition() == remaining);
644*c8dee2aaSAndroid Build Coastguard Worker     };
645*c8dee2aaSAndroid Build Coastguard Worker 
646*c8dee2aaSAndroid Build Coastguard Worker 
647*c8dee2aaSAndroid Build Coastguard Worker     std::function<void (SkStream* stream, bool recurse)> test_all;
648*c8dee2aaSAndroid Build Coastguard Worker     test_all = [&](SkStream* stream, bool recurse) {
649*c8dee2aaSAndroid Build Coastguard Worker         REPORTER_ASSERT(r, stream->getLength() == remaining);
650*c8dee2aaSAndroid Build Coastguard Worker         REPORTER_ASSERT(r, stream->getPosition() == 0);
651*c8dee2aaSAndroid Build Coastguard Worker 
652*c8dee2aaSAndroid Build Coastguard Worker         test_full_read(stream);
653*c8dee2aaSAndroid Build Coastguard Worker         test_rewind(stream);
654*c8dee2aaSAndroid Build Coastguard Worker         test_move(stream);
655*c8dee2aaSAndroid Build Coastguard Worker         test_seek(stream);
656*c8dee2aaSAndroid Build Coastguard Worker         test_seek_beginning(stream);
657*c8dee2aaSAndroid Build Coastguard Worker         test_seek_end(stream);
658*c8dee2aaSAndroid Build Coastguard Worker 
659*c8dee2aaSAndroid Build Coastguard Worker         if (recurse) {
660*c8dee2aaSAndroid Build Coastguard Worker             // Duplicate shares the original offset.
661*c8dee2aaSAndroid Build Coastguard Worker             auto duplicate = stream->duplicate();
662*c8dee2aaSAndroid Build Coastguard Worker             if (!duplicate) {
663*c8dee2aaSAndroid Build Coastguard Worker                 ERRORF(r, "Failed to duplicate the stream!");
664*c8dee2aaSAndroid Build Coastguard Worker             } else {
665*c8dee2aaSAndroid Build Coastguard Worker                 test_all(duplicate.get(), false);
666*c8dee2aaSAndroid Build Coastguard Worker             }
667*c8dee2aaSAndroid Build Coastguard Worker 
668*c8dee2aaSAndroid Build Coastguard Worker             // Fork shares the original offset, too.
669*c8dee2aaSAndroid Build Coastguard Worker             auto fork = stream->fork();
670*c8dee2aaSAndroid Build Coastguard Worker             if (!fork) {
671*c8dee2aaSAndroid Build Coastguard Worker                 ERRORF(r, "Failed to fork the stream!");
672*c8dee2aaSAndroid Build Coastguard Worker             } else {
673*c8dee2aaSAndroid Build Coastguard Worker                 REPORTER_ASSERT(r, fork->isAtEnd());
674*c8dee2aaSAndroid Build Coastguard Worker                 REPORTER_ASSERT(r, fork->getLength() == remaining);
675*c8dee2aaSAndroid Build Coastguard Worker                 REPORTER_ASSERT(r, fork->rewind());
676*c8dee2aaSAndroid Build Coastguard Worker 
677*c8dee2aaSAndroid Build Coastguard Worker                 test_all(fork.get(), false);
678*c8dee2aaSAndroid Build Coastguard Worker             }
679*c8dee2aaSAndroid Build Coastguard Worker         }
680*c8dee2aaSAndroid Build Coastguard Worker     };
681*c8dee2aaSAndroid Build Coastguard Worker 
682*c8dee2aaSAndroid Build Coastguard Worker     test_all(&stream2, true);
683*c8dee2aaSAndroid Build Coastguard Worker }
684*c8dee2aaSAndroid Build Coastguard Worker 
DEF_TEST(RBuffer,reporter)685*c8dee2aaSAndroid Build Coastguard Worker DEF_TEST(RBuffer, reporter) {
686*c8dee2aaSAndroid Build Coastguard Worker     int32_t value = 0;
687*c8dee2aaSAndroid Build Coastguard Worker     SkRBuffer buffer(&value, 4);
688*c8dee2aaSAndroid Build Coastguard Worker     REPORTER_ASSERT(reporter, buffer.isValid());
689*c8dee2aaSAndroid Build Coastguard Worker 
690*c8dee2aaSAndroid Build Coastguard Worker     int32_t tmp;
691*c8dee2aaSAndroid Build Coastguard Worker     REPORTER_ASSERT(reporter, buffer.read(&tmp, 4));
692*c8dee2aaSAndroid Build Coastguard Worker     REPORTER_ASSERT(reporter, buffer.isValid());
693*c8dee2aaSAndroid Build Coastguard Worker 
694*c8dee2aaSAndroid Build Coastguard Worker     REPORTER_ASSERT(reporter, !buffer.read(&tmp, 4));
695*c8dee2aaSAndroid Build Coastguard Worker     REPORTER_ASSERT(reporter, !buffer.isValid());
696*c8dee2aaSAndroid Build Coastguard Worker }
697