mirror of https://github.com/adamdruppe/arsd.git
bug fixes and hack to read corrupted files
This commit is contained in:
parent
83a0c3db07
commit
eab34ab324
46
bmp.d
46
bmp.d
|
@ -26,8 +26,16 @@ MemoryImage readBmp(string filename) {
|
||||||
return readBmpIndirect(&specialFread);
|
return readBmpIndirect(&specialFread);
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Reads a bitmap out of an in-memory array of data. For example, that returned from [std.file.read].
|
/++
|
||||||
MemoryImage readBmp(in ubyte[] data, bool lookForFileHeader = true) {
|
Reads a bitmap out of an in-memory array of data. For example, from the data returned from [std.file.read].
|
||||||
|
|
||||||
|
It forwards the arguments to [readBmpIndirect], so see that for more details.
|
||||||
|
|
||||||
|
If you are given a raw pointer to some data, you might just slice it: bytes 2-6 of the file header (if present)
|
||||||
|
are a little-endian uint giving the file size. You might slice only to that, or you could slice right to `int.max`
|
||||||
|
and trust the library to bounds check for you based on data integrity checks.
|
||||||
|
+/
|
||||||
|
MemoryImage readBmp(in ubyte[] data, bool lookForFileHeader = true, bool hackAround64BitLongs = false) {
|
||||||
const(ubyte)[] current = data;
|
const(ubyte)[] current = data;
|
||||||
void specialFread(void* tgt, size_t size) {
|
void specialFread(void* tgt, size_t size) {
|
||||||
while(size) {
|
while(size) {
|
||||||
|
@ -39,14 +47,16 @@ MemoryImage readBmp(in ubyte[] data, bool lookForFileHeader = true) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return readBmpIndirect(&specialFread, lookForFileHeader);
|
return readBmpIndirect(&specialFread, lookForFileHeader, hackAround64BitLongs);
|
||||||
}
|
}
|
||||||
|
|
||||||
/++
|
/++
|
||||||
Reads using a delegate to read instead of assuming a direct file
|
Reads using a delegate to read instead of assuming a direct file. View the source of `readBmp`'s overloads for fairly simple examples of how you can use it
|
||||||
|
|
||||||
History:
|
History:
|
||||||
The `lookForFileHeader` param was added in July 2020.
|
The `lookForFileHeader` param was added in July 2020.
|
||||||
|
|
||||||
|
The `hackAround64BitLongs` param was added December 21, 2020. You should probably never use this unless you know for sure you have a file corrupted in this specific way. View the source to see a comment inside the file to describe it a bit more.
|
||||||
+/
|
+/
|
||||||
MemoryImage readBmpIndirect(scope void delegate(void*, size_t) fread, bool lookForFileHeader = true, bool hackAround64BitLongs = false) {
|
MemoryImage readBmpIndirect(scope void delegate(void*, size_t) fread, bool lookForFileHeader = true, bool hackAround64BitLongs = false) {
|
||||||
uint read4() { uint what; fread(&what, 4); return what; }
|
uint read4() { uint what; fread(&what, 4); return what; }
|
||||||
|
@ -65,7 +75,19 @@ MemoryImage readBmpIndirect(scope void delegate(void*, size_t) fread, bool lookF
|
||||||
return le;
|
return le;
|
||||||
}
|
}
|
||||||
ushort read2(){ ushort what; fread(&what, 2); return what; }
|
ushort read2(){ ushort what; fread(&what, 2); return what; }
|
||||||
ubyte read1(){ ubyte what; fread(&what, 1); return what; }
|
|
||||||
|
bool headerRead = false;
|
||||||
|
int hackCounter;
|
||||||
|
|
||||||
|
ubyte read1() {
|
||||||
|
if(hackAround64BitLongs && headerRead && hackCounter < 16) {
|
||||||
|
hackCounter++;
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
ubyte what;
|
||||||
|
fread(&what, 1);
|
||||||
|
return what;
|
||||||
|
}
|
||||||
|
|
||||||
void require1(ubyte t, size_t line = __LINE__) {
|
void require1(ubyte t, size_t line = __LINE__) {
|
||||||
if(read1() != t)
|
if(read1() != t)
|
||||||
|
@ -177,12 +199,15 @@ MemoryImage readBmpIndirect(scope void delegate(void*, size_t) fread, bool lookF
|
||||||
version(arsd_debug_bitmap_loader) { import core.stdc.stdio; printf("header bytes left: %u\n", cast(uint)sizeOfBitmapInfoHeader); }
|
version(arsd_debug_bitmap_loader) { import core.stdc.stdio; printf("header bytes left: %u\n", cast(uint)sizeOfBitmapInfoHeader); }
|
||||||
foreach (skip; 0..sizeOfBitmapInfoHeader) read1();
|
foreach (skip; 0..sizeOfBitmapInfoHeader) read1();
|
||||||
|
|
||||||
|
headerRead = true;
|
||||||
|
|
||||||
if(bitsPerPixel <= 8) {
|
if(bitsPerPixel <= 8) {
|
||||||
// indexed image
|
// indexed image
|
||||||
version(arsd_debug_bitmap_loader) { import core.stdc.stdio; printf("colorsUsed=%u; colorsImportant=%u\n", colorsUsed, colorsImportant); }
|
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);
|
if (colorsUsed == 0 || colorsUsed > (1 << bitsPerPixel)) colorsUsed = (1 << bitsPerPixel);
|
||||||
auto img = new IndexedImage(width, height);
|
auto img = new IndexedImage(width, height);
|
||||||
img.palette.reserve(1 << bitsPerPixel);
|
img.palette.reserve(1 << bitsPerPixel);
|
||||||
|
|
||||||
foreach(idx; 0 .. /*(1 << bitsPerPixel)*/colorsUsed) {
|
foreach(idx; 0 .. /*(1 << bitsPerPixel)*/colorsUsed) {
|
||||||
auto b = read1();
|
auto b = read1();
|
||||||
auto g = read1();
|
auto g = read1();
|
||||||
|
@ -260,6 +285,7 @@ MemoryImage readBmpIndirect(scope void delegate(void*, size_t) fread, bool lookF
|
||||||
++bytesRead;
|
++bytesRead;
|
||||||
}
|
}
|
||||||
bytesRead = 0;
|
bytesRead = 0;
|
||||||
|
|
||||||
for(int x = 0; x < width; x++) {
|
for(int x = 0; x < width; x++) {
|
||||||
auto b = read1();
|
auto b = read1();
|
||||||
++bytesRead;
|
++bytesRead;
|
||||||
|
@ -400,7 +426,9 @@ void writeBmp(MemoryImage img, string filename) {
|
||||||
throw new Exception("can't open save file");
|
throw new Exception("can't open save file");
|
||||||
scope(exit) fclose(fp);
|
scope(exit) fclose(fp);
|
||||||
|
|
||||||
|
int written;
|
||||||
void my_fwrite(ubyte b) {
|
void my_fwrite(ubyte b) {
|
||||||
|
written++;
|
||||||
fputc(b, fp);
|
fputc(b, fp);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -465,15 +493,15 @@ void writeBmpIndirect(MemoryImage img, scope void delegate(ubyte) fwrite, bool p
|
||||||
ushort offsetToBits;
|
ushort offsetToBits;
|
||||||
if(bitsPerPixel == 8)
|
if(bitsPerPixel == 8)
|
||||||
offsetToBits = 1078;
|
offsetToBits = 1078;
|
||||||
if (bitsPerPixel == 24 || bitsPerPixel == 16)
|
else if (bitsPerPixel == 24 || bitsPerPixel == 16)
|
||||||
offsetToBits = 54;
|
offsetToBits = 54;
|
||||||
else
|
else
|
||||||
offsetToBits = cast(ushort)(54 + 4 * 1 << bitsPerPixel); // room for the palette...
|
offsetToBits = cast(ushort)(54 * (1 << bitsPerPixel)); // room for the palette...
|
||||||
|
|
||||||
uint fileSize = offsetToBits;
|
uint fileSize = offsetToBits;
|
||||||
if(bitsPerPixel == 8)
|
if(bitsPerPixel == 8) {
|
||||||
fileSize += height * (width + width%4);
|
fileSize += height * (width + width%4);
|
||||||
else if(bitsPerPixel == 24)
|
} else if(bitsPerPixel == 24)
|
||||||
fileSize += height * ((width * 3) + (!((width*3)%4) ? 0 : 4-((width*3)%4)));
|
fileSize += height * ((width * 3) + (!((width*3)%4) ? 0 : 4-((width*3)%4)));
|
||||||
else assert(0, "not implemented"); // FIXME
|
else assert(0, "not implemented"); // FIXME
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue