imagecore/utils/memorystream.h (147 lines of code) (raw):

/* * MIT License * * Copyright (c) 2017 Twitter * * Permission is hereby granted, free of charge, to any person obtaining a copy * of this software and associated documentation files (the "Software"), to deal * in the Software without restriction, including without limitation the rights * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell * copies of the Software, and to permit persons to whom the Software is * furnished to do so, subject to the following conditions: * * The above copyright notice and this permission notice shall be included in all * copies or substantial portions of the Software. * * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE * SOFTWARE. */ #pragma once #include "imagecore/imagecore.h" #include "imagecore/utils/endianutils.h" class MemoryStreamReader { public: MemoryStreamReader(const uint8_t* data, uint32_t size, bool is_be) : m_dataStart(data), m_dataEnd(data + size), m_dataCur(data), m_isBe(is_be), m_isLastReadValid(true) {} /* reading */ inline uint8_t get_byte() { uint8_t result = 0; m_isLastReadValid = (m_dataCur + sizeof(uint8_t) <= m_dataEnd); if(m_isLastReadValid) { result = m_dataCur[0]; } return result; } inline uint8_t get_byte_advance() { uint8_t res = get_byte(); m_dataCur += sizeof(uint8_t); return res; } inline uint16_t get_short() { m_isLastReadValid = (m_dataCur + sizeof(uint16_t) <= m_dataEnd); uint16_t result = 0; if(m_isLastReadValid) { if (m_isBe ) { result = (unsigned int)m_dataCur[0] << 8 | ((unsigned int)m_dataCur[1]); } else { result = (unsigned int)m_dataCur[1] << 8 | ((unsigned int)m_dataCur[0]); } } return result; } inline uint16_t get_short_advance() { uint16_t res = get_short(); m_dataCur += sizeof(uint16_t); return res; } inline uint32_t get_uint() { m_isLastReadValid = (m_dataCur + sizeof(uint32_t) <= m_dataEnd); unsigned int result = 0; if(m_isLastReadValid) { if( m_isBe ) { result = (m_dataCur[0] << 24) | (m_dataCur[1] << 16) | (m_dataCur[2] << 8) | (m_dataCur[3]); } else { result = (m_dataCur[3] << 24) | (m_dataCur[2] << 16) | (m_dataCur[1] << 8) | (m_dataCur[0]); } } return result; } inline uint32_t get_uint_advance() { uint32_t res = get_uint(); m_dataCur += sizeof(uint32_t); return res; } inline void advance(uint16_t length) { m_dataCur += length; } inline void seek(uint16_t length) { m_dataCur = m_dataStart + length; } inline bool isLastReadValid() const { return m_isLastReadValid; } private: const uint8_t* m_dataStart; const uint8_t* m_dataEnd; const uint8_t* m_dataCur; bool m_isBe; bool m_isLastReadValid; }; class MemoryStreamWriter { public: MemoryStreamWriter(uint8_t* data, uint32_t size, bool is_be) : m_dataStart(data), m_dataEnd(data + size), m_dataCur(data), m_isBe(is_be) {} /* writing */ inline void put_byte(uint8_t value) { SECURE_ASSERT(m_dataCur + sizeof(uint8_t) <= m_dataEnd); m_dataCur[0] = value; } inline void put_byte_advance(uint8_t value) { put_byte(value); m_dataCur += sizeof(uint8_t); } inline void put_short(uint16_t value) { SECURE_ASSERT(m_dataCur + sizeof(uint16_t) <= m_dataEnd); if (m_isBe ) { value = htobe16(value); } memcpy(m_dataCur, &value, sizeof(uint16_t)); } inline void put_short_advance(uint16_t value) { put_short(value); m_dataCur += sizeof(uint16_t); } inline void put_uint(uint32_t value) { SECURE_ASSERT(m_dataCur + sizeof(uint32_t) <= m_dataEnd); if (m_isBe ) { value = htobe32(value); } memcpy(m_dataCur, &value, sizeof(uint32_t)); } inline void put_uint_advance(uint32_t value) { put_uint(value); m_dataCur += sizeof(uint32_t); } inline void put_raw_data_advance(const uint8_t* data, uint32_t size) { SECURE_ASSERT(m_dataCur + size <= m_dataEnd); memcpy(m_dataCur, data, size); m_dataCur += size; } inline void advance(uint16_t length) { m_dataCur += length; } inline uint32_t get_offset() { return (uint32_t)(m_dataCur - m_dataStart); } inline const uint8_t* getData() { return m_dataStart; } inline uint32_t getSize() { return (uint32_t)(m_dataCur - m_dataStart); } private: const uint8_t* m_dataStart; const uint8_t* m_dataEnd; uint8_t* m_dataCur; bool m_isBe; };