Blame img.ldr.tga.inc.c

ea5919
ea5919
ea5919
#pragma pack(push,1)
ea5919
typedef struct {
ea5919
  unsigned char  idLength;
ea5919
  unsigned char  colormapType;
ea5919
  unsigned char  imageType;
a6eedf
  unsigned char  colormapIndex[2];
a6eedf
  unsigned char  colormapLength[2];
ea5919
  unsigned char  colormapSize;
a6eedf
  unsigned char  xOrigin[2];
a6eedf
  unsigned char  yOrigin[2];
a6eedf
  unsigned char  width[2];
a6eedf
  unsigned char  height[2];
ea5919
  unsigned char  pixelSize;
ea5919
  unsigned char  attributes;
ea5919
} TgaHeader;
ea5919
#pragma pack(pop)
ea5919
ea5919
a6eedf
unsigned short getLEU16(unsigned char *c)
a6eedf
    { return c[0] | ((unsigned short)c[1] << 8); }
a6eedf
void setLEU16(unsigned char *c, unsigned short i)
a6eedf
    { c[0] = i; c[1] = i >> 8; }
a6eedf
a6eedf
ea5919
int imageLoadTga(const char *path, int *outWidth, int *outHeight, unsigned char **pixels) {
ea5919
  *outWidth = *outHeight = 0; *pixels = NULL;
ea5919
a6eedf
  FILE *f = fopen(path, "rb");
a6eedf
    if (!f) {
ea5919
    fprintf(stderr, "Cannot open for read: %s\n", path);
ea5919
    fflush(stderr);
ea5919
    return FALSE;
ea5919
  }
ea5919
a6eedf
  TgaHeader header = {};
a6eedf
  fread(&header, sizeof(header), 1, f); // todo: endianess
ea5919
a6eedf
  //printf("%d %d %d %d %d\n", header.imageType, header.width, header.height, header.colormapType, header.pixelSize);
ea5919
a6eedf
  if ( header.imageType != 2
a6eedf
    || !header.width
a6eedf
    || !header.height
a6eedf
    || header.colormapType
a6eedf
    || (header.pixelSize != 24 && header.pixelSize != 32)
ea5919
  ) {
ea5919
    fprintf(stderr, "Unsupported format. Only uncompressed 24 or 32 bits TGA are supported: %s\n", path);
ea5919
    fflush(stderr);
ea5919
    return FALSE;
a6eedf
  }
ea5919
ea5919
  fseek(f, header.idLength, SEEK_CUR);
ea5919
a6eedf
  int w = *outWidth = getLEU16(header.width);
a6eedf
  int h = *outHeight = getLEU16(header.height);
a6eedf
  int rowSize = w*4;
a6eedf
  int size = h*rowSize;
ea5919
  *pixels = (unsigned char*)calloc(1, size);
ea5919
  unsigned char *row = *pixels + size;
a6eedf
  for(unsigned short r = h; r; --r, row -= rowSize) {
ea5919
    for(unsigned char *c = row - rowSize; c < row; c += 4) {
ea5919
      c[2] = fgetc(f);
ea5919
      c[1] = fgetc(f);
ea5919
      c[0] = fgetc(f);
ea5919
      c[3] = header.pixelSize == 32 ? fgetc(f) : 255;
ea5919
    }
ea5919
  }
a6eedf
  fclose(f);
ea5919
a6eedf
  return TRUE;
ea5919
}
ea5919
ea5919
ea5919
int imageSaveTga(const char *path, int width, int height, const unsigned char *pixels) {
a6eedf
  if (width <= 0 || height <= 0 || width > 0xffff || height > 0xffff || !pixels) return FALSE;
ea5919
a6eedf
  FILE *f = fopen(path, "wb");
a6eedf
  if (!f) {
ea5919
    fprintf(stderr, "Cannot open file for write: %s\n", path);
ea5919
    fflush(stderr);
ea5919
    return FALSE;
ea5919
  }
ea5919
a6eedf
  TgaHeader header = {};
a6eedf
  header.imageType = 2;
a6eedf
  setLEU16(header.width, width);
a6eedf
  setLEU16(header.height, height);
a6eedf
  header.pixelSize = 32;
a6eedf
  fwrite(&header, sizeof(header), 1, f); // todo: endianess
ea5919
ea5919
  int rowSize = width*4;
ea5919
  int size = height*rowSize;
ea5919
  const unsigned char *row = pixels + size;
a6eedf
  for(unsigned short r = height; r; --r, row -= rowSize) {
ea5919
    for(const unsigned char *c = row - rowSize; c < row; c += 4) {
ea5919
      fputc(c[2], f);
ea5919
      fputc(c[1], f);
ea5919
      fputc(c[0], f);
ea5919
      fputc(c[3], f);
ea5919
    }
ea5919
  }
a6eedf
  fclose(f);
ea5919
a6eedf
  return TRUE;
ea5919
}
ea5919