Improved performance for converting non-compressed BMPs to the Be Bitmap format, reorganized the code and cleaned it up a bit.

git-svn-id: file:///srv/svn/repos/haiku/trunk/current@363 a95241bf-73f2-0310-859d-f6bbb57e9c96
This commit is contained in:
Matthew Wilber 2002-07-21 04:57:49 +00:00
parent 014ff00176
commit ea91930473

View File

@ -212,16 +212,13 @@ status_t identify_bmp_header(BPositionIO *inSource, translator_info *outInfo,
return B_NO_TRANSLATOR; return B_NO_TRANSLATOR;
if (infoHeader.planes != 1) if (infoHeader.planes != 1)
return B_NO_TRANSLATOR; return B_NO_TRANSLATOR;
if (infoHeader.bitsperpixel != 1 && if ((infoHeader.bitsperpixel != 1 || infoHeader.compression != BMP_NO_COMPRESS) &&
infoHeader.bitsperpixel != 4 && (infoHeader.bitsperpixel != 4 || infoHeader.compression != BMP_NO_COMPRESS) &&
infoHeader.bitsperpixel != 8 && (infoHeader.bitsperpixel != 4 || infoHeader.compression != BMP_RLE4_COMPRESS) &&
infoHeader.bitsperpixel != 16 && (infoHeader.bitsperpixel != 8 || infoHeader.compression != BMP_NO_COMPRESS) &&
infoHeader.bitsperpixel != 24 && (infoHeader.bitsperpixel != 8 || infoHeader.compression != BMP_RLE8_COMPRESS) &&
infoHeader.bitsperpixel != 32) (infoHeader.bitsperpixel != 24 || infoHeader.compression != BMP_NO_COMPRESS) &&
return B_NO_TRANSLATOR; (infoHeader.bitsperpixel != 32 || infoHeader.compression != BMP_NO_COMPRESS))
if (infoHeader.compression != BMP_NO_COMPRESS &&
infoHeader.compression != BMP_RLE8_COMPRESS &&
infoHeader.compression != BMP_RLE4_COMPRESS)
return B_NO_TRANSLATOR; return B_NO_TRANSLATOR;
if (infoHeader.colorsimportant > infoHeader.colorsused) if (infoHeader.colorsimportant > infoHeader.colorsused)
return B_NO_TRANSLATOR; return B_NO_TRANSLATOR;
@ -437,89 +434,48 @@ status_t translate_from_bits(BPositionIO *inSource, ssize_t amtread,
return B_NO_TRANSLATOR; return B_NO_TRANSLATOR;
} }
status_t translate_from_bmp24_to_bits(BPositionIO *inSource, BPositionIO *outDestination, status_t translate_from_bmpnpal_to_bits(BPositionIO *inSource, BPositionIO *outDestination,
int32 datasize, BMPInfoHeader &infoHeader) int32 datasize, BMPInfoHeader &infoHeader)
{ {
char *bitspixels = new char[datasize];
if (!bitspixels)
return B_ERROR;
char pixel[4] = { 0xFF, 0xFF, 0xFF, 0xFF };
int32 bitsRowBytes = infoHeader.width * 4; int32 bitsRowBytes = infoHeader.width * 4;
int32 bmpbytesperpixel = infoHeader.bitsperpixel / 8; int32 bmpBytesPerPixel = infoHeader.bitsperpixel / 8;
int32 padding = (infoHeader.width * bmpbytesperpixel) % 4; int32 padding = (infoHeader.width * bmpBytesPerPixel) % 4;
if (padding) if (padding)
padding = 4 - padding; padding = 4 - padding;
uint32 bmppixcol = 0, bmppixrow = 0; int32 bmpRowBytes = (infoHeader.width * bmpBytesPerPixel) + padding;
ssize_t rd = inSource->Read(pixel, bmpbytesperpixel); uint32 bmppixrow = 0;
while (rd > 0) { off_t bmpoffset = ((infoHeader.height - 1) * bmpRowBytes);
int32 bitsoffset = ((infoHeader.height - inSource->Seek(bmpoffset, SEEK_CUR);
(bmppixrow + 1)) * bitsRowBytes) + char *bmpRowData = new char[bmpRowBytes];
(bmppixcol * 4); if (!bmpRowData)
memcpy(bitspixels + bitsoffset, pixel, 4); return B_ERROR;
bmppixcol++; char *bitsRowData = new char[bitsRowBytes];
if (!bitsRowData) {
if (bmppixcol == infoHeader.width) { delete[] bmpRowData;
bmppixcol = 0;
bmppixrow++;
// If padding, skip over the padding
if (padding)
inSource->Seek(padding, SEEK_CUR);
// if I've read all of the pixel data, break
// out of the loop so I don't try to read
// non-pixel data
if (bmppixrow == infoHeader.height)
break;
}
rd = inSource->Read(pixel, bmpbytesperpixel);
}
outDestination->Write(bitspixels, datasize);
delete[] bitspixels;
return B_OK;
}
status_t translate_from_bmp8_to_bits(BPositionIO *inSource, BPositionIO *outDestination,
int32 datasize, BMPInfoHeader &infoHeader, const char *palette)
{
uint8 index;
char *bitspixels = new char[datasize];
if (!bitspixels)
return B_ERROR; return B_ERROR;
memset(bitspixels, 0xffffffff, datasize);
int32 padding = infoHeader.width % 4;
if (padding)
padding = 4 - padding;
int32 bitsRowBytes = infoHeader.width * 4;
uint32 bmppixcol = 0, bmppixrow = 0;
ssize_t rd = inSource->Read(&index, 1);
while (rd > 0) {
int32 bitsoffset = ((infoHeader.height -
(bmppixrow + 1)) * bitsRowBytes) +
(bmppixcol * 4);
memcpy(bitspixels + bitsoffset, palette + (index * 4), 3);
bmppixcol++;
if (bmppixcol == infoHeader.width) {
bmppixcol = 0;
bmppixrow++;
// If padding, read in the bytes of padding
if (padding)
inSource->Seek(padding, SEEK_CUR);
// if I've read all of the pixel data, break
// out of the loop so I don't try to read
// non-pixel data
if (bmppixrow == infoHeader.height)
break;
}
rd = inSource->Read(&index, 1);
} }
outDestination->Write(bitspixels, datasize); memset(bitsRowData, 0xffffffff, bitsRowBytes);
ssize_t rd = inSource->Read(bmpRowData, bmpRowBytes);
while (rd == bmpRowBytes) {
for (uint32 i = 0; i < infoHeader.width; i++)
memcpy(bitsRowData + (i * 4),
bmpRowData + (i * bmpBytesPerPixel), 3);
outDestination->Write(bitsRowData, bitsRowBytes);
bmppixrow++;
// if I've read all of the pixel data, break
// out of the loop so I don't try to read
// non-pixel data
if (bmppixrow == infoHeader.height)
break;
inSource->Seek(-(bmpRowBytes * 2), SEEK_CUR);
rd = inSource->Read(bmpRowData, bmpRowBytes);
}
delete[] bmpRowData;
delete[] bitsRowData;
delete[] bitspixels;
return B_OK; return B_OK;
} }
@ -675,64 +631,6 @@ status_t translate_from_bmp8r_to_bits(BPositionIO *inSource, BPositionIO *outDes
return B_OK; return B_OK;
} }
status_t translate_from_bmp4_to_bits(BPositionIO *inSource, BPositionIO *outDestination,
int32 datasize, BMPInfoHeader &infoHeader, const char *palette)
{
uint8 indices;
char *bitspixels = new char[datasize];
if (!bitspixels)
return B_ERROR;
memset(bitspixels, 0xffffffff, datasize);
int32 padding;
if (!(infoHeader.width % 2))
padding = (infoHeader.width / 2) % 4;
else
padding = ((infoHeader.width + 2 - (infoHeader.width % 2)) / 2) % 4;
if (padding)
padding = 4 - padding;
int32 bitsRowBytes = infoHeader.width * 4;
uint32 bmppixcol = 0, bmppixrow = 0;
ssize_t rd = inSource->Read(&indices, 1);
while (rd > 0) {
uint32 hbytesUsed = min(2,
infoHeader.width - bmppixcol);
for (uint32 i = 0; i < hbytesUsed; i++) {
uint8 index;
if (!i)
index = (indices>>4) & 0xf;
else
index = indices & 0xf;
// should be either zero or one
int32 bitsoffset = ((infoHeader.height -
(bmppixrow + 1)) * bitsRowBytes) +
(bmppixcol * 4);
memcpy(bitspixels + bitsoffset, palette + (index * 4), 3);
bmppixcol++;
}
if (bmppixcol == infoHeader.width) {
bmppixcol = 0;
bmppixrow++;
// If padding, read in the bytes of padding
if (padding)
inSource->Seek(padding, SEEK_CUR);
// if I've read all of the pixel data, break
// out of the loop so I don't try to read
// non-pixel data
if (bmppixrow == infoHeader.height)
break;
}
rd = inSource->Read(&indices, 1);
}
outDestination->Write(bitspixels, datasize);
delete[] bitspixels;
return B_OK;
}
status_t translate_from_bmp4r_to_bits(BPositionIO *inSource, BPositionIO *outDestination, status_t translate_from_bmp4r_to_bits(BPositionIO *inSource, BPositionIO *outDestination,
int32 datasize, BMPInfoHeader &infoHeader, const char *palette) int32 datasize, BMPInfoHeader &infoHeader, const char *palette)
{ {
@ -899,57 +797,68 @@ status_t translate_from_bmp4r_to_bits(BPositionIO *inSource, BPositionIO *outDes
return B_OK; return B_OK;
} }
status_t translate_from_bmp1_to_bits(BPositionIO *inSource, BPositionIO *outDestination, status_t translate_from_bmppal_to_bits(BPositionIO *inSource, BPositionIO *outDestination,
int32 datasize, BMPInfoHeader &infoHeader, const char *palette) int32 datasize, BMPInfoHeader &infoHeader, const char *palette)
{ {
uint8 indices; uint16 pixelsPerByte = 8 / infoHeader.bitsperpixel;
char *bitspixels = new char[datasize]; uint16 bitsPerPixel = infoHeader.bitsperpixel;
if (!bitspixels)
return B_ERROR; uint8 mask = 1;
memset(bitspixels, 0xffffffff, datasize); for (uint16 i = 0; i < bitsPerPixel; i++)
mask *= 2;
mask -= 1;
int32 padding; int32 padding;
if (!(infoHeader.width % 8)) if (!(infoHeader.width % pixelsPerByte))
padding = (infoHeader.width / 8) % 4; padding = (infoHeader.width / pixelsPerByte) % 4;
else else
padding = ((infoHeader.width + 8 - (infoHeader.width % 8)) / 8) % 4; padding = ((infoHeader.width + pixelsPerByte -
(infoHeader.width % pixelsPerByte)) / pixelsPerByte) % 4;
if (padding) if (padding)
padding = 4 - padding; padding = 4 - padding;
int32 bmpRowBytes = (infoHeader.width / pixelsPerByte) +
((infoHeader.width % pixelsPerByte) ? 1 : 0) + padding;
uint32 bmppixrow = 0;
off_t bmpoffset = ((infoHeader.height - 1) * bmpRowBytes);
inSource->Seek(bmpoffset, SEEK_CUR);
char *bmpRowData = new char[bmpRowBytes];
if (!bmpRowData)
return B_ERROR;
int32 bitsRowBytes = infoHeader.width * 4; int32 bitsRowBytes = infoHeader.width * 4;
uint32 bmppixcol = 0, bmppixrow = 0; char *bitsRowData = new char[bitsRowBytes];
ssize_t rd = inSource->Read(&indices, 1); if (!bitsRowData) {
while (rd > 0) { delete[] bmpRowData;
uint32 bitsUsed = min(8, return B_ERROR;
infoHeader.width - bmppixcol);
for (uint32 i = 0; i < bitsUsed; i++) {
uint8 index = (indices>>(7 - i)) & 1;
// should be either zero or one
int32 bitsoffset = ((infoHeader.height -
(bmppixrow + 1)) * bitsRowBytes) +
(bmppixcol * 4);
memcpy(bitspixels + bitsoffset, palette + (index * 4), 3);
bmppixcol++;
}
if (bmppixcol == infoHeader.width) {
bmppixcol = 0;
bmppixrow++;
// If padding, read in the bytes of padding
if (padding )
inSource->Seek(padding, SEEK_CUR);
// if I've read all of the pixel data, break
// out of the loop so I don't try to read
// non-pixel data
if (bmppixrow == infoHeader.height)
break;
}
rd = inSource->Read(&indices, 1);
} }
outDestination->Write(bitspixels, datasize); memset(bitsRowData, 0xffffffff, bitsRowBytes);
ssize_t rd = inSource->Read(bmpRowData, bmpRowBytes);
while (rd == bmpRowBytes) {
for (uint32 i = 0; i < infoHeader.width; i++) {
uint8 indices = (bmpRowData + (i / pixelsPerByte))[0];
uint8 index;
index = (indices >>
(bitsPerPixel * ((pixelsPerByte - 1) - (i % pixelsPerByte)))) & mask;
memcpy(bitsRowData + (i * 4),
palette + (4 * index), 3);
}
outDestination->Write(bitsRowData, bitsRowBytes);
bmppixrow++;
// if I've read all of the pixel data, break
// out of the loop so I don't try to read
// non-pixel data
if (bmppixrow == infoHeader.height)
break;
inSource->Seek(-(bmpRowBytes * 2), SEEK_CUR);
rd = inSource->Read(bmpRowData, bmpRowBytes);
}
delete[] bmpRowData;
delete[] bitsRowData;
delete[] bitspixels;
return B_OK; return B_OK;
} }
@ -1088,13 +997,13 @@ status_t translate_from_bmp(BPositionIO *inSource, ssize_t amtread,
switch (infoHeader.bitsperpixel) { switch (infoHeader.bitsperpixel) {
case 32: case 32:
case 24: case 24:
return translate_from_bmp24_to_bits(inSource, outDestination, return translate_from_bmpnpal_to_bits(inSource, outDestination,
datasize, infoHeader); datasize, infoHeader);
case 8: case 8:
// 8 bit BMP with NO compression // 8 bit BMP with NO compression
if (infoHeader.compression == BMP_NO_COMPRESS) if (infoHeader.compression == BMP_NO_COMPRESS)
return translate_from_bmp8_to_bits(inSource, return translate_from_bmppal_to_bits(inSource,
outDestination, datasize, infoHeader, bmppalette); outDestination, datasize, infoHeader, bmppalette);
// 8 bit RLE compressed BMP // 8 bit RLE compressed BMP
@ -1107,7 +1016,7 @@ status_t translate_from_bmp(BPositionIO *inSource, ssize_t amtread,
case 4: case 4:
// 4 bit BMP with NO compression // 4 bit BMP with NO compression
if (!infoHeader.compression) if (!infoHeader.compression)
return translate_from_bmp4_to_bits(inSource, return translate_from_bmppal_to_bits(inSource,
outDestination, datasize, infoHeader, bmppalette); outDestination, datasize, infoHeader, bmppalette);
// 4 bit RLE compressed BMP // 4 bit RLE compressed BMP
@ -1118,7 +1027,7 @@ status_t translate_from_bmp(BPositionIO *inSource, ssize_t amtread,
return B_NO_TRANSLATOR; return B_NO_TRANSLATOR;
case 1: case 1:
return translate_from_bmp1_to_bits(inSource, return translate_from_bmppal_to_bits(inSource,
outDestination, datasize, infoHeader, bmppalette); outDestination, datasize, infoHeader, bmppalette);
default: default: