SkStream.h revision 8d0b5770f8fcfdeb8ad9808e58c49116f14b6190
1/*
2 * Copyright (C) 2006 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 *      http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
17#ifndef SkStream_DEFINED
18#define SkStream_DEFINED
19
20#include "SkRefCnt.h"
21#include "SkScalar.h"
22
23class SkData;
24
25class SK_API SkStream : public SkRefCnt {
26public:
27    virtual ~SkStream();
28    /** Called to rewind to the beginning of the stream. If this cannot be
29        done, return false.
30    */
31    virtual bool rewind() = 0;
32    /** If this stream represents a file, this method returns the file's name.
33        If it does not, it returns NULL (the default behavior).
34    */
35    virtual const char* getFileName();
36    /** Called to read or skip size number of bytes.
37        If buffer is NULL and size > 0, skip that many bytes, returning how many were skipped.
38        If buffer is NULL and size == 0, return the total length of the stream.
39        If buffer != NULL, copy the requested number of bytes into buffer, returning how many were copied.
40        @param buffer   If buffer is NULL, ignore and just skip size bytes, otherwise copy size bytes into buffer
41        @param size The number of bytes to skip or copy
42        @return bytes read on success
43    */
44    virtual size_t read(void* buffer, size_t size) = 0;
45
46    /** Return the total length of the stream.
47    */
48    size_t getLength() { return this->read(NULL, 0); }
49
50    /** Skip the specified number of bytes, returning the actual number
51        of bytes that could be skipped.
52    */
53    size_t skip(size_t bytes);
54
55    /** If the stream is backed by RAM, this method returns the starting
56        address for the data. If not (i.e. it is backed by a file or other
57        structure), this method returns NULL.
58        The default implementation returns NULL.
59    */
60    virtual const void* getMemoryBase();
61
62    int8_t   readS8();
63    int16_t  readS16();
64    int32_t  readS32();
65
66    uint8_t  readU8() { return (uint8_t)this->readS8(); }
67    uint16_t readU16() { return (uint16_t)this->readS16(); }
68    uint32_t readU32() { return (uint32_t)this->readS32(); }
69
70    bool     readBool() { return this->readU8() != 0; }
71    SkScalar readScalar();
72    size_t   readPackedUInt();
73};
74
75class SK_API SkWStream : SkNoncopyable {
76public:
77    virtual ~SkWStream();
78
79    /** Called to write bytes to a SkWStream. Returns true on success
80        @param buffer the address of at least size bytes to be written to the stream
81        @param size The number of bytes in buffer to write to the stream
82        @return true on success
83    */
84    virtual bool write(const void* buffer, size_t size) = 0;
85    virtual void newline();
86    virtual void flush();
87
88    // helpers
89
90    bool    write8(U8CPU);
91    bool    write16(U16CPU);
92    bool    write32(uint32_t);
93
94    bool    writeText(const char text[]);
95    bool    writeDecAsText(int32_t);
96    bool    writeBigDecAsText(int64_t, int minDigits = 0);
97    bool    writeHexAsText(uint32_t, int minDigits = 0);
98    bool    writeScalarAsText(SkScalar);
99
100    bool    writeBool(bool v) { return this->write8(v); }
101    bool    writeScalar(SkScalar);
102    bool    writePackedUInt(size_t);
103
104    bool writeStream(SkStream* input, size_t length);
105};
106
107////////////////////////////////////////////////////////////////////////////////////////
108
109#include "SkString.h"
110
111struct SkFILE;
112
113/** A stream that reads from a FILE*, which is opened in the constructor and
114    closed in the destructor
115 */
116class SkFILEStream : public SkStream {
117public:
118    /** Initialize the stream by calling fopen on the specified path. Will be
119        closed in the destructor.
120     */
121    explicit SkFILEStream(const char path[] = NULL);
122    virtual ~SkFILEStream();
123
124    /** Returns true if the current path could be opened.
125    */
126    bool isValid() const { return fFILE != NULL; }
127    /** Close the current file, and open a new file with the specified
128        path. If path is NULL, just close the current file.
129    */
130    void setPath(const char path[]);
131
132    virtual bool rewind();
133    virtual size_t read(void* buffer, size_t size);
134    virtual const char* getFileName();
135
136private:
137    SkFILE*     fFILE;
138    SkString    fName;
139};
140
141/** A stream that reads from a file descriptor
142 */
143class SkFDStream : public SkStream {
144public:
145    /** Initialize the stream with a dup() of the specified file descriptor.
146        If closeWhenDone is true, then the descriptor will be closed in the
147        destructor.
148     */
149    SkFDStream(int fileDesc, bool closeWhenDone);
150    virtual ~SkFDStream();
151
152    /** Returns true if the current path could be opened.
153     */
154    bool isValid() const { return fFD >= 0; }
155
156    virtual bool rewind();
157    virtual size_t read(void* buffer, size_t size);
158    virtual const char* getFileName() { return NULL; }
159
160private:
161    int     fFD;
162    bool    fCloseWhenDone;
163};
164
165class SkMemoryStream : public SkStream {
166public:
167    SkMemoryStream();
168    /** We allocate (and free) the memory. Write to it via getMemoryBase()
169    */
170    SkMemoryStream(size_t length);
171    /** if copyData is true, the stream makes a private copy of the data
172    */
173    SkMemoryStream(const void* data, size_t length, bool copyData = false);
174    virtual ~SkMemoryStream();
175
176    /** Resets the stream to the specified data and length,
177        just like the constructor.
178        if copyData is true, the stream makes a private copy of the data
179    */
180    virtual void setMemory(const void* data, size_t length,
181                           bool copyData = false);
182    /** Replace any memory buffer with the specified buffer. The caller
183        must have allocated data with sk_malloc or sk_realloc, since it
184        will be freed with sk_free.
185    */
186    void setMemoryOwned(const void* data, size_t length);
187    void skipToAlign4();
188    virtual bool rewind();
189    virtual size_t read(void* buffer, size_t size);
190    virtual const void* getMemoryBase();
191    const void* getAtPos();
192    size_t seek(size_t offset);
193    size_t peek() const { return fOffset; }
194
195private:
196    const void* fSrc;
197    size_t fSize, fOffset;
198    SkBool8 fWeOwnTheData;
199};
200
201/** \class SkBufferStream
202    This is a wrapper class that adds buffering to another stream.
203    The caller can provide the buffer, or ask SkBufferStream to allocated/free
204    it automatically.
205*/
206class SkBufferStream : public SkStream {
207public:
208    /** Provide the stream to be buffered (proxy), and the size of the buffer that
209        should be used. This will be allocated and freed automatically. If bufferSize is 0,
210        a default buffer size will be used.
211        The proxy stream is referenced, and will be unreferenced in when the
212        bufferstream is destroyed.
213    */
214    SkBufferStream(SkStream* proxy, size_t bufferSize = 0);
215    /** Provide the stream to be buffered (proxy), and a buffer and size to be used.
216        This buffer is owned by the caller, and must be at least bufferSize bytes big.
217        Passing NULL for buffer will cause the buffer to be allocated/freed automatically.
218        If buffer is not NULL, it is an error for bufferSize to be 0.
219     The proxy stream is referenced, and will be unreferenced in when the
220     bufferstream is destroyed.
221    */
222    SkBufferStream(SkStream* proxy, void* buffer, size_t bufferSize);
223    virtual ~SkBufferStream();
224
225    virtual bool        rewind();
226    virtual const char* getFileName();
227    virtual size_t      read(void* buffer, size_t size);
228    virtual const void* getMemoryBase();
229
230private:
231    enum {
232        kDefaultBufferSize  = 128
233    };
234    // illegal
235    SkBufferStream(const SkBufferStream&);
236    SkBufferStream& operator=(const SkBufferStream&);
237
238    SkStream*   fProxy;
239    char*       fBuffer;
240    size_t      fOrigBufferSize, fBufferSize, fBufferOffset;
241    bool        fWeOwnTheBuffer;
242
243    void    init(void*, size_t);
244};
245
246/////////////////////////////////////////////////////////////////////////////////////////////
247
248class SkFILEWStream : public SkWStream {
249public:
250            SkFILEWStream(const char path[]);
251    virtual ~SkFILEWStream();
252
253    /** Returns true if the current path could be opened.
254    */
255    bool isValid() const { return fFILE != NULL; }
256
257    virtual bool write(const void* buffer, size_t size);
258    virtual void flush();
259private:
260    SkFILE* fFILE;
261};
262
263class SkMemoryWStream : public SkWStream {
264public:
265    SkMemoryWStream(void* buffer, size_t size);
266    virtual bool write(const void* buffer, size_t size);
267
268private:
269    char*   fBuffer;
270    size_t  fMaxLength;
271    size_t  fBytesWritten;
272};
273
274class SK_API SkDynamicMemoryWStream : public SkWStream {
275public:
276    SkDynamicMemoryWStream();
277    virtual ~SkDynamicMemoryWStream();
278    virtual bool write(const void* buffer, size_t size);
279    // random access write
280    // modifies stream and returns true if offset + size is less than or equal to getOffset()
281    bool write(const void* buffer, size_t offset, size_t size);
282    bool read(void* buffer, size_t offset, size_t size);
283    size_t getOffset() const { return fBytesWritten; }
284
285    // copy what has been written to the stream into dst
286    void    copyTo(void* dst) const;
287    /*  return a cache of the flattened data returned by copyTo().
288        This copy is only valid until the next call to write().
289        The memory is managed by the stream class.
290    */
291    const char* getStream() const;
292
293    // same as getStream, but additionally detach the flattened datat
294    // DEPRECATED : use copyToData() instead
295    const char* detach() const;
296
297    /**
298     *  Return a copy of the data written so far
299     */
300    SkData* copyToData() const;
301
302    // reset the stream to its original state
303    void reset();
304    void padToAlign4();
305private:
306    struct Block;
307    Block*  fHead;
308    Block*  fTail;
309    size_t  fBytesWritten;
310    mutable char*   fCopyToCache;
311};
312
313
314class SkDebugWStream : public SkWStream {
315public:
316    // overrides
317    virtual bool write(const void* buffer, size_t size);
318    virtual void newline();
319};
320
321// for now
322typedef SkFILEStream SkURLStream;
323
324#endif
325