ketmar rle8 bmp loader

This commit is contained in:
Adam D. Ruppe 2017-03-15 09:17:15 -04:00
parent 35d7806ad3
commit 2f406ae2ad
1 changed files with 160 additions and 61 deletions

221
bmp.d
View File

@ -3,6 +3,9 @@ module arsd.bmp;
import arsd.color;
//version = arsd_debug_bitmap_loader;
MemoryImage readBmp(string filename) {
import core.stdc.stdio;
@ -12,6 +15,7 @@ MemoryImage readBmp(string filename) {
scope(exit) fclose(fp);
void specialFread(void* tgt, size_t size) {
version(arsd_debug_bitmap_loader) { import core.stdc.stdio; printf("ofs: 0x%08x\n", cast(uint)ftell(fp)); }
fread(tgt, size, 1, fp);
}
@ -60,51 +64,92 @@ MemoryImage readBmpIndirect(void delegate(void*, size_t) fread) {
require2(0); // reserved
auto offsetToBits = read4();
version(arsd_debug_bitmap_loader) { import core.stdc.stdio; printf("pixel data offset: 0x%08x\n", cast(uint)offsetToBits); }
auto sizeOfBitmapInfoHeader = read4();
if (sizeOfBitmapInfoHeader < 12) throw new Exception("invalid bitmap header size");
auto width = read4();
auto height = read4();
int width, height, rdheight;
if (sizeOfBitmapInfoHeader == 12) {
width = read2();
rdheight = cast(short)read2();
} else {
if (sizeOfBitmapInfoHeader < 16) throw new Exception("invalid bitmap header size");
sizeOfBitmapInfoHeader -= 4; // hack!
width = read4();
rdheight = cast(int)read4();
}
height = (rdheight < 0 ? -rdheight : rdheight);
rdheight = (rdheight < 0 ? 1 : -1); // so we can use it as delta (note the inverted sign)
if (width < 1 || height < 1) throw new Exception("invalid bitmap dimensions");
require2(1); // planes
auto bitsPerPixel = read2();
switch (bitsPerPixel) {
case 1: case 2: case 4: case 8: case 16: case 24: case 32: break;
default: throw new Exception("invalid bitmap depth");
}
/*
0 = BI_RGB
1 = BI_RLE8 RLE 8-bit/pixel Can be used only with 8-bit/pixel bitmaps
2 = BI_RLE4 RLE 4-bit/pixel Can be used only with 4-bit/pixel bitmaps
3 = BI_BITFIELDS
*/
auto compression = read4();
auto sizeOfUncompressedData = read4();
uint compression = 0;
uint sizeOfUncompressedData = 0;
uint xPixelsPerMeter = 0;
uint yPixelsPerMeter = 0;
uint colorsUsed = 0;
uint colorsImportant = 0;
auto xPixelsPerMeter = read4();
auto yPixelsPerMeter = read4();
auto colorsUsed = read4();
auto colorsImportant = read4();
sizeOfBitmapInfoHeader -= 12;
if (sizeOfBitmapInfoHeader > 0) {
if (sizeOfBitmapInfoHeader < 6*4) throw new Exception("invalid bitmap header size");
sizeOfBitmapInfoHeader -= 6*4;
compression = read4();
sizeOfUncompressedData = read4();
xPixelsPerMeter = read4();
yPixelsPerMeter = read4();
colorsUsed = read4();
colorsImportant = read4();
}
if (compression > 3) throw new Exception("invalid bitmap compression");
if (compression == 1 && bitsPerPixel != 8) throw new Exception("invalid bitmap compression");
if (compression == 2 && bitsPerPixel != 4) throw new Exception("invalid bitmap compression");
version(arsd_debug_bitmap_loader) { import core.stdc.stdio; printf("compression: %u; bpp: %u\n", compression, cast(uint)bitsPerPixel); }
int additionalRead = 0;
uint redMask;
uint greenMask;
uint blueMask;
uint alphaMask;
if(compression == 3) {
if (compression == 3) {
if (sizeOfBitmapInfoHeader < 4*4) throw new Exception("invalid bitmap compression");
sizeOfBitmapInfoHeader -= 4*4;
redMask = read4();
greenMask = read4();
blueMask = read4();
alphaMask = read4();
additionalRead += 4 * 4;
}
// FIXME: we could probably handle RLE as well
// FIXME: we could probably handle RLE4 as well
// I don't know about the rest of the header, so I'm just skipping it.
// 40 is the size of the basic info header that I did read.
foreach(skip; 0 .. sizeOfBitmapInfoHeader - 40 - additionalRead)
read1();
version(arsd_debug_bitmap_loader) { import core.stdc.stdio; printf("header bytes left: %u\n", cast(uint)sizeOfBitmapInfoHeader); }
foreach (skip; 0..sizeOfBitmapInfoHeader) read1();
if(bitsPerPixel <= 8) {
// indexed image
version(arsd_debug_bitmap_loader) { import core.stdc.stdio; printf("colorsUsed=%u; colorsImportant=%u\n", colorsUsed, colorsImportant); }
if (colorsUsed == 0 || colorsUsed > (1 << bitsPerPixel)) colorsUsed = (1 << bitsPerPixel);
auto img = new IndexedImage(width, height);
foreach(idx; 0 .. (1 << bitsPerPixel)) {
img.palette.reserve(1 << bitsPerPixel);
foreach(idx; 0 .. /*(1 << bitsPerPixel)*/colorsUsed) {
auto b = read1();
auto g = read1();
auto r = read1();
@ -112,64 +157,118 @@ MemoryImage readBmpIndirect(void delegate(void*, size_t) fread) {
img.palette ~= Color(r, g, b);
}
while (img.palette.length < (1 << bitsPerPixel)) img.palette ~= Color.transparent;
// and the data
int bytesPerPixel = 1;
auto offsetStart = width * height * bytesPerPixel;
auto offsetStart = (rdheight > 0 ? 0 : width * height * bytesPerPixel);
int bytesRead = 0;
for(int y = height; y > 0; y--) {
offsetStart -= width * bytesPerPixel;
int offset = offsetStart;
int bytesRead = 0;
for(int x = 0; x < width; x++) {
auto b = read1();
bytesRead++;
if(bitsPerPixel == 8) {
img.data[offset++] = b;
} else if(bitsPerPixel == 4) {
img.data[offset++] = (b&0xf0) >> 4;
x++;
if(offset == img.data.length)
if (compression == 1) {
// this is complicated
assert(bitsPerPixel == 8); // always
int x = 0, y = (rdheight > 0 ? 0 : height-1);
void setpix (int v) {
if (x >= 0 && y >= 0 && x < width && y < height) img.data.ptr[y*width+x] = v&0xff;
++x;
}
version(arsd_debug_bitmap_loader) { import core.stdc.stdio; printf("width=%d; height=%d; rdheight=%d\n", width, height, rdheight); }
for (;;) {
ubyte codelen = read1();
ubyte codecode = read1();
version(arsd_debug_bitmap_loader) { import core.stdc.stdio; printf("x=%d; y=%d; len=%u; code=%u\n", x, y, cast(uint)codelen, cast(uint)codecode); }
bytesRead += 2;
if (codelen == 0) {
// special code
if (codecode == 0) {
// end of line
version(arsd_debug_bitmap_loader) { import core.stdc.stdio; printf(" EOL\n"); }
while (x < width) setpix(1);
x = 0;
y += rdheight;
if (y < 0 || y >= height) break; // ooops
} else if (codecode == 1) {
version(arsd_debug_bitmap_loader) { import core.stdc.stdio; printf(" EOB\n"); }
// end of bitmap
break;
img.data[offset++] = (b&0x0f);
} else if(bitsPerPixel == 2) {
img.data[offset++] = (b & 0b11000000) >> 6;
x++;
if(offset == img.data.length)
break;
img.data[offset++] = (b & 0b00110000) >> 4;
x++;
if(offset == img.data.length)
break;
img.data[offset++] = (b & 0b00001100) >> 2;
x++;
if(offset == img.data.length)
break;
img.data[offset++] = (b & 0b00000011) >> 0;
} else if(bitsPerPixel == 1) {
foreach(lol; 0 .. 8) {
img.data[offset++] = (b & (1 << lol)) >> (7 - lol);
} else if (codecode == 2) {
// delta
int xofs = read1();
int yofs = read1();
version(arsd_debug_bitmap_loader) { import core.stdc.stdio; printf(" deltax=%d; deltay=%d\n", xofs, yofs); }
bytesRead += 2;
x += xofs;
y += yofs*rdheight;
if (y < 0 || y >= height) break; // ooops
} else {
version(arsd_debug_bitmap_loader) { import core.stdc.stdio; printf(" LITERAL: %u\n", cast(uint)codecode); }
// literal copy
while (codecode-- > 0) {
setpix(read1());
++bytesRead;
}
version(arsd_debug_bitmap_loader) if (bytesRead%2) { import core.stdc.stdio; printf(" LITERAL SKIP\n"); }
if (bytesRead%2) { read1(); ++bytesRead; }
assert(bytesRead%2 == 0);
}
} else {
while (codelen-- > 0) setpix(codecode);
}
}
} else if (compression == 2) {
throw new Exception("4RLE for bitmaps aren't supported yet");
} else {
for(int y = height; y > 0; y--) {
if (rdheight < 0) offsetStart -= width * bytesPerPixel;
int offset = offsetStart;
while (bytesRead%4 != 0) {
read1();
++bytesRead;
}
bytesRead = 0;
for(int x = 0; x < width; x++) {
auto b = read1();
++bytesRead;
if(bitsPerPixel == 8) {
img.data[offset++] = b;
} else if(bitsPerPixel == 4) {
img.data[offset++] = (b&0xf0) >> 4;
x++;
if(offset == img.data.length)
break;
}
x--; // we do this once too many times in the loop
} else assert(0);
// I don't think these happen in the wild but I could be wrong, my bmp knowledge is somewhat outdated
img.data[offset++] = (b&0x0f);
} else if(bitsPerPixel == 2) {
img.data[offset++] = (b & 0b11000000) >> 6;
x++;
if(offset == img.data.length)
break;
img.data[offset++] = (b & 0b00110000) >> 4;
x++;
if(offset == img.data.length)
break;
img.data[offset++] = (b & 0b00001100) >> 2;
x++;
if(offset == img.data.length)
break;
img.data[offset++] = (b & 0b00000011) >> 0;
} else if(bitsPerPixel == 1) {
foreach(lol; 0 .. 8) {
img.data[offset++] = (b & (1 << lol)) >> (7 - lol);
x++;
if(offset == img.data.length)
break;
}
x--; // we do this once too many times in the loop
} else assert(0);
// I don't think these happen in the wild but I could be wrong, my bmp knowledge is somewhat outdated
}
if (rdheight > 0) offsetStart += width * bytesPerPixel;
}
int w = bytesRead%4;
if(w)
for(int a = 0; a < 4-w; a++)
require1(0); // pad until divisible by four
}
return img;
} else {
if (compression != 0) throw new Exception("invalid bitmap compression");
// true color image
auto img = new TrueColorImage(width, height);