colobot/colobot-base/common/ioutils.h

168 lines
3.9 KiB
C++

/*
* This file is part of the Colobot: Gold Edition source code
* Copyright (C) 2001-2023, Daniel Roux, EPSITEC SA & TerranovaTeam
* http://epsitec.ch; http://colobot.info; http://github.com/colobot
*
* 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 http://gnu.org/licenses
*/
/**
* \file common/ioutils.h
* \brief Functions for binary I/O
*/
#pragma once
#include <iostream>
#include <cstring>
namespace IOUtils
{
//! Writes a binary number to output stream
/**
* \c T is a numeric type (int, unsigned int, etc.)
* \c N is number of bytes
* Write order is little-endian
*/
template<int N, typename T>
void WriteBinary(T value, std::ostream &ostr)
{
for (int i = 0; i < N; ++i)
{
unsigned char byte = (value >> (i*8)) & 0xFF;
ostr.write(reinterpret_cast<char*>(&byte), 1);
}
}
//! Reads a binary number from input stream
/**
* \c T is a numeric type (int, unsigned int, etc.)
* \c N is number of bytes
* Read order is little-endian
*/
template<int N, typename T>
T ReadBinary(std::istream &istr)
{
T value = 0;
for (int i = 0; i < N; ++i)
{
unsigned char byte = 0;
istr.read(reinterpret_cast<char*>(&byte), 1);
value |= byte << (i*8);
}
return value;
}
//! Writes a binary 1-byte boolean
/**
* false is 0; true is 1.
*/
inline void WriteBinaryBool(bool value, std::ostream &ostr)
{
unsigned char v = value ? 1 : 0;
IOUtils::WriteBinary<1, unsigned char>(v, ostr);
}
//! Reads a binary 1-byte boolean
/**
* 0 is false; other values are true.
*/
inline bool ReadBinaryBool(std::istream &istr)
{
int v = IOUtils::ReadBinary<1, unsigned char>(istr);
return v != 0;
}
//! Writes a binary 32-bit float to output stream
/**
* Write order is little-endian
* NOTE: code is probably not portable as there are platforms with other float representations.
*/
inline void WriteBinaryFloat(float value, std::ostream &ostr)
{
union FloatCast
{
float fValue;
unsigned int iValue;
};
FloatCast u;
u.fValue = 0.0f;
u.iValue = 0;
u.fValue = value;
IOUtils::WriteBinary<4, unsigned int>(u.iValue, ostr);
}
//! Reads a binary 32-bit float from input stream
/**
* Read order is little-endian
* NOTE: code is probably not portable as there are platforms with other float representations.
*/
inline float ReadBinaryFloat(std::istream &istr)
{
union FloatCast
{
float fValue;
unsigned int iValue;
};
FloatCast u;
u.fValue = 0.0f;
u.iValue = 0;
u.iValue = IOUtils::ReadBinary<4, unsigned int>(istr);
return u.fValue;
}
//! Writes a variable binary string to output stream
/**
* The string is written by first writing string length
* in \c N byte binary number and then the string bytes.
*/
template<int N>
void WriteBinaryString(const std::string &value, std::ostream &ostr)
{
int length = value.size();
WriteBinary<N, int>(length, ostr);
for (int i = 0; i < length; ++i)
ostr.put(value[i]);
}
//! Reads a variable binary string from output stream
/**
* The string is read by first reading string length
* in \c N byte binary number and then the string bytes.
*/
template<int N>
std::string ReadBinaryString(std::istream &istr)
{
int length = ReadBinary<N, int>(istr);
std::string str;
char c = 0;
for (int i = 0; i < length; ++i)
{
istr.read(&c, 1);
str += c;
}
return str;
}
} // namespace IOUtils