- targa_header.pixel_size = fgetc(fin);
- targa_header.attributes = fgetc(fin);
-
- if (targa_header.image_type!=2
- && targa_header.image_type!=10)
- Host_Error ("LoadTGA: Only type 2 and 10 targa RGB images supported\n");
-
- if (targa_header.colormap_type !=0
- || (targa_header.pixel_size!=32 && targa_header.pixel_size!=24))
- Host_Error ("LoadTGA: Only 32 or 24 bit images supported (no colormaps)\n");
-
- columns = targa_header.width;
- rows = targa_header.height;
- numPixels = columns * rows;
-
- image_rgba = qmalloc(numPixels*4);
-
- if (targa_header.id_length != 0)
- fseek(fin, targa_header.id_length, SEEK_CUR); // skip TARGA image comment
-
- if (targa_header.image_type==2) { // Uncompressed, RGB images
- for(row=rows-1; row>=0; row--) {
- pixbuf = image_rgba + row*columns*4;
- for(column=0; column<columns; column++) {
- unsigned char red = 0,green = 0,blue = 0,alphabyte = 0;
- switch (targa_header.pixel_size) {
- case 24:
-
- blue = getc(fin);
- green = getc(fin);
- red = getc(fin);
- *pixbuf++ = red;
- *pixbuf++ = green;
- *pixbuf++ = blue;
- *pixbuf++ = 255;
- break;
- case 32:
- blue = getc(fin);
- green = getc(fin);
- red = getc(fin);
- alphabyte = getc(fin);
+ }
+ if ((matchwidth && image_width != matchwidth) || (matchheight && image_height != matchheight))
+ return NULL;
+ targa_header.pixel_size = f[16];
+ targa_header.attributes = f[17];
+
+ // advance to end of header
+ fin = f + 18;
+
+ // skip TARGA image comment (usually 0 bytes)
+ fin += targa_header.id_length;
+
+ // read/skip the colormap if present (note: according to the TARGA spec it
+ // can be present even on truecolor or greyscale images, just not used by
+ // the image data)
+ if (targa_header.colormap_type)
+ {
+ if (targa_header.colormap_length > 256)
+ {
+ Con_Print("LoadTGA: only up to 256 colormap_length supported\n");
+ PrintTargaHeader(&targa_header);
+ return NULL;
+ }
+ if (targa_header.colormap_index)
+ {
+ Con_Print("LoadTGA: colormap_index not supported\n");
+ PrintTargaHeader(&targa_header);
+ return NULL;
+ }
+ if (targa_header.colormap_size == 24)
+ {
+ for (x = 0;x < targa_header.colormap_length;x++)
+ {
+ palette[x*4+2] = *fin++;
+ palette[x*4+1] = *fin++;
+ palette[x*4+0] = *fin++;
+ palette[x*4+3] = 255;
+ }
+ }
+ else if (targa_header.colormap_size == 32)
+ {
+ for (x = 0;x < targa_header.colormap_length;x++)
+ {
+ palette[x*4+2] = *fin++;
+ palette[x*4+1] = *fin++;
+ palette[x*4+0] = *fin++;
+ palette[x*4+3] = *fin++;
+ }
+ }
+ else
+ {
+ Con_Print("LoadTGA: Only 32 and 24 bit colormap_size supported\n");
+ PrintTargaHeader(&targa_header);
+ return NULL;
+ }
+ }
+
+ // check our pixel_size restrictions according to image_type
+ switch (targa_header.image_type & ~8)
+ {
+ case 2:
+ if (targa_header.pixel_size != 24 && targa_header.pixel_size != 32)
+ {
+ Con_Print("LoadTGA: only 24bit and 32bit pixel sizes supported for type 2 and type 10 images\n");
+ PrintTargaHeader(&targa_header);
+ return NULL;
+ }
+ break;
+ case 3:
+ // set up a palette to make the loader easier
+ for (x = 0;x < 256;x++)
+ {
+ palette[x*4+2] = x;
+ palette[x*4+1] = x;
+ palette[x*4+0] = x;
+ palette[x*4+3] = 255;
+ }
+ // fall through to colormap case
+ case 1:
+ if (targa_header.pixel_size != 8)
+ {
+ Con_Print("LoadTGA: only 8bit pixel size for type 1, 3, 9, and 11 images supported\n");
+ PrintTargaHeader(&targa_header);
+ return NULL;
+ }
+ break;
+ default:
+ Con_Printf("LoadTGA: Only type 1, 2, 3, 9, 10, and 11 targa RGB images supported, image_type = %i\n", targa_header.image_type);
+ PrintTargaHeader(&targa_header);
+ return NULL;
+ }
+
+ if (targa_header.attributes & 0x10)
+ {
+ Con_Print("LoadTGA: origin must be in top left or bottom left, top right and bottom right are not supported\n");
+ return NULL;
+ }
+
+ // number of attribute bits per pixel, we only support 0 or 8
+ alphabits = targa_header.attributes & 0x0F;
+ if (alphabits != 8 && alphabits != 0)
+ {
+ Con_Print("LoadTGA: only 0 or 8 attribute (alpha) bits supported\n");
+ return NULL;
+ }
+
+ image_rgba = (unsigned char *)Mem_Alloc(tempmempool, image_width * image_height * 4);
+ if (!image_rgba)
+ {
+ Con_Printf("LoadTGA: not enough memory for %i by %i image\n", image_width, image_height);
+ return NULL;
+ }
+
+ // If bit 5 of attributes isn't set, the image has been stored from bottom to top
+ if ((targa_header.attributes & 0x20) == 0)
+ {
+ pixbuf = image_rgba + (image_height - 1)*image_width*4;
+ row_inc = -image_width*4*2;
+ }
+ else
+ {
+ pixbuf = image_rgba;
+ row_inc = 0;
+ }
+
+ x = 0;
+ y = 0;
+ red = green = blue = alpha = 255;
+ pix_inc = 1;
+ if ((targa_header.image_type & ~8) == 2)
+ pix_inc = targa_header.pixel_size / 8;
+ switch (targa_header.image_type)
+ {
+ case 1: // colormapped, uncompressed
+ case 3: // greyscale, uncompressed
+ if (fin + image_width * image_height * pix_inc > enddata)
+ break;
+ for (y = 0;y < image_height;y++, pixbuf += row_inc)
+ {
+ for (x = 0;x < image_width;x++)
+ {
+ p = palette + *fin++ * 4;
+ *pixbuf++ = p[0];
+ *pixbuf++ = p[1];
+ *pixbuf++ = p[2];
+ *pixbuf++ = p[3];
+ }
+ }
+ break;
+ case 2:
+ // BGR or BGRA, uncompressed
+ if (fin + image_width * image_height * pix_inc > enddata)
+ break;
+ if (targa_header.pixel_size == 32 && alphabits)
+ {
+ for (y = 0;y < image_height;y++, pixbuf += row_inc)
+ {
+ for (x = 0;x < image_width;x++, fin += pix_inc)
+ {
+ *pixbuf++ = fin[2];
+ *pixbuf++ = fin[1];
+ *pixbuf++ = fin[0];
+ *pixbuf++ = fin[3];
+ }
+ }
+ }
+ else
+ {
+ for (y = 0;y < image_height;y++, pixbuf += row_inc)
+ {
+ for (x = 0;x < image_width;x++, fin += pix_inc)
+ {
+ *pixbuf++ = fin[2];
+ *pixbuf++ = fin[1];
+ *pixbuf++ = fin[0];
+ *pixbuf++ = 255;
+ }
+ }
+ }
+ break;
+ case 9: // colormapped, RLE
+ case 11: // greyscale, RLE
+ for (y = 0;y < image_height;y++, pixbuf += row_inc)
+ {
+ for (x = 0;x < image_width;)
+ {
+ if (fin >= enddata)
+ break; // error - truncated file
+ runlen = *fin++;
+ if (runlen & 0x80)
+ {
+ // RLE - all pixels the same color
+ runlen += 1 - 0x80;
+ if (fin + pix_inc > enddata)
+ break; // error - truncated file
+ if (x + runlen > image_width)
+ break; // error - line exceeds width
+ p = palette + *fin++ * 4;
+ red = p[0];
+ green = p[1];
+ blue = p[2];
+ alpha = p[3];
+ for (;runlen--;x++)
+ {
+ *pixbuf++ = red;
+ *pixbuf++ = green;
+ *pixbuf++ = blue;
+ *pixbuf++ = alpha;
+ }
+ }
+ else
+ {
+ // uncompressed - all pixels different color
+ runlen++;
+ if (fin + pix_inc * runlen > enddata)
+ break; // error - truncated file
+ if (x + runlen > image_width)
+ break; // error - line exceeds width
+ for (;runlen--;x++)
+ {
+ p = palette + *fin++ * 4;
+ *pixbuf++ = p[0];
+ *pixbuf++ = p[1];
+ *pixbuf++ = p[2];
+ *pixbuf++ = p[3];
+ }
+ }
+ }
+ }
+ break;
+ case 10:
+ // BGR or BGRA, RLE
+ if (targa_header.pixel_size == 32 && alphabits)
+ {
+ for (y = 0;y < image_height;y++, pixbuf += row_inc)
+ {
+ for (x = 0;x < image_width;)
+ {
+ if (fin >= enddata)
+ break; // error - truncated file
+ runlen = *fin++;
+ if (runlen & 0x80)
+ {
+ // RLE - all pixels the same color
+ runlen += 1 - 0x80;
+ if (fin + pix_inc > enddata)
+ break; // error - truncated file
+ if (x + runlen > image_width)
+ break; // error - line exceeds width
+ red = fin[2];
+ green = fin[1];
+ blue = fin[0];
+ alpha = fin[3];
+ fin += pix_inc;
+ for (;runlen--;x++)
+ {