InputSource_private.hh 3.51 KB
#ifndef QPDF_INPUTSOURCE_PRIVATE_HH
#define QPDF_INPUTSOURCE_PRIVATE_HH

#include <qpdf/Buffer.hh>
#include <qpdf/InputSource.hh>
#include <qpdf/Util.hh>

#include <limits>
#include <sstream>
#include <stdexcept>

namespace qpdf::is
{
    class OffsetBuffer final: public InputSource
    {
      public:
        OffsetBuffer(
            std::string const& description,
            std::string_view view,
            qpdf_offset_t global_offset = 0) :
            description(description),
            view_(view),
            global_offset(global_offset)
        {
            if (global_offset < 0) {
                throw std::logic_error("is::OffsetBuffer constructed with negative offset");
            }
            last_offset = global_offset;
        }

        OffsetBuffer(std::string const& description, Buffer* buf, qpdf_offset_t global_offset = 0) :
            OffsetBuffer(
                description,
                {buf && buf->getSize()
                     ? std::string_view(
                           reinterpret_cast<const char*>(buf->getBuffer()), buf->getSize())
                     : std::string_view()},
                global_offset)
        {
        }

        ~OffsetBuffer() final = default;

        qpdf_offset_t findAndSkipNextEOL() final;

        std::string const&
        getName() const final
        {
            return description;
        }

        qpdf_offset_t
        tell() final
        {
            return pos + global_offset;
        }

        void seek(qpdf_offset_t offset, int whence) final;

        void
        rewind() final
        {
            pos = 0;
        }

        size_t read(char* buffer, size_t length) final;

        void
        unreadCh(char ch) final
        {
            if (pos > 0) {
                --pos;
            }
        }

      private:
        std::string description;
        qpdf_offset_t pos{0};
        std::string_view view_;
        qpdf_offset_t global_offset;
    };

} // namespace qpdf::is

inline size_t
InputSource::read(std::string& str, size_t count, qpdf_offset_t at)
{
    if (at >= 0) {
        seek(at, SEEK_SET);
    }
    str.resize(count);
    str.resize(read(str.data(), count));
    return str.size();
}

inline std::string
InputSource::read(size_t count, qpdf_offset_t at)
{
    std::string result(count, '\0');
    (void)read(result, count, at);
    return result;
}

inline void
InputSource::loadBuffer()
{
    buf_idx = 0;
    buf_len = qpdf_offset_t(read(buffer, buf_size));
    // NB read sets last_offset
    buf_start = last_offset;
}

inline qpdf_offset_t
InputSource::fastTell()
{
    if (buf_len == 0) {
        loadBuffer();
    } else {
        auto curr = tell();
        if (curr < buf_start || curr >= (buf_start + buf_len)) {
            loadBuffer();
        } else {
            last_offset = curr;
            buf_idx = curr - buf_start;
        }
    }
    return last_offset;
}

inline bool
InputSource::fastRead(char& ch)
{
    // Before calling fastRead, fastTell must be called to prepare the buffer. Once reading is
    // complete, fastUnread must be called to set the correct file position.
    if (buf_idx < buf_len) {
        ch = buffer[buf_idx];
        ++(buf_idx);
        ++(last_offset);
        return true;

    } else if (buf_len == 0) {
        return false;
    } else {
        seek(buf_start + buf_len, SEEK_SET);
        fastTell();
        return fastRead(ch);
    }
}

inline void
InputSource::fastUnread(bool back)
{
    last_offset -= back ? 1 : 0;
    seek(last_offset, SEEK_SET);
}

#endif // QPDF_INPUTSOURCE_PRIVATE_HH