mirror of
https://github.com/OpenRCT2/OpenRCT2
synced 2026-01-17 12:03:07 +01:00
195 lines
4.8 KiB
C++
195 lines
4.8 KiB
C++
#pragma region Copyright (c) 2014-2016 OpenRCT2 Developers
|
|
/*****************************************************************************
|
|
* OpenRCT2, an open source clone of Roller Coaster Tycoon 2.
|
|
*
|
|
* OpenRCT2 is the work of many authors, a full list can be found in contributors.md
|
|
* For more information, visit https://github.com/OpenRCT2/OpenRCT2
|
|
*
|
|
* OpenRCT2 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.
|
|
*
|
|
* A full copy of the GNU General Public License can be found in licence.txt
|
|
*****************************************************************************/
|
|
#pragma endregion
|
|
|
|
#include "Math.hpp"
|
|
#include "Memory.hpp"
|
|
#include "MemoryStream.h"
|
|
|
|
MemoryStream::MemoryStream()
|
|
{
|
|
_access = MEMORY_ACCESS_READ |
|
|
MEMORY_ACCESS_WRITE |
|
|
MEMORY_ACCESS_OWNER;
|
|
_dataCapacity = 0;
|
|
_dataSize = 0;
|
|
_data = nullptr;
|
|
_position = nullptr;
|
|
}
|
|
|
|
MemoryStream::MemoryStream(const MemoryStream ©)
|
|
{
|
|
_access = copy._access;
|
|
_dataCapacity = copy._dataCapacity;
|
|
_dataSize = copy._dataSize;
|
|
|
|
if (_access == MEMORY_ACCESS_OWNER)
|
|
{
|
|
_data = Memory::Duplicate(copy._data, _dataCapacity);
|
|
_position = (void*)((uintptr_t)_data + copy.GetPosition());
|
|
}
|
|
}
|
|
|
|
MemoryStream::MemoryStream(size_t capacity)
|
|
{
|
|
_access = MEMORY_ACCESS_READ |
|
|
MEMORY_ACCESS_WRITE |
|
|
MEMORY_ACCESS_OWNER;
|
|
_dataCapacity = capacity;
|
|
_dataSize = 0;
|
|
_data = Memory::Allocate<void>(capacity);
|
|
_position = _data;
|
|
}
|
|
|
|
MemoryStream::MemoryStream(void * data, size_t dataSize, uint32 access)
|
|
{
|
|
_access = access;
|
|
_dataCapacity = dataSize;
|
|
_dataSize = dataSize;
|
|
_data = data;
|
|
_position = _data;
|
|
}
|
|
|
|
MemoryStream::MemoryStream(const void * data, size_t dataSize)
|
|
: MemoryStream((void *)data, dataSize, MEMORY_ACCESS_READ)
|
|
{
|
|
}
|
|
|
|
MemoryStream::~MemoryStream()
|
|
{
|
|
if (_access & MEMORY_ACCESS_OWNER)
|
|
{
|
|
Memory::Free(_data);
|
|
}
|
|
_dataCapacity = 0;
|
|
_dataSize = 0;
|
|
_data = nullptr;
|
|
}
|
|
|
|
void * MemoryStream::GetData() const
|
|
{
|
|
return Memory::Duplicate(_data, _dataSize);
|
|
}
|
|
|
|
void * MemoryStream::TakeData()
|
|
{
|
|
_access &= ~MEMORY_ACCESS_OWNER;
|
|
return _data;
|
|
}
|
|
|
|
bool MemoryStream::CanRead() const
|
|
{
|
|
return (_access & MEMORY_ACCESS_READ) != 0;
|
|
}
|
|
|
|
bool MemoryStream::CanWrite() const
|
|
{
|
|
return (_access & MEMORY_ACCESS_WRITE) != 0;
|
|
}
|
|
|
|
uint64 MemoryStream::GetLength() const
|
|
{
|
|
return _dataSize;
|
|
}
|
|
|
|
uint64 MemoryStream::GetPosition() const
|
|
{
|
|
return (uint64)((uintptr_t)_position - (uintptr_t)_data);
|
|
}
|
|
|
|
void MemoryStream::SetPosition(uint64 position)
|
|
{
|
|
Seek(position, STREAM_SEEK_BEGIN);
|
|
}
|
|
|
|
void MemoryStream::Seek(sint64 offset, int origin)
|
|
{
|
|
uint64 newPosition;
|
|
switch (origin) {
|
|
default:
|
|
case STREAM_SEEK_BEGIN:
|
|
newPosition = offset;
|
|
break;
|
|
case STREAM_SEEK_CURRENT:
|
|
newPosition = GetPosition() + offset;
|
|
break;
|
|
case STREAM_SEEK_END:
|
|
newPosition = _dataSize + offset;
|
|
break;
|
|
}
|
|
|
|
if (newPosition > _dataSize)
|
|
{
|
|
throw IOException("New position out of bounds.");
|
|
}
|
|
_position = (void*)((uintptr_t)_data + (uintptr_t)newPosition);
|
|
}
|
|
|
|
void MemoryStream::Read(void * buffer, uint64 length)
|
|
{
|
|
uint64 position = GetPosition();
|
|
if (position + length > _dataSize)
|
|
{
|
|
throw IOException("Attempted to read past end of stream.");
|
|
}
|
|
|
|
Memory::Copy(buffer, _position, (size_t)length);
|
|
_position = (void*)((uintptr_t)_position + length);
|
|
}
|
|
|
|
uint64 MemoryStream::TryRead(void * buffer, uint64 length)
|
|
{
|
|
uint64 remainingBytes = GetLength() - GetPosition();
|
|
uint64 bytesToRead = Math::Min(length, remainingBytes);
|
|
Read(buffer, bytesToRead);
|
|
return bytesToRead;
|
|
}
|
|
|
|
void MemoryStream::Write(const void * buffer, uint64 length)
|
|
{
|
|
uint64 position = GetPosition();
|
|
uint64 nextPosition = position + length;
|
|
if (nextPosition > _dataCapacity)
|
|
{
|
|
if (_access & MEMORY_ACCESS_OWNER)
|
|
{
|
|
EnsureCapacity((size_t)nextPosition);
|
|
}
|
|
else
|
|
{
|
|
throw IOException("Attempted to write past end of stream.");
|
|
}
|
|
}
|
|
|
|
Memory::Copy(_position, buffer, (size_t)length);
|
|
_position = (void*)((uintptr_t)_position + length);
|
|
_dataSize = Math::Max<size_t>(_dataSize, (size_t)nextPosition);
|
|
}
|
|
|
|
void MemoryStream::EnsureCapacity(size_t capacity)
|
|
{
|
|
if (_dataCapacity < capacity)
|
|
{
|
|
size_t newCapacity = Math::Max<size_t>(8, _dataCapacity);
|
|
while (newCapacity < capacity)
|
|
{
|
|
newCapacity *= 2;
|
|
}
|
|
|
|
_dataCapacity = newCapacity;
|
|
Memory::Reallocate(_data, _dataCapacity);
|
|
}
|
|
}
|