mirror of https://github.com/adamdruppe/arsd.git
ketmar memory management patch
This commit is contained in:
parent
a82ab16df8
commit
f48d7f4e9a
115
png.d
115
png.d
|
@ -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;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue