/* rgb.c */ /* Vis5D version 4.3 */ /* Vis5D system for visualizing five dimensional gridded data sets Copyright (C) 1990 - 1997 Bill Hibbard, Johan Kellum, Brian Paul, Dave Santek, and Andre Battaiola. This program 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 1, or (at your option) any later version. This program 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 this program; if not, write to the Free Software Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. */ /* * Code to read SGI .rgb image files. Borrowed from the SGI OpenGL TK * source code. */ #include #include #include #include "rgb.h" #ifndef SEEK_SET # define SEEK_SET 0 #endif typedef struct _rawImageRec { unsigned short imagic; unsigned short type; unsigned short dim; unsigned short sizeX, sizeY, sizeZ; unsigned long min, max; unsigned long wasteBytes; char name[80]; unsigned long colorMap; FILE *file; unsigned char *tmp, *tmpR, *tmpG, *tmpB; unsigned long rleEnd; unsigned int *rowStart; int *rowSize; } rawImageRec; static void ConvertShort(unsigned short *array, long length) { unsigned long b1, b2; unsigned char *ptr; ptr = (unsigned char *)array; while (length--) { b1 = *ptr++; b2 = *ptr++; *array++ = (b1 << 8) | (b2); } } static void ConvertLong( unsigned int *array, long length) { unsigned long b1, b2, b3, b4; unsigned char *ptr; ptr = (unsigned char *)array; while (length--) { b1 = *ptr++; b2 = *ptr++; b3 = *ptr++; b4 = *ptr++; *array++ = (b1 << 24) | (b2 << 16) | (b3 << 8) | (b4); } } static rawImageRec *RawImageOpen(char *fileName) { union { int testWord; char testByte[4]; } endianTest; rawImageRec *raw; int swapFlag; int x; endianTest.testWord = 1; if (endianTest.testByte[0] == 1) { swapFlag = 1; } else { swapFlag = 0; } raw = (rawImageRec *)malloc(sizeof(rawImageRec)); if (raw == NULL) { fprintf(stderr, "Out of memory!\n"); return NULL; } if ((raw->file = fopen(fileName, "rb")) == NULL) { perror(fileName); return NULL; } fread(raw, 1, 12, raw->file); if (swapFlag) { ConvertShort(&raw->imagic, 6); } raw->tmp = (unsigned char *)malloc(raw->sizeX*256); raw->tmpR = (unsigned char *)malloc(raw->sizeX*256); raw->tmpG = (unsigned char *)malloc(raw->sizeX*256); raw->tmpB = (unsigned char *)malloc(raw->sizeX*256); if (raw->tmp == NULL || raw->tmpR == NULL || raw->tmpG == NULL || raw->tmpB == NULL) { fprintf(stderr, "Out of memory!\n"); return NULL; } if ((raw->type & 0xFF00) == 0x0100) { x = raw->sizeY * raw->sizeZ * sizeof(unsigned int); raw->rowStart = (unsigned int *)malloc(x); raw->rowSize = (int *)malloc(x); if (raw->rowStart == NULL || raw->rowSize == NULL) { fprintf(stderr, "Out of memory!\n"); return NULL; } raw->rleEnd = 512 + (2 * x); fseek(raw->file, 512, SEEK_SET); fread(raw->rowStart, 1, x, raw->file); fread(raw->rowSize, 1, x, raw->file); if (swapFlag) { ConvertLong(raw->rowStart, x/sizeof(unsigned int)); ConvertLong((unsigned int *)raw->rowSize, x/sizeof(int)); } } return raw; } static void RawImageClose(rawImageRec *raw) { fclose(raw->file); free(raw->tmp); free(raw->tmpR); free(raw->tmpG); free(raw->tmpB); free(raw); } static void RawImageGetRow(rawImageRec *raw, unsigned char *buf, int y, int z) { unsigned char *iPtr, *oPtr, pixel; int count; if ((raw->type & 0xFF00) == 0x0100) { fseek(raw->file, raw->rowStart[y+z*raw->sizeY], SEEK_SET); fread(raw->tmp, 1, (unsigned int)raw->rowSize[y+z*raw->sizeY], raw->file); iPtr = raw->tmp; oPtr = buf; while (1) { pixel = *iPtr++; count = (int)(pixel & 0x7F); if (!count) { return; } if (pixel & 0x80) { while (count--) { *oPtr++ = *iPtr++; } } else { pixel = *iPtr++; while (count--) { *oPtr++ = pixel; } } } } else { fseek(raw->file, 512+(y*raw->sizeX)+(z*raw->sizeX*raw->sizeY), SEEK_SET); fread(buf, 1, raw->sizeX, raw->file); } } static void RawImageGetData(rawImageRec *raw, RGB_IMAGE *final) { unsigned char *ptr; int i, j; final->data = (unsigned char *)malloc((raw->sizeX+1)*(raw->sizeY+1)*4); if (final->data == NULL) { fprintf(stderr, "Out of memory!\n"); return; } ptr = final->data; for (i = 0; i < (int)(raw->sizeY); i++) { int ii = raw->sizeY - i - 1; /* flip vertically */ RawImageGetRow(raw, raw->tmpR, ii, 0); RawImageGetRow(raw, raw->tmpG, ii, 1); RawImageGetRow(raw, raw->tmpB, ii, 2); for (j = 0; j < (int)(raw->sizeX); j++) { #ifdef SGI_GL /* IRIS GL uses BGR order for textures */ /* WLH 17 Oct 97 *ptr++ = *(raw->tmpR + j); *ptr++ = *(raw->tmpG + j); *ptr++ = *(raw->tmpB + j); */ *ptr++ = 0; *ptr++ = *(raw->tmpB + j); *ptr++ = *(raw->tmpG + j); *ptr++ = *(raw->tmpR + j); #else /* OpenGL, Mesa, etc use RGB order for textures */ /* WLH 17 Oct 97 *ptr++ = *(raw->tmpB + j); *ptr++ = *(raw->tmpG + j); *ptr++ = *(raw->tmpR + j); */ *ptr++ = *(raw->tmpR + j); *ptr++ = *(raw->tmpG + j); *ptr++ = *(raw->tmpB + j); *ptr++ = 0; #endif /* WLH 17 Oct 97 *ptr++ = 0; */ } } } RGB_IMAGE *ReadRGB(char *fileName) { rawImageRec *raw; RGB_IMAGE *final; raw = RawImageOpen(fileName); if (!raw) { return NULL; } final = (RGB_IMAGE *)malloc(sizeof(RGB_IMAGE)); if (final == NULL) { fprintf(stderr, "Out of memory!\n"); return NULL; } final->sizeX = raw->sizeX; final->sizeY = raw->sizeY; RawImageGetData(raw, final); RawImageClose(raw); return final; } void FreeRGB( RGB_IMAGE *image ) { if (image) { free(image->data); free(image); } } .