/** * Copyright 2003-2006 The Apache Software Foundation * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ #ifndef BUFFEREDSTREAM_H #define BUFFEREDSTREAM_H #include "streambase.h" #include "inputstreambuffer.h" namespace jstreams { template class BufferedInputStream : public StreamBase { private: bool finishedWritingToBuffer; InputStreamBuffer buffer; void writeToBuffer(int32_t minsize); int32_t read_(const T*& start, int32_t min, int32_t max); protected: /** * This function must be implemented by the subclasses. * It should write a maximum of @p space characters at the buffer * position pointed to by @p start. If no more data is avaiable due to * end of file, -1 should be returned. If an error occurs, the status * should be set to Error, an error message should be set and the function * must return -1. **/ virtual int32_t fillBuffer(T* start, int32_t space) = 0; // this function might be useful if you want to reuse a bufferedstream void resetBuffer() {printf("implement 'resetBuffer'\n");} public: BufferedInputStream(); int32_t read(const T*& start, int32_t min, int32_t max); int64_t mark(int32_t readlimit); int64_t reset(int64_t); virtual int64_t skip(int64_t ntoskip); }; template BufferedInputStream::BufferedInputStream() { finishedWritingToBuffer = false; } template void BufferedInputStream::writeToBuffer(int32_t ntoread) { int32_t missing = ntoread - buffer.avail; int32_t nwritten = 0; while (missing > 0 && nwritten >= 0) { int32_t space; space = buffer.makeSpace(missing); T* start = buffer.readPos + buffer.avail; nwritten = fillBuffer(start, space); if (nwritten > 0) { buffer.avail += nwritten; missing = ntoread - buffer.avail; } } if (nwritten < 0) { finishedWritingToBuffer = true; } } template int32_t BufferedInputStream::read(const T*& start, int32_t min, int32_t max) { if (StreamBase::status == Error) return -2; if (StreamBase::status == Eof) return -1; // do we need to read data into the buffer? if (!finishedWritingToBuffer && min > buffer.avail) { // do we have enough space in the buffer? writeToBuffer(min); if (StreamBase::status == Error) return -2; if (StreamBase::status == Eof) return -1; } int32_t nread = buffer.read(start, max); /* if (nread == 0) { printf("bis: start %p min %i max %i nread %i avail %i bsize %i pos %lli size %lli\n", start, min, max, nread, buffer.avail, buffer.size, BufferedInputStream::position, BufferedInputStream::size); printf("buf: start %p readpos %p marpos %p\n", buffer.start, buffer.readPos, buffer.markPos); }*/ BufferedInputStream::position += nread; if (BufferedInputStream::status == Ok && buffer.avail == 0 && finishedWritingToBuffer) { BufferedInputStream::status = Eof; // save one call to read() by already returning -1 if no data is there if (nread == 0) nread = -1; } return nread; } template int64_t BufferedInputStream::mark(int32_t readlimit) { buffer.mark(readlimit); return StreamBase::position; } template int64_t BufferedInputStream::reset(int64_t newpos) { if (StreamBase::status == Error) return -2; // check to see if we have this position int64_t d = BufferedInputStream::position - newpos; if (buffer.readPos - d >= buffer.start && -d < buffer.avail) { BufferedInputStream::position -= d; buffer.avail += d; buffer.readPos -= d; StreamBase::status = Ok; } return StreamBase::position; } template int64_t BufferedInputStream::skip(int64_t ntoskip) { const T *begin; int32_t nread; int64_t skipped = 0; while (ntoskip) { int32_t step = (int32_t)((ntoskip > buffer.size) ?buffer.size :ntoskip); nread = read(begin, 1, step); if (nread <= 0) { return skipped; } ntoskip -= nread; skipped += nread; } return skipped; } } #endif