ketmar memory management patch

This commit is contained in:
Adam D. Ruppe 2016-06-17 10:21:22 -04:00
parent a82ab16df8
commit f48d7f4e9a
1 changed files with 78 additions and 37 deletions

115
png.d
View File

@ -44,7 +44,7 @@ void main(string[] args) {
// By Adam D. Ruppe, 2009-2010, released into the public domain // By Adam D. Ruppe, 2009-2010, released into the public domain
//import std.file; //import std.file;
import std.zlib; //import std.zlib;
public import arsd.color; public import arsd.color;
@ -404,6 +404,7 @@ void writeImageToPngFile(in char[] filename, TrueColorImage image) {
PNG* png; PNG* png;
ubyte[] com; ubyte[] com;
{ {
import std.zlib;
PngHeader h; PngHeader h;
h.width = image.width; h.width = image.width;
h.height = image.height; h.height = image.height;
@ -622,6 +623,7 @@ void addImageDatastreamToPng(const(ubyte)[] data, PNG* png) {
// we need to go through the lines and add the filter byte // we need to go through the lines and add the filter byte
// then compress it into an IDAT chunk // then compress it into an IDAT chunk
// then add the IEND chunk // then add the IEND chunk
import std.zlib;
PngHeader h = getHeader(png); PngHeader h = getHeader(png);
@ -661,6 +663,7 @@ deprecated alias PngHeader PNGHeader;
// bKGD - palette entry for background or the RGB (16 bits each) for that. or 16 bits of grey // bKGD - palette entry for background or the RGB (16 bits each) for that. or 16 bits of grey
ubyte[] getDatastream(PNG* p) { ubyte[] getDatastream(PNG* p) {
import std.zlib;
ubyte[] compressed; ubyte[] compressed;
foreach(c; p.chunks) { foreach(c; p.chunks) {
@ -1138,7 +1141,6 @@ struct LazyPngFile(LazyPngChunksProvider)
if(isInputRange!(LazyPngChunksProvider) && if(isInputRange!(LazyPngChunksProvider) &&
is(ElementType!(LazyPngChunksProvider) == Chunk)) is(ElementType!(LazyPngChunksProvider) == Chunk))
{ {
LazyPngChunksProvider chunks; LazyPngChunksProvider chunks;
this(LazyPngChunksProvider chunks) { this(LazyPngChunksProvider chunks) {
@ -1210,58 +1212,98 @@ struct LazyPngFile(LazyPngChunksProvider)
chunkSize = bytesPerLine(); chunkSize = bytesPerLine();
struct DatastreamByChunk(T) { struct DatastreamByChunk(T) {
std.zlib.UnCompress decompressor; private import etc.c.zlib;
z_stream zs;
int chunkSize; int chunkSize;
int bufpos;
int plpos; // bytes eaten in current chunk payload
T chunks; T chunks;
bool eoz;
this(int cs, T chunks) { this(int cs, T chunks) {
decompressor = new std.zlib.UnCompress(); import core.stdc.stdlib : malloc;
this.chunkSize = cs; this.chunkSize = cs;
this.chunks = chunks; this.chunks = chunks;
assert(chunkSize > 0);
buffer = (cast(ubyte*)malloc(chunkSize))[0..chunkSize];
pkbuf = (cast(ubyte*)malloc(32768))[0..32768]; // arbitrary number
zs.avail_in = 0;
zs.avail_out = 0;
auto res = inflateInit2(&zs, 15);
assert(res == Z_OK);
popFront(); // priming popFront(); // priming
} }
ubyte[] front() { ~this () {
assert(current.length == chunkSize); version(arsdpng_debug) { import core.stdc.stdio : printf; printf("destroying lazy PNG reader...\n"); }
return current; import core.stdc.stdlib : free;
inflateEnd(&zs);
if (pkbuf.ptr !is null) free(pkbuf.ptr);
if (buffer.ptr !is null) free(buffer.ptr);
} }
ubyte[] current; @disable this (this); // no copies!
ubyte[] front () { return (bufpos > 0 ? buffer[0..bufpos] : null); }
ubyte[] buffer; ubyte[] buffer;
ubyte[] pkbuf; // we will keep some packed data here in case payload moves, lol
void popFront() { void popFront () {
while(buffer.length < chunkSize) { bufpos = 0;
if(chunks.front().stype != "IDAT") { while (plpos != plpos.max && bufpos < chunkSize) {
buffer ~= cast(ubyte[]) decompressor.flush(); // do we have some bytes in zstream?
if(buffer.length != 0) { if (zs.avail_in > 0) {
// FIXME why was this here? // just unpack
//buffer ~= cast(ubyte[]) zs.next_out = cast(typeof(zs.next_out))(buffer.ptr+bufpos);
//decompressor.uncompress(chunks.front().payload); int rd = chunkSize-bufpos;
continue; zs.avail_out = rd;
auto err = inflate(&zs, Z_SYNC_FLUSH);
if (err != Z_STREAM_END && err != Z_OK) throw new Exception("PNG unpack error");
if (err == Z_STREAM_END) {
assert(zs.avail_in == 0);
eoz = true;
} }
current = null; bufpos += rd-zs.avail_out;
buffer = null; continue;
return;
} }
// no more zstream bytes; do we have something in current chunk?
buffer ~= cast(ubyte[]) if (plpos == plpos.max || plpos >= chunks.front.payload.length) {
decompressor.uncompress(chunks.front().payload); // current chunk is complete, do we have more chunks?
chunks.popFront(); if (chunks.front.stype != "IDAT") break; // this chunk is not IDAT, that means that... alas
chunks.popFront(); // remove current IDAT
plpos = 0;
if (chunks.empty || chunks.front.stype != "IDAT") plpos = plpos.max; // special value
continue;
}
if (plpos < chunks.front.payload.length) {
// current chunk is not complete, get some more bytes from it
int rd = cast(int)(chunks.front.payload.length-plpos <= pkbuf.length ? chunks.front.payload.length-plpos : pkbuf.length);
assert(rd > 0);
pkbuf[0..rd] = chunks.front.payload[plpos..plpos+rd];
plpos += rd;
if (eoz) {
// we did hit end-of-stream, reinit zlib (well, well, i know that we can reset it... meh)
inflateEnd(&zs);
zs.avail_in = 0;
zs.avail_out = 0;
auto res = inflateInit2(&zs, 15);
assert(res == Z_OK);
eoz = false;
}
// setup read pointer
zs.next_in = cast(typeof(zs.next_in))pkbuf.ptr;
zs.avail_in = cast(uint)rd;
continue;
}
assert(0, "wtf?! we should not be here!");
} }
assert(chunkSize <= buffer.length, format("%s !<= %s remaining data: \n%s", chunkSize, buffer.length, buffer));
current = buffer[0 .. chunkSize];
buffer = buffer[chunkSize .. $];
} }
bool empty() { bool empty () { return (bufpos == 0); }
return (current.length == 0);
}
} }
auto range = DatastreamByChunk!(typeof(chunks))(chunkSize, chunks); return DatastreamByChunk!(typeof(chunks))(chunkSize, chunks);
return range;
} }
// FIXME: no longer compiles // FIXME: no longer compiles
@ -1514,7 +1556,7 @@ struct BufferedInputRange(Range)
/* PNG file format implementation */ /* PNG file format implementation */
import std.zlib; //import std.zlib;
import std.math; import std.math;
/// All PNG files are supposed to open with these bytes according to the spec /// All PNG files are supposed to open with these bytes according to the spec
@ -1682,6 +1724,7 @@ void writePngLazy(OutputRange, InputRange)(ref OutputRange where, InputRange ima
isInputRange!(InputRange) && isInputRange!(InputRange) &&
is(ElementType!InputRange == RgbaScanline)) is(ElementType!InputRange == RgbaScanline))
{ {
import std.zlib;
where.put(PNG_MAGIC_NUMBER); where.put(PNG_MAGIC_NUMBER);
PngHeader header; PngHeader header;
@ -1812,5 +1855,3 @@ int bytesPerPixel(PngHeader header) {
return (bitsPerPixel + 7) / 8; return (bitsPerPixel + 7) / 8;
} }