Coverage Report

Created: 2026-06-15 06:21

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/qpdf/libqpdf/BufferInputSource.cc
Line
Count
Source
1
#include <qpdf/BufferInputSource.hh>
2
#include <qpdf/InputSource_private.hh>
3
4
#include <qpdf/Buffer.hh>
5
#include <qpdf/QIntC.hh>
6
7
#include <algorithm>
8
#include <cstring>
9
#include <sstream>
10
11
using namespace qpdf;
12
13
#ifndef QPDF_FUTURE
14
15
BufferInputSource::BufferInputSource(std::string const& description, Buffer* buf, bool own_memory) :
16
124k
    own_memory(own_memory),
17
124k
    description(description),
18
124k
    buf(buf),
19
124k
    cur_offset(0),
20
124k
    max_offset(buf ? QIntC::to_offset(buf->getSize()) : 0)
21
124k
{
22
124k
}
23
24
BufferInputSource::BufferInputSource(std::string const& description, std::string const& contents) :
25
0
    own_memory(true),
26
0
    description(description),
27
0
    buf(new Buffer(contents.length())),
28
0
    cur_offset(0),
29
0
    max_offset(QIntC::to_offset(buf->getSize()))
30
0
{
31
0
    memcpy(buf->getBuffer(), contents.c_str(), contents.length());
32
0
}
33
34
BufferInputSource::~BufferInputSource()
35
124k
{
36
124k
    if (own_memory) {
37
0
        delete buf;
38
0
    }
39
124k
}
40
41
qpdf_offset_t
42
BufferInputSource::findAndSkipNextEOL()
43
13.6M
{
44
13.6M
    util::internal_error_if(cur_offset < 0, "BufferInputSource offset < 0");
45
13.6M
    qpdf_offset_t end_pos = max_offset;
46
13.6M
    if (cur_offset >= end_pos) {
47
36.3k
        last_offset = end_pos;
48
36.3k
        cur_offset = end_pos;
49
36.3k
        return end_pos;
50
36.3k
    }
51
52
13.6M
    qpdf_offset_t result = 0;
53
13.6M
    unsigned char const* buffer = buf->getBuffer();
54
13.6M
    unsigned char const* end = buffer + end_pos;
55
13.6M
    unsigned char const* p = buffer + cur_offset;
56
57
1.23G
    while ((p < end) && !((*p == '\r') || (*p == '\n'))) {
58
1.22G
        ++p;
59
1.22G
    }
60
13.6M
    if (p < end) {
61
13.5M
        result = p - buffer;
62
13.5M
        cur_offset = result + 1;
63
13.5M
        ++p;
64
14.8M
        while ((cur_offset < end_pos) && ((*p == '\r') || (*p == '\n'))) {
65
1.24M
            ++p;
66
1.24M
            ++cur_offset;
67
1.24M
        }
68
13.5M
    } else {
69
76.0k
        cur_offset = end_pos;
70
76.0k
        result = end_pos;
71
76.0k
    }
72
13.6M
    return result;
73
13.6M
}
74
75
std::string const&
76
BufferInputSource::getName() const
77
10.2M
{
78
10.2M
    return description;
79
10.2M
}
80
81
qpdf_offset_t
82
BufferInputSource::tell()
83
231M
{
84
231M
    return cur_offset;
85
231M
}
86
87
void
88
BufferInputSource::seek(qpdf_offset_t offset, int whence)
89
74.9M
{
90
74.9M
    switch (whence) {
91
74.0M
    case SEEK_SET:
92
74.0M
        cur_offset = offset;
93
74.0M
        break;
94
95
231k
    case SEEK_END:
96
231k
        QIntC::range_check(max_offset, offset);
97
231k
        cur_offset = max_offset + offset;
98
231k
        break;
99
100
705k
    default:
101
705k
        util::assertion(whence == SEEK_CUR, "invalid argument to BufferInputSource::seek");
102
705k
        QIntC::range_check(cur_offset, offset);
103
705k
        cur_offset += offset;
104
74.9M
    }
105
106
74.9M
    if (cur_offset < 0) {
107
1.94k
        throw std::runtime_error(description + ": seek before beginning of buffer");
108
1.94k
    }
109
74.9M
}
110
111
void
112
BufferInputSource::rewind()
113
0
{
114
0
    cur_offset = 0;
115
0
}
116
117
size_t
118
BufferInputSource::read(char* buffer, size_t length)
119
15.0M
{
120
15.0M
    util::internal_error_if(cur_offset < 0, "BufferInputSource offset < 0");
121
15.0M
    qpdf_offset_t end_pos = max_offset;
122
15.0M
    if (cur_offset >= end_pos) {
123
379k
        last_offset = end_pos;
124
379k
        return 0;
125
379k
    }
126
127
14.6M
    last_offset = cur_offset;
128
14.6M
    size_t len = std::min(QIntC::to_size(end_pos - cur_offset), length);
129
14.6M
    memcpy(buffer, buf->getBuffer() + cur_offset, len);
130
14.6M
    cur_offset += QIntC::to_offset(len);
131
14.6M
    return len;
132
15.0M
}
133
134
void
135
BufferInputSource::unreadCh(char ch)
136
91.3k
{
137
91.3k
    if (cur_offset > 0) {
138
91.3k
        --cur_offset;
139
91.3k
    }
140
91.3k
}
141
142
#else
143
144
class BufferInputSource::Members
145
{
146
  public:
147
    Members(std::string const& description, Buffer* buf, bool own_memory) :
148
123k
        buf(own_memory ? buf : nullptr),
149
123k
        is(description,
150
123k
           buf && buf->getSize() > 0
151
123k
               ? std::string_view(reinterpret_cast<const char*>(buf->getBuffer()), buf->getSize())
152
123k
               : std::string_view())
153
123k
    {
154
123k
    }
155
156
    Members(std::string const& description, std::string const& str) :
157
0
        content(str),
158
0
        is(description, content)
159
0
    {
160
0
    }
161
162
123k
    ~Members() = default;
163
164
    std::unique_ptr<Buffer> buf{nullptr};
165
    std::string content;
166
    is::OffsetBuffer is;
167
};
168
169
BufferInputSource::BufferInputSource(std::string const& description, Buffer* buf, bool own_memory) :
170
    m(std::make_unique<Members>(description, buf, own_memory))
