#include "constrainedfiledatastream.hpp" #include "lowlevelfile.hpp" #include <stdexcept> #include <cassert> #include <libs/platform/stdint.h> namespace { class ConstrainedDataStream : public Ogre::DataStream { public: static const size_t sBufferSize = 4096; // somewhat arbitrary though 64KB buffers didn't seem to improve performance any static const size_t sBufferThreshold = 1024; // reads larger than this bypass buffering as cost of memcpy outweighs cost of system call ConstrainedDataStream(const Ogre::String &fname, size_t start, size_t length) { mFile.open (fname.c_str ()); mSize = length != 0xFFFFFFFF ? length : mFile.size () - start; mPos = 0; mOrigin = start; mExtent = start + mSize; mBufferOrigin = 0; mBufferExtent = 0; } size_t read(void* buf, size_t count) { assert (mPos <= mSize); uint8_t * out = reinterpret_cast <uint8_t *> (buf); size_t posBeg = mOrigin + mPos; size_t posEnd = posBeg + count; if (posEnd > mExtent) posEnd = mExtent; size_t posCur = posBeg; while (posCur != posEnd) { size_t readLeft = posEnd - posCur; if (posCur < mBufferOrigin || posCur >= mBufferExtent) { if (readLeft >= sBufferThreshold || (posCur == mOrigin && posEnd == mExtent)) { assert (mFile.tell () == mBufferExtent); if (posCur != mBufferExtent) mFile.seek (posCur); posCur += mFile.read (out, readLeft); mBufferOrigin = mBufferExtent = posCur; mPos = posCur - mOrigin; return posCur - posBeg; } else { size_t newBufferOrigin; if ((posCur < mBufferOrigin) && (mBufferOrigin - posCur < sBufferSize)) newBufferOrigin = std::max (mOrigin, mBufferOrigin > sBufferSize ? mBufferOrigin - sBufferSize : 0); else newBufferOrigin = posCur; fill (newBufferOrigin); } } size_t xfer = std::min (readLeft, mBufferExtent - posCur); memcpy (out, mBuffer + (posCur - mBufferOrigin), xfer); posCur += xfer; out += xfer; } count = posEnd - posBeg; mPos += count; return count; } void skip(long count) { assert (mPos <= mSize); if((count >= 0 && (size_t)count <= mSize-mPos) || (count < 0 && (size_t)-count <= mPos)) mPos += count; } void seek(size_t pos) { assert (mPos <= mSize); if (pos < mSize) mPos = pos; } virtual size_t tell() const { assert (mPos <= mSize); return mPos; } virtual bool eof() const { assert (mPos <= mSize); return mPos == mSize; } virtual void close() { mFile.close(); } private: void fill (size_t newOrigin) { assert (mFile.tell () == mBufferExtent); size_t newExtent = newOrigin + sBufferSize; if (newExtent > mExtent) newExtent = mExtent; size_t oldExtent = mBufferExtent; if (newOrigin != oldExtent) mFile.seek (newOrigin); mBufferOrigin = mBufferExtent = newOrigin; size_t amountRequested = newExtent - newOrigin; size_t amountRead = mFile.read (mBuffer, amountRequested); if (amountRead != amountRequested) throw std::runtime_error ("An unexpected condition occurred while reading from a file."); mBufferExtent = newExtent; } LowLevelFile mFile; size_t mOrigin; size_t mExtent; size_t mPos; uint8_t mBuffer [sBufferSize]; size_t mBufferOrigin; size_t mBufferExtent; }; } // end of unnamed namespace Ogre::DataStreamPtr openConstrainedFileDataStream (char const * filename, size_t offset, size_t length) { return Ogre::DataStreamPtr(new ConstrainedDataStream(filename, offset, length)); }