From 6343bf04ca08d68b4d8dfa0b7fd6a847bc1b278a Mon Sep 17 00:00:00 2001 From: "Adam D. Ruppe" Date: Tue, 12 Jul 2022 14:23:34 -0400 Subject: [PATCH] didnt handle < 8bpp small images well --- png.d | 29 ++++++++++++++++++++--------- 1 file changed, 20 insertions(+), 9 deletions(-) diff --git a/png.d b/png.d index 5db45a3..c4f27bf 100644 --- a/png.d +++ b/png.d @@ -334,7 +334,14 @@ PNG* pngFromImage(IndexedImage i) { addImageDatastreamToPng(i.data, png); } else { // gotta convert it - ubyte[] datastream = new ubyte[cast(size_t)i.width * i.height * h.depth / 8]; // FIXME? + + auto bitsPerLine = i.width * h.depth; + if(bitsPerLine % 8 != 0) + bitsPerLine = bitsPerLine / 8 + 1; + else + bitsPerLine = bitsPerLine / 8; + + ubyte[] datastream = new ubyte[bitsPerLine * i.height]; int shift = 0; switch(h.depth) { @@ -704,28 +711,32 @@ void addImageDatastreamToPng(const(ubyte)[] data, PNG* png, bool addIend = true) if(h.width == 0) throw new Exception("width zero?!!?!?!"); + int multiplier; size_t bytesPerLine; switch(h.type) { case 0: - // FIXME: < 8 depth not supported here but should be - bytesPerLine = cast(size_t)h.width * 1 * h.depth / 8; + multiplier = 1; break; case 2: - bytesPerLine = cast(size_t)h.width * 3 * h.depth / 8; + multiplier = 3; break; case 3: - bytesPerLine = cast(size_t)h.width * 1 * h.depth / 8; + multiplier = 1; break; case 4: - // FIXME: < 8 depth not supported here but should be - bytesPerLine = cast(size_t)h.width * 2 * h.depth / 8; + multiplier = 2; break; case 6: - bytesPerLine = cast(size_t)h.width * 4 * h.depth / 8; + multiplier = 4; break; default: assert(0); - } + + bytesPerLine = h.width * multiplier * h.depth / 8; + if((h.width * multiplier * h.depth) % 8 != 0) + bytesPerLine += 1; + + assert(bytesPerLine >= 1); Chunk dat; dat.type = ['I', 'D', 'A', 'T']; size_t pos = 0;