mirror of https://github.com/adamdruppe/arsd.git
ketmar module: .tga file loading
This commit is contained in:
parent
351eded297
commit
e22ed09eff
78
image.d
78
image.d
|
@ -5,6 +5,7 @@ public import arsd.color;
|
|||
public import arsd.png;
|
||||
public import arsd.jpeg;
|
||||
public import arsd.bmp;
|
||||
public import arsd.targa;
|
||||
|
||||
static if (__traits(compiles, { import iv.vfs; })) enum ArsdImageHasIVVFS = true; else enum ArsdImageHasIVVFS = false;
|
||||
|
||||
|
@ -27,6 +28,8 @@ enum ImageFileFormat {
|
|||
Png, ///
|
||||
Bmp, ///
|
||||
Jpeg, ///
|
||||
Tga, ///
|
||||
Gif, /// we can't load it yet, but we can at least detect it
|
||||
}
|
||||
|
||||
|
||||
|
@ -44,12 +47,15 @@ public ImageFileFormat guessImageFormatFromExtension (const(char)[] filename) {
|
|||
if (strEquCI(ext, "png")) return ImageFileFormat.Png;
|
||||
if (strEquCI(ext, "bmp")) return ImageFileFormat.Bmp;
|
||||
if (strEquCI(ext, "jpg") || strEquCI(ext, "jpeg")) return ImageFileFormat.Jpeg;
|
||||
if (strEquCI(ext, "gif")) return ImageFileFormat.Gif;
|
||||
if (strEquCI(ext, "tga")) return ImageFileFormat.Tga;
|
||||
return ImageFileFormat.Unknown;
|
||||
}
|
||||
|
||||
|
||||
/// Try to guess image format by first data bytes.
|
||||
public ImageFileFormat guessImageFormatFromMemory (const(void)[] membuf) {
|
||||
enum TargaSign = "TRUEVISION-XFILE.\x00";
|
||||
auto buf = cast(const(ubyte)[])membuf;
|
||||
if (buf.length == 0) return ImageFileFormat.Unknown;
|
||||
// detect file format
|
||||
|
@ -64,11 +70,79 @@ public ImageFileFormat guessImageFormatFromMemory (const(void)[] membuf) {
|
|||
uint datasize = buf.ptr[2]|(buf.ptr[3]<<8)|(buf.ptr[4]<<16)|(buf.ptr[5]<<24);
|
||||
if (datasize > 6 && datasize <= buf.length) return ImageFileFormat.Bmp;
|
||||
}
|
||||
// gif
|
||||
if (buf.length > 5 && buf.ptr[0] == 'G' && buf.ptr[1] == 'I' && buf.ptr[2] == 'F' &&
|
||||
buf.ptr[3] == '8' && (buf.ptr[4] == '7' || buf.ptr[4] == '9'))
|
||||
{
|
||||
return ImageFileFormat.Gif;
|
||||
}
|
||||
// jpg
|
||||
try {
|
||||
int width, height, components;
|
||||
if (detect_jpeg_image_from_memory(buf, width, height, components)) return ImageFileFormat.Jpeg;
|
||||
} catch (Exception e) {} // sorry
|
||||
// tga (sorry, targas without footer, i don't love you)
|
||||
if (buf.length > TargaSign.length+4*2 && cast(const(char)[])(buf[$-TargaSign.length..$]) == TargaSign) {
|
||||
// more guesswork
|
||||
switch (buf.ptr[2]) {
|
||||
case 1: case 2: case 3: case 9: case 10: case 11: return ImageFileFormat.Tga;
|
||||
default:
|
||||
}
|
||||
}
|
||||
// ok, try to guess targa by validating some header fields
|
||||
bool guessTarga () nothrow @trusted @nogc {
|
||||
if (buf.length < 45) return false; // minimal 1x1 tga
|
||||
immutable ubyte idlength = buf.ptr[0];
|
||||
immutable ubyte bColorMapType = buf.ptr[1];
|
||||
immutable ubyte type = buf.ptr[2];
|
||||
immutable ushort wColorMapFirstEntryIndex = cast(ushort)(buf.ptr[3]|(buf.ptr[4]<<8));
|
||||
immutable ushort wColorMapLength = cast(ushort)(buf.ptr[5]|(buf.ptr[6]<<8));
|
||||
immutable ubyte bColorMapEntrySize = buf.ptr[7];
|
||||
immutable ushort wOriginX = cast(ushort)(buf.ptr[8]|(buf.ptr[9]<<8));
|
||||
immutable ushort wOriginY = cast(ushort)(buf.ptr[10]|(buf.ptr[11]<<8));
|
||||
immutable ushort wImageWidth = cast(ushort)(buf.ptr[12]|(buf.ptr[13]<<8));
|
||||
immutable ushort wImageHeight = cast(ushort)(buf.ptr[14]|(buf.ptr[15]<<8));
|
||||
immutable ubyte bPixelDepth = buf.ptr[16];
|
||||
immutable ubyte bImageDescriptor = buf.ptr[17];
|
||||
if (wImageWidth < 1 || wImageHeight < 1 || wImageWidth > 16384 || wImageHeight > 16384) return false; // arbitrary limit
|
||||
immutable uint pixelsize = (bPixelDepth>>3);
|
||||
switch (type) {
|
||||
case 2: // truecolor, raw
|
||||
case 10: // truecolor, rle
|
||||
switch (pixelsize) {
|
||||
case 2: case 3: case 4: break;
|
||||
default: return false;
|
||||
}
|
||||
break;
|
||||
case 1: // paletted, raw
|
||||
case 9: // paletted, rle
|
||||
if (pixelsize != 1) return false;
|
||||
break;
|
||||
case 3: // b/w, raw
|
||||
case 11: // b/w, rle
|
||||
if (pixelsize != 1 && pixelsize != 2) return false;
|
||||
break;
|
||||
default: // invalid type
|
||||
return false;
|
||||
}
|
||||
// check for valid colormap
|
||||
switch (bColorMapType) {
|
||||
case 0:
|
||||
if (wColorMapFirstEntryIndex != 0 || wColorMapLength != 0 || bColorMapEntrySize != 0) return 0;
|
||||
break;
|
||||
case 1:
|
||||
if (bColorMapEntrySize != 15 && bColorMapEntrySize != 16 && bColorMapEntrySize != 24 && bColorMapEntrySize != 32) return false;
|
||||
if (wColorMapLength == 0) return false;
|
||||
break;
|
||||
default: // invalid colormap type
|
||||
return false;
|
||||
}
|
||||
if (((bImageDescriptor>>6)&3) != 0) return false;
|
||||
// this *looks* like a tga
|
||||
return true;
|
||||
}
|
||||
if (guessTarga()) return ImageFileFormat.Tga;
|
||||
// dunno
|
||||
return ImageFileFormat.Unknown;
|
||||
}
|
||||
|
||||
|
@ -83,6 +157,8 @@ public MemoryImage loadImageFromFile(T:const(char)[]) (T filename) {
|
|||
case ImageFileFormat.Png: static if (is(T == string)) return readPng(filename); else return readPng(filename.idup);
|
||||
case ImageFileFormat.Bmp: static if (is(T == string)) return readBmp(filename); else return readBmp(filename.idup);
|
||||
case ImageFileFormat.Jpeg: return readJpeg(filename);
|
||||
case ImageFileFormat.Gif: throw new Exception("arsd has no GIF loader yet");
|
||||
case ImageFileFormat.Tga: import std.stdio; return loadTga(File(filename));
|
||||
}
|
||||
}
|
||||
}
|
||||
|
@ -95,6 +171,8 @@ public MemoryImage loadImageFromMemory (const(void)[] membuf) {
|
|||
case ImageFileFormat.Png: return imageFromPng(readPng(cast(const(ubyte)[])membuf));
|
||||
case ImageFileFormat.Bmp: return readBmp(cast(const(ubyte)[])membuf);
|
||||
case ImageFileFormat.Jpeg: return readJpegFromMemory(cast(const(ubyte)[])membuf);
|
||||
case ImageFileFormat.Gif: throw new Exception("arsd has no GIF loader yet");
|
||||
case ImageFileFormat.Tga: return loadTgaMem(membuf);
|
||||
}
|
||||
}
|
||||
|
||||
|
|
|
@ -0,0 +1,696 @@
|
|||
//ketmar: Adam didn't wrote this, don't blame him!
|
||||
module arsd.targa;
|
||||
|
||||
import arsd.color;
|
||||
import std.stdio : File; // sorry
|
||||
|
||||
static if (__traits(compiles, { import iv.vfs; })) enum ArsdTargaHasIVVFS = true; else enum ArsdTargaHasIVVFS = false;
|
||||
static if (ArsdTargaHasIVVFS) import iv.vfs;
|
||||
|
||||
|
||||
// ////////////////////////////////////////////////////////////////////////// //
|
||||
public MemoryImage loadTgaMem (const(void)[] buf, const(char)[] filename=null) {
|
||||
static struct MemRO {
|
||||
const(ubyte)[] data;
|
||||
long pos;
|
||||
|
||||
this (const(void)[] abuf) { data = cast(const(ubyte)[])abuf; }
|
||||
|
||||
@property long tell () { return pos; }
|
||||
@property long size () { return data.length; }
|
||||
|
||||
void seek (long offset, int whence=Seek.Set) {
|
||||
switch (whence) {
|
||||
case Seek.Set:
|
||||
if (offset < 0 || offset > data.length) throw new Exception("invalid offset");
|
||||
pos = offset;
|
||||
break;
|
||||
case Seek.Cur:
|
||||
if (offset < -pos || offset > data.length-pos) throw new Exception("invalid offset");
|
||||
pos += offset;
|
||||
break;
|
||||
case Seek.End:
|
||||
pos = data.length+offset;
|
||||
if (pos < 0 || pos > data.length) throw new Exception("invalid offset");
|
||||
break;
|
||||
default:
|
||||
throw new Exception("invalid offset origin");
|
||||
}
|
||||
}
|
||||
|
||||
ptrdiff_t read (void* buf, size_t count) {
|
||||
if (pos >= data.length) return 0;
|
||||
if (count > 0) {
|
||||
import core.stdc.string : memcpy;
|
||||
long rlen = data.length-pos;
|
||||
if (rlen >= count) rlen = count;
|
||||
assert(rlen != 0);
|
||||
memcpy(buf, data.ptr+pos, cast(size_t)rlen);
|
||||
pos += rlen;
|
||||
return cast(ptrdiff_t)rlen;
|
||||
} else {
|
||||
return 0;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
auto rd = MemRO(buf);
|
||||
return loadTga(rd, filename);
|
||||
}
|
||||
|
||||
static if (ArsdTargaHasIVVFS) public MemoryImage loadTga (VFile fl) { return loadTgaImpl(fl, fl.name); }
|
||||
public MemoryImage loadTga (File fl) { return loadTgaImpl(fl, fl.name); }
|
||||
public MemoryImage loadTga(T:const(char)[]) (T fname) {
|
||||
static if (is(T == typeof(null))) {
|
||||
throw new Exception("cannot load nameless tga");
|
||||
} else {
|
||||
static if (ArsdTargaHasIVVFS) {
|
||||
return loadTga(VFile(fname));
|
||||
} else static if (is(T == string)) {
|
||||
return loadTga(File(fname), fname);
|
||||
} else {
|
||||
return loadTga(File(fname.idup), fname);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// pass filename to ease detection
|
||||
// hack around "has scoped destruction, cannot build closure"
|
||||
public MemoryImage loadTga(ST) (auto ref ST fl, const(char)[] filename=null) if (isReadableStream!ST && isSeekableStream!ST) { return loadTgaImpl(fl, filename); }
|
||||
|
||||
private MemoryImage loadTgaImpl(ST) (auto ref ST fl, const(char)[] filename) {
|
||||
enum TGAFILESIGNATURE = "TRUEVISION-XFILE.\x00";
|
||||
|
||||
static immutable ubyte[32] cmap16 = [0,8,16,25,33,41,49,58,66,74,82,90,99,107,115,123,132,140,148,156,165,173,181,189,197,206,214,222,230,239,247,255];
|
||||
|
||||
static struct Header {
|
||||
ubyte idsize;
|
||||
ubyte cmapType;
|
||||
ubyte imgType;
|
||||
ushort cmapFirstIdx;
|
||||
ushort cmapSize;
|
||||
ubyte cmapElementSize;
|
||||
ushort originX;
|
||||
ushort originY;
|
||||
ushort width;
|
||||
ushort height;
|
||||
ubyte bpp;
|
||||
ubyte imgdsc;
|
||||
|
||||
@property bool zeroBits () const pure nothrow @safe @nogc { return ((imgdsc&0xc0) == 0); }
|
||||
@property bool xflip () const pure nothrow @safe @nogc { return ((imgdsc&0b010000) != 0); }
|
||||
@property bool yflip () const pure nothrow @safe @nogc { return ((imgdsc&0b100000) == 0); }
|
||||
}
|
||||
|
||||
static struct ExtFooter {
|
||||
uint extofs;
|
||||
uint devdirofs;
|
||||
char[18] sign=0;
|
||||
}
|
||||
|
||||
static struct Extension {
|
||||
ushort size;
|
||||
char[41] author=0;
|
||||
char[324] comments=0;
|
||||
ushort month, day, year;
|
||||
ushort hour, minute, second;
|
||||
char[41] jid=0;
|
||||
ushort jhours, jmins, jsecs;
|
||||
char[41] producer=0;
|
||||
ushort prodVer;
|
||||
ubyte prodSubVer;
|
||||
ubyte keyR, keyG, keyB, keyZero;
|
||||
ushort pixratioN, pixratioD;
|
||||
ushort gammaN, gammaD;
|
||||
uint ccofs;
|
||||
uint wtfofs;
|
||||
uint scanlineofs;
|
||||
ubyte attrType;
|
||||
}
|
||||
|
||||
ExtFooter extfooter;
|
||||
uint rleBC, rleDC;
|
||||
ubyte[4] rleLast;
|
||||
Color[256] cmap;
|
||||
|
||||
void readPixel(bool asRLE, uint bytesPerPixel) (ubyte[] pixel, scope ubyte delegate () readByte) {
|
||||
static if (asRLE) {
|
||||
if (rleDC > 0) {
|
||||
// still counting
|
||||
static if (bytesPerPixel == 1) pixel.ptr[0] = rleLast.ptr[0];
|
||||
else pixel.ptr[0..bytesPerPixel] = rleLast.ptr[0..bytesPerPixel];
|
||||
--rleDC;
|
||||
return;
|
||||
}
|
||||
if (rleBC > 0) {
|
||||
--rleBC;
|
||||
} else {
|
||||
ubyte b = readByte();
|
||||
if (b&0x80) rleDC = (b&0x7f); else rleBC = (b&0x7f);
|
||||
}
|
||||
foreach (immutable idx; 0..bytesPerPixel) rleLast.ptr[idx] = pixel.ptr[idx] = readByte();
|
||||
} else {
|
||||
foreach (immutable idx; 0..bytesPerPixel) pixel.ptr[idx] = readByte();
|
||||
}
|
||||
}
|
||||
|
||||
// 8 bit color-mapped row
|
||||
Color readColorCM8(bool asRLE) (scope ubyte delegate () readByte) {
|
||||
ubyte[1] pixel = void;
|
||||
readPixel!(asRLE, 1)(pixel[], readByte);
|
||||
auto cmp = cast(const(ubyte)*)(cmap.ptr+pixel.ptr[0]);
|
||||
return Color(cmp[0], cmp[1], cmp[2]);
|
||||
}
|
||||
|
||||
// 8 bit greyscale
|
||||
Color readColorBM8(bool asRLE) (scope ubyte delegate () readByte) {
|
||||
ubyte[1] pixel = void;
|
||||
readPixel!(asRLE, 1)(pixel[], readByte);
|
||||
return Color(pixel.ptr[0], pixel.ptr[0], pixel.ptr[0]);
|
||||
}
|
||||
|
||||
// 16 bit greyscale
|
||||
Color readColorBM16(bool asRLE) (scope ubyte delegate () readByte) {
|
||||
ubyte[2] pixel = void;
|
||||
readPixel!(asRLE, 2)(pixel[], readByte);
|
||||
immutable ubyte v = cast(ubyte)((pixel.ptr[0]|(pixel.ptr[1]<<8))>>8);
|
||||
return Color(v, v, v);
|
||||
}
|
||||
|
||||
// 16 bit
|
||||
Color readColor16(bool asRLE) (scope ubyte delegate () readByte) {
|
||||
ubyte[2] pixel = void;
|
||||
readPixel!(asRLE, 2)(pixel[], readByte);
|
||||
immutable v = pixel.ptr[0]+(pixel.ptr[1]<<8);
|
||||
return Color(cmap16.ptr[(v>>10)&0x1f], cmap16.ptr[(v>>5)&0x1f], cmap16.ptr[v&0x1f]);
|
||||
}
|
||||
|
||||
// 24 bit or 32 bit
|
||||
Color readColorTrue(bool asRLE, uint bytesPerPixel) (scope ubyte delegate () readByte) {
|
||||
ubyte[bytesPerPixel] pixel = void;
|
||||
readPixel!(asRLE, bytesPerPixel)(pixel[], readByte);
|
||||
static if (bytesPerPixel == 4) {
|
||||
return Color(pixel.ptr[2], pixel.ptr[1], pixel.ptr[0], pixel.ptr[3]);
|
||||
} else {
|
||||
return Color(pixel.ptr[2], pixel.ptr[1], pixel.ptr[0]);
|
||||
}
|
||||
}
|
||||
|
||||
bool isGoodExtension (const(char)[] filename) {
|
||||
if (filename.length >= 4) {
|
||||
// try extension
|
||||
auto ext = filename[$-4..$];
|
||||
if (ext[0] == '.' && (ext[1] == 'T' || ext[1] == 't') && (ext[2] == 'G' || ext[2] == 'g') && (ext[3] == 'A' || ext[3] == 'a')) return true;
|
||||
}
|
||||
// try signature
|
||||
return false;
|
||||
}
|
||||
|
||||
bool detect(ST) (auto ref ST fl, const(char)[] filename) if (isReadableStream!ST && isSeekableStream!ST) {
|
||||
bool goodext = false;
|
||||
if (fl.size < 45) return false; // minimal 1x1 tga
|
||||
if (filename.length) { goodext = isGoodExtension(filename); if (!goodext) return false; }
|
||||
// try footer
|
||||
fl.seek(-(4*2+18), Seek.End);
|
||||
extfooter.extofs = fl.readNum!uint;
|
||||
extfooter.devdirofs = fl.readNum!uint;
|
||||
fl.rawReadExact(extfooter.sign[]);
|
||||
if (extfooter.sign != TGAFILESIGNATURE) {
|
||||
//if (!goodext) return false;
|
||||
extfooter = extfooter.init;
|
||||
return true; // alas, footer is optional
|
||||
}
|
||||
return true;
|
||||
}
|
||||
|
||||
if (!detect(fl, filename)) throw new Exception("not a TGA");
|
||||
fl.seek(0);
|
||||
Header hdr;
|
||||
fl.readStruct(hdr);
|
||||
// parse header
|
||||
// arbitrary size limits
|
||||
if (hdr.width == 0 || hdr.width > 16384) throw new Exception("invalid tga width");
|
||||
if (hdr.height == 0 || hdr.height > 16384) throw new Exception("invalid tga height");
|
||||
switch (hdr.bpp) {
|
||||
case 1: case 2: case 4: case 8: case 15: case 16: case 24: case 32: break;
|
||||
default: throw new Exception("invalid tga bpp");
|
||||
}
|
||||
uint bytesPerPixel = ((hdr.bpp)>>3);
|
||||
if (bytesPerPixel == 0 || bytesPerPixel > 4) throw new Exception("invalid tga pixel size");
|
||||
bool loadCM = false;
|
||||
// get the row reading function
|
||||
ubyte readByte () { ubyte b; fl.rawReadExact((&b)[0..1]); return b; }
|
||||
scope Color delegate (scope ubyte delegate () readByte) readColor;
|
||||
switch (hdr.imgType) {
|
||||
case 2: // true color, no rle
|
||||
switch (bytesPerPixel) {
|
||||
case 2: readColor = &readColor16!false; break;
|
||||
case 3: readColor = &readColorTrue!(false, 3); break;
|
||||
case 4: readColor = &readColorTrue!(false, 4); break;
|
||||
default: throw new Exception("invalid tga pixel size");
|
||||
}
|
||||
break;
|
||||
case 10: // true color, rle
|
||||
switch (bytesPerPixel) {
|
||||
case 2: readColor = &readColor16!true; break;
|
||||
case 3: readColor = &readColorTrue!(true, 3); break;
|
||||
case 4: readColor = &readColorTrue!(true, 4); break;
|
||||
default: throw new Exception("invalid tga pixel size");
|
||||
}
|
||||
break;
|
||||
case 3: // black&white, no rle
|
||||
switch (bytesPerPixel) {
|
||||
case 1: readColor = &readColorBM8!false; break;
|
||||
case 2: readColor = &readColorBM16!false; break;
|
||||
default: throw new Exception("invalid tga pixel size");
|
||||
}
|
||||
break;
|
||||
case 11: // black&white, rle
|
||||
switch (bytesPerPixel) {
|
||||
case 1: readColor = &readColorBM8!true; break;
|
||||
case 2: readColor = &readColorBM16!true; break;
|
||||
default: throw new Exception("invalid tga pixel size");
|
||||
}
|
||||
break;
|
||||
case 1: // colormap, no rle
|
||||
if (bytesPerPixel != 1) throw new Exception("invalid tga pixel size");
|
||||
loadCM = true;
|
||||
readColor = &readColorCM8!false;
|
||||
break;
|
||||
case 9: // colormap, rle
|
||||
if (bytesPerPixel != 1) throw new Exception("invalid tga pixel size");
|
||||
loadCM = true;
|
||||
readColor = &readColorCM8!true;
|
||||
break;
|
||||
default: throw new Exception("invalid tga format");
|
||||
}
|
||||
// check for valid colormap
|
||||
switch (hdr.cmapType) {
|
||||
case 0:
|
||||
if (hdr.cmapFirstIdx != 0 || hdr.cmapSize != 0 || hdr.cmapElementSize != 0) throw new Exception("invalid tga colormap type");
|
||||
break;
|
||||
case 1:
|
||||
if (hdr.cmapElementSize != 15 && hdr.cmapElementSize != 16 && hdr.cmapElementSize != 24 && hdr.cmapElementSize != 32) throw new Exception("invalid tga colormap type");
|
||||
if (hdr.cmapSize == 0) throw new Exception("invalid tga colormap type");
|
||||
break;
|
||||
default: throw new Exception("invalid tga colormap type");
|
||||
}
|
||||
if (!hdr.zeroBits) throw new Exception("invalid tga header");
|
||||
void loadColormap () {
|
||||
if (hdr.cmapType != 1) throw new Exception("invalid tga colormap type");
|
||||
// calculate color map size
|
||||
uint colorEntryBytes = 0;
|
||||
switch (hdr.cmapElementSize) {
|
||||
case 15:
|
||||
case 16: colorEntryBytes = 2; break;
|
||||
case 24: colorEntryBytes = 3; break;
|
||||
case 32: colorEntryBytes = 4; break;
|
||||
default: throw new Exception("invalid tga colormap type");
|
||||
}
|
||||
uint colorMapBytes = colorEntryBytes*hdr.cmapSize;
|
||||
if (colorMapBytes == 0) throw new Exception("invalid tga colormap type");
|
||||
// if we're going to use the color map, read it in.
|
||||
if (loadCM) {
|
||||
if (hdr.cmapFirstIdx+hdr.cmapSize > 256) throw new Exception("invalid tga colormap type");
|
||||
ubyte readCMB () {
|
||||
if (colorMapBytes == 0) return 0;
|
||||
--colorMapBytes;
|
||||
return readByte;
|
||||
}
|
||||
cmap[] = Color.black;
|
||||
auto cmp = cmap.ptr;
|
||||
switch (colorEntryBytes) {
|
||||
case 2:
|
||||
foreach (immutable n; 0..hdr.cmapSize) {
|
||||
uint v = readCMB();
|
||||
v |= readCMB()<<8;
|
||||
cmp.b = cmap16.ptr[v&0x1f];
|
||||
cmp.g = cmap16.ptr[(v>>5)&0x1f];
|
||||
cmp.r = cmap16.ptr[(v>>10)&0x1f];
|
||||
++cmp;
|
||||
}
|
||||
break;
|
||||
case 3:
|
||||
foreach (immutable n; 0..hdr.cmapSize) {
|
||||
cmp.b = readCMB();
|
||||
cmp.g = readCMB();
|
||||
cmp.r = readCMB();
|
||||
++cmp;
|
||||
}
|
||||
break;
|
||||
case 4:
|
||||
foreach (immutable n; 0..hdr.cmapSize) {
|
||||
cmp.b = readCMB();
|
||||
cmp.g = readCMB();
|
||||
cmp.r = readCMB();
|
||||
cmp.a = readCMB();
|
||||
++cmp;
|
||||
}
|
||||
break;
|
||||
default: throw new Exception("invalid tga colormap type");
|
||||
}
|
||||
} else {
|
||||
// skip colormap
|
||||
fl.seek(colorMapBytes, Seek.Cur);
|
||||
}
|
||||
}
|
||||
|
||||
// now load the data
|
||||
fl.seek(hdr.idsize, Seek.Cur);
|
||||
if (hdr.cmapType != 0) loadColormap();
|
||||
|
||||
// we don't know if alpha is premultiplied yet
|
||||
bool hasAlpha = (bytesPerPixel == 4);
|
||||
bool validAlpha = hasAlpha;
|
||||
bool premult = false;
|
||||
|
||||
auto tcimg = new TrueColorImage(hdr.width, hdr.height);
|
||||
scope(failure) delete tcimg;
|
||||
|
||||
{
|
||||
// read image data
|
||||
immutable bool xflip = hdr.xflip, yflip = hdr.yflip;
|
||||
Color* pixdata = tcimg.imageData.colors.ptr;
|
||||
if (yflip) pixdata += (hdr.height-1)*hdr.width;
|
||||
foreach (immutable y; 0..hdr.height) {
|
||||
auto d = pixdata;
|
||||
if (xflip) d += hdr.width-1;
|
||||
foreach (immutable x; 0..hdr.width) {
|
||||
*d = readColor(&readByte);
|
||||
if (xflip) --d; else ++d;
|
||||
}
|
||||
if (yflip) pixdata -= hdr.width; else pixdata += hdr.width;
|
||||
}
|
||||
}
|
||||
|
||||
if (hasAlpha) {
|
||||
if (extfooter.extofs != 0) {
|
||||
Extension ext;
|
||||
fl.seek(extfooter.extofs);
|
||||
fl.readStruct(ext);
|
||||
// some idiotic writers set 494 instead 495, tolerate that
|
||||
if (ext.size < 494) throw new Exception("invalid tga extension record");
|
||||
if (ext.attrType == 4) {
|
||||
// premultiplied alpha
|
||||
foreach (ref Color clr; tcimg.imageData.colors) {
|
||||
if (clr.a != 0) {
|
||||
clr.r = cast(ubyte)(clr.r*255/clr.a);
|
||||
clr.g = cast(ubyte)(clr.g*255/clr.a);
|
||||
clr.b = cast(ubyte)(clr.b*255/clr.a);
|
||||
}
|
||||
}
|
||||
} else if (ext.attrType != 3) {
|
||||
validAlpha = false;
|
||||
}
|
||||
} else {
|
||||
// some writers sets all alphas to zero, check for that
|
||||
validAlpha = false;
|
||||
foreach (ref Color clr; tcimg.imageData.colors) if (clr.a != 0) { validAlpha = true; break; }
|
||||
}
|
||||
if (!validAlpha) foreach (ref Color clr; tcimg.imageData.colors) clr.a = 255;
|
||||
}
|
||||
return tcimg;
|
||||
}
|
||||
|
||||
|
||||
// ////////////////////////////////////////////////////////////////////////// //
|
||||
private:
|
||||
static if (!ArsdTargaHasIVVFS) {
|
||||
import core.stdc.stdio : SEEK_SET, SEEK_CUR, SEEK_END;
|
||||
|
||||
enum Seek : int {
|
||||
Set = SEEK_SET,
|
||||
Cur = SEEK_CUR,
|
||||
End = SEEK_END,
|
||||
}
|
||||
|
||||
|
||||
// ////////////////////////////////////////////////////////////////////////// //
|
||||
// augmentation checks
|
||||
// is this "low-level" stream that can be read?
|
||||
enum isLowLevelStreamR(T) = is(typeof((inout int=0) {
|
||||
auto t = T.init;
|
||||
ubyte[1] b;
|
||||
ptrdiff_t r = t.read(b.ptr, 1);
|
||||
}));
|
||||
|
||||
// is this "low-level" stream that can be written?
|
||||
enum isLowLevelStreamW(T) = is(typeof((inout int=0) {
|
||||
auto t = T.init;
|
||||
ubyte[1] b;
|
||||
ptrdiff_t w = t.write(b.ptr, 1);
|
||||
}));
|
||||
|
||||
|
||||
// is this "low-level" stream that can be seeked?
|
||||
enum isLowLevelStreamS(T) = is(typeof((inout int=0) {
|
||||
auto t = T.init;
|
||||
long p = t.lseek(0, 0);
|
||||
}));
|
||||
|
||||
|
||||
// ////////////////////////////////////////////////////////////////////////// //
|
||||
// augment low-level streams with `rawRead`
|
||||
T[] rawRead(ST, T) (auto ref ST st, T[] buf) if (isLowLevelStreamR!ST && !is(T == const) && !is(T == immutable)) {
|
||||
if (buf.length > 0) {
|
||||
auto res = st.read(buf.ptr, buf.length*T.sizeof);
|
||||
if (res == -1 || res%T.sizeof != 0) throw new Exception("read error");
|
||||
return buf[0..res/T.sizeof];
|
||||
} else {
|
||||
return buf[0..0];
|
||||
}
|
||||
}
|
||||
|
||||
// augment low-level streams with `rawWrite`
|
||||
void rawWrite(ST, T) (auto ref ST st, in T[] buf) if (isLowLevelStreamW!ST) {
|
||||
if (buf.length > 0) {
|
||||
auto res = st.write(buf.ptr, buf.length*T.sizeof);
|
||||
if (res == -1 || res%T.sizeof != 0) throw new Exception("write error");
|
||||
}
|
||||
}
|
||||
|
||||
// read exact size or throw error
|
||||
T[] rawReadExact(ST, T) (auto ref ST st, T[] buf) if (isReadableStream!ST && !is(T == const) && !is(T == immutable)) {
|
||||
if (buf.length == 0) return buf;
|
||||
auto left = buf.length*T.sizeof;
|
||||
auto dp = cast(ubyte*)buf.ptr;
|
||||
while (left > 0) {
|
||||
auto res = st.rawRead(cast(void[])(dp[0..left]));
|
||||
if (res.length == 0) throw new Exception("read error");
|
||||
dp += res.length;
|
||||
left -= res.length;
|
||||
}
|
||||
return buf;
|
||||
}
|
||||
|
||||
// write exact size or throw error (just for convenience)
|
||||
void rawWriteExact(ST, T) (auto ref ST st, in T[] buf) if (isWriteableStream!ST) { st.rawWrite(buf); }
|
||||
|
||||
// if stream doesn't have `.size`, but can be seeked, emulate it
|
||||
long size(ST) (auto ref ST st) if (!isSeekableStream!ST && !streamHasSize!ST) {
|
||||
auto opos = st.tell;
|
||||
st.seek(0, Seek.End);
|
||||
auto res = st.tell;
|
||||
st.seek(opos);
|
||||
return res;
|
||||
}
|
||||
|
||||
|
||||
// ////////////////////////////////////////////////////////////////////////// //
|
||||
// check if a given stream supports `eof`
|
||||
enum streamHasEof(T) = is(typeof((inout int=0) {
|
||||
auto t = T.init;
|
||||
bool n = t.eof;
|
||||
}));
|
||||
|
||||
// check if a given stream supports `seek`
|
||||
enum streamHasSeek(T) = is(typeof((inout int=0) {
|
||||
import core.stdc.stdio : SEEK_END;
|
||||
auto t = T.init;
|
||||
t.seek(0);
|
||||
t.seek(0, SEEK_END);
|
||||
}));
|
||||
|
||||
// check if a given stream supports `tell`
|
||||
enum streamHasTell(T) = is(typeof((inout int=0) {
|
||||
auto t = T.init;
|
||||
long pos = t.tell;
|
||||
}));
|
||||
|
||||
// check if a given stream supports `size`
|
||||
enum streamHasSize(T) = is(typeof((inout int=0) {
|
||||
auto t = T.init;
|
||||
long pos = t.size;
|
||||
}));
|
||||
|
||||
// check if a given stream supports `rawRead()`.
|
||||
// it's enough to support `void[] rawRead (void[] buf)`
|
||||
enum isReadableStream(T) = is(typeof((inout int=0) {
|
||||
auto t = T.init;
|
||||
ubyte[1] b;
|
||||
auto v = cast(void[])b;
|
||||
t.rawRead(v);
|
||||
}));
|
||||
|
||||
// check if a given stream supports `rawWrite()`.
|
||||
// it's enough to support `inout(void)[] rawWrite (inout(void)[] buf)`
|
||||
enum isWriteableStream(T) = is(typeof((inout int=0) {
|
||||
auto t = T.init;
|
||||
ubyte[1] b;
|
||||
t.rawWrite(cast(void[])b);
|
||||
}));
|
||||
|
||||
// check if a given stream supports `.seek(ofs, [whence])`, and `.tell`
|
||||
enum isSeekableStream(T) = (streamHasSeek!T && streamHasTell!T);
|
||||
|
||||
// check if we can get size of a given stream.
|
||||
// this can be done either with `.size`, or with `.seek` and `.tell`
|
||||
enum isSizedStream(T) = (streamHasSize!T || isSeekableStream!T);
|
||||
|
||||
// ////////////////////////////////////////////////////////////////////////// //
|
||||
private enum isGoodEndianness(string s) = (s == "LE" || s == "le" || s == "BE" || s == "be");
|
||||
|
||||
private template isLittleEndianness(string s) if (isGoodEndianness!s) {
|
||||
enum isLittleEndianness = (s == "LE" || s == "le");
|
||||
}
|
||||
|
||||
private template isBigEndianness(string s) if (isGoodEndianness!s) {
|
||||
enum isLittleEndianness = (s == "BE" || s == "be");
|
||||
}
|
||||
|
||||
private template isSystemEndianness(string s) if (isGoodEndianness!s) {
|
||||
version(LittleEndian) {
|
||||
enum isSystemEndianness = isLittleEndianness!s;
|
||||
} else {
|
||||
enum isSystemEndianness = isBigEndianness!s;
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
// ////////////////////////////////////////////////////////////////////////// //
|
||||
// write integer value of the given type, with the given endianness (default: little-endian)
|
||||
// usage: st.writeNum!ubyte(10)
|
||||
void writeNum(T, string es="LE", ST) (auto ref ST st, T n) if (isGoodEndianness!es && isWriteableStream!ST && __traits(isIntegral, T)) {
|
||||
static assert(T.sizeof <= 8); // just in case
|
||||
static if (isSystemEndianness!es) {
|
||||
st.rawWriteExact((&n)[0..1]);
|
||||
} else {
|
||||
ubyte[T.sizeof] b = void;
|
||||
version(LittleEndian) {
|
||||
// convert to big-endian
|
||||
foreach_reverse (ref x; b) { x = n&0xff; n >>= 8; }
|
||||
} else {
|
||||
// convert to little-endian
|
||||
foreach (ref x; b) { x = n&0xff; n >>= 8; }
|
||||
}
|
||||
st.rawWriteExact(b[]);
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
// read integer value of the given type, with the given endianness (default: little-endian)
|
||||
// usage: auto v = st.readNum!ubyte
|
||||
T readNum(T, string es="LE", ST) (auto ref ST st) if (isGoodEndianness!es && isReadableStream!ST && __traits(isIntegral, T)) {
|
||||
static assert(T.sizeof <= 8); // just in case
|
||||
static if (isSystemEndianness!es) {
|
||||
T v = void;
|
||||
st.rawReadExact((&v)[0..1]);
|
||||
return v;
|
||||
} else {
|
||||
ubyte[T.sizeof] b = void;
|
||||
st.rawReadExact(b[]);
|
||||
T v = 0;
|
||||
version(LittleEndian) {
|
||||
// convert from big-endian
|
||||
foreach (ubyte x; b) { v <<= 8; v |= x; }
|
||||
} else {
|
||||
// conver from little-endian
|
||||
foreach_reverse (ubyte x; b) { v <<= 8; v |= x; }
|
||||
}
|
||||
return v;
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
private enum reverseBytesMixin = "
|
||||
foreach (idx; 0..b.length/2) {
|
||||
ubyte t = b[idx];
|
||||
b[idx] = b[$-idx-1];
|
||||
b[$-idx-1] = t;
|
||||
}
|
||||
";
|
||||
|
||||
|
||||
// write floating value of the given type, with the given endianness (default: little-endian)
|
||||
// usage: st.writeNum!float(10)
|
||||
void writeNum(T, string es="LE", ST) (auto ref ST st, T n) if (isGoodEndianness!es && isWriteableStream!ST && __traits(isFloating, T)) {
|
||||
static assert(T.sizeof <= 8);
|
||||
static if (isSystemEndianness!es) {
|
||||
st.rawWriteExact((&n)[0..1]);
|
||||
} else {
|
||||
import core.stdc.string : memcpy;
|
||||
ubyte[T.sizeof] b = void;
|
||||
memcpy(b.ptr, &v, T.sizeof);
|
||||
mixin(reverseBytesMixin);
|
||||
st.rawWriteExact(b[]);
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
// read floating value of the given type, with the given endianness (default: little-endian)
|
||||
// usage: auto v = st.readNum!float
|
||||
T readNum(T, string es="LE", ST) (auto ref ST st) if (isGoodEndianness!es && isReadableStream!ST && __traits(isFloating, T)) {
|
||||
static assert(T.sizeof <= 8);
|
||||
T v = void;
|
||||
static if (isSystemEndianness!es) {
|
||||
st.rawReadExact((&v)[0..1]);
|
||||
} else {
|
||||
import core.stdc.string : memcpy;
|
||||
ubyte[T.sizeof] b = void;
|
||||
st.rawReadExact(b[]);
|
||||
mixin(reverseBytesMixin);
|
||||
memcpy(&v, b.ptr, T.sizeof);
|
||||
}
|
||||
return v;
|
||||
}
|
||||
|
||||
|
||||
// ////////////////////////////////////////////////////////////////////////// //
|
||||
void readStruct(string es="LE", SS, ST) (auto ref ST fl, ref SS st)
|
||||
if (is(SS == struct) && isGoodEndianness!es && isReadableStream!ST)
|
||||
{
|
||||
void unserData(T) (ref T v) {
|
||||
import std.traits : Unqual;
|
||||
alias UT = Unqual!T;
|
||||
static if (is(T : V[], V)) {
|
||||
// array
|
||||
static if (__traits(isStaticArray, T)) {
|
||||
foreach (ref it; v) unserData(it);
|
||||
} else static if (is(UT == char)) {
|
||||
// special case: dynamic `char[]` array will be loaded as asciiz string
|
||||
char c;
|
||||
for (;;) {
|
||||
if (fl.rawRead((&c)[0..1]).length == 0) break; // don't require trailing zero on eof
|
||||
if (c == 0) break;
|
||||
v ~= c;
|
||||
}
|
||||
} else {
|
||||
assert(0, "cannot load dynamic arrays yet");
|
||||
}
|
||||
} else static if (is(T : V[K], K, V)) {
|
||||
assert(0, "cannot load associative arrays yet");
|
||||
} else static if (__traits(isIntegral, UT) || __traits(isFloating, UT)) {
|
||||
// this takes care of `*char` and `bool` too
|
||||
v = cast(UT)fl.readNum!(UT, es);
|
||||
} else static if (is(T == struct)) {
|
||||
// struct
|
||||
import std.traits : FieldNameTuple, hasUDA;
|
||||
foreach (string fldname; FieldNameTuple!T) {
|
||||
unserData(__traits(getMember, v, fldname));
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
unserData(st);
|
||||
}
|
||||
}
|
Loading…
Reference in New Issue