poco/Foundation/include/Poco/MemoryStream.h

253 lines
5.7 KiB
C
Raw Normal View History

//
// MemoryStream.h
//
// Library: Foundation
// Package: Streams
// Module: MemoryStream
//
// Definition of MemoryStreamBuf, MemoryInputStream, MemoryOutputStream
//
// Copyright (c) 2009, Applied Informatics Software Engineering GmbH.
// and Contributors.
//
// SPDX-License-Identifier: BSL-1.0
//
#ifndef Foundation_MemoryStream_INCLUDED
#define Foundation_MemoryStream_INCLUDED
#include "Poco/Bugcheck.h"
#include "Poco/Foundation.h"
#include "Poco/StreamUtil.h"
#include <streambuf>
#include <iosfwd>
#include <ios>
#include <istream>
#include <ostream>
namespace Poco {
2022-07-07 11:18:20 +02:00
template <typename ch, typename tr>
class BasicMemoryStreamBuf: public std::basic_streambuf<ch, tr>
2022-07-07 11:18:20 +02:00
/// BasicMemoryStreamBuf is a simple implementation of a
/// stream buffer for reading and writing from a memory area.
///
/// This streambuf only supports unidirectional streams.
/// In other words, the BasicMemoryStreamBuf can be
/// used for the implementation of an istream or an
/// ostream, but not for an iostream.
{
protected:
typedef std::basic_streambuf<ch, tr> Base;
typedef std::basic_ios<ch, tr> IOS;
typedef ch char_type;
typedef tr char_traits;
typedef typename Base::int_type int_type;
typedef typename Base::pos_type pos_type;
typedef typename Base::off_type off_type;
public:
BasicMemoryStreamBuf(char_type* pBuffer, std::streamsize bufferSize):
_pBuffer(pBuffer),
_bufferSize(bufferSize)
{
this->setg(_pBuffer, _pBuffer, _pBuffer + _bufferSize);
this->setp(_pBuffer, _pBuffer + _bufferSize);
}
~BasicMemoryStreamBuf()
{
}
virtual int_type overflow(int_type /*c*/)
{
return char_traits::eof();
}
virtual int_type underflow()
{
return char_traits::eof();
}
virtual pos_type seekoff(off_type off, std::ios_base::seekdir way, std::ios_base::openmode which = std::ios_base::in | std::ios_base::out)
{
const pos_type fail = off_type(-1);
off_type newoff = off_type(-1);
if ((which & std::ios_base::in) != 0)
{
if (this->gptr() == 0)
return fail;
if (way == std::ios_base::beg)
{
newoff = 0;
}
else if (way == std::ios_base::cur)
{
// cur is not valid if both in and out are specified (Condition 3)
if ((which & std::ios_base::out) != 0)
return fail;
newoff = this->gptr() - this->eback();
}
else if (way == std::ios_base::end)
{
newoff = this->egptr() - this->eback();
}
else
{
poco_bugcheck();
}
if ((newoff + off) < 0 || (this->egptr() - this->eback()) < (newoff + off))
return fail;
this->setg(this->eback(), this->eback() + newoff + off, this->egptr());
}
if ((which & std::ios_base::out) != 0)
{
if (this->pptr() == 0)
return fail;
if (way == std::ios_base::beg)
{
newoff = 0;
}
else if (way == std::ios_base::cur)
{
// cur is not valid if both in and out are specified (Condition 3)
if ((which & std::ios_base::in) != 0)
return fail;
newoff = this->pptr() - this->pbase();
}
else if (way == std::ios_base::end)
{
newoff = this->epptr() - this->pbase();
}
else
{
poco_bugcheck();
}
if (newoff + off < 0 || (this->epptr() - this->pbase()) < newoff + off)
return fail;
this->pbump((int)(newoff + off - (this->pptr() - this->pbase())));
}
return newoff;
}
virtual pos_type seekpos(pos_type pos, std::ios_base::openmode which = std::ios_base::in | std::ios_base::out)
{
const off_type off = pos;
return seekoff(off, std::ios::beg, which);
}
virtual int sync()
{
return 0;
}
2022-07-07 11:18:20 +02:00
std::streamsize charsWritten() const
{
return static_cast<std::streamsize>(this->pptr() - this->pbase());
}
void reset()
/// Resets the buffer so that current read and write positions
/// will be set to the beginning of the buffer.
{
this->setg(_pBuffer, _pBuffer, _pBuffer + _bufferSize);
this->setp(_pBuffer, _pBuffer + _bufferSize);
}
2022-07-07 11:18:20 +02:00
private:
char_type* _pBuffer;
std::streamsize _bufferSize;
BasicMemoryStreamBuf();
BasicMemoryStreamBuf(const BasicMemoryStreamBuf&);
BasicMemoryStreamBuf& operator = (const BasicMemoryStreamBuf&);
};
//
// We provide an instantiation for char
//
typedef BasicMemoryStreamBuf<char, std::char_traits<char>> MemoryStreamBuf;
class Foundation_API MemoryIOS: public virtual std::ios
/// The base class for MemoryInputStream and MemoryOutputStream.
///
/// This class is needed to ensure the correct initialization
/// order of the stream buffer and base classes.
{
public:
MemoryIOS(char* pBuffer, std::streamsize bufferSize);
/// Creates the basic stream.
2022-07-07 11:18:20 +02:00
~MemoryIOS();
/// Destroys the stream.
MemoryStreamBuf* rdbuf();
/// Returns a pointer to the underlying streambuf.
2022-07-07 11:18:20 +02:00
protected:
MemoryStreamBuf _buf;
};
class Foundation_API MemoryInputStream: public MemoryIOS, public std::istream
/// An input stream for reading from a memory area.
{
public:
MemoryInputStream(const char* pBuffer, std::streamsize bufferSize);
/// Creates a MemoryInputStream for the given memory area,
/// ready for reading.
2022-07-07 11:18:20 +02:00
~MemoryInputStream();
/// Destroys the MemoryInputStream.
};
class Foundation_API MemoryOutputStream: public MemoryIOS, public std::ostream
/// An input stream for reading from a memory area.
{
public:
MemoryOutputStream(char* pBuffer, std::streamsize bufferSize);
/// Creates a MemoryOutputStream for the given memory area,
/// ready for writing.
2022-07-07 11:18:20 +02:00
~MemoryOutputStream();
/// Destroys the MemoryInputStream.
std::streamsize charsWritten() const;
/// Returns the number of chars written to the buffer.
};
//
// inlines
//
inline MemoryStreamBuf* MemoryIOS::rdbuf()
{
return &_buf;
}
inline std::streamsize MemoryOutputStream::charsWritten() const
{
return _buf.charsWritten();
}
} // namespace Poco
#endif // Foundation_MemoryStream_INCLUDED