171
{
172
}
173
174
BufferInputSource::BufferInputSource(std::string const& description, std::string const& contents) :
175
    m(std::make_unique<Members>(description, contents))
176
{
177
}
178
BufferInputSource::~BufferInputSource() = default;
179
180
qpdf_offset_t
181
BufferInputSource::findAndSkipNextEOL()
182
{
183
    auto result = m->is.findAndSkipNextEOL();
184
    last_offset = m->is.getLastOffset();
185
    return result;
186
}
187
std::string const&
188
BufferInputSource::getName() const
189
{
190
    return m->is.getName();
191
}
192
qpdf_offset_t
193
BufferInputSource::tell()
194
{
195
    return m->is.tell();
196
}
197
void
198
BufferInputSource::seek(qpdf_offset_t offset, int whence)
199
{
200
    m->is.seek(offset, whence);
201
}
202
void
203
BufferInputSource::rewind()
204
{
205
    m->is.rewind();
206
}
207
size_t
208
BufferInputSource::read(char* buffer, size_t length)
209
{
210
    auto result = m->is.read(buffer, length);
211
    last_offset = m->is.getLastOffset();
212
    return result;
213
}
214
void
215
BufferInputSource::unreadCh(char ch)
216
{
217
    m->is.unreadCh(ch);
218
}
219
220
#endif // QPDF_FUTURE
221
222
qpdf_offset_t
223
is::OffsetBuffer::findAndSkipNextEOL()
224
10.4M
{
225
10.4M
    util::internal_error_if(pos < 0, "is::OffsetBuffer offset < 0");
226
10.4M
    auto end_pos = static_cast<qpdf_offset_t>(view_.size());
227
10.4M
    if (pos >= end_pos) {
228
36.4k
        last_offset = end_pos + global_offset;
229
36.4k
        pos = end_pos;
230
36.4k
        return end_pos + global_offset;
231
36.4k
    }
232
233
10.3M
    qpdf_offset_t result = 0;
234
10.3M
    auto buffer = view_.begin();
235
10.3M
    auto end = view_.end();
236
10.3M
    auto p = buffer + static_cast<std::ptrdiff_t>(pos);
237
238
1.26G
    while (p < end && !(*p == '\r' || *p == '\n')) {
239
1.25G
        ++p;
240
1.25G
    }
241
10.3M
    if (p < end) {
242
10.3M
        result = p - buffer;
243
10.3M
        pos = result + 1;
244
10.3M
        ++p;
245
13.2M
        while (pos < end_pos && (*p == '\r' || *p == '\n')) {
246
2.90M
            ++p;
247
2.90M
            ++pos;
248
2.90M
        }
249
10.3M
    } else {
250
74.4k
        pos = end_pos;
251
74.4k
        result = end_pos;
252
74.4k
    }
253
10.3M
    return result + global_offset;
254
10.4M
}
255
256
void
257
is::OffsetBuffer::seek(qpdf_offset_t offset, int whence)
258
186M
{
259
186M
    switch (whence) {
260
185M
    case SEEK_SET:
261
185M
        pos = offset - global_offset;
262
185M
        break;
263
264
243k
    case SEEK_END:
265
243k
        QIntC::range_check(static_cast<qpdf_offset_t>(view_.size()), offset);
266
243k
        pos = static_cast<qpdf_offset_t>(view_.size()) + offset;
267
243k
        break;
268
269
624k
    default:
270
624k
        util::assertion(whence == SEEK_CUR, "invalid argument to BufferInputSource::seek");
271
624k
        QIntC::range_check(pos, offset);
272
624k
        pos += offset;
273
186M
    }
274
275
186M
    if (pos < 0) {
276
1.85k
        throw std::runtime_error(description + ": seek before beginning of buffer");
277
1.85k
    }
278
186M
}
279
280
size_t
281
is::OffsetBuffer::read(char* buffer, size_t length)
282
22.8M
{
283
22.8M
    util::internal_error_if(pos < 0, "is::OffsetBuffer offset < 0");
284
22.8M
    auto end_pos = static_cast<qpdf_offset_t>(view_.size());
285
22.8M
    if (pos >= end_pos) {
286
579k
        last_offset = end_pos + global_offset;
287
579k
        return 0;
288
579k
    }
289
290
22.2M
    last_offset = pos + global_offset;
291
22.2M
    size_t len = std::min(QIntC::to_size(end_pos - pos), length);
292
22.2M
    memcpy(buffer, view_.data() + pos, len);
293
22.2M
    pos += QIntC::to_offset(len);
294
22.2M
    return len;
295
22.8M
}