/* * xvbmp.c - i/o routines for .BMP files (MS Windows 3.x) * * LoadBMP(fname, numcols) * WriteBMP(fp, pic, ptype, w, h, r, g, b, numcols, style); */ /* Copyright Notice * ================ * Copyright 1989, 1990, 1991, 1992, 1993 by John Bradley * * Permission to use, copy, and distribute XV in its entirety, for * non-commercial purposes, is hereby granted without fee, provided that * this license information and copyright notice appear in all copies. * * Note that distributing XV 'bundled' in with ANY product is considered * to be a 'commercial purpose'. * * Also note that any copies of XV that are distributed MUST be built * and/or configured to be in their 'unregistered copy' mode, so that it * is made obvious to the user that XV is shareware, and that they should * consider donating, or at least reading this License Info. * * The software may be modified for your own purposes, but modified * versions may NOT be distributed without prior consent of the author. * * This software is provided 'as-is', without any express or implied * warranty. In no event will the author be held liable for any damages * arising from the use of this software. * * If you would like to do something with XV that this copyright * prohibits (such as distributing it with a commercial product, * using portions of the source in some other program, etc.), please * contact the author (preferably via email). Arrangements can * probably be worked out. * * XV is shareware for PERSONAL USE only. You may use XV for your own * amusement, and if you find it nifty, useful, generally cool, or of * some value to you, your non-deductable donation would be greatly * appreciated. $25 is the suggested donation, though, of course, * larger donations are quite welcome. Folks who donate $25 or more * can receive a Real Nice bound copy of the XV manual for no extra * charge. * * Commercial, government, and institutional users MUST register their * copies of XV, for the exceedingly REASONABLE price of just $25 per * workstation/X terminal. Site licenses are available for those who * wish to run XV on a large number of machines. Contact the author * for more details. * * The author may be contacted via: * US Mail: John Bradley * 1053 Floyd Terrace * Bryn Mawr, PA 19010 * * Phone: (215) 898-8813 * EMail: bradley@cis.upenn.edu */ #include "xv.h" /* comments on error handling: a truncated file is not considered a Major Error. The file is loaded, the rest of the pic is filled with 0's. a file with garbage characters in it is an unloadable file. All allocated stuff is tossed, and LoadPBM returns non-zero not being able to malloc is a Fatal Error. The program is aborted. */ #define BI_RGB 0 #define BI_RLE8 1 #define BI_RLE4 2 static long filesize; #ifdef __STDC__ static int loadBMP1(FILE *, byte *, int, int); static int loadBMP4(FILE *, byte *, int, int, int); static int loadBMP8(FILE *, byte *, int, int, int); static int loadBMP24(FILE *, byte *, int, int); static unsigned int getshort(FILE *); static unsigned int getint(FILE *); static void putshort(FILE *, int); static void putint(FILE *, int); static void writeBMP1(FILE *, byte *, int, int); static void writeBMP4(FILE *, byte *, int, int); static void writeBMP8(FILE *, byte *, int, int); static void writeBMP24(FILE *, byte *, int, int); static int bmpError(char *, char *); #else static int loadBMP1(), loadBMP4(), loadBMP8(), loadBMP24(); static unsigned int getshort(), getint(); static void putshort(), putint(); static void writeBMP1(), writeBMP4(), writeBMP8(), writeBMP24(); static int bmpError(); #endif /*******************************************/ int LoadBMP(fname, pinfo) char *fname; PICINFO *pinfo; /*******************************************/ { FILE *fp; int i, c, c1, rv; unsigned int bfSize, bfOffBits, biSize, biWidth, biHeight, biPlanes; unsigned int biBitCount, biCompression, biSizeImage, biXPelsPerMeter; unsigned int biYPelsPerMeter, biClrUsed, biClrImportant; char *cmpstr; byte *pic24, *pic8; char buf[512], *bname; /* returns '1' on success */ pic8 = pic24 = (byte *) NULL; bname = BaseName(fname); fp=fopen(fname,"r"); if (!fp) return (bmpError(bname, "couldn't open file")); fseek(fp, 0L, 2); /* figure out the file size */ filesize = ftell(fp); fseek(fp, 0L, 0); /* read the file type (first two bytes) */ c = getc(fp); c1 = getc(fp); if (c!='B' || c1!='M') { bmpError(bname,"file type != 'BM'"); goto ERROR; } bfSize = getint(fp); getshort(fp); /* reserved and ignored */ getshort(fp); bfOffBits = getint(fp); biSize = getint(fp); biWidth = getint(fp); biHeight = getint(fp); biPlanes = getshort(fp); biBitCount = getshort(fp); biCompression = getint(fp); biSizeImage = getint(fp); biXPelsPerMeter = getint(fp); biYPelsPerMeter = getint(fp); biClrUsed = getint(fp); biClrImportant = getint(fp); if (DEBUG>1) { fprintf(stderr,"\nLoadBMP:\tbfSize=%d, bfOffBits=%d\n",bfSize,bfOffBits); fprintf(stderr,"\t\tbiSize=%d, biWidth=%d, biHeight=%d, biPlanes=%d\n", biSize, biWidth, biHeight, biPlanes); fprintf(stderr,"\t\tbiBitCount=%d, biCompression=%d, biSizeImage=%d\n", biBitCount, biCompression, biSizeImage); fprintf(stderr,"\t\tbiX,YPelsPerMeter=%d,%d biClrUsed=%d, biClrImp=%d\n", biXPelsPerMeter, biYPelsPerMeter, biClrUsed, biClrImportant); } if (ferror(fp)) { bmpError(bname,"EOF reached in file header"); goto ERROR; } /* error checking */ if ((biBitCount!=1 && biBitCount!=4 && biBitCount!=8 && biBitCount!=24) || biPlanes!=1 || biCompression>BI_RLE4) { sprintf(buf,"Bogus BMP File! (bitCount=%d, Planes=%d, Compression=%d)", biBitCount, biPlanes, biCompression); bmpError(bname, buf); goto ERROR; } if (((biBitCount==1 || biBitCount==24) && biCompression != BI_RGB) || (biBitCount==4 && biCompression==BI_RLE8) || (biBitCount==8 && biCompression==BI_RLE4)) { sprintf(buf,"Bogus BMP File! (bitCount=%d, Compression=%d)", biBitCount, biCompression); bmpError(bname, buf); goto ERROR; } /* skip ahead to colormap, using biSize */ c = biSize - 40; /* 40 bytes read from biSize to biClrImportant */ for (i=0; ib[i] = getc(fp); pinfo->g[i] = getc(fp); pinfo->r[i] = getc(fp); getc(fp); /* unused */ } if (ferror(fp)) { bmpError(bname,"EOF reached in BMP colormap"); goto ERROR; } if (DEBUG>1) { fprintf(stderr,"LoadBMP: BMP colormap: (RGB order)\n"); for (i=0; ir[i],pinfo->g[i],pinfo->b[i]); } fprintf(stderr,"\n\n"); } } /* create pic8 or pic24 */ if (biBitCount==24) { pic24 = (byte *) calloc(biWidth * biHeight * 3, 1); if (!pic24) return (bmpError(bname, "couldn't malloc 'pic24'")); } else { pic8 = (byte *) calloc(biWidth * biHeight, 1); if (!pic8) return(bmpError(bname, "couldn't malloc 'pic8'")); } WaitCursor(); /* load up the image */ if (biBitCount == 1) rv = loadBMP1(fp,pic8,biWidth,biHeight); else if (biBitCount == 4) rv = loadBMP4(fp,pic8,biWidth,biHeight, biCompression); else if (biBitCount == 8) rv = loadBMP8(fp,pic8,biWidth,biHeight, biCompression); else rv = loadBMP24(fp,pic24,biWidth,biHeight); if (rv) bmpError(bname, "File appears truncated. Winging it.\n"); fclose(fp); if (biBitCount == 24) { pinfo->pic = pic24; pinfo->type = PIC24; } else { pinfo->pic = pic8; pinfo->type = PIC8; } cmpstr = ""; if (biCompression == BI_RLE4) cmpstr = ", RLE4 compressed"; else if (biCompression == BI_RLE8) cmpstr = ", RLE8 compressed"; pinfo->w = biWidth; pinfo->h = biHeight; pinfo->frmType = F_BMP; pinfo->colType = F_FULLCOLOR; sprintf(pinfo->fullInfo, "BMP, %d bit%s per pixel%s. (%ld bytes)", biBitCount, (biBitCount == 1) ? "" : "s", cmpstr, filesize); sprintf(pinfo->shrtInfo, "%dx%d BMP.", biWidth, biHeight); pinfo->comment = (char *) NULL; return 1; ERROR: fclose(fp); return 0; } /*******************************************/ static int loadBMP1(fp, pic8, w, h) FILE *fp; byte *pic8; int w,h; { int i,j,c,bitnum,padw; byte *pp; c = 0; padw = ((w + 31)/32) * 32; /* 'w', padded to be a multiple of 32 */ for (i=h-1; i>=0; i--) { pp = pic8 + (i * w); if ((i&0x3f)==0) WaitCursor(); for (j=bitnum=0; j=0; i--) { pp = pic8 + (i * w); if ((i&0x3f)==0) WaitCursor(); for (j=nybnum=0; j> 4; c <<= 4; } } if (ferror(fp)) break; } } else if (comp == BI_RLE4) { /* read RLE4 compressed data */ x = y = 0; pp = pic8 + x + (h-y-1)*w; while (y>4)&0x0f); } else { /* c==0x00 : escape codes */ c = getc(fp); if (c == EOF) { rv = 1; break; } if (c == 0x00) { /* end of line */ x=0; y++; pp = pic8 + x + (h-y-1)*w; } else if (c == 0x01) break; /* end of pic8 */ else if (c == 0x02) { /* delta */ c = getc(fp); x += c; c = getc(fp); y += c; pp = pic8 + x + (h-y-1)*w; } else { /* absolute mode */ for (i=0; i>4)&0x0f); } if (((c&3)==1) || ((c&3)==2)) getc(fp); /* read pad byte */ } } /* escape processing */ if (ferror(fp)) break; } /* while */ } else { fprintf(stderr,"unknown BMP compression type 0x%0x\n", comp); } if (ferror(fp)) rv = 1; return rv; } /*******************************************/ static int loadBMP8(fp, pic8, w, h, comp) FILE *fp; byte *pic8; int w,h,comp; { int i,j,c,c1,padw,x,y,rv; byte *pp; rv = 0; if (comp == BI_RGB) { /* read uncompressed data */ padw = ((w + 3)/4) * 4; /* 'w' padded to a multiple of 4pix (32 bits) */ for (i=h-1; i>=0; i--) { pp = pic8 + (i * w); if ((i&0x3f)==0) WaitCursor(); for (j=0; j=0; i--) { pp = pic24 + (i * w * 3); if ((i&0x3f)==0) WaitCursor(); for (j=0; j>8) & 0xff; putc(c, fp); putc(c1,fp); } /*******************************************/ static void putint(fp, i) FILE *fp; int i; { int c, c1, c2, c3; c = ((unsigned int ) i) & 0xff; c1 = (((unsigned int) i)>>8) & 0xff; c2 = (((unsigned int) i)>>16) & 0xff; c3 = (((unsigned int) i)>>24) & 0xff; putc(c, fp); putc(c1,fp); putc(c2,fp); putc(c3,fp); } static byte pc2nc[256],r1[256],g1[256],b1[256]; /*******************************************/ int WriteBMP(fp,pic824,ptype,w,h,rmap,gmap,bmap,numcols,colorstyle) FILE *fp; byte *pic824; int ptype,w,h; byte *rmap, *gmap, *bmap; int numcols, colorstyle; { /* * if PIC8, and colorstyle == F_FULLCOLOR, F_GREYSCALE, or F_REDUCED, * the program writes an uncompressed 4- or 8-bit image (depending on * the value of numcols) * * if PIC24, and colorstyle == F_FULLCOLOR, program writes an uncompressed * 24-bit image * if PIC24 and colorstyle = F_GREYSCALE, program writes an uncompressed * 8-bit image * note that PIC24 and F_BWDITHER/F_REDUCED won't happen * * if colorstyle == F_BWDITHER, it writes a 1-bit image * */ int i,j, nc, nbits, bperlin, cmaplen; byte *graypic, *sp, *dp, graymap[256]; nc = nbits = cmaplen = 0; graypic = NULL; if (ptype == PIC24 && colorstyle == F_GREYSCALE) { /* generate a faked 8-bit per pixel image with a grayscale cmap, so that it can just fall through existing 8-bit code */ graypic = (byte *) malloc(w*h); if (!graypic) FatalError("unable to malloc in WriteBMP()"); for (i=0,sp=pic824,dp=graypic; i=0; i--) { pp = pic8 + (i * w); if ((i&0x3f)==0) WaitCursor(); for (j=bitnum=c=0; j<=padw; j++,bitnum++) { if (bitnum == 8) { /* write the next byte */ putc(c,fp); bitnum = c = 0; } c <<= 1; if (j=0; i--) { pp = pic8 + (i * w); if ((i&0x3f)==0) WaitCursor(); for (j=nybnum=c=0; j<=padw; j++,nybnum++) { if (nybnum == 2) { /* write next byte */ putc((c&0xff), fp); nybnum = c = 0; } c <<= 4; if (j=0; i--) { pp = pic8 + (i * w); if ((i&0x3f)==0) WaitCursor(); for (j=0; j=0; i--) { pp = pic24 + (i * w * 3); if ((i&0x3f)==0) WaitCursor(); for (j=0; j