• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  * Copyright 2016 Google Inc.
3  *
4  * Use of this source code is governed by a BSD-style license that can be
5  * found in the LICENSE file.
6  */
7 
8 #ifndef SkStreamBuffer_DEFINED
9 #define SkStreamBuffer_DEFINED
10 
11 #include "include/core/SkData.h"
12 #include "include/core/SkStream.h"
13 #include "include/core/SkTypes.h"
14 #include "include/private/SkTHash.h"
15 
16 /**
17  *  Helper class for reading from a stream that may not have all its data
18  *  available yet.
19  *
20  *  Used by GIFImageReader, and currently set up for that use case.
21  *
22  *  Buffers up to 256 * 3 bytes (256 colors, with 3 bytes each) to support GIF.
23  *  FIXME (scroggo): Make this more general purpose?
24  */
25 class SkStreamBuffer : SkNoncopyable {
26 public:
27     SkStreamBuffer(std::unique_ptr<SkStream>);
28     ~SkStreamBuffer();
29 
30     /**
31      *  Return a pointer the buffered data.
32      *
33      *  The number of bytes buffered is the number passed to buffer()
34      *  after the last call to flush().
35      */
36     const char* get() const;
37 
38     /**
39      *  Buffer from the stream into our buffer.
40      *
41      *  If this call returns true, get() can be used to access |bytes| bytes
42      *  from the stream. In addition, markPosition() can be called to mark this
43      *  position and enable calling getAtPosition() later to retrieve |bytes|
44      *  bytes.
45      *
46      *  @param bytes Total number of bytes desired.
47      *
48      *  @return Whether all bytes were successfully buffered.
49      */
50     bool buffer(size_t bytes);
51 
52     /**
53      *  Flush the buffer.
54      *
55      *  After this call, no bytes are buffered.
56      */
flush()57     void flush() {
58         if (fHasLengthAndPosition) {
59             if (fTrulyBuffered < fBytesBuffered) {
60                 fStream->move(fBytesBuffered - fTrulyBuffered);
61             }
62             fTrulyBuffered = 0;
63         }
64         fPosition += fBytesBuffered;
65         fBytesBuffered = 0;
66     }
67 
68     /**
69      *  Mark the current position in the stream to return to it later.
70      *
71      *  This is the position of the start of the buffer. After this call, a
72      *  a client can call getDataAtPosition to retrieve all the bytes currently
73      *  buffered.
74      *
75      *  @return size_t Position which can be passed to getDataAtPosition later
76      *      to retrieve the data currently buffered.
77      */
78     size_t markPosition();
79 
80     /**
81      *  Retrieve data at position, as previously marked by markPosition().
82      *
83      *  @param position Position to retrieve data, as marked by markPosition().
84      *  @param length   Amount of data required at position.
85      *  @return SkData The data at position.
86      */
87     sk_sp<SkData> getDataAtPosition(size_t position, size_t length);
88 
89 private:
90     inline static constexpr size_t kMaxSize = 256 * 3;
91 
92     std::unique_ptr<SkStream>   fStream;
93     size_t                      fPosition;
94     char                        fBuffer[kMaxSize];
95     size_t                      fBytesBuffered;
96     // If the stream has a length and position, we can make two optimizations:
97     // - We can skip buffering
98     // - During parsing, we can store the position and size of data that is
99     //   needed later during decoding.
100     const bool                  fHasLengthAndPosition;
101     // When fHasLengthAndPosition is true, we do not need to actually buffer
102     // inside buffer(). We'll buffer inside get(). This keeps track of how many
103     // bytes we've buffered inside get(), for the (non-existent) case of:
104     //  buffer(n)
105     //  get()
106     //  buffer(n + u)
107     //  get()
108     // The second call to get() needs to only truly buffer the part that was
109     // not already buffered.
110     mutable size_t              fTrulyBuffered;
111     // Only used if !fHasLengthAndPosition. In that case, markPosition will
112     // copy into an SkData, stored here.
113     SkTHashMap<size_t, SkData*> fMarkedData;
114 };
115 #endif // SkStreamBuffer_DEFINED
116 
117