// ========================================================== // BMP Loader and Writer // // Design and implementation by // - Floris van den Berg (flvdberg@wxs.nl) // - Markus Loibl (markus.loibl@epost.de) // - Martin Weber (martweb@gmx.net) // - Hervé Drolon (drolon@infonie.fr) // - Michal Novotny (michal@etc.cz) // // This file is part of FreeImage 3 // // COVERED CODE IS PROVIDED UNDER THIS LICENSE ON AN "AS IS" BASIS, WITHOUT WARRANTY // OF ANY KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING, WITHOUT LIMITATION, WARRANTIES // THAT THE COVERED CODE IS FREE OF DEFECTS, MERCHANTABLE, FIT FOR A PARTICULAR PURPOSE // OR NON-INFRINGING. THE ENTIRE RISK AS TO THE QUALITY AND PERFORMANCE OF THE COVERED // CODE IS WITH YOU. SHOULD ANY COVERED CODE PROVE DEFECTIVE IN ANY RESPECT, YOU (NOT // THE INITIAL DEVELOPER OR ANY OTHER CONTRIBUTOR) ASSUME THE COST OF ANY NECESSARY // SERVICING, REPAIR OR CORRECTION. THIS DISCLAIMER OF WARRANTY CONSTITUTES AN ESSENTIAL // PART OF THIS LICENSE. NO USE OF ANY COVERED CODE IS AUTHORIZED HEREUNDER EXCEPT UNDER // THIS DISCLAIMER. // // Use at your own risk! // ========================================================== #include "FreeImage.h" #include "Utilities.h" // ---------------------------------------------------------- // Constants + headers // ---------------------------------------------------------- static const BYTE RLE_COMMAND = 0; static const BYTE RLE_ENDOFLINE = 0; static const BYTE RLE_ENDOFBITMAP = 1; static const BYTE RLE_DELTA = 2; static const BYTE BI_RGB = 0; static const BYTE BI_RLE8 = 1; static const BYTE BI_RLE4 = 2; static const BYTE BI_BITFIELDS = 3; // ---------------------------------------------------------- #ifdef _WIN32 #pragma pack(push, 1) #else #pragma pack(1) #endif typedef struct tagBITMAPCOREHEADER { DWORD bcSize; WORD bcWidth; WORD bcHeight; WORD bcPlanes; WORD bcBitCnt; } BITMAPCOREHEADER, *PBITMAPCOREHEADER; typedef struct tagBITMAPINFOOS2_1X_HEADER { DWORD biSize; WORD biWidth; WORD biHeight; WORD biPlanes; WORD biBitCount; } BITMAPINFOOS2_1X_HEADER, *PBITMAPINFOOS2_1X_HEADER; typedef struct tagBITMAPFILEHEADER { WORD bfType; DWORD bfSize; WORD bfReserved1; WORD bfReserved2; DWORD bfOffBits; } BITMAPFILEHEADER, *PBITMAPFILEHEADER; #ifdef _WIN32 #pragma pack(pop) #else #pragma pack() #endif // ========================================================== // Plugin Interface // ========================================================== static int s_format_id; // ========================================================== // Internal functions // ========================================================== #ifdef FREEIMAGE_BIGENDIAN static void SwapInfoHeader(BITMAPINFOHEADER *header) { SwapLong(&header->biSize); SwapLong((DWORD *)&header->biWidth); SwapLong((DWORD *)&header->biHeight); SwapShort(&header->biPlanes); SwapShort(&header->biBitCount); SwapLong(&header->biCompression); SwapLong(&header->biSizeImage); SwapLong((DWORD *)&header->biXPelsPerMeter); SwapLong((DWORD *)&header->biYPelsPerMeter); SwapLong(&header->biClrUsed); SwapLong(&header->biClrImportant); } static void SwapCoreHeader(BITMAPCOREHEADER *header) { SwapLong(&header->bcSize); SwapShort(&header->bcWidth); SwapShort(&header->bcHeight); SwapShort(&header->bcPlanes); SwapShort(&header->bcBitCnt); } static void SwapOS21XHeader(BITMAPINFOOS2_1X_HEADER *header) { SwapLong(&header->biSize); SwapShort(&header->biWidth); SwapShort(&header->biHeight); SwapShort(&header->biPlanes); SwapShort(&header->biBitCount); } static void SwapFileHeader(BITMAPFILEHEADER *header) { SwapShort(&header->bfType); SwapLong(&header->bfSize); SwapShort(&header->bfReserved1); SwapShort(&header->bfReserved2); SwapLong(&header->bfOffBits); } #endif // -------------------------------------------------------------------------- /** Load uncompressed image pixels for 1-, 4-, 8-, 16-, 24- and 32-bit dib @param io FreeImage IO @param handle FreeImage IO handle @param dib Image to be loaded @param height Image height @param pitch Image pitch @param bit_count Image bit-depth (1-, 4-, 8-, 16-, 24- or 32-bit) @return Returns TRUE if successful, returns FALSE otherwise */ static BOOL LoadPixelData(FreeImageIO *io, fi_handle handle, FIBITMAP *dib, int height, unsigned pitch, unsigned bit_count) { unsigned count = 0; // Load pixel data // NB: height can be < 0 for BMP data if (height > 0) { count = io->read_proc((void *)FreeImage_GetBits(dib), height * pitch, 1, handle); if(count != 1) { return FALSE; } } else { int positiveHeight = abs(height); for (int c = 0; c < positiveHeight; ++c) { count = io->read_proc((void *)FreeImage_GetScanLine(dib, positiveHeight - c - 1), pitch, 1, handle); if(count != 1) { return FALSE; } } } // swap as needed #ifdef FREEIMAGE_BIGENDIAN if (bit_count == 16) { for(unsigned y = 0; y < FreeImage_GetHeight(dib); y++) { WORD *pixel = (WORD *)FreeImage_GetScanLine(dib, y); for(unsigned x = 0; x < FreeImage_GetWidth(dib); x++) { SwapShort(pixel); pixel++; } } } #endif #if FREEIMAGE_COLORORDER == FREEIMAGE_COLORORDER_RGB if (bit_count == 24 || bit_count == 32) { for(unsigned y = 0; y < FreeImage_GetHeight(dib); y++) { BYTE *pixel = FreeImage_GetScanLine(dib, y); for(unsigned x = 0; x < FreeImage_GetWidth(dib); x++) { INPLACESWAP(pixel[0], pixel[2]); pixel += (bit_count >> 3); } } } #endif return TRUE; } /** Load image pixels for 4-bit RLE compressed dib @param io FreeImage IO @param handle FreeImage IO handle @param width Image width @param height Image height @param dib Image to be loaded @return Returns TRUE if successful, returns FALSE otherwise */ static BOOL LoadPixelDataRLE4(FreeImageIO *io, fi_handle handle, int width, int height, FIBITMAP *dib) { int status_byte = 0; BYTE second_byte = 0; int bits = 0; BYTE *pixels = NULL; // temporary 8-bit buffer try { height = abs(height); pixels = (BYTE*)malloc(width * height * sizeof(BYTE)); if(!pixels) throw(1); memset(pixels, 0, width * height * sizeof(BYTE)); BYTE *q = pixels; BYTE *end = pixels + height * width; for (int scanline = 0; scanline < height; ) { if (q < pixels || q >= end) { break; } if(io->read_proc(&status_byte, sizeof(BYTE), 1, handle) != 1) { throw(1); } if (status_byte != 0) { status_byte = (int)MIN((size_t)status_byte, (size_t)(end - q)); // Encoded mode if(io->read_proc(&second_byte, sizeof(BYTE), 1, handle) != 1) { throw(1); } for (int i = 0; i < status_byte; i++) { *q++=(BYTE)((i & 0x01) ? (second_byte & 0x0f) : ((second_byte >> 4) & 0x0f)); } bits += status_byte; } else { // Escape mode if(io->read_proc(&status_byte, sizeof(BYTE), 1, handle) != 1) { throw(1); } switch (status_byte) { case RLE_ENDOFLINE: { // End of line bits = 0; scanline++; q = pixels + scanline*width; } break; case RLE_ENDOFBITMAP: // End of bitmap q = end; break; case RLE_DELTA: { // read the delta values BYTE delta_x = 0; BYTE delta_y = 0; if(io->read_proc(&delta_x, sizeof(BYTE), 1, handle) != 1) { throw(1); } if(io->read_proc(&delta_y, sizeof(BYTE), 1, handle) != 1) { throw(1); } // apply them bits += delta_x; scanline += delta_y; q = pixels + scanline*width+bits; } break; default: { // Absolute mode status_byte = (int)MIN((size_t)status_byte, (size_t)(end - q)); for (int i = 0; i < status_byte; i++) { if ((i & 0x01) == 0) { if(io->read_proc(&second_byte, sizeof(BYTE), 1, handle) != 1) { throw(1); } } *q++=(BYTE)((i & 0x01) ? (second_byte & 0x0f) : ((second_byte >> 4) & 0x0f)); } bits += status_byte; // Read pad byte if (((status_byte & 0x03) == 1) || ((status_byte & 0x03) == 2)) { BYTE padding = 0; if(io->read_proc(&padding, sizeof(BYTE), 1, handle) != 1) { throw(1); } } } break; } } } { // Convert to 4-bit for(int y = 0; y < height; y++) { const BYTE *src = (BYTE*)pixels + y * width; BYTE *dst = FreeImage_GetScanLine(dib, y); BOOL hinibble = TRUE; for (int cols = 0; cols < width; cols++){ if (hinibble) { dst[cols >> 1] = (src[cols] << 4); } else { dst[cols >> 1] |= src[cols]; } hinibble = !hinibble; } } } free(pixels); return TRUE; } catch(int) { if(pixels) free(pixels); return FALSE; } } /** Load image pixels for 8-bit RLE compressed dib @param io FreeImage IO @param handle FreeImage IO handle @param width Image width @param height Image height @param dib Image to be loaded @return Returns TRUE if successful, returns FALSE otherwise */ static BOOL LoadPixelDataRLE8(FreeImageIO *io, fi_handle handle, int width, int height, FIBITMAP *dib) { BYTE status_byte = 0; BYTE second_byte = 0; int scanline = 0; int bits = 0; for (;;) { if( io->read_proc(&status_byte, sizeof(BYTE), 1, handle) != 1) { return FALSE; } switch (status_byte) { case RLE_COMMAND : if(io->read_proc(&status_byte, sizeof(BYTE), 1, handle) != 1) { return FALSE; } switch (status_byte) { case RLE_ENDOFLINE : bits = 0; scanline++; break; case RLE_ENDOFBITMAP : return TRUE; case RLE_DELTA : { // read the delta values BYTE delta_x = 0; BYTE delta_y = 0; if(io->read_proc(&delta_x, sizeof(BYTE), 1, handle) != 1) { return FALSE; } if(io->read_proc(&delta_y, sizeof(BYTE), 1, handle) != 1) { return FALSE; } // apply them bits += delta_x; scanline += delta_y; break; } default : { if(scanline >= abs(height)) { return TRUE; } int count = MIN((int)status_byte, width - bits); BYTE *sline = FreeImage_GetScanLine(dib, scanline); if(io->read_proc((void *)(sline + bits), sizeof(BYTE) * count, 1, handle) != 1) { return FALSE; } // align run length to even number of bytes if ((status_byte & 1) == 1) { if(io->read_proc(&second_byte, sizeof(BYTE), 1, handle) != 1) { return FALSE; } } bits += status_byte; break; } } break; default : { if(scanline >= abs(height)) { return TRUE; } int count = MIN((int)status_byte, width - bits); BYTE *sline = FreeImage_GetScanLine(dib, scanline); if(io->read_proc(&second_byte, sizeof(BYTE), 1, handle) != 1) { return FALSE; } for (int i = 0; i < count; i++) { *(sline + bits) = second_byte; bits++; } break; } } } } // -------------------------------------------------------------------------- static FIBITMAP * LoadWindowsBMP(FreeImageIO *io, fi_handle handle, int flags, unsigned bitmap_bits_offset) { FIBITMAP *dib = NULL; try { BOOL header_only = (flags & FIF_LOAD_NOPIXELS) == FIF_LOAD_NOPIXELS; // load the info header BITMAPINFOHEADER bih; io->read_proc(&bih, sizeof(BITMAPINFOHEADER), 1, handle); #ifdef FREEIMAGE_BIGENDIAN SwapInfoHeader(&bih); #endif // keep some general information about the bitmap unsigned used_colors = bih.biClrUsed; int width = bih.biWidth; int height = bih.biHeight; // WARNING: height can be < 0 => check each call using 'height' as a parameter unsigned bit_count = bih.biBitCount; unsigned compression = bih.biCompression; unsigned pitch = CalculatePitch(CalculateLine(width, bit_count)); switch (bit_count) { case 1 : case 4 : case 8 : { if ((used_colors == 0) || (used_colors > CalculateUsedPaletteEntries(bit_count))) used_colors = CalculateUsedPaletteEntries(bit_count); // allocate enough memory to hold the bitmap (header, palette, pixels) and read the palette dib = FreeImage_AllocateHeader(header_only, width, height, bit_count); if (dib == NULL) { throw FI_MSG_ERROR_DIB_MEMORY; } // set resolution information FreeImage_SetDotsPerMeterX(dib, bih.biXPelsPerMeter); FreeImage_SetDotsPerMeterY(dib, bih.biYPelsPerMeter); // load the palette io->read_proc(FreeImage_GetPalette(dib), used_colors * sizeof(RGBQUAD), 1, handle); #if FREEIMAGE_COLORORDER == FREEIMAGE_COLORORDER_RGB RGBQUAD *pal = FreeImage_GetPalette(dib); for(int i = 0; i < used_colors; i++) { INPLACESWAP(pal[i].rgbRed, pal[i].rgbBlue); } #endif if(header_only) { // header only mode return dib; } // seek to the actual pixel data. // this is needed because sometimes the palette is larger than the entries it contains predicts if (bitmap_bits_offset > (sizeof(BITMAPFILEHEADER) + sizeof(BITMAPINFOHEADER) + (used_colors * sizeof(RGBQUAD)))) io->seek_proc(handle, bitmap_bits_offset, SEEK_SET); // read the pixel data switch (compression) { case BI_RGB : if( LoadPixelData(io, handle, dib, height, pitch, bit_count) ) { return dib; } else { throw "Error encountered while decoding BMP data"; } break; case BI_RLE4 : if( LoadPixelDataRLE4(io, handle, width, height, dib) ) { return dib; } else { throw "Error encountered while decoding RLE4 BMP data"; } break; case BI_RLE8 : if( LoadPixelDataRLE8(io, handle, width, height, dib) ) { return dib; } else { throw "Error encountered while decoding RLE8 BMP data"; } break; default : throw FI_MSG_ERROR_UNSUPPORTED_COMPRESSION; } } break; // 1-, 4-, 8-bit case 16 : { if (bih.biCompression == BI_BITFIELDS) { DWORD bitfields[3]; io->read_proc(bitfields, 3 * sizeof(DWORD), 1, handle); dib = FreeImage_AllocateHeader(header_only, width, height, bit_count, bitfields[0], bitfields[1], bitfields[2]); } else { dib = FreeImage_AllocateHeader(header_only, width, height, bit_count, FI16_555_RED_MASK, FI16_555_GREEN_MASK, FI16_555_BLUE_MASK); } if (dib == NULL) { throw FI_MSG_ERROR_DIB_MEMORY; } // set resolution information FreeImage_SetDotsPerMeterX(dib, bih.biXPelsPerMeter); FreeImage_SetDotsPerMeterY(dib, bih.biYPelsPerMeter); if(header_only) { // header only mode return dib; } // load pixel data and swap as needed if OS is Big Endian if (bitmap_bits_offset > (sizeof(BITMAPFILEHEADER) + sizeof(BITMAPINFOHEADER))) { // seek to the actual pixel data io->seek_proc(handle, bitmap_bits_offset, SEEK_SET); } LoadPixelData(io, handle, dib, height, pitch, bit_count); return dib; } break; // 16-bit case 24 : case 32 : { if (bih.biCompression == BI_BITFIELDS) { DWORD bitfields[3]; io->read_proc(bitfields, 3 * sizeof(DWORD), 1, handle); dib = FreeImage_AllocateHeader(header_only, width, height, bit_count, bitfields[0], bitfields[1], bitfields[2]); } else { if( bit_count == 32 ) { dib = FreeImage_AllocateHeader(header_only, width, height, bit_count, FI_RGBA_RED_MASK, FI_RGBA_GREEN_MASK, FI_RGBA_BLUE_MASK); } else { dib = FreeImage_AllocateHeader(header_only, width, height, bit_count, FI_RGBA_RED_MASK, FI_RGBA_GREEN_MASK, FI_RGBA_BLUE_MASK); } } if (dib == NULL) { throw FI_MSG_ERROR_DIB_MEMORY; } // set resolution information FreeImage_SetDotsPerMeterX(dib, bih.biXPelsPerMeter); FreeImage_SetDotsPerMeterY(dib, bih.biYPelsPerMeter); if(header_only) { // header only mode return dib; } // Skip over the optional palette // A 24 or 32 bit DIB may contain a palette for faster color reduction if (FreeImage_GetColorsUsed(dib) > 0) { io->seek_proc(handle, FreeImage_GetColorsUsed(dib) * sizeof(RGBQUAD), SEEK_CUR); } else if ((bih.biCompression != BI_BITFIELDS) && (bitmap_bits_offset > sizeof(BITMAPFILEHEADER) + sizeof(BITMAPINFOHEADER))) { io->seek_proc(handle, bitmap_bits_offset, SEEK_SET); } // read in the bitmap bits // load pixel data and swap as needed if OS is Big Endian LoadPixelData(io, handle, dib, height, pitch, bit_count); // check if the bitmap contains transparency, if so enable it in the header FreeImage_SetTransparent(dib, (FreeImage_GetColorType(dib) == FIC_RGBALPHA)); return dib; } break; // 24-, 32-bit } } catch(const char *message) { if(dib) { FreeImage_Unload(dib); } if(message) { FreeImage_OutputMessageProc(s_format_id, message); } } return NULL; } // -------------------------------------------------------------------------- static FIBITMAP * LoadOS22XBMP(FreeImageIO *io, fi_handle handle, int flags, unsigned bitmap_bits_offset) { FIBITMAP *dib = NULL; try { BOOL header_only = (flags & FIF_LOAD_NOPIXELS) == FIF_LOAD_NOPIXELS; // load the info header BITMAPINFOHEADER bih; io->read_proc(&bih, sizeof(BITMAPINFOHEADER), 1, handle); #ifdef FREEIMAGE_BIGENDIAN SwapInfoHeader(&bih); #endif // keep some general information about the bitmap unsigned used_colors = bih.biClrUsed; int width = bih.biWidth; int height = bih.biHeight; // WARNING: height can be < 0 => check each read_proc using 'height' as a parameter unsigned bit_count = bih.biBitCount; unsigned compression = bih.biCompression; unsigned pitch = CalculatePitch(CalculateLine(width, bit_count)); switch (bit_count) { case 1 : case 4 : case 8 : { if ((used_colors == 0) || (used_colors > CalculateUsedPaletteEntries(bit_count))) used_colors = CalculateUsedPaletteEntries(bit_count); // allocate enough memory to hold the bitmap (header, palette, pixels) and read the palette dib = FreeImage_AllocateHeader(header_only, width, height, bit_count); if (dib == NULL) { throw FI_MSG_ERROR_DIB_MEMORY; } // set resolution information FreeImage_SetDotsPerMeterX(dib, bih.biXPelsPerMeter); FreeImage_SetDotsPerMeterY(dib, bih.biYPelsPerMeter); // load the palette // note that it may contain RGB or RGBA values : we will calculate this unsigned pal_size = (bitmap_bits_offset - sizeof(BITMAPFILEHEADER) - bih.biSize) / used_colors; io->seek_proc(handle, sizeof(BITMAPFILEHEADER) + bih.biSize, SEEK_SET); RGBQUAD *pal = FreeImage_GetPalette(dib); if(pal_size == 4) { for (unsigned count = 0; count < used_colors; count++) { FILE_BGRA bgra; io->read_proc(&bgra, sizeof(FILE_BGRA), 1, handle); pal[count].rgbRed = bgra.r; pal[count].rgbGreen = bgra.g; pal[count].rgbBlue = bgra.b; } } else if(pal_size == 3) { for (unsigned count = 0; count < used_colors; count++) { FILE_BGR bgr; io->read_proc(&bgr, sizeof(FILE_BGR), 1, handle); pal[count].rgbRed = bgr.r; pal[count].rgbGreen = bgr.g; pal[count].rgbBlue = bgr.b; } } if(header_only) { // header only mode return dib; } // seek to the actual pixel data. // this is needed because sometimes the palette is larger than the entries it contains predicts if (bitmap_bits_offset > (sizeof(BITMAPFILEHEADER) + sizeof(BITMAPINFOHEADER) + (used_colors * 3))) io->seek_proc(handle, bitmap_bits_offset, SEEK_SET); // read the pixel data switch (compression) { case BI_RGB : // load pixel data LoadPixelData(io, handle, dib, height, pitch, bit_count); return dib; case BI_RLE4 : if( LoadPixelDataRLE4(io, handle, width, height, dib) ) { return dib; } else { throw "Error encountered while decoding RLE4 BMP data"; } break; case BI_RLE8 : if( LoadPixelDataRLE8(io, handle, width, height, dib) ) { return dib; } else { throw "Error encountered while decoding RLE8 BMP data"; } break; default : throw FI_MSG_ERROR_UNSUPPORTED_COMPRESSION; } } case 16 : { if (bih.biCompression == 3) { DWORD bitfields[3]; io->read_proc(bitfields, 3 * sizeof(DWORD), 1, handle); dib = FreeImage_AllocateHeader(header_only, width, height, bit_count, bitfields[0], bitfields[1], bitfields[2]); } else { dib = FreeImage_AllocateHeader(header_only, width, height, bit_count, FI16_555_RED_MASK, FI16_555_GREEN_MASK, FI16_555_BLUE_MASK); } if (dib == NULL) { throw FI_MSG_ERROR_DIB_MEMORY; } // set resolution information FreeImage_SetDotsPerMeterX(dib, bih.biXPelsPerMeter); FreeImage_SetDotsPerMeterY(dib, bih.biYPelsPerMeter); if(header_only) { // header only mode return dib; } if (bitmap_bits_offset > (sizeof(BITMAPFILEHEADER) + sizeof(BITMAPINFOHEADER) + (used_colors * 3))) { io->seek_proc(handle, bitmap_bits_offset, SEEK_SET); } // load pixel data and swap as needed if OS is Big Endian LoadPixelData(io, handle, dib, height, pitch, bit_count); return dib; } case 24 : case 32 : { if( bit_count == 32 ) { dib = FreeImage_AllocateHeader(header_only, width, height, bit_count, FI_RGBA_RED_MASK, FI_RGBA_GREEN_MASK, FI_RGBA_BLUE_MASK); } else { dib = FreeImage_AllocateHeader(header_only, width, height, bit_count, FI_RGBA_RED_MASK, FI_RGBA_GREEN_MASK, FI_RGBA_BLUE_MASK); } if (dib == NULL) { throw FI_MSG_ERROR_DIB_MEMORY; } // set resolution information FreeImage_SetDotsPerMeterX(dib, bih.biXPelsPerMeter); FreeImage_SetDotsPerMeterY(dib, bih.biYPelsPerMeter); if(header_only) { // header only mode return dib; } // Skip over the optional palette // A 24 or 32 bit DIB may contain a palette for faster color reduction if (bitmap_bits_offset > (sizeof(BITMAPFILEHEADER) + sizeof(BITMAPINFOHEADER) + (used_colors * 3))) io->seek_proc(handle, bitmap_bits_offset, SEEK_SET); // read in the bitmap bits // load pixel data and swap as needed if OS is Big Endian LoadPixelData(io, handle, dib, height, pitch, bit_count); // check if the bitmap contains transparency, if so enable it in the header FreeImage_SetTransparent(dib, (FreeImage_GetColorType(dib) == FIC_RGBALPHA)); return dib; } } } catch(const char *message) { if(dib) FreeImage_Unload(dib); FreeImage_OutputMessageProc(s_format_id, message); } return NULL; } // -------------------------------------------------------------------------- static FIBITMAP * LoadOS21XBMP(FreeImageIO *io, fi_handle handle, int flags, unsigned bitmap_bits_offset) { FIBITMAP *dib = NULL; try { BOOL header_only = (flags & FIF_LOAD_NOPIXELS) == FIF_LOAD_NOPIXELS; BITMAPINFOOS2_1X_HEADER bios2_1x; io->read_proc(&bios2_1x, sizeof(BITMAPINFOOS2_1X_HEADER), 1, handle); #ifdef FREEIMAGE_BIGENDIAN SwapOS21XHeader(&bios2_1x); #endif // keep some general information about the bitmap unsigned used_colors = 0; unsigned width = bios2_1x.biWidth; unsigned height = bios2_1x.biHeight; // WARNING: height can be < 0 => check each read_proc using 'height' as a parameter unsigned bit_count = bios2_1x.biBitCount; unsigned pitch = CalculatePitch(CalculateLine(width, bit_count)); switch (bit_count) { case 1 : case 4 : case 8 : { used_colors = CalculateUsedPaletteEntries(bit_count); // allocate enough memory to hold the bitmap (header, palette, pixels) and read the palette dib = FreeImage_AllocateHeader(header_only, width, height, bit_count); if (dib == NULL) { throw FI_MSG_ERROR_DIB_MEMORY; } // set resolution information to default values (72 dpi in english units) FreeImage_SetDotsPerMeterX(dib, 2835); FreeImage_SetDotsPerMeterY(dib, 2835); // load the palette RGBQUAD *pal = FreeImage_GetPalette(dib); for (unsigned count = 0; count < used_colors; count++) { FILE_BGR bgr; io->read_proc(&bgr, sizeof(FILE_BGR), 1, handle); pal[count].rgbRed = bgr.r; pal[count].rgbGreen = bgr.g; pal[count].rgbBlue = bgr.b; } if(header_only) { // header only mode return dib; } // Skip over the optional palette // A 24 or 32 bit DIB may contain a palette for faster color reduction io->seek_proc(handle, bitmap_bits_offset, SEEK_SET); // read the pixel data // load pixel data LoadPixelData(io, handle, dib, height, pitch, bit_count); return dib; } case 16 : { dib = FreeImage_AllocateHeader(header_only, width, height, bit_count, FI16_555_RED_MASK, FI16_555_GREEN_MASK, FI16_555_BLUE_MASK); if (dib == NULL) { throw FI_MSG_ERROR_DIB_MEMORY; } // set resolution information to default values (72 dpi in english units) FreeImage_SetDotsPerMeterX(dib, 2835); FreeImage_SetDotsPerMeterY(dib, 2835); if(header_only) { // header only mode return dib; } // load pixel data and swap as needed if OS is Big Endian LoadPixelData(io, handle, dib, height, pitch, bit_count); return dib; } case 24 : case 32 : { if( bit_count == 32 ) { dib = FreeImage_AllocateHeader(header_only, width, height, bit_count, FI_RGBA_RED_MASK, FI_RGBA_GREEN_MASK, FI_RGBA_BLUE_MASK); } else { dib = FreeImage_AllocateHeader(header_only, width, height, bit_count, FI_RGBA_RED_MASK, FI_RGBA_GREEN_MASK, FI_RGBA_BLUE_MASK); } if (dib == NULL) { throw FI_MSG_ERROR_DIB_MEMORY; } // set resolution information to default values (72 dpi in english units) FreeImage_SetDotsPerMeterX(dib, 2835); FreeImage_SetDotsPerMeterY(dib, 2835); if(header_only) { // header only mode return dib; } // Skip over the optional palette // A 24 or 32 bit DIB may contain a palette for faster color reduction // load pixel data and swap as needed if OS is Big Endian LoadPixelData(io, handle, dib, height, pitch, bit_count); // check if the bitmap contains transparency, if so enable it in the header FreeImage_SetTransparent(dib, (FreeImage_GetColorType(dib) == FIC_RGBALPHA)); return dib; } } } catch(const char *message) { if(dib) FreeImage_Unload(dib); FreeImage_OutputMessageProc(s_format_id, message); } return NULL; } // ========================================================== // Plugin Implementation // ========================================================== static const char * DLL_CALLCONV Format() { return "BMP"; } static const char * DLL_CALLCONV Description() { return "Windows or OS/2 Bitmap"; } static const char * DLL_CALLCONV Extension() { return "bmp"; } static const char * DLL_CALLCONV RegExpr() { return "^BM"; } static const char * DLL_CALLCONV MimeType() { return "image/bmp"; } static BOOL DLL_CALLCONV Validate(FreeImageIO *io, fi_handle handle) { BYTE bmp_signature1[] = { 0x42, 0x4D }; BYTE bmp_signature2[] = { 0x42, 0x41 }; BYTE signature[2] = { 0, 0 }; io->read_proc(signature, 1, sizeof(bmp_signature1), handle); if (memcmp(bmp_signature1, signature, sizeof(bmp_signature1)) == 0) return TRUE; if (memcmp(bmp_signature2, signature, sizeof(bmp_signature2)) == 0) return TRUE; return FALSE; } static BOOL DLL_CALLCONV SupportsExportDepth(int depth) { return ( (depth == 1) || (depth == 4) || (depth == 8) || (depth == 16) || (depth == 24) || (depth == 32) ); } static BOOL DLL_CALLCONV SupportsExportType(FREE_IMAGE_TYPE type) { return (type == FIT_BITMAP) ? TRUE : FALSE; } static BOOL DLL_CALLCONV SupportsNoPixels() { return TRUE; } // ---------------------------------------------------------- static FIBITMAP * DLL_CALLCONV Load(FreeImageIO *io, fi_handle handle, int page, int flags, void *data) { if (handle != NULL) { BITMAPFILEHEADER bitmapfileheader; DWORD type = 0; // we use this offset value to make seemingly absolute seeks relative in the file long offset_in_file = io->tell_proc(handle); // read the fileheader io->read_proc(&bitmapfileheader, sizeof(BITMAPFILEHEADER), 1, handle); #ifdef FREEIMAGE_BIGENDIAN SwapFileHeader(&bitmapfileheader); #endif // check the signature if((bitmapfileheader.bfType != 0x4D42) && (bitmapfileheader.bfType != 0x4142)) { FreeImage_OutputMessageProc(s_format_id, FI_MSG_ERROR_MAGIC_NUMBER); return NULL; } // read the first byte of the infoheader io->read_proc(&type, sizeof(DWORD), 1, handle); io->seek_proc(handle, 0 - (long)sizeof(DWORD), SEEK_CUR); #ifdef FREEIMAGE_BIGENDIAN SwapLong(&type); #endif // call the appropriate load function for the found bitmap type switch(type) { case 12: // OS/2 and also all Windows versions since Windows 3.0 return LoadOS21XBMP(io, handle, flags, offset_in_file + bitmapfileheader.bfOffBits); case 64: // OS/2 return LoadOS22XBMP(io, handle, flags, offset_in_file + bitmapfileheader.bfOffBits); case 40: // BITMAPINFOHEADER - all Windows versions since Windows 3.0 return LoadWindowsBMP(io, handle, flags, offset_in_file + bitmapfileheader.bfOffBits); case 52: // BITMAPV2INFOHEADER (undocumented) break; case 56: // BITMAPV3INFOHEADER (undocumented) break; case 108: // BITMAPV4HEADER - all Windows versions since Windows 95/NT4 (not supported) break; case 124: // BITMAPV5HEADER - Windows 98/2000 and newer (not supported) break; default: break; } FreeImage_OutputMessageProc(s_format_id, "unknown bmp subtype with id %d", type); } return NULL; } // ---------------------------------------------------------- /** Encode a 8-bit source buffer into a 8-bit target buffer using a RLE compression algorithm. The size of the target buffer must be equal to the size of the source buffer. On return, the function will return the real size of the target buffer, which should be less that or equal to the source buffer size. @param target 8-bit Target buffer @param source 8-bit Source buffer @param size Source/Target input buffer size @return Returns the target buffer size */ static int RLEEncodeLine(BYTE *target, BYTE *source, int size) { BYTE buffer[256]; int buffer_size = 0; int target_pos = 0; for (int i = 0; i < size; ++i) { if ((i < size - 1) && (source[i] == source[i + 1])) { // find a solid block of same bytes int j = i + 1; int jmax = 254 + i; while ((j < size - 1) && (j < jmax) && (source[j] == source[j + 1])) ++j; // if the block is larger than 3 bytes, use it // else put the data into the larger pool if (((j - i) + 1) > 3) { // don't forget to write what we already have in the buffer switch(buffer_size) { case 0 : break; case RLE_DELTA : target[target_pos++] = 1; target[target_pos++] = buffer[0]; target[target_pos++] = 1; target[target_pos++] = buffer[1]; break; case RLE_ENDOFBITMAP : target[target_pos++] = (BYTE)buffer_size; target[target_pos++] = buffer[0]; break; default : target[target_pos++] = RLE_COMMAND; target[target_pos++] = (BYTE)buffer_size; memcpy(target + target_pos, buffer, buffer_size); // prepare for next run target_pos += buffer_size; if ((buffer_size & 1) == 1) target_pos++; break; } // write the continuous data target[target_pos++] = (BYTE)((j - i) + 1); target[target_pos++] = source[i]; buffer_size = 0; } else { for (int k = 0; k < (j - i) + 1; ++k) { buffer[buffer_size++] = source[i + k]; if (buffer_size == 254) { // write what we have target[target_pos++] = RLE_COMMAND; target[target_pos++] = (BYTE)buffer_size; memcpy(target + target_pos, buffer, buffer_size); // prepare for next run target_pos += buffer_size; buffer_size = 0; } } } i = j; } else { buffer[buffer_size++] = source[i]; } // write the buffer if it's full if (buffer_size == 254) { target[target_pos++] = RLE_COMMAND; target[target_pos++] = (BYTE)buffer_size; memcpy(target + target_pos, buffer, buffer_size); // prepare for next run target_pos += buffer_size; buffer_size = 0; } } // write the last bytes switch(buffer_size) { case 0 : break; case RLE_DELTA : target[target_pos++] = 1; target[target_pos++] = buffer[0]; target[target_pos++] = 1; target[target_pos++] = buffer[1]; break; case RLE_ENDOFBITMAP : target[target_pos++] = (BYTE)buffer_size; target[target_pos++] = buffer[0]; break; default : target[target_pos++] = RLE_COMMAND; target[target_pos++] = (BYTE)buffer_size; memcpy(target + target_pos, buffer, buffer_size); // prepare for next run target_pos += buffer_size; if ((buffer_size & 1) == 1) target_pos++; break; } // write the END_OF_LINE marker target[target_pos++] = RLE_COMMAND; target[target_pos++] = RLE_ENDOFLINE; // return the written size return target_pos; } static BOOL DLL_CALLCONV Save(FreeImageIO *io, FIBITMAP *dib, fi_handle handle, int page, int flags, void *data) { if ((dib != NULL) && (handle != NULL)) { // write the file header BITMAPFILEHEADER bitmapfileheader; bitmapfileheader.bfType = 0x4D42; bitmapfileheader.bfOffBits = sizeof(BITMAPFILEHEADER) + sizeof(BITMAPINFOHEADER) + FreeImage_GetColorsUsed(dib) * sizeof(RGBQUAD); bitmapfileheader.bfSize = bitmapfileheader.bfOffBits + FreeImage_GetHeight(dib) * FreeImage_GetPitch(dib); bitmapfileheader.bfReserved1 = 0; bitmapfileheader.bfReserved2 = 0; // take care of the bit fields data of any bool bit_fields = (FreeImage_GetBPP(dib) == 16); if (bit_fields) { bitmapfileheader.bfSize += 3 * sizeof(DWORD); bitmapfileheader.bfOffBits += 3 * sizeof(DWORD); } #ifdef FREEIMAGE_BIGENDIAN SwapFileHeader(&bitmapfileheader); #endif if (io->write_proc(&bitmapfileheader, sizeof(BITMAPFILEHEADER), 1, handle) != 1) return FALSE; // update the bitmap info header BITMAPINFOHEADER bih; memcpy(&bih, FreeImage_GetInfoHeader(dib), sizeof(BITMAPINFOHEADER)); if (bit_fields) bih.biCompression = BI_BITFIELDS; else if ((bih.biBitCount == 8) && (flags & BMP_SAVE_RLE)) bih.biCompression = BI_RLE8; else bih.biCompression = BI_RGB; // write the bitmap info header #ifdef FREEIMAGE_BIGENDIAN SwapInfoHeader(&bih); #endif if (io->write_proc(&bih, sizeof(BITMAPINFOHEADER), 1, handle) != 1) return FALSE; // write the bit fields when we are dealing with a 16 bit BMP if (bit_fields) { DWORD d; d = FreeImage_GetRedMask(dib); if (io->write_proc(&d, sizeof(DWORD), 1, handle) != 1) return FALSE; d = FreeImage_GetGreenMask(dib); if (io->write_proc(&d, sizeof(DWORD), 1, handle) != 1) return FALSE; d = FreeImage_GetBlueMask(dib); if (io->write_proc(&d, sizeof(DWORD), 1, handle) != 1) return FALSE; } // write the palette if (FreeImage_GetPalette(dib) != NULL) { RGBQUAD *pal = FreeImage_GetPalette(dib); FILE_BGRA bgra; for(unsigned i = 0; i < FreeImage_GetColorsUsed(dib); i++ ) { bgra.b = pal[i].rgbBlue; bgra.g = pal[i].rgbGreen; bgra.r = pal[i].rgbRed; bgra.a = pal[i].rgbReserved; if (io->write_proc(&bgra, sizeof(FILE_BGRA), 1, handle) != 1) return FALSE; } } // write the bitmap data... if RLE compression is enable, use it unsigned bpp = FreeImage_GetBPP(dib); if ((bpp == 8) && (flags & BMP_SAVE_RLE)) { BYTE *buffer = (BYTE*)malloc(FreeImage_GetPitch(dib) * 2 * sizeof(BYTE)); for (DWORD i = 0; i < FreeImage_GetHeight(dib); ++i) { int size = RLEEncodeLine(buffer, FreeImage_GetScanLine(dib, i), FreeImage_GetLine(dib)); if (io->write_proc(buffer, size, 1, handle) != 1) { free(buffer); return FALSE; } } buffer[0] = RLE_COMMAND; buffer[1] = RLE_ENDOFBITMAP; if (io->write_proc(buffer, 2, 1, handle) != 1) { free(buffer); return FALSE; } free(buffer); #ifdef FREEIMAGE_BIGENDIAN } else if (bpp == 16) { int padding = FreeImage_GetPitch(dib) - FreeImage_GetWidth(dib) * sizeof(WORD); WORD pad = 0; WORD pixel; for(unsigned y = 0; y < FreeImage_GetHeight(dib); y++) { BYTE *line = FreeImage_GetScanLine(dib, y); for(unsigned x = 0; x < FreeImage_GetWidth(dib); x++) { pixel = ((WORD *)line)[x]; SwapShort(&pixel); if (io->write_proc(&pixel, sizeof(WORD), 1, handle) != 1) return FALSE; } if(padding != 0) { if(io->write_proc(&pad, padding, 1, handle) != 1) { return FALSE; } } } #endif #if FREEIMAGE_COLORORDER == FREEIMAGE_COLORORDER_RGB } else if (bpp == 24) { int padding = FreeImage_GetPitch(dib) - FreeImage_GetWidth(dib) * sizeof(FILE_BGR); DWORD pad = 0; FILE_BGR bgr; for(unsigned y = 0; y < FreeImage_GetHeight(dib); y++) { BYTE *line = FreeImage_GetScanLine(dib, y); for(unsigned x = 0; x < FreeImage_GetWidth(dib); x++) { RGBTRIPLE *triple = ((RGBTRIPLE *)line)+x; bgr.b = triple->rgbtBlue; bgr.g = triple->rgbtGreen; bgr.r = triple->rgbtRed; if (io->write_proc(&bgr, sizeof(FILE_BGR), 1, handle) != 1) return FALSE; } if(padding != 0) { if(io->write_proc(&pad, padding, 1, handle) != 1) { return FALSE; } } } } else if (bpp == 32) { FILE_BGRA bgra; for(unsigned y = 0; y < FreeImage_GetHeight(dib); y++) { BYTE *line = FreeImage_GetScanLine(dib, y); for(unsigned x = 0; x < FreeImage_GetWidth(dib); x++) { RGBQUAD *quad = ((RGBQUAD *)line)+x; bgra.b = quad->rgbBlue; bgra.g = quad->rgbGreen; bgra.r = quad->rgbRed; bgra.a = quad->rgbReserved; if (io->write_proc(&bgra, sizeof(FILE_BGRA), 1, handle) != 1) return FALSE; } } #endif } else if (io->write_proc(FreeImage_GetBits(dib), FreeImage_GetHeight(dib) * FreeImage_GetPitch(dib), 1, handle) != 1) { return FALSE; } return TRUE; } else { return FALSE; } } // ========================================================== // Init // ========================================================== void DLL_CALLCONV InitBMP(Plugin *plugin, int format_id) { s_format_id = format_id; plugin->format_proc = Format; plugin->description_proc = Description; plugin->extension_proc = Extension; plugin->regexpr_proc = RegExpr; plugin->open_proc = NULL; plugin->close_proc = NULL; plugin->pagecount_proc = NULL; plugin->pagecapability_proc = NULL; plugin->load_proc = Load; plugin->save_proc = Save; plugin->validate_proc = Validate; plugin->mime_proc = MimeType; plugin->supports_export_bpp_proc = SupportsExportDepth; plugin->supports_export_type_proc = SupportsExportType; plugin->supports_icc_profiles_proc = NULL; // not implemented yet; plugin->supports_no_pixels_proc = SupportsNoPixels; }