116 lines
3.5 KiB
C
116 lines
3.5 KiB
C
|
// Tencent is pleased to support the open source community by making RapidJSON available.
|
||
|
//
|
||
|
// Copyright (C) 2015 THL A29 Limited, a Tencent company, and Milo Yip. All rights reserved.
|
||
|
//
|
||
|
// Licensed under the MIT License (the "License"); you may not use this file except
|
||
|
// in compliance with the License. You may obtain a copy of the License at
|
||
|
//
|
||
|
// http://opensource.org/licenses/MIT
|
||
|
//
|
||
|
// Unless required by applicable law or agreed to in writing, software distributed
|
||
|
// under the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR
|
||
|
// CONDITIONS OF ANY KIND, either express or implied. See the License for the
|
||
|
// specific language governing permissions and limitations under the License.
|
||
|
|
||
|
#ifndef RAPIDJSON_ISTREAMWRAPPER_H_
|
||
|
#define RAPIDJSON_ISTREAMWRAPPER_H_
|
||
|
|
||
|
#include "stream.h"
|
||
|
#include <iosfwd>
|
||
|
|
||
|
#ifdef __clang__
|
||
|
RAPIDJSON_DIAG_PUSH
|
||
|
RAPIDJSON_DIAG_OFF(padded)
|
||
|
#endif
|
||
|
|
||
|
#ifdef _MSC_VER
|
||
|
RAPIDJSON_DIAG_PUSH
|
||
|
RAPIDJSON_DIAG_OFF(4351) // new behavior: elements of array 'array' will be default initialized
|
||
|
#endif
|
||
|
|
||
|
RAPIDJSON_NAMESPACE_BEGIN
|
||
|
|
||
|
//! Wrapper of \c std::basic_istream into RapidJSON's Stream concept.
|
||
|
/*!
|
||
|
The classes can be wrapped including but not limited to:
|
||
|
|
||
|
- \c std::istringstream
|
||
|
- \c std::stringstream
|
||
|
- \c std::wistringstream
|
||
|
- \c std::wstringstream
|
||
|
- \c std::ifstream
|
||
|
- \c std::fstream
|
||
|
- \c std::wifstream
|
||
|
- \c std::wfstream
|
||
|
|
||
|
\tparam StreamType Class derived from \c std::basic_istream.
|
||
|
*/
|
||
|
|
||
|
template <typename StreamType>
|
||
|
class BasicIStreamWrapper {
|
||
|
public:
|
||
|
typedef typename StreamType::char_type Ch;
|
||
|
BasicIStreamWrapper(StreamType& stream) : stream_(stream), count_(), peekBuffer_() {}
|
||
|
|
||
|
Ch Peek() const {
|
||
|
typename StreamType::int_type c = stream_.peek();
|
||
|
return RAPIDJSON_LIKELY(c != StreamType::traits_type::eof()) ? static_cast<Ch>(c) : '\0';
|
||
|
}
|
||
|
|
||
|
Ch Take() {
|
||
|
typename StreamType::int_type c = stream_.get();
|
||
|
if (RAPIDJSON_LIKELY(c != StreamType::traits_type::eof())) {
|
||
|
count_++;
|
||
|
return static_cast<Ch>(c);
|
||
|
}
|
||
|
else
|
||
|
return '\0';
|
||
|
}
|
||
|
|
||
|
// tellg() may return -1 when failed. So we count by ourself.
|
||
|
size_t Tell() const { return count_; }
|
||
|
|
||
|
Ch* PutBegin() { RAPIDJSON_ASSERT(false); return 0; }
|
||
|
void Put(Ch) { RAPIDJSON_ASSERT(false); }
|
||
|
void Flush() { RAPIDJSON_ASSERT(false); }
|
||
|
size_t PutEnd(Ch*) { RAPIDJSON_ASSERT(false); return 0; }
|
||
|
|
||
|
// For encoding detection only.
|
||
|
const Ch* Peek4() const {
|
||
|
RAPIDJSON_ASSERT(sizeof(Ch) == 1); // Only usable for byte stream.
|
||
|
int i;
|
||
|
bool hasError = false;
|
||
|
for (i = 0; i < 4; ++i) {
|
||
|
typename StreamType::int_type c = stream_.get();
|
||
|
if (c == StreamType::traits_type::eof()) {
|
||
|
hasError = true;
|
||
|
stream_.clear();
|
||
|
break;
|
||
|
}
|
||
|
peekBuffer_[i] = static_cast<Ch>(c);
|
||
|
}
|
||
|
for (--i; i >= 0; --i)
|
||
|
stream_.putback(peekBuffer_[i]);
|
||
|
return !hasError ? peekBuffer_ : 0;
|
||
|
}
|
||
|
|
||
|
private:
|
||
|
BasicIStreamWrapper(const BasicIStreamWrapper&);
|
||
|
BasicIStreamWrapper& operator=(const BasicIStreamWrapper&);
|
||
|
|
||
|
StreamType& stream_;
|
||
|
size_t count_; //!< Number of characters read. Note:
|
||
|
mutable Ch peekBuffer_[4];
|
||
|
};
|
||
|
|
||
|
typedef BasicIStreamWrapper<std::istream> IStreamWrapper;
|
||
|
typedef BasicIStreamWrapper<std::wistream> WIStreamWrapper;
|
||
|
|
||
|
#if defined(__clang__) || defined(_MSC_VER)
|
||
|
RAPIDJSON_DIAG_POP
|
||
|
#endif
|
||
|
|
||
|
RAPIDJSON_NAMESPACE_END
|
||
|
|
||
|
#endif // RAPIDJSON_ISTREAMWRAPPER_H_
|