dxx-rebirth/d1x-rebirth/iff/iff.c
2013-10-19 17:03:11 +00:00

1008 lines
24 KiB
C

/*
THE COMPUTER CODE CONTAINED HEREIN IS THE SOLE PROPERTY OF PARALLAX
SOFTWARE CORPORATION ("PARALLAX"). PARALLAX, IN DISTRIBUTING THE CODE TO
END-USERS, AND SUBJECT TO ALL OF THE TERMS AND CONDITIONS HEREIN, GRANTS A
ROYALTY-FREE, PERPETUAL LICENSE TO SUCH END-USERS FOR USE BY SUCH END-USERS
IN USING, DISPLAYING, AND CREATING DERIVATIVE WORKS THEREOF, SO LONG AS
SUCH USE, DISPLAY OR CREATION IS FOR NON-COMMERCIAL, ROYALTY OR REVENUE
FREE PURPOSES. IN NO EVENT SHALL THE END-USER USE THE COMPUTER CODE
CONTAINED HEREIN FOR REVENUE-BEARING PURPOSES. THE END-USER UNDERSTANDS
AND AGREES TO THE TERMS HEREIN AND ACCEPTS THE SAME BY USE OF THIS FILE.
COPYRIGHT 1993-1998 PARALLAX SOFTWARE CORPORATION. ALL RIGHTS RESERVED.
*/
/*
*
* Routines for reading and writing IFF files
*
*/
#define COMPRESS 1 //do the RLE or not? (for debugging mostly)
#define WRITE_TINY 0 //should we write a TINY chunk?
#define MIN_COMPRESS_WIDTH 65 //don't compress if less than this wide
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include "u_mem.h"
#include "iff.h"
#include "dxxerror.h"
#include "physfsx.h"
//Internal constants and structures for this library
//Type values for bitmaps
#define TYPE_PBM 0
#define TYPE_ILBM 1
//Compression types
#define cmpNone 0
#define cmpByteRun1 1
//Masking types
#define mskNone 0
#define mskHasMask 1
#define mskHasTransparentColor 2
//Palette entry structure
typedef struct pal_entry {sbyte r,g,b;} pal_entry;
//structure of the header in the file
typedef struct iff_bitmap_header {
short w,h; //width and height of this bitmap
short x,y; //generally unused
short type; //see types above
short transparentcolor; //which color is transparent (if any)
short pagewidth,pageheight; //width & height of source screen
sbyte nplanes; //number of planes (8 for 256 color image)
sbyte masking,compression; //see constants above
sbyte xaspect,yaspect; //aspect ratio (usually 5/6)
pal_entry palette[256]; //the palette for this bitmap
ubyte *raw_data; //ptr to array of data
short row_size; //offset to next row
} iff_bitmap_header;
ubyte iff_transparent_color;
ubyte iff_has_transparency; // 0=no transparency, 1=iff_transparent_color is valid
#define MAKE_SIG(a,b,c,d) (((int32_t)(a)<<24)+((int32_t)(b)<<16)+((c)<<8)+(d))
#define form_sig MAKE_SIG('F','O','R','M')
#define ilbm_sig MAKE_SIG('I','L','B','M')
#define body_sig MAKE_SIG('B','O','D','Y')
#define pbm_sig MAKE_SIG('P','B','M',' ')
#define bmhd_sig MAKE_SIG('B','M','H','D')
#define anhd_sig MAKE_SIG('A','N','H','D')
#define cmap_sig MAKE_SIG('C','M','A','P')
#define tiny_sig MAKE_SIG('T','I','N','Y')
#define anim_sig MAKE_SIG('A','N','I','M')
#define dlta_sig MAKE_SIG('D','L','T','A')
int32_t get_sig(PHYSFS_file *f)
{
int s;
PHYSFS_readSBE32(f, &s);
return s;
}
#define put_sig(sig, f) PHYSFS_writeSBE32(f, sig)
int parse_bmhd(PHYSFS_file *ifile,long len,iff_bitmap_header *bmheader)
{
len++; /* so no "parm not used" warning */
// debug("parsing bmhd len=%ld\n",len);
PHYSFS_readSBE16(ifile, &bmheader->w);
PHYSFS_readSBE16(ifile, &bmheader->h);
PHYSFS_readSBE16(ifile, &bmheader->x);
PHYSFS_readSBE16(ifile, &bmheader->y);
bmheader->nplanes = PHYSFSX_readByte(ifile);
bmheader->masking = PHYSFSX_readByte(ifile);
bmheader->compression = PHYSFSX_readByte(ifile);
PHYSFSX_readByte(ifile); /* skip pad */
PHYSFS_readSBE16(ifile, &bmheader->transparentcolor);
bmheader->xaspect = PHYSFSX_readByte(ifile);
bmheader->yaspect = PHYSFSX_readByte(ifile);
PHYSFS_readSBE16(ifile, &bmheader->pagewidth);
PHYSFS_readSBE16(ifile, &bmheader->pageheight);
iff_transparent_color = (unsigned char)bmheader->transparentcolor;
iff_has_transparency = 0;
if (bmheader->masking == mskHasTransparentColor)
iff_has_transparency = 1;
else if (bmheader->masking != mskNone && bmheader->masking != mskHasMask)
return IFF_UNKNOWN_MASK;
// debug("w,h=%d,%d x,y=%d,%d\n",w,h,x,y);
// debug("nplanes=%d, masking=%d ,compression=%d, transcolor=%d\n",nplanes,masking,compression,transparentcolor);
return IFF_NO_ERROR;
}
// the buffer pointed to by raw_data is stuffed with a pointer to decompressed pixel data
int parse_body(PHYSFS_file *ifile,long len,iff_bitmap_header *bmheader)
{
unsigned char *p=bmheader->raw_data;
int width,depth;
signed char n;
int nn,wid_cnt,end_cnt,plane;
unsigned char *data_end;
int end_pos;
#ifndef NDEBUG
int row_count=0;
#endif
width=0;
depth=0;
end_pos = PHYSFS_tell(ifile) + len;
if (len&1)
end_pos++;
if (bmheader->type == TYPE_PBM) {
width=bmheader->w;
depth=1;
} else if (bmheader->type == TYPE_ILBM) {
width = (bmheader->w+7)/8;
depth=bmheader->nplanes;
}
end_cnt = (width&1)?-1:0;
data_end = p + width*bmheader->h*depth;
if (bmheader->compression == cmpNone) { /* no compression */
int y;
for (y=bmheader->h;y;y--) {
PHYSFS_read(ifile, p, width, depth);
p += bmheader->w;
if (bmheader->masking == mskHasMask)
PHYSFSX_fseek(ifile, width, SEEK_CUR); //skip mask!
if (bmheader->w & 1) PHYSFSX_fgetc(ifile);
}
//cnt = len - bmheader->h * ((bmheader->w+1)&~1);
}
else if (bmheader->compression == cmpByteRun1)
for (wid_cnt=width,plane=0; PHYSFS_tell(ifile) < end_pos && p<data_end;) {
unsigned char c;
if (wid_cnt == end_cnt) {
wid_cnt = width;
plane++;
if ((bmheader->masking == mskHasMask && plane==depth+1) ||
(bmheader->masking != mskHasMask && plane==depth))
plane=0;
}
Assert(wid_cnt > end_cnt);
n=PHYSFSX_fgetc(ifile);
if (n >= 0) { // copy next n+1 bytes from source, they are not compressed
nn = (int) n+1;
wid_cnt -= nn;
if (wid_cnt==-1) {--nn; Assert(width&1);}
if (plane==depth) //masking row
PHYSFSX_fseek(ifile, nn, SEEK_CUR);
else
{
PHYSFS_read(ifile, p, nn, 1);
p += nn;
}
if (wid_cnt==-1) PHYSFSX_fseek(ifile, 1, SEEK_CUR);
}
else if (n>=-127) { // next -n + 1 bytes are following byte
c=PHYSFSX_fgetc(ifile);
nn = (int) -n+1;
wid_cnt -= nn;
if (wid_cnt==-1) {--nn; Assert(width&1);}
if (plane!=depth) //not masking row
{memset(p,c,nn); p+=nn;}
}
#ifndef NDEBUG
if ((p-bmheader->raw_data) % width == 0)
row_count++;
Assert((p-bmheader->raw_data) - (width*row_count) < width);
#endif
}
if (bmheader->masking==mskHasMask && p==data_end && PHYSFS_tell(ifile)==end_pos-2) //I don't know why...
PHYSFSX_fseek(ifile, 1, SEEK_CUR); //...but if I do this it works
if (p==data_end && PHYSFS_tell(ifile)==end_pos-1) //must be a pad byte
//ignore = PHYSFSX_fgetc(ifile); //get pad byte
PHYSFSX_fseek(ifile, 1, SEEK_CUR);
else
if (PHYSFS_tell(ifile)!=end_pos || p!=data_end) {
// debug("IFF Error: p=%x, data_end=%x, cnt=%d\n",p,data_end,cnt);
return IFF_CORRUPT;
}
return IFF_NO_ERROR;
}
//modify passed bitmap
int parse_delta(PHYSFS_file *ifile,long len,iff_bitmap_header *bmheader)
{
unsigned char *p=bmheader->raw_data;
int y;
long chunk_end = PHYSFS_tell(ifile) + len;
PHYSFSX_fseek(ifile, 4, SEEK_CUR); //longword, seems to be equal to 4. Don't know what it is
for (y=0;y<bmheader->h;y++) {
ubyte n_items;
int cnt = bmheader->w;
ubyte code;
n_items = PHYSFSX_readByte(ifile);
while (n_items--) {
code = PHYSFSX_readByte(ifile);
if (code==0) { //repeat
ubyte rep,val;
rep = PHYSFSX_readByte(ifile);
val = PHYSFSX_readByte(ifile);
cnt -= rep;
if (cnt==-1)
rep--;
while (rep--)
*p++ = val;
}
else if (code > 0x80) { //skip
cnt -= (code-0x80);
p += (code-0x80);
if (cnt==-1)
p--;
}
else { //literal
cnt -= code;
if (cnt==-1)
code--;
while (code--)
*p++ = PHYSFSX_readByte(ifile);
if (cnt==-1)
PHYSFSX_readByte(ifile);
}
}
if (cnt == -1) {
if (!bmheader->w&1)
return IFF_CORRUPT;
}
else if (cnt)
return IFF_CORRUPT;
}
if (PHYSFS_tell(ifile) == chunk_end-1) //pad
PHYSFSX_fseek(ifile, 1, SEEK_CUR);
if (PHYSFS_tell(ifile) != chunk_end)
return IFF_CORRUPT;
else
return IFF_NO_ERROR;
}
// the buffer pointed to by raw_data is stuffed with a pointer to bitplane pixel data
void skip_chunk(PHYSFS_file *ifile,long len)
{
int ilen;
ilen = (len+1) & ~1;
PHYSFSX_fseek(ifile,ilen,SEEK_CUR);
}
//read an ILBM or PBM file
// Pass pointer to opened file, and to empty bitmap_header structure, and form length
int iff_parse_ilbm_pbm(PHYSFS_file *ifile,long form_type,iff_bitmap_header *bmheader,int form_len,grs_bitmap *prev_bm)
{
int sig,len;
long start_pos,end_pos;
start_pos = PHYSFS_tell(ifile);
end_pos = start_pos-4+form_len;
if (form_type == pbm_sig)
bmheader->type = TYPE_PBM;
else
bmheader->type = TYPE_ILBM;
while ((PHYSFS_tell(ifile) < end_pos) && (sig=get_sig(ifile)) != EOF) {
if (PHYSFS_readSBE32(ifile, &len)==EOF) break;
switch (sig) {
case bmhd_sig: {
int ret;
int save_w=bmheader->w,save_h=bmheader->h;
ret = parse_bmhd(ifile,len,bmheader);
if (ret != IFF_NO_ERROR)
return ret;
if (bmheader->raw_data) {
if (save_w != bmheader->w || save_h != bmheader->h)
return IFF_BM_MISMATCH;
}
else {
MALLOC( bmheader->raw_data, ubyte, bmheader->w * bmheader->h );
if (!bmheader->raw_data)
return IFF_NO_MEM;
}
break;
}
case anhd_sig:
if (!prev_bm) return IFF_CORRUPT;
// initialized the new bitmap
gr_init_bitmap ( (grs_bitmap*) bmheader, prev_bm->bm_type, 0, 0, prev_bm->bm_w, prev_bm->bm_h, prev_bm->bm_rowsize, 0);
// and copy
gr_bm_bitblt(prev_bm->bm_w, prev_bm->bm_h, 0, 0, 0, 0, prev_bm, (grs_bitmap*) bmheader);
skip_chunk(ifile,len);
break;
case cmap_sig:
{
int ncolors=(int) (len/3),cnum;
unsigned char r,g,b;
for (cnum=0;cnum<ncolors;cnum++) {
r=PHYSFSX_fgetc(ifile);
g=PHYSFSX_fgetc(ifile);
b=PHYSFSX_fgetc(ifile);
r >>= 2; bmheader->palette[cnum].r = r;
g >>= 2; bmheader->palette[cnum].g = g;
b >>= 2; bmheader->palette[cnum].b = b;
}
if (len & 1) PHYSFSX_fgetc(ifile);
break;
}
case body_sig:
{
int r;
if ((r=parse_body(ifile,len,bmheader))!=IFF_NO_ERROR)
return r;
break;
}
case dlta_sig:
{
int r;
if ((r=parse_delta(ifile,len,bmheader))!=IFF_NO_ERROR)
return r;
break;
}
default:
skip_chunk(ifile,len);
break;
}
}
if (PHYSFS_tell(ifile) != start_pos-4+form_len)
return IFF_CORRUPT;
return IFF_NO_ERROR; /* ok! */
}
//convert an ILBM file to a PBM file
int convert_ilbm_to_pbm(iff_bitmap_header *bmheader)
{
int x,y,p;
sbyte *new_data, *destptr, *rowptr;
int bytes_per_row,byteofs;
ubyte checkmask,newbyte,setbit;
MALLOC(new_data, sbyte, bmheader->w * bmheader->h);
if (new_data == NULL) return IFF_NO_MEM;
destptr = new_data;
bytes_per_row = 2*((bmheader->w+15)/16);
for (y=0;y<bmheader->h;y++) {
rowptr = (signed char *) &bmheader->raw_data[y * bytes_per_row * bmheader->nplanes];
for (x=0,checkmask=0x80;x<bmheader->w;x++) {
byteofs = x >> 3;
for (p=newbyte=0,setbit=1;p<bmheader->nplanes;p++) {
if (rowptr[bytes_per_row * p + byteofs] & checkmask)
newbyte |= setbit;
setbit <<= 1;
}
*destptr++ = newbyte;
if ((checkmask >>= 1) == 0) checkmask=0x80;
}
}
d_free(bmheader->raw_data);
bmheader->raw_data = (unsigned char *) new_data;
bmheader->type = TYPE_PBM;
return IFF_NO_ERROR;
}
#define INDEX_TO_15BPP(i) ((short)((((palptr[(i)].r/2)&31)<<10)+(((palptr[(i)].g/2)&31)<<5)+((palptr[(i)].b/2 )&31)))
int convert_rgb15(grs_bitmap *bm,iff_bitmap_header *bmheader)
{
int x,y;
// int newptr = 0;
pal_entry *palptr;
palptr = bmheader->palette;
gr_init_bitmap (bm, bm->bm_type, 0, 0, bm->bm_w, bm->bm_h, bm->bm_rowsize, 0);
for (y=0; y<bm->bm_h; y++) {
for (x=0; x<bmheader->w; x++)
gr_bm_pixel (bm, x, y, INDEX_TO_15BPP(bmheader->raw_data[y*bmheader->w+x]));
}
return IFF_NO_ERROR;
}
//copy an iff header structure to a grs_bitmap structure
void copy_iff_to_grs(grs_bitmap *bm,iff_bitmap_header *bmheader)
{
gr_init_bitmap (bm, bmheader->type, 0, 0, bmheader->w, bmheader->h, bmheader->w, bmheader->raw_data);
}
//if bm->bm_data is set, use it (making sure w & h are correct), else
//allocate the memory
int iff_parse_bitmap(PHYSFS_file *ifile, grs_bitmap *bm, int bitmap_type, sbyte *palette, grs_bitmap *prev_bm)
{
int ret; //return code
iff_bitmap_header bmheader;
int sig,form_len;
long form_type;
bmheader.raw_data = bm->bm_data;
if (bmheader.raw_data) {
bmheader.w = bm->bm_w;
bmheader.h = bm->bm_h;
}//added 05/17/99 Matt Mueller - don't just leave them unitialized
else{
bmheader.w=bmheader.h=0;
}
//end addition -MM
sig=get_sig(ifile);
if (sig != form_sig) {
ret = IFF_NOT_IFF;
goto done;
}
PHYSFS_readSBE32(ifile, &form_len);
form_type = get_sig(ifile);
if (form_type == anim_sig)
ret = IFF_FORM_ANIM;
else if ((form_type == pbm_sig) || (form_type == ilbm_sig))
ret = iff_parse_ilbm_pbm(ifile,form_type,&bmheader,form_len,prev_bm);
else
ret = IFF_UNKNOWN_FORM;
if (ret != IFF_NO_ERROR) { //got an error parsing
if (bmheader.raw_data) d_free(bmheader.raw_data);
goto done;
}
//If IFF file is ILBM, convert to PPB
if (bmheader.type == TYPE_ILBM) {
ret = convert_ilbm_to_pbm(&bmheader);
if (ret != IFF_NO_ERROR) goto done;
}
//Copy data from iff_bitmap_header structure into grs_bitmap structure
copy_iff_to_grs(bm,&bmheader);
if (palette) memcpy(palette,&bmheader.palette,sizeof(bmheader.palette));
//Now do post-process if required
if (bitmap_type == BM_RGB15) {
ret = convert_rgb15(bm,&bmheader);
if (ret != IFF_NO_ERROR) goto done;
}
done:
return ret;
}
//returns error codes - see IFF.H. see GR.H for bitmap_type
int iff_read_bitmap(const char *ifilename,grs_bitmap *bm,int bitmap_type,ubyte *palette)
{
int ret; //return code
PHYSFS_file *ifile;
ifile = PHYSFSX_openReadBuffered(ifilename);
if (ifile == NULL)
return IFF_NO_FILE;
bm->bm_data = NULL;
ret = iff_parse_bitmap(ifile,bm,bitmap_type,(signed char *) palette,NULL);
PHYSFS_close(ifile);
return ret;
}
//like iff_read_bitmap(), but reads into a bitmap that already exists,
//without allocating memory for the bitmap.
int iff_read_into_bitmap(const char *ifilename, grs_bitmap *bm, sbyte *palette)
{
int ret; //return code
PHYSFS_file *ifile;
ifile = PHYSFSX_openReadBuffered(ifilename);
if (ifile == NULL)
return IFF_NO_FILE;
ret = iff_parse_bitmap(ifile,bm,bm->bm_type,palette,NULL);
PHYSFS_close(ifile);
return ret;
}
#define BMHD_SIZE 20
int write_bmhd(PHYSFS_file *ofile,iff_bitmap_header *bitmap_header)
{
put_sig(bmhd_sig,ofile);
PHYSFS_writeSBE32(ofile, BMHD_SIZE);
PHYSFS_writeSBE16(ofile, bitmap_header->w);
PHYSFS_writeSBE16(ofile, bitmap_header->h);
PHYSFS_writeSBE16(ofile, bitmap_header->x);
PHYSFS_writeSBE16(ofile, bitmap_header->y);
PHYSFSX_writeU8(ofile, bitmap_header->nplanes);
PHYSFSX_writeU8(ofile, bitmap_header->masking);
PHYSFSX_writeU8(ofile, bitmap_header->compression);
PHYSFSX_writeU8(ofile, 0); /* pad */
PHYSFS_writeSBE16(ofile, bitmap_header->transparentcolor);
PHYSFSX_writeU8(ofile, bitmap_header->xaspect);
PHYSFSX_writeU8(ofile, bitmap_header->yaspect);
PHYSFS_writeSBE16(ofile, bitmap_header->pagewidth);
PHYSFS_writeSBE16(ofile, bitmap_header->pageheight);
return IFF_NO_ERROR;
}
int write_pal(PHYSFS_file *ofile,iff_bitmap_header *bitmap_header)
{
int i;
int n_colors = 1<<bitmap_header->nplanes;
put_sig(cmap_sig,ofile);
// PHYSFS_writeSBE32(sizeof(pal_entry) * n_colors,ofile);
PHYSFS_writeSBE32(ofile, 3 * n_colors);
for (i=0; i<256; i++) {
unsigned char r,g,b;
r = bitmap_header->palette[i].r * 4 + (bitmap_header->palette[i].r?3:0);
g = bitmap_header->palette[i].g * 4 + (bitmap_header->palette[i].g?3:0);
b = bitmap_header->palette[i].b * 4 + (bitmap_header->palette[i].b?3:0);
PHYSFSX_writeU8(ofile, r);
PHYSFSX_writeU8(ofile, g);
PHYSFSX_writeU8(ofile, b);
}
return IFF_NO_ERROR;
}
int rle_span(ubyte *dest,ubyte *src,int len)
{
int n,lit_cnt,rep_cnt;
ubyte last,*cnt_ptr,*dptr;
cnt_ptr=0;
dptr = dest;
last=src[0]; lit_cnt=1;
for (n=1;n<len;n++) {
if (src[n] == last) {
rep_cnt = 2;
n++;
while (n<len && rep_cnt<128 && src[n]==last) {n++; rep_cnt++;}
if (rep_cnt > 2 || lit_cnt < 2) {
if (lit_cnt > 1) {*cnt_ptr = lit_cnt-2; --dptr;} //save old lit count
*dptr++ = -(rep_cnt-1);
*dptr++ = last;
last = src[n];
lit_cnt = (n<len)?1:0;
continue; //go to next char
} else n--;
}
{
if (lit_cnt==1) {
cnt_ptr = dptr++; //save place for count
*dptr++=last; //store first char
}
*dptr++ = last = src[n];
if (lit_cnt == 127) {
*cnt_ptr = lit_cnt;
//cnt_ptr = dptr++;
lit_cnt = 1;
last = src[++n];
}
else lit_cnt++;
}
}
if (lit_cnt==1) {
*dptr++ = 0;
*dptr++=last; //store first char
}
else if (lit_cnt > 1)
*cnt_ptr = lit_cnt-1;
return dptr-dest;
}
#define EVEN(a) ((a+1)&0xfffffffel)
//returns length of chunk
int write_body(PHYSFS_file *ofile,iff_bitmap_header *bitmap_header,int compression_on)
{
int w=bitmap_header->w,h=bitmap_header->h;
int y,odd=w&1;
long len = EVEN(w) * h,newlen,total_len=0;
ubyte *p=bitmap_header->raw_data,*new_span;
long save_pos;
put_sig(body_sig,ofile);
save_pos = PHYSFS_tell(ofile);
PHYSFS_writeSBE32(ofile, len);
//if (! (new_span = malloc(bitmap_header->w+(bitmap_header->w/128+2)*2))) return IFF_NO_MEM;
MALLOC( new_span, ubyte, bitmap_header->w + (bitmap_header->w/128+2));//adb: removed *2
if (new_span == NULL) return IFF_NO_MEM;
for (y=bitmap_header->h;y--;) {
if (compression_on) {
total_len += newlen = rle_span(new_span,p,bitmap_header->w+odd);
PHYSFS_write(ofile,new_span,newlen,1);
}
else
PHYSFS_write(ofile,p,bitmap_header->w+odd,1);
p+=bitmap_header->row_size; //bitmap_header->w;
}
if (compression_on) { //write actual data length
Assert(PHYSFSX_fseek(ofile,save_pos,SEEK_SET)==0);
(void)save_pos;
PHYSFS_writeSBE32(ofile, total_len);
Assert(PHYSFSX_fseek(ofile,total_len,SEEK_CUR)==0);
if (total_len&1) PHYSFSX_writeU8(ofile, 0); //pad to even
}
d_free(new_span);
return ((compression_on) ? (EVEN(total_len)+8) : (len+8));
}
#if WRITE_TINY
//write a small representation of a bitmap. returns size
int write_tiny(PHYSFS_file *ofile,iff_bitmap_header *bitmap_header,int compression_on)
{
int skip;
int new_w,new_h;
int len,total_len=0,newlen;
int x,y,xofs,odd;
ubyte *p = bitmap_header->raw_data;
ubyte tspan[80],new_span[80*2];
long save_pos;
skip = max((bitmap_header->w+79)/80,(bitmap_header->h+63)/64);
new_w = bitmap_header->w / skip;
new_h = bitmap_header->h / skip;
odd = new_w & 1;
len = new_w * new_h + 4;
put_sig(tiny_sig,ofile);
save_pos = PHYSFS_tell(ofile);
PHYSFS_writeSBE32(ofile, EVEN(len));
PHYSFS_writeSBE16(ofile, new_w);
PHYSFS_writeSBE16(ofile, new_h);
for (y=0;y<new_h;y++) {
for (x=xofs=0;x<new_w;x++,xofs+=skip)
tspan[x] = p[xofs];
if (compression_on) {
total_len += newlen = rle_span(new_span,tspan,new_w+odd);
PHYSFS_write(ofile,new_span,newlen,1);
}
else
PHYSFS_write(ofile,p,new_w+odd,1);
p += skip * bitmap_header->row_size; //bitmap_header->w;
}
if (compression_on) {
Assert(PHYSFSX_fseek(ofile,save_pos,SEEK_SET)==0);
(void)save_pos;
PHYSFS_writeSBE32(ofile, 4+total_len);
Assert(PHYSFSX_fseek(ofile,4+total_len,SEEK_CUR)==0);
if (total_len&1) PHYSFSX_writeU8(ofile, 0); //pad to even
}
return ((compression_on) ? (EVEN(total_len)+8+4) : (len+8));
}
#endif
int write_pbm(PHYSFS_file *ofile,iff_bitmap_header *bitmap_header,int compression_on) /* writes a pbm iff file */
{
int ret;
long raw_size = EVEN(bitmap_header->w) * bitmap_header->h;
long body_size,tiny_size,pbm_size = 4 + BMHD_SIZE + 8 + EVEN(raw_size) + sizeof(pal_entry)*(1<<bitmap_header->nplanes)+8;
long save_pos;
put_sig(form_sig,ofile);
save_pos = PHYSFS_tell(ofile);
PHYSFS_writeSBE32(ofile, pbm_size+8);
put_sig(pbm_sig,ofile);
ret = write_bmhd(ofile,bitmap_header);
if (ret != IFF_NO_ERROR) return ret;
ret = write_pal(ofile,bitmap_header);
if (ret != IFF_NO_ERROR) return ret;
#if WRITE_TINY
tiny_size = write_tiny(ofile,bitmap_header,compression_on);
#else
tiny_size = 0;
#endif
body_size = write_body(ofile,bitmap_header,compression_on);
pbm_size = 4 + BMHD_SIZE + body_size + tiny_size + sizeof(pal_entry)*(1<<bitmap_header->nplanes)+8;
Assert(PHYSFSX_fseek(ofile,save_pos,SEEK_SET)==0);
(void)save_pos;
PHYSFS_writeSBE32(ofile, pbm_size+8);
Assert(PHYSFSX_fseek(ofile,pbm_size+8,SEEK_CUR)==0);
return ret;
}
//writes an IFF file from a grs_bitmap structure. writes palette if not null
//returns error codes - see IFF.H.
int iff_write_bitmap(const char *ofilename,grs_bitmap *bm,ubyte *palette)
{
PHYSFS_file *ofile;
iff_bitmap_header bmheader;
int ret;
int compression_on;
if (bm->bm_type == BM_RGB15) return IFF_BAD_BM_TYPE;
#if COMPRESS
compression_on = (bm->bm_w>=MIN_COMPRESS_WIDTH);
#else
compression_on = 0;
#endif
//fill in values in bmheader
bmheader.x = bmheader.y = 0;
bmheader.w = bm->bm_w;
bmheader.h = bm->bm_h;
bmheader.type = TYPE_PBM;
bmheader.transparentcolor = iff_transparent_color;
bmheader.pagewidth = bm->bm_w; //I don't think it matters what I write
bmheader.pageheight = bm->bm_h;
bmheader.nplanes = 8;
bmheader.masking = mskNone;
if (iff_has_transparency) {
bmheader.masking |= mskHasTransparentColor;
}
bmheader.compression = (compression_on?cmpByteRun1:cmpNone);
bmheader.xaspect = bmheader.yaspect = 1; //I don't think it matters what I write
bmheader.raw_data = bm->bm_data;
bmheader.row_size = bm->bm_rowsize;
if (palette) memcpy(&bmheader.palette,palette,256*3);
//open file and write
if ((ofile = PHYSFS_openWrite(ofilename)) == NULL)
return IFF_NO_FILE;
ret = write_pbm(ofile,&bmheader,compression_on);
PHYSFS_close(ofile);
return ret;
}
//read in many brushes. fills in array of pointers, and n_bitmaps.
//returns iff error codes
int iff_read_animbrush(const char *ifilename,grs_bitmap **bm_list,int max_bitmaps,int *n_bitmaps,ubyte *palette)
{
int ret = IFF_NO_ERROR; //return code
PHYSFS_file *ifile;
int sig,form_len;
long form_type;
*n_bitmaps=0;
ifile = PHYSFSX_openReadBuffered(ifilename);
if (ifile == NULL)
return IFF_NO_FILE;
sig=get_sig(ifile);
PHYSFS_readSBE32(ifile, &form_len);
if (sig != form_sig) {
ret = IFF_NOT_IFF;
goto done;
}
form_type = get_sig(ifile);
if ((form_type == pbm_sig) || (form_type == ilbm_sig))
ret = IFF_FORM_BITMAP;
else if (form_type == anim_sig) {
int anim_end = PHYSFS_tell(ifile) + form_len - 4;
while (PHYSFS_tell(ifile) < anim_end && *n_bitmaps < max_bitmaps) {
grs_bitmap *prev_bm;
prev_bm = *n_bitmaps>0?bm_list[*n_bitmaps-1]:NULL;
MALLOC(bm_list[*n_bitmaps] , grs_bitmap, 1 );
gr_init_bitmap_data ((grs_bitmap *) &bm_list[*n_bitmaps]);
ret = iff_parse_bitmap(ifile,bm_list[*n_bitmaps],form_type,*n_bitmaps>0?NULL:(signed char *)palette,prev_bm);
if (ret != IFF_NO_ERROR)
goto done;
(*n_bitmaps)++;
}
if (PHYSFS_tell(ifile) < anim_end) //ran out of room
ret = IFF_TOO_MANY_BMS;
}
else
ret = IFF_UNKNOWN_FORM;
done:
PHYSFS_close(ifile);
return ret;
}
//text for error messges
static const char error_messages[] = {
"No error.\0"
"Not enough mem for loading or processing bitmap.\0"
"IFF file has unknown FORM type.\0"
"Not an IFF file.\0"
"Cannot open file.\0"
"Tried to save invalid type, like BM_RGB15.\0"
"Bad data in file.\0"
"ANIM file cannot be loaded with normal bitmap loader.\0"
"Normal bitmap file cannot be loaded with anim loader.\0"
"Array not big enough on anim brush read.\0"
"Unknown mask type in bitmap header.\0"
"Error reading file.\0"
};
//function to return pointer to error message
const char *iff_errormsg(int error_number)
{
const char *p = error_messages;
while (error_number--) {
if (!p) return NULL;
p += strlen(p)+1;
}
return p;
}