// Copyright (c) 2012-2015, The CryptoNote developers, The Bytecoin developers // // This file is part of Bytecoin. // // Bytecoin is free software: you can redistribute it and/or modify // it under the terms of the GNU Lesser General Public License as published by // the Free Software Foundation, either version 3 of the License, or // (at your option) any later version. // // Bytecoin is distributed in the hope that it will be useful, // but WITHOUT ANY WARRANTY; without even the implied warranty of // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the // GNU Lesser General Public License for more details. // // You should have received a copy of the GNU Lesser General Public License // along with Bytecoin. If not, see . #include "BinaryInputStreamSerializer.h" #include "SerializationOverloads.h" #include #include #include namespace { template::digits> typename std::enable_if::value && std::is_unsigned::value, size_t>::type readVarint(std::istream& s, T &i) { size_t read = 0; i = 0; for (int shift = 0;; shift += 7) { if (s.eof()) { return read; } uint8_t byte = s.get(); if (!s) { throw std::runtime_error("Stream read error"); } ++read; if (shift + 7 >= bits && byte >= 1 << (bits - shift)) { throw std::runtime_error("Varint overflow"); } if (byte == 0 && shift != 0) { throw std::runtime_error("Non-canonical varint representation"); } i |= static_cast(byte & 0x7f) << shift; if ((byte & 0x80) == 0) { break; } } return read; } template void readVarintAs(std::istream& s, T &i) { StorageType v; readVarint(s, v); i = static_cast(v); } } namespace CryptoNote { ISerializer::SerializerType BinaryInputStreamSerializer::type() const { return ISerializer::INPUT; } bool BinaryInputStreamSerializer::beginObject(Common::StringView name) { return true; } void BinaryInputStreamSerializer::endObject() { } bool BinaryInputStreamSerializer::beginArray(std::size_t& size, Common::StringView name) { readVarintAs(stream, size); return true; } void BinaryInputStreamSerializer::endArray() { } bool BinaryInputStreamSerializer::operator()(uint8_t& value, Common::StringView name) { readVarint(stream, value); return true; } bool BinaryInputStreamSerializer::operator()(uint32_t& value, Common::StringView name) { readVarint(stream, value); return true; } bool BinaryInputStreamSerializer::operator()(int32_t& value, Common::StringView name) { readVarintAs(stream, value); return true; } bool BinaryInputStreamSerializer::operator()(int64_t& value, Common::StringView name) { readVarintAs(stream, value); return true; } bool BinaryInputStreamSerializer::operator()(uint64_t& value, Common::StringView name) { readVarint(stream, value); return true; } bool BinaryInputStreamSerializer::operator()(bool& value, Common::StringView name) { value = stream.get() != 0; return true; } bool BinaryInputStreamSerializer::operator()(std::string& value, Common::StringView name) { uint64_t size; readVarint(stream, size); if (size > 0) { std::vector temp; temp.resize(size); checkedRead(&temp[0], size); value.reserve(size); value.assign(&temp[0], size); } else { value.clear(); } return true; } bool BinaryInputStreamSerializer::binary(void* value, std::size_t size, Common::StringView name) { checkedRead(static_cast(value), size); return true; } bool BinaryInputStreamSerializer::binary(std::string& value, Common::StringView name) { return (*this)(value, name); } bool BinaryInputStreamSerializer::operator()(double& value, Common::StringView name) { assert(false); //the method is not supported for this type of serialization throw std::runtime_error("double serialization is not supported in BinaryInputStreamSerializer"); return false; } void BinaryInputStreamSerializer::checkedRead(char* buf, size_t size) { if (stream.read(buf, size).gcount() != size) { throw std::runtime_error("Stream read error"); } } }