mirror of
https://github.com/qpdf/qpdf.git
synced 2025-01-02 22:50:20 +00:00
Pl_Buffer: reduce memory growth (fixes #228)
Rather than keeping a list of buffers for every write, accumulate bytes in a single buffer, doubling the size of the buffer when needed to accommodate new data. This is not the best possible implementation, but the change was implemented in this way to avoid changing the shape of Pl_Buffer and thus breaking backward compatibility.
This commit is contained in:
parent
48331b4bdc
commit
3d6615b276
@ -1,3 +1,11 @@
|
|||||||
|
2018-08-12 Jay Berkenbilt <ejb@ql.org>
|
||||||
|
|
||||||
|
* Rewrite the internals of Pl_Buffer to be much more efficient in
|
||||||
|
use of memory at a very slight performance cost. The old
|
||||||
|
implementation could cause memory usage to go out of control for
|
||||||
|
files with large images compressed using the TIFF predictor.
|
||||||
|
Fixes #228.
|
||||||
|
|
||||||
2018-08-05 Jay Berkenbilt <ejb@ql.org>
|
2018-08-05 Jay Berkenbilt <ejb@ql.org>
|
||||||
|
|
||||||
* Bug fix: end of line characters were not properly handled inside
|
* Bug fix: end of line characters were not properly handled inside
|
||||||
|
12
TODO
12
TODO
@ -31,6 +31,16 @@ Soon
|
|||||||
- See ../misc/broken-files
|
- See ../misc/broken-files
|
||||||
|
|
||||||
|
|
||||||
|
Next ABI
|
||||||
|
========
|
||||||
|
|
||||||
|
Do these things next time we have to break binary compatibility
|
||||||
|
|
||||||
|
* Pl_Buffer's internal structure is not right for what it does. It
|
||||||
|
was modified for greater efficiency, but it was done in a way that
|
||||||
|
preserved binary compatibility, so the implementation is a bit
|
||||||
|
convoluted.
|
||||||
|
|
||||||
Lexical
|
Lexical
|
||||||
=======
|
=======
|
||||||
|
|
||||||
@ -72,6 +82,8 @@ directory or that are otherwise not publicly accessible. This includes
|
|||||||
things sent to me by email that are specifically not public. Even so,
|
things sent to me by email that are specifically not public. Even so,
|
||||||
I find it useful to make reference to them in this list
|
I find it useful to make reference to them in this list
|
||||||
|
|
||||||
|
* Pl_TIFFPredictor is pretty slow.
|
||||||
|
|
||||||
* Some test cases on bad fails fail because qpdf is unable to find
|
* Some test cases on bad fails fail because qpdf is unable to find
|
||||||
the root dictionary when it fails to read the trailer. Recovery
|
the root dictionary when it fails to read the trailer. Recovery
|
||||||
could find the root dictionary and even the info dictionary in
|
could find the root dictionary and even the info dictionary in
|
||||||
|
@ -17,11 +17,32 @@ Pl_Buffer::~Pl_Buffer()
|
|||||||
void
|
void
|
||||||
Pl_Buffer::write(unsigned char* buf, size_t len)
|
Pl_Buffer::write(unsigned char* buf, size_t len)
|
||||||
{
|
{
|
||||||
Buffer* b = new Buffer(len);
|
PointerHolder<Buffer> cur_buf;
|
||||||
memcpy(b->getBuffer(), buf, len);
|
size_t cur_size = 0;
|
||||||
this->data.push_back(b);
|
if (! this->data.empty())
|
||||||
|
{
|
||||||
|
cur_buf = this->data.back();
|
||||||
|
cur_size = cur_buf->getSize();
|
||||||
|
}
|
||||||
|
size_t left = cur_size - this->total_size;
|
||||||
|
if (left < len)
|
||||||
|
{
|
||||||
|
size_t new_size = std::max(this->total_size + len, 2 * cur_size);
|
||||||
|
Buffer* b = new Buffer(new_size);
|
||||||
|
if (cur_buf.getPointer())
|
||||||
|
{
|
||||||
|
memcpy(b->getBuffer(), cur_buf->getBuffer(), this->total_size);
|
||||||
|
}
|
||||||
|
this->data.clear();
|
||||||
|
cur_buf = b;
|
||||||
|
this->data.push_back(cur_buf);
|
||||||
|
}
|
||||||
|
if (len)
|
||||||
|
{
|
||||||
|
memcpy(cur_buf->getBuffer() + this->total_size, buf, len);
|
||||||
|
this->total_size += len;
|
||||||
|
}
|
||||||
this->ready = false;
|
this->ready = false;
|
||||||
this->total_size += len;
|
|
||||||
|
|
||||||
if (getNext(true))
|
if (getNext(true))
|
||||||
{
|
{
|
||||||
@ -49,17 +70,13 @@ Pl_Buffer::getBuffer()
|
|||||||
|
|
||||||
Buffer* b = new Buffer(this->total_size);
|
Buffer* b = new Buffer(this->total_size);
|
||||||
unsigned char* p = b->getBuffer();
|
unsigned char* p = b->getBuffer();
|
||||||
while (! this->data.empty())
|
if (! this->data.empty())
|
||||||
{
|
{
|
||||||
PointerHolder<Buffer> bp = this->data.front();
|
PointerHolder<Buffer> bp = this->data.back();
|
||||||
this->data.pop_front();
|
this->data.clear();
|
||||||
size_t bytes = bp->getSize();
|
memcpy(p, bp->getBuffer(), this->total_size);
|
||||||
memcpy(p, bp->getBuffer(), bytes);
|
|
||||||
p += bytes;
|
|
||||||
this->total_size -= bytes;
|
|
||||||
}
|
}
|
||||||
|
this->total_size = 0;
|
||||||
assert(this->total_size == 0);
|
|
||||||
this->ready = false;
|
this->ready = false;
|
||||||
|
|
||||||
return b;
|
return b;
|
||||||
|
Loading…
Reference in New Issue
Block a user