1 // Tencent is pleased to support the open source community by making RapidJSON available.
2 //
3 // Copyright (C) 2015 THL A29 Limited, a Tencent company, and Milo Yip.
4 //
5 // Licensed under the MIT License (the "License"); you may not use this file except
6 // in compliance with the License. You may obtain a copy of the License at
7 //
8 // http://opensource.org/licenses/MIT
9 //
10 // Unless required by applicable law or agreed to in writing, software distributed
11 // under the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR
12 // CONDITIONS OF ANY KIND, either express or implied. See the License for the
13 // specific language governing permissions and limitations under the License.
14 
15 #ifndef RAPIDJSON_ISTREAMWRAPPER_H_
16 #define RAPIDJSON_ISTREAMWRAPPER_H_
17 
18 #include "stream.h"
19 #include <iosfwd>
20 #include <ios>
21 
22 #ifdef __clang__
23 RAPIDJSON_DIAG_PUSH
RAPIDJSON_DIAG_OFF(padded)24 RAPIDJSON_DIAG_OFF(padded)
25 #elif defined(_MSC_VER)
26 RAPIDJSON_DIAG_PUSH
27 RAPIDJSON_DIAG_OFF(4351) // new behavior: elements of array 'array' will be default initialized
28 #endif
29 
30 RAPIDJSON_NAMESPACE_BEGIN
31 
32 //! Wrapper of \c std::basic_istream into RapidJSON's Stream concept.
33 /*!
34     The classes can be wrapped including but not limited to:
35 
36     - \c std::istringstream
37     - \c std::stringstream
38     - \c std::wistringstream
39     - \c std::wstringstream
40     - \c std::ifstream
41     - \c std::fstream
42     - \c std::wifstream
43     - \c std::wfstream
44 
45     \tparam StreamType Class derived from \c std::basic_istream.
46 */
47 
48 template <typename StreamType>
49 class BasicIStreamWrapper {
50 public:
51     typedef typename StreamType::char_type Ch;
52 
53     //! Constructor.
54     /*!
55         \param stream stream opened for read.
56     */
57     BasicIStreamWrapper(StreamType &stream) : stream_(stream), buffer_(peekBuffer_), bufferSize_(4), bufferLast_(0), current_(buffer_), readCount_(0), count_(0), eof_(false) {
58         Read();
59     }
60 
61     //! Constructor.
62     /*!
63         \param stream stream opened for read.
64         \param buffer user-supplied buffer.
65         \param bufferSize size of buffer in bytes. Must >=4 bytes.
66     */
67     BasicIStreamWrapper(StreamType &stream, char* buffer, size_t bufferSize) : stream_(stream), buffer_(buffer), bufferSize_(bufferSize), bufferLast_(0), current_(buffer_), readCount_(0), count_(0), eof_(false) {
68         RAPIDJSON_ASSERT(bufferSize >= 4);
69         Read();
70     }
71 
72     Ch Peek() const { return *current_; }
73     Ch Take() { Ch c = *current_; Read(); return c; }
74     size_t Tell() const { return count_ + static_cast<size_t>(current_ - buffer_); }
75 
76     // Not implemented
77     void Put(Ch) { RAPIDJSON_ASSERT(false); }
78     void Flush() { RAPIDJSON_ASSERT(false); }
79     Ch* PutBegin() { RAPIDJSON_ASSERT(false); return 0; }
80     size_t PutEnd(Ch*) { RAPIDJSON_ASSERT(false); return 0; }
81 
82     // For encoding detection only.
83     const Ch* Peek4() const {
84         return (current_ + 4 - !eof_ <= bufferLast_) ? current_ : 0;
85     }
86 
87 private:
88     BasicIStreamWrapper();
89     BasicIStreamWrapper(const BasicIStreamWrapper&);
90     BasicIStreamWrapper& operator=(const BasicIStreamWrapper&);
91 
92     void Read() {
93         if (current_ < bufferLast_)
94             ++current_;
95         else if (!eof_) {
96             count_ += readCount_;
97             readCount_ = bufferSize_;
98             bufferLast_ = buffer_ + readCount_ - 1;
99             current_ = buffer_;
100 
101             if (!stream_.read(buffer_, static_cast<std::streamsize>(bufferSize_))) {
102                 readCount_ = static_cast<size_t>(stream_.gcount());
103                 *(bufferLast_ = buffer_ + readCount_) = '\0';
104                 eof_ = true;
105             }
106         }
107     }
108 
109     StreamType &stream_;
110     Ch peekBuffer_[4], *buffer_;
111     size_t bufferSize_;
112     Ch *bufferLast_;
113     Ch *current_;
114     size_t readCount_;
115     size_t count_;  //!< Number of characters read
116     bool eof_;
117 };
118 
119 typedef BasicIStreamWrapper<std::istream> IStreamWrapper;
120 typedef BasicIStreamWrapper<std::wistream> WIStreamWrapper;
121 
122 #if defined(__clang__) || defined(_MSC_VER)
123 RAPIDJSON_DIAG_POP
124 #endif
125 
126 RAPIDJSON_NAMESPACE_END
127 
128 #endif // RAPIDJSON_ISTREAMWRAPPER_H_
129