/* This file is part of cpp-ethereum. cpp-ethereum is free software: you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation, either version 2 of the License, or (at your option) any later version. Foobar 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 General Public License for more details. You should have received a copy of the GNU General Public License along with Foobar. If not, see . */ /** @file RLP.cpp * @author Gav Wood * @date 2014 */ #include "RLP.h" using namespace std; using namespace eth; bytes eth::RLPNull = rlp(""); bytes eth::RLPEmptyList = rlpList(); RLP::iterator& RLP::iterator::operator++() { if (m_remaining) { m_lastItem.retarget(m_lastItem.next().data(), m_remaining); m_lastItem = m_lastItem.cropped(0, RLP(m_lastItem).actualSize()); m_remaining -= std::min(m_remaining, m_lastItem.size()); } else m_lastItem.retarget(m_lastItem.next().data(), 0); return *this; } RLP::iterator::iterator(RLP const& _parent, bool _begin) { if (_begin && _parent.isList()) { auto pl = _parent.payload(); m_lastItem = pl.cropped(0, RLP(pl).actualSize()); uint t = 0; for (uint i = 0; i < _parent.itemCount(); ++i) t += _parent[i].actualSize(); if (pl.size() != t) cout << _parent.itemCount() << " " << asHex(pl); assert(pl.size() == t); m_remaining = pl.size() - m_lastItem.size(); } else { m_lastItem = _parent.data().cropped(_parent.data().size()); m_remaining = 0; } } RLP RLP::operator[](uint _i) const { if (!isList() || itemCount() <= _i) return RLP(); if (_i < m_lastIndex) { m_lastEnd = RLP(payload()).actualSize(); m_lastItem = payload().cropped(0, m_lastEnd); m_lastIndex = 0; } for (; m_lastIndex < _i; ++m_lastIndex) { m_lastItem = payload().cropped(m_lastEnd); m_lastItem = m_lastItem.cropped(0, RLP(m_lastItem).actualSize()); m_lastEnd += m_lastItem.size(); } return RLP(m_lastItem); } RLPs RLP::toList() const { RLPs ret; if (!isList()) return ret; uint64_t c = items(); for (uint64_t i = 0; i < c; ++i) ret.push_back(operator[](i)); return ret; } eth::uint RLP::actualSize() const { if (isNull()) return 0; if (isSingleByte()) return 1; if (isData()) return payload().data() - m_data.data() + items(); if (isList()) { bytesConstRef d = payload(); uint64_t c = items(); for (uint64_t i = 0; i < c; ++i, d = d.cropped(RLP(d).actualSize())) {} return d.data() - m_data.data(); } return 0; } bool RLP::isInt() const { byte n = m_data[0]; if (n < c_rlpDataImmLenStart) return !!n; else if (n <= c_rlpDataImmLenStart + c_rlpDataImmLenCount) return m_data[1]; else if (n < c_rlpListStart) return m_data[1 + n - c_rlpDataIndLenZero]; else return false; return false; } eth::uint RLP::items() const { uint ret = 0; byte n = m_data[0]; if (n < c_rlpDataImmLenStart) return 1; else if (n <= c_rlpDataImmLenStart + c_rlpDataImmLenCount) return n - c_rlpDataImmLenStart; else if (n < c_rlpListStart) for (int i = 0; i < n - c_rlpDataIndLenZero; ++i) ret = (ret << 8) | m_data[i + 1]; else if (n <= c_rlpListStart + c_rlpDataImmLenCount) return n - c_rlpListStart; else for (int i = 0; i < n - c_rlpListIndLenZero; ++i) ret = (ret << 8) | m_data[i + 1]; return ret; } RLPStream& RLPStream::appendRaw(bytesConstRef _s) { uint os = m_out.size(); m_out.resize(os + _s.size()); memcpy(m_out.data() + os, _s.data(), _s.size()); return *this; } RLPStream& RLPStream::appendList(uint _count) { if (_count < c_rlpListImmLenCount) m_out.push_back((byte)(_count + c_rlpListStart)); else pushCount(_count, c_rlpListIndLenZero); return *this; } RLPStream& RLPStream::append(bytesConstRef _s, bool _compact) { uint s = _s.size(); byte const* d = _s.data(); if (_compact) for (unsigned i = 0; i < _s.size() && !*d; ++i, --s, ++d) {} if (s == 1 && *d < c_rlpDataImmLenStart) m_out.push_back(*d); else { if (s < c_rlpDataImmLenCount) m_out.push_back((byte)(s + c_rlpDataImmLenStart)); else pushCount(s, c_rlpDataIndLenZero); appendRaw(bytesConstRef(d, s)); } return *this; } RLPStream& RLPStream::append(bigint _i) { if (!_i) m_out.push_back(c_rlpDataImmLenStart); else if (_i < c_rlpDataImmLenStart) m_out.push_back((byte)_i); else { uint br = bytesRequired(_i); if (br < c_rlpDataImmLenCount) m_out.push_back((byte)(br + c_rlpDataImmLenStart)); else { auto brbr = bytesRequired(br); m_out.push_back((byte)(c_rlpDataIndLenZero + brbr)); pushInt(br, brbr); } pushInt(_i, br); } return *this; } void RLPStream::pushCount(uint _count, byte _base) { auto br = bytesRequired(_count); m_out.push_back((byte)(br + _base)); // max 8 bytes. pushInt(_count, br); } std::ostream& eth::operator<<(std::ostream& _out, eth::RLP const& _d) { if (_d.isNull()) _out << "null"; else if (_d.isInt()) _out << std::showbase << std::hex << std::nouppercase << _d.toInt(RLP::LaisezFaire) << dec; else if (_d.isData()) _out << eth::escaped(_d.toString(), false); else if (_d.isList()) { _out << "["; int j = 0; for (auto i: _d) _out << (j++ ? ", " : " ") << i; _out << " ]"; } return _out; }