/* Copyright (c) 2013 yvt This file is part of OpenSpades. OpenSpades is free software: you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation, either version 3 of the License, or (at your option) any later version. OpenSpades is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with OpenSpades. If not, see . */ #include "IBitmapCodec.h" #include "Bitmap.h" #include "Debug.h" #include "Exception.h" #include "IStream.h" #include typedef struct _TargaHeader { unsigned char id_length, colormap_type, image_type; unsigned short colormap_index, colormap_length; unsigned char colormap_size; unsigned short x_origin, y_origin, width, height; unsigned char pixel_size, attributes; } TargaHeader; // TODO: endian conversion #define LittleShort(a) (a) typedef unsigned char byte; namespace spades { class TargaReader: public IBitmapCodec { public: virtual bool CanLoad(){ return true; } virtual bool CanSave(){ return false; } virtual bool CheckExtension(const std::string& filename){ return EndsWith(filename, ".tga"); } virtual std::string GetName(){ static std::string name("Quake 3 Derived Targa Importer"); return name; } virtual Bitmap *Load(IStream *str){ SPADES_MARK_FUNCTION(); unsigned columns, rows, numPixels; byte *pixbuf; int row, column; byte *buf_p; byte *end; union { byte *b; void *v; } buffer; TargaHeader targa_header; byte *targa_rgba; int length; std::vector bufferStorage; // // load the file // length = (int)(str->GetLength() - str->GetPosition()); if(length < 18) { SPRaise("LoadTGA: header too short"); } bufferStorage.resize(length); str->Read(bufferStorage.data(), (size_t)length); buffer.v = bufferStorage.data(); buf_p = buffer.b; end = buffer.b + length; targa_header.id_length = buf_p[0]; targa_header.colormap_type = buf_p[1]; targa_header.image_type = buf_p[2]; memcpy(&targa_header.colormap_index, &buf_p[3], 2); memcpy(&targa_header.colormap_length, &buf_p[5], 2); targa_header.colormap_size = buf_p[7]; memcpy(&targa_header.x_origin, &buf_p[8], 2); memcpy(&targa_header.y_origin, &buf_p[10], 2); memcpy(&targa_header.width, &buf_p[12], 2); memcpy(&targa_header.height, &buf_p[14], 2); targa_header.pixel_size = buf_p[16]; targa_header.attributes = buf_p[17]; targa_header.colormap_index = LittleShort(targa_header.colormap_index); targa_header.colormap_length = LittleShort(targa_header.colormap_length); targa_header.x_origin = LittleShort(targa_header.x_origin); targa_header.y_origin = LittleShort(targa_header.y_origin); targa_header.width = LittleShort(targa_header.width); targa_header.height = LittleShort(targa_header.height); buf_p += 18; if (targa_header.image_type!=2 && targa_header.image_type!=10 && targa_header.image_type != 3 ) { SPRaise ( "LoadTGA: Only type 2 (RGB), 3 (gray), and 10 (RGB) TGA images supported"); } if ( targa_header.colormap_type != 0 ) { SPRaise("LoadTGA: colormaps not supported" ); } if ( ( targa_header.pixel_size != 32 && targa_header.pixel_size != 24 ) && targa_header.image_type != 3 ) { SPRaise( "LoadTGA: Only 32 or 24 bit images supported (no colormaps)"); } columns = targa_header.width; rows = targa_header.height; numPixels = columns * rows * 4; if(!columns || !rows || numPixels > 0x7FFFFFFF || numPixels / columns / 4 != rows) { SPRaise ( "LoadTGA: invalid image size"); } std::vector outPixels; outPixels.resize(numPixels); targa_rgba = outPixels.data(); if (targa_header.id_length != 0) { if (buf_p + targa_header.id_length > end) SPRaise( "LoadTGA: header too short" ); buf_p += targa_header.id_length; // skip TARGA image comment } if ( targa_header.image_type==2 || targa_header.image_type == 3 ) { if(buf_p + columns*rows*targa_header.pixel_size/8 > end) { SPRaise( "LoadTGA: file truncated"); } // Uncompressed RGB or gray scale image for(row=rows-1; row>=0; row--) { pixbuf = targa_rgba + row*columns*4; for(column=0; column=0; row--) { pixbuf = targa_rgba + row*columns*4; for(column=0; column end) SPRaise ( "LoadTGA: file truncated"); packetHeader= *buf_p++; packetSize = 1 + (packetHeader & 0x7f); if (packetHeader & 0x80) { // run-length packet if(buf_p + targa_header.pixel_size/8 > end) SPRaise( "LoadTGA: file truncated" ); switch (targa_header.pixel_size) { case 24: blue = *buf_p++; green = *buf_p++; red = *buf_p++; alphabyte = 255; break; case 32: blue = *buf_p++; green = *buf_p++; red = *buf_p++; alphabyte = *buf_p++; break; default: SPRaise( "LoadTGA: illegal pixel_size '%d' ", targa_header.pixel_size ); break; } for(j=0;j0) row--; else goto breakOut; pixbuf = targa_rgba + row*columns*4; } } } else { // non run-length packet if(buf_p + targa_header.pixel_size/8*packetSize > end) SPRaise("LoadTGA: file truncated"); for(j=0;j0) row--; else goto breakOut; pixbuf = targa_rgba + row*columns*4; } } } } breakOut:; } } #if 0 // TTimo: this is the chunk of code to ensure a behavior that meets TGA specs // bit 5 set => top-down if (targa_header.attributes & 0x20) { unsigned char *flip = (unsigned char*)malloc (columns*4); unsigned char *src, *dst; for (row = 0; row < rows/2; row++) { src = targa_rgba + row * 4 * columns; dst = targa_rgba + (rows - row - 1) * 4 * columns; memcpy (flip, src, columns*4); memcpy (src, dst, columns*4); memcpy (dst, flip, columns*4); } free (flip); } #endif // instead we just print a warning if (targa_header.attributes & 0x20) { //SPRaise( PRINT_WARNING, "WARNING: '%s' TGA file header declares top-down image, ignoring\n", name); } Bitmap *bmp = new Bitmap(columns, rows); byte *out = (byte *)bmp->GetPixels(); for(int i = 0; i