/* XMRig * Copyright (c) 2012-2013 The Cryptonote developers * Copyright (c) 2014-2021 The Monero Project * Copyright (c) 2018-2021 SChernykh * Copyright (c) 2016-2021 XMRig , * * This program 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 3 of the License, or * (at your option) any later version. * * This program 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 this program. If not, see . */ #ifndef XMRIG_BLOBREADER_H #define XMRIG_BLOBREADER_H #include #include #include namespace xmrig { template class BlobReader { public: inline BlobReader(const uint8_t *data, size_t size) : m_size(size), m_data(data) {} inline bool operator()(uint64_t &data) { return getVarint(data); } inline bool operator()(uint8_t &data) { return getByte(data); } inline size_t index() const { return m_index; } inline size_t remaining() const { return m_size - m_index; } inline bool skip(size_t n) { if (m_index + n > m_size) { return outOfRange(); } m_index += n; return true; } template inline bool operator()(uint8_t(&data)[N]) { if (m_index + N > m_size) { return outOfRange(); } memcpy(data, m_data + m_index, N); m_index += N; return true; } template inline bool operator()(T &data, size_t n) { if (m_index + n > m_size) { return outOfRange(); } data = { m_data + m_index, n }; m_index += n; return true; } private: inline bool getByte(uint8_t &data) { if (m_index >= m_size) { return outOfRange(); } data = m_data[m_index++]; return true; } inline bool getVarint(uint64_t &data) { uint64_t result = 0; uint8_t t; int shift = 0; do { if (!getByte(t)) { return outOfRange(); } result |= static_cast(t & 0x7F) << shift; shift += 7; } while (t & 0x80); data = result; return true; } inline bool outOfRange() { if (EXCEPTIONS) { throw std::out_of_range("Blob read out of range"); } return false; } const size_t m_size; const uint8_t *m_data; size_t m_index = 0; }; } /* namespace xmrig */ #endif /* XMRIG_BLOBREADER_H */