mirror of
https://github.com/libSDL2pp/libSDL2pp.git
synced 2025-08-03 10:55:57 -04:00
Make StreamRWops work with istream/ostream
Streams do not generally work well with RWops because * streams have separate read and write pointers * ostream doesn't allow you to determine how many bytes were actually written * istream and ostream have separate set of functions Try my best to support streams in RWops though, engaging some template magic: * provide separate template implementations of all operations which depend on whether stream is an istream or ostream. This allows to e.g. return 0 immediately for an attempt to write() to istream. * disallow StreamRWops for classes which are both istream and ostream to not run into ambiguity of separate read/write pointers * for read failure, but partially read object back to the stream to not lose data for following read (not sure that e.g. fread behaves so though; I'll anyway expect user to Seek() after read or write failure) * for write failure, there's no way to avoid leaking partial data to the stream In general, it is best to use this container as read-only. Also add tests for StreamRWops
This commit is contained in:
parent
7ba131a913
commit
53aa26dec5
@ -26,16 +26,97 @@
|
||||
|
||||
#include <stdexcept>
|
||||
#include <iostream>
|
||||
#include <type_traits>
|
||||
|
||||
namespace SDL2pp {
|
||||
|
||||
template<typename S>
|
||||
template <class S>
|
||||
class StreamRWops : public CustomRWops {
|
||||
// since STL streams have different pointers for reading and writing,
|
||||
// supporting both at the same time is impossible
|
||||
static_assert(!(std::is_base_of<std::istream, S>::value && std::is_base_of<std::ostream, S>::value), "StreamRWops does not support reading and writing at the same time");
|
||||
|
||||
protected:
|
||||
S& stream_;
|
||||
|
||||
private:
|
||||
template <class SS>
|
||||
typename std::enable_if<std::is_base_of<std::istream, SS>::value && !std::is_base_of<std::ostream, SS>::value, void>::type SeekHelper(off_t off, std::ios_base::seekdir dir) {
|
||||
stream_.seekg(off, dir);
|
||||
}
|
||||
|
||||
template <class SS>
|
||||
typename std::enable_if<!std::is_base_of<std::istream, SS>::value && std::is_base_of<std::ostream, SS>::value, void>::type SeekHelper(off_t off, std::ios_base::seekdir dir) {
|
||||
stream_.seekp(off, dir);
|
||||
}
|
||||
|
||||
template <class SS>
|
||||
typename std::enable_if<std::is_base_of<std::istream, SS>::value && !std::is_base_of<std::ostream, SS>::value, off_t>::type TellHelper() {
|
||||
return stream_.tellg();
|
||||
}
|
||||
|
||||
template <class SS>
|
||||
typename std::enable_if<!std::is_base_of<std::istream, SS>::value && std::is_base_of<std::ostream, SS>::value, off_t>::type TellHelper() {
|
||||
return stream_.tellp();
|
||||
}
|
||||
|
||||
template <class SS>
|
||||
typename std::enable_if<std::is_base_of<std::istream, SS>::value, size_t>::type ReadHelper(void* ptr, size_t size, size_t maxnum) {
|
||||
stream_.read(static_cast<char*>(ptr), size * maxnum);
|
||||
size_t nread = stream_.gcount();
|
||||
|
||||
// eof is OK
|
||||
if (stream_.rdstate() == (std::ios_base::eofbit | std::ios_base::failbit))
|
||||
stream_.clear();
|
||||
|
||||
if (nread != size * maxnum) {
|
||||
// short read
|
||||
char* pos = static_cast<char*>(ptr);
|
||||
pos += nread;
|
||||
|
||||
int count = nread % size;
|
||||
|
||||
// put partially read object back into the stream
|
||||
while (--count >= 0)
|
||||
stream_.putback(*--pos);
|
||||
}
|
||||
|
||||
return nread / size;
|
||||
}
|
||||
|
||||
template <class SS>
|
||||
typename std::enable_if<!std::is_base_of<std::istream, SS>::value, size_t>::type ReadHelper(void*, size_t, size_t) {
|
||||
return 0;
|
||||
}
|
||||
|
||||
template <class SS>
|
||||
typename std::enable_if<std::is_base_of<std::ostream, SS>::value, size_t>::type WriteHelper(const void* ptr, size_t size, size_t maxnum) {
|
||||
stream_.write(static_cast<const char*>(ptr), size * maxnum);
|
||||
// XXX: there seem to be no reliable way to tell how much
|
||||
// was actually written
|
||||
if (stream_.rdstate() & std::ios_base::badbit)
|
||||
return 0;
|
||||
return maxnum;
|
||||
}
|
||||
|
||||
template <class SS>
|
||||
typename std::enable_if<!std::is_base_of<std::ostream, SS>::value, size_t>::type WriteHelper(const void*, size_t, size_t) {
|
||||
return 0;
|
||||
}
|
||||
|
||||
template <class SS>
|
||||
typename std::enable_if<std::is_base_of<std::ostream, SS>::value, int>::type CloseHelper() {
|
||||
stream_.flush();
|
||||
return stream_.rdstate() & std::ios_base::badbit;
|
||||
}
|
||||
|
||||
template <class SS>
|
||||
typename std::enable_if<!std::is_base_of<std::ostream, SS>::value, int>::type CloseHelper() {
|
||||
return 0;
|
||||
}
|
||||
|
||||
public:
|
||||
StreamRWops(const S& stream) : stream_(stream) {
|
||||
StreamRWops(S& stream) : stream_(stream) {
|
||||
}
|
||||
|
||||
StreamRWops(const StreamRWops<S>&) = default;
|
||||
@ -46,52 +127,30 @@ public:
|
||||
virtual Sint64 Seek(Sint64 offset, int whence) override {
|
||||
switch (whence) {
|
||||
case RW_SEEK_SET:
|
||||
stream_.seekg(offset, std::ios_base::beg);
|
||||
SeekHelper<S>(offset, std::ios_base::beg);
|
||||
break;
|
||||
case RW_SEEK_CUR:
|
||||
stream_.seekg(offset, std::ios_base::cur);
|
||||
SeekHelper<S>(offset, std::ios_base::cur);
|
||||
break;
|
||||
case RW_SEEK_END:
|
||||
stream_.seekg(offset, std::ios_base::end);
|
||||
SeekHelper<S>(offset, std::ios_base::end);
|
||||
break;
|
||||
default:
|
||||
throw Exception("Unexpected whence value for StreamRWops::Seek");
|
||||
throw std::logic_error("Unexpected whence value for StreamRWops::Seek");
|
||||
}
|
||||
return stream_.tellg();
|
||||
return TellHelper<S>();
|
||||
}
|
||||
|
||||
virtual size_t Read(void* ptr, size_t size, size_t maxnum) override {
|
||||
stream_.read(ptr, size * maxnum);
|
||||
size_t nread = stream_.gcount();
|
||||
|
||||
if (nread != size * maxnum) {
|
||||
// short read
|
||||
unsigned char* pos = static_cast<unsigned char*>(ptr);
|
||||
pos += nread + 1;
|
||||
|
||||
int count = nread % size;
|
||||
|
||||
// put partially read object back into the stream
|
||||
while (--count >= 0)
|
||||
stream_.putback(*--pos);
|
||||
|
||||
stream_.seekg(-count, std::ios_base::cur);
|
||||
}
|
||||
return nread / size;
|
||||
return ReadHelper<S>(ptr, size, maxnum);
|
||||
}
|
||||
|
||||
virtual size_t Write(const void* ptr, size_t size, size_t maxnum) override {
|
||||
stream_.write(ptr, size * maxnum);
|
||||
// XXX: there seem to be no reliable way to tell how much
|
||||
// was actually written
|
||||
if (stream_.restate() & std::ios_base::badbit)
|
||||
return 0;
|
||||
return maxnum;
|
||||
return WriteHelper<S>(ptr, size, maxnum);
|
||||
}
|
||||
|
||||
virtual int Close() override {
|
||||
stream_.flush();
|
||||
return stream_.restate() & std::ios_base::badbit;
|
||||
return CloseHelper<S>();
|
||||
}
|
||||
};
|
||||
|
||||
|
@ -2,6 +2,7 @@
|
||||
|
||||
#include <SDL2pp/Exception.hh>
|
||||
#include <SDL2pp/ContainerRWops.hh>
|
||||
#include <SDL2pp/StreamRWops.hh>
|
||||
#include <SDL2pp/RWops.hh>
|
||||
|
||||
#include "testing.h"
|
||||
@ -141,6 +142,66 @@ BEGIN_TEST()
|
||||
}
|
||||
}
|
||||
|
||||
// Test for StreamRWops
|
||||
{
|
||||
{
|
||||
// write test
|
||||
|
||||
std::stringstream test;
|
||||
RWops rw((StreamRWops<std::ostream>(test)));
|
||||
|
||||
char buf[4] = { 'a', 'b', 'c', 'd' };
|
||||
EXPECT_TRUE(rw.Write(buf, 1, 4) == 4);
|
||||
|
||||
EXPECT_TRUE(rw.Seek(0, RW_SEEK_CUR) == 4);
|
||||
|
||||
EXPECT_TRUE(rw.Seek(2, RW_SEEK_SET) == 2);
|
||||
|
||||
EXPECT_TRUE(rw.Write(buf, 1, 4) == 4);
|
||||
|
||||
EXPECT_EQUAL(test.str(), "ababcd");
|
||||
}
|
||||
|
||||
{
|
||||
// read test
|
||||
|
||||
std::stringstream test("abcdef");
|
||||
RWops rw((StreamRWops<std::istream>(test)));
|
||||
|
||||
char buf[4];
|
||||
EXPECT_EQUAL(rw.Read(buf, 1, 4), 4UL);
|
||||
|
||||
EXPECT_EQUAL(std::string(buf, 4), "abcd");
|
||||
|
||||
EXPECT_EQUAL(rw.Seek(0, RW_SEEK_CUR), 4);
|
||||
|
||||
EXPECT_EQUAL(rw.Seek(2, RW_SEEK_SET), 2);
|
||||
|
||||
EXPECT_EQUAL(rw.Read(buf, 1, 4), 4UL);
|
||||
|
||||
EXPECT_EQUAL(std::string(buf, 4), "cdef");
|
||||
|
||||
// short read
|
||||
EXPECT_EQUAL(rw.Seek(4, RW_SEEK_SET), 4);
|
||||
|
||||
EXPECT_EQUAL(rw.Read(buf, 1, 4), 2UL);
|
||||
|
||||
EXPECT_EQUAL(std::string(buf, 2), "ef");
|
||||
|
||||
// short object read
|
||||
EXPECT_EQUAL(rw.Seek(4, RW_SEEK_SET), 4);
|
||||
|
||||
EXPECT_EQUAL(rw.Read(buf, 4, 1), 0UL);
|
||||
|
||||
EXPECT_EQUAL(rw.Seek(0, RW_SEEK_CUR), 4);
|
||||
|
||||
// read end
|
||||
EXPECT_EQUAL(rw.Read(buf, 1, 2), 2UL);
|
||||
|
||||
EXPECT_EQUAL(std::string(buf, 2), "ef");
|
||||
}
|
||||
}
|
||||
|
||||
// SDL file read test
|
||||
{
|
||||
RWops rw = RWops::FromFile(TESTDATA_DIR "/test.txt");
|
||||
|
Loading…
x
Reference in New Issue
Block a user