1.1 --- /dev/null Thu Jan 01 00:00:00 1970 +0000
1.2 +++ b/os/graphics/graphicstools/gdi_tools/bmconv/PBMTOBM.CPP Fri Jun 15 03:10:57 2012 +0200
1.3 @@ -0,0 +1,815 @@
1.4 +// Copyright (c) 1997-2009 Nokia Corporation and/or its subsidiary(-ies).
1.5 +// All rights reserved.
1.6 +// This component and the accompanying materials are made available
1.7 +// under the terms of "Eclipse Public License v1.0"
1.8 +// which accompanies this distribution, and is available
1.9 +// at the URL "http://www.eclipse.org/legal/epl-v10.html".
1.10 +//
1.11 +// Initial Contributors:
1.12 +// Nokia Corporation - initial contribution.
1.13 +//
1.14 +// Contributors:
1.15 +//
1.16 +// Description:
1.17 +//
1.18 +
1.19 +#include <assert.h>
1.20 +#include "BMCONV.H"
1.21 +
1.22 +EpocLoader::EpocLoader():
1.23 + iPbmBits(NULL)
1.24 + {}
1.25 +
1.26 +EpocLoader::~EpocLoader()
1.27 + {
1.28 + delete iPbmBits;
1.29 + }
1.30 +
1.31 +int EpocLoader::EpocBitmapCount(char* aFileName, int& aCount, int& isRomFormat)
1.32 + {
1.33 +#if defined(__MSVCDOTNET__) || defined(__TOOLS2__)
1.34 + fstream file(aFileName, ios::in | ios::binary);
1.35 +#else //!__MSVCDOTNET__
1.36 + fstream file(aFileName, ios::in | ios::binary | ios::nocreate);
1.37 +#endif //__MSVCDOTNET__
1.38 +
1.39 + if (file.is_open()==0)
1.40 + return Files;
1.41 +
1.42 + long int wordbuffer;
1.43 + file.read((char *)&wordbuffer,4);
1.44 + if (file.gcount()!=4)
1.45 + return SourceFile;
1.46 + if (wordbuffer==KMultiBitmapRomImageUid)
1.47 + {
1.48 + // ROM format file - next 4 bytes are the number of bitmaps
1.49 + isRomFormat=1;
1.50 + }
1.51 + else
1.52 + {
1.53 + if (wordbuffer!=KWriteOnceFileStoreUid)
1.54 + return SourceFile;
1.55 + isRomFormat=0;
1.56 + file.read((char *)&wordbuffer,4);
1.57 + if (file.gcount()!=4 || wordbuffer!=KMultiBitmapFileImageUid)
1.58 + return SourceFile;
1.59 + file.read((char *)&wordbuffer,4);
1.60 + if (file.gcount()!=4 || wordbuffer!=0)
1.61 + return SourceFile;
1.62 + file.read((char *)&wordbuffer,4);
1.63 + if (file.gcount()!=4 || wordbuffer!=KMultiBitmapFileImageChecksum)
1.64 + return SourceFile;
1.65 + file.read((char *)&wordbuffer,4);
1.66 + if (file.gcount()!=4)
1.67 + return SourceFile;
1.68 + file.seekg(wordbuffer,ios::beg);
1.69 + }
1.70 + file.read((char *)&wordbuffer,4);
1.71 + if (file.gcount()!=4)
1.72 + return SourceFile;
1.73 + aCount=wordbuffer;
1.74 + return NoError;
1.75 + }
1.76 +
1.77 +int EpocLoader::LoadEpocBitmap(char* aFileName,int aIndex)
1.78 + {
1.79 +#if defined(__MSVCDOTNET__) || defined(__TOOLS2__)
1.80 + fstream file(aFileName, ios::in | ios::binary);
1.81 +#else //!__MSVCDOTNET__
1.82 + fstream file(aFileName, ios::in | ios::binary | ios::nocreate);
1.83 +#endif //__MSVCDOTNET__
1.84 +
1.85 + if (file.is_open()==0)
1.86 + return Files;
1.87 +
1.88 + long int wordbuffer;
1.89 + file.read((char *)&wordbuffer,4);
1.90 + if (file.gcount()!=4)
1.91 + return SourceFile;
1.92 + file.close();
1.93 +
1.94 + if (wordbuffer==KMultiBitmapRomImageUid)
1.95 + return LoadRom(aFileName,aIndex);
1.96 +
1.97 + return LoadFile(aFileName,aIndex);
1.98 + }
1.99 +
1.100 +int EpocLoader::LoadFile(char* aFileName,int aIndex)
1.101 + {
1.102 +#if defined(__MSVCDOTNET__) || defined(__TOOLS2__)
1.103 + fstream file(aFileName, ios::in | ios::binary);
1.104 +#else //!__MSVCDOTNET__
1.105 + fstream file(aFileName, ios::in | ios::binary | ios::nocreate);
1.106 +#endif //__MSVCDOTNET__
1.107 +
1.108 + if (file.is_open()==0)
1.109 + return Files;
1.110 +
1.111 + long int wordbuffer;
1.112 + file.read((char *)&wordbuffer,4);
1.113 + if (file.gcount()!=4 || wordbuffer!=KWriteOnceFileStoreUid)
1.114 + return SourceFile;
1.115 + file.read((char *)&wordbuffer,4);
1.116 + if (file.gcount()!=4 || wordbuffer!=KMultiBitmapFileImageUid)
1.117 + return SourceFile;
1.118 + file.read((char *)&wordbuffer,4);
1.119 + if (file.gcount()!=4 || wordbuffer!=0)
1.120 + return SourceFile;
1.121 + file.read((char *)&wordbuffer,4);
1.122 + if (file.gcount()!=4 || wordbuffer!=KMultiBitmapFileImageChecksum)
1.123 + return SourceFile;
1.124 + file.read((char *)&wordbuffer,4);
1.125 + if (file.gcount()!=4)
1.126 + return SourceFile;
1.127 + file.seekg(wordbuffer,ios::beg);
1.128 + file.read((char *)&wordbuffer,4);
1.129 + if (file.gcount()!=4)
1.130 + return SourceFile;
1.131 +
1.132 + int numsources=wordbuffer;
1.133 + if (aIndex >= numsources)
1.134 + return OutOfRange;
1.135 + file.seekg(aIndex*4,ios::cur);
1.136 + file.read((char *)&wordbuffer,4);
1.137 + if (file.gcount()!=4)
1.138 + return SourceFile;
1.139 + file.seekg(wordbuffer,ios::beg);
1.140 + int ret = DoLoadFile(file);
1.141 + file.close();
1.142 + return ret;
1.143 + }
1.144 +
1.145 +int EpocLoader::LoadRom(char* aFileName,int aIndex)
1.146 + {
1.147 +#if defined(__MSVCDOTNET__) || defined(__TOOLS2__)
1.148 + fstream file(aFileName, ios::in | ios::binary);
1.149 +#else //!__MSVCDOTNET__
1.150 + fstream file(aFileName, ios::in | ios::binary | ios::nocreate);
1.151 +#endif //__MSVCDOTNET__
1.152 +
1.153 + if (file.is_open()==0)
1.154 + return Files;
1.155 +
1.156 + long int wordbuffer;
1.157 + file.read((char *)&wordbuffer,4);
1.158 + if (file.gcount()!=4)
1.159 + return SourceFile;
1.160 + if (wordbuffer!=KMultiBitmapRomImageUid)
1.161 + return SourceFile;
1.162 + file.read((char *)&wordbuffer,4);
1.163 + if (file.gcount()!=4)
1.164 + return SourceFile;
1.165 + if (aIndex>=wordbuffer)
1.166 + return OutOfRange;
1.167 + file.seekg(aIndex*4,ios::cur);
1.168 + file.read((char *)&wordbuffer,4);
1.169 + if (file.gcount()!=4)
1.170 + return SourceFile;
1.171 + file.seekg(wordbuffer,ios::beg);
1.172 + int ret=DoLoadRom(file);
1.173 + file.close();
1.174 + return ret;
1.175 + }
1.176 +
1.177 +int EpocLoader::DoLoadFile(fstream& aFile)
1.178 + {
1.179 + long size = sizeof(SEpocBitmapHeader);
1.180 + aFile.read((char *)&iPbmHeader,size);
1.181 + if (aFile.gcount()!=size)
1.182 + return SourceFile;
1.183 + iOriginalPbmHeader = iPbmHeader;
1.184 + size=iPbmHeader.iBitmapSize-iPbmHeader.iStructSize;
1.185 +
1.186 + iPbmBits=new char[size];
1.187 + if (iPbmBits==NULL)
1.188 + return NoMemory;
1.189 + memset(iPbmBits,0xff,size);
1.190 + aFile.read(iPbmBits,size);
1.191 + aFile.close();
1.192 + if (aFile.gcount() != size)
1.193 + return SourceFile;
1.194 +
1.195 + if (iPbmHeader.iCompression != ENoBitmapCompression)
1.196 + {
1.197 + return Decompress(size);
1.198 + }
1.199 +
1.200 + return NoError;
1.201 + }
1.202 +
1.203 +int EpocLoader::DoLoadRom(fstream& aFile)
1.204 + {
1.205 + Bitmap bmp;
1.206 + long size=sizeof(Bitmap);
1.207 + aFile.read((char*)&bmp,size);
1.208 + if (aFile.gcount() != size)
1.209 + return SourceFile;
1.210 +
1.211 + size = bmp.iHeader.iBitmapSize - sizeof(SEpocBitmapHeader);
1.212 + iPbmBits = new char[size];
1.213 + if (iPbmBits == NULL)
1.214 + return NoMemory;
1.215 + memset(iPbmBits,0xff,size);
1.216 +
1.217 + aFile.read(iPbmBits,size);
1.218 + if (aFile.gcount() != size)
1.219 + return SourceFile;
1.220 + iPbmHeader = bmp.iHeader;
1.221 + iOriginalPbmHeader = iPbmHeader;
1.222 +
1.223 + if (iPbmHeader.iCompression != ENoBitmapCompression)
1.224 + {
1.225 + return Decompress(size);
1.226 + }
1.227 +
1.228 + return NoError;
1.229 + }
1.230 +
1.231 +int EpocLoader::Decompress(int aSize)
1.232 + {
1.233 + int byteWidth = BitmapUtils::ByteWidth(iPbmHeader.iWidthInPixels,iPbmHeader.iBitsPerPixel);
1.234 + int expandedsize = byteWidth * iPbmHeader.iHeightInPixels;
1.235 + char* newbits = new char[expandedsize];
1.236 + if (newbits == NULL)
1.237 + return NoMemory;
1.238 + memset(newbits,0xff,expandedsize);
1.239 + int ret = NoError;
1.240 + switch (iPbmHeader.iCompression)
1.241 + {
1.242 + case EByteRLECompression:
1.243 + ret = ExpandByteRLEData(newbits,expandedsize,iPbmBits,aSize);
1.244 + break;
1.245 + case ETwelveBitRLECompression:
1.246 + ret = ExpandTwelveBitRLEData(newbits,expandedsize,iPbmBits,aSize);
1.247 + break;
1.248 + case ESixteenBitRLECompression:
1.249 + ret = ExpandSixteenBitRLEData(newbits,expandedsize,iPbmBits,aSize);
1.250 + break;
1.251 + case ETwentyFourBitRLECompression:
1.252 + ret = ExpandTwentyFourBitRLEData(newbits,expandedsize,iPbmBits,aSize);
1.253 + break;
1.254 + case EThirtyTwoUBitRLECompression:
1.255 + ret = ExpandThirtyTwoUBitRLEData(newbits,expandedsize,iPbmBits,aSize);
1.256 + break;
1.257 + case EThirtyTwoABitRLECompression:
1.258 + ret = ExpandThirtyTwoABitRLEData(newbits,expandedsize,iPbmBits,aSize);
1.259 + break;
1.260 + default:
1.261 + ret = UnknownCompression;
1.262 + break;
1.263 + }
1.264 + delete iPbmBits;
1.265 + iPbmBits = newbits;
1.266 + iPbmHeader.iCompression = ENoBitmapCompression;
1.267 + iPbmHeader.iBitmapSize += expandedsize-aSize;
1.268 + return ret;
1.269 + }
1.270 +
1.271 +int EpocLoader::ExpandByteRLEData(char* aDest,int aDestSize,char* aSrce,int aSrceSize)
1.272 + {
1.273 + char* srcelimit=aSrce+aSrceSize;
1.274 + char* destlimit=aDest+aDestSize;
1.275 + while(aSrce<srcelimit && aDest<destlimit)
1.276 + {
1.277 + char count=*aSrce++;
1.278 + if (count<0)
1.279 + {
1.280 + int runLength=-count;
1.281 + memcpy(aDest,aSrce,runLength);
1.282 + aSrce+=runLength;
1.283 + aDest+=runLength;
1.284 + }
1.285 + else
1.286 + {
1.287 + char value=*aSrce++;
1.288 + while(count>=0)
1.289 + {
1.290 + *aDest++=value;
1.291 + count--;
1.292 + }
1.293 + }
1.294 + }
1.295 + if (aSrce!=srcelimit || aDest!=destlimit)
1.296 + return DecompressionError;
1.297 + return NoError;
1.298 + }
1.299 +
1.300 +int EpocLoader::ExpandTwelveBitRLEData(char* aDest,int aDestSizeInBytes,char* aSrce,int aSrceSizeInBytes)
1.301 + {
1.302 + unsigned short* srcePtr = (unsigned short*)aSrce;
1.303 + unsigned short* destPtr = (unsigned short*)aDest;
1.304 + unsigned short* srcePtrLimit = srcePtr + (aSrceSizeInBytes / 2);
1.305 + unsigned short* destPtrLimit = destPtr + (aDestSizeInBytes / 2);
1.306 +
1.307 + while (srcePtr < srcePtrLimit && destPtr < destPtrLimit)
1.308 + {
1.309 + unsigned short value = *srcePtr++;
1.310 + int runLength = (value >> 12) + 1;
1.311 + value &= 0x0fff;
1.312 +
1.313 + for (;runLength > 0;runLength--)
1.314 + *destPtr++ = value;
1.315 + }
1.316 +
1.317 + if (srcePtr != srcePtrLimit || destPtr != destPtrLimit)
1.318 + return DecompressionError;
1.319 +
1.320 + return NoError;
1.321 + }
1.322 +
1.323 +int EpocLoader::ExpandSixteenBitRLEData(char* aDest,int aDestSizeInBytes,char* aSrce,int aSrceSizeInBytes)
1.324 + {
1.325 + char* srcePtrLimit = aSrce + aSrceSizeInBytes;
1.326 + unsigned short* destPtr = (unsigned short*)aDest;
1.327 + unsigned short* destPtrLimit = destPtr + (aDestSizeInBytes / 2);
1.328 +
1.329 + while (aSrce < srcePtrLimit && destPtr < destPtrLimit)
1.330 + {
1.331 + int runLength = *aSrce++;
1.332 +
1.333 + if (runLength >= 0)
1.334 + {
1.335 + unsigned short value = *((unsigned short*)(aSrce));
1.336 + aSrce += 2;
1.337 + for (runLength++; runLength > 0; runLength--)
1.338 + *destPtr++ = value;
1.339 + }
1.340 + else
1.341 + {
1.342 + runLength = -runLength;
1.343 + int byteLength = runLength * 2;
1.344 + memcpy(destPtr,aSrce,byteLength);
1.345 + aSrce += byteLength;
1.346 + destPtr += runLength;
1.347 + }
1.348 + }
1.349 +
1.350 + if (aSrce != srcePtrLimit || destPtr != destPtrLimit)
1.351 + return DecompressionError;
1.352 +
1.353 + return NoError;
1.354 + }
1.355 +
1.356 +int EpocLoader::ExpandTwentyFourBitRLEData(char* aDest,int aDestSizeInBytes,char* aSrce,int aSrceSizeInBytes)
1.357 + {
1.358 + char* srcePtrLimit = aSrce + aSrceSizeInBytes;
1.359 + char* destPtrLimit = aDest + aDestSizeInBytes;
1.360 +
1.361 + while (aSrce < srcePtrLimit && aDest < destPtrLimit)
1.362 + {
1.363 + int runLength = *aSrce++;
1.364 +
1.365 + if (runLength >= 0)
1.366 + {
1.367 + char component1 = *aSrce++;
1.368 + char component2 = *aSrce++;
1.369 + char component3 = *aSrce++;
1.370 + for (runLength++; runLength > 0; runLength--)
1.371 + {
1.372 + *aDest++ = component1;
1.373 + *aDest++ = component2;
1.374 + *aDest++ = component3;
1.375 + }
1.376 + }
1.377 + else
1.378 + {
1.379 + runLength = -runLength;
1.380 + int byteLength = runLength * 3;
1.381 + memcpy(aDest,aSrce,byteLength);
1.382 + aSrce += byteLength;
1.383 + aDest += byteLength;
1.384 + }
1.385 + }
1.386 +
1.387 + if (aSrce != srcePtrLimit || aDest != destPtrLimit)
1.388 + return DecompressionError;
1.389 +
1.390 + return NoError;
1.391 + }
1.392 +
1.393 +/** The function decodes 24-bit compressed pixel buffer into the 32-bit buffer, where top bytes are unused*/
1.394 +int EpocLoader::ExpandThirtyTwoUBitRLEData(char* aDest,int aDestSizeInBytes,char* aSrce,int aSrceSizeInBytes)
1.395 + {
1.396 + char* srcePtrLimit = aSrce + aSrceSizeInBytes;
1.397 + char* destPtrLimit = aDest + aDestSizeInBytes;
1.398 +
1.399 + while (aSrce < srcePtrLimit && aDest < destPtrLimit)
1.400 + {
1.401 + int runLength = *aSrce++;
1.402 +
1.403 + if (runLength >= 0)
1.404 + {
1.405 + char component1 = *aSrce++;
1.406 + char component2 = *aSrce++;
1.407 + char component3 = *aSrce++;
1.408 + for (runLength++; runLength > 0; runLength--)
1.409 + {
1.410 + *aDest++ = component1;
1.411 + *aDest++ = component2;
1.412 + *aDest++ = component3;
1.413 + aDest++;
1.414 + }
1.415 + }
1.416 + else
1.417 + {
1.418 + runLength = -runLength;
1.419 + for(int ii = 0; ii < runLength; ii++)
1.420 + {
1.421 + memcpy(aDest,aSrce,3);
1.422 + aSrce += 3;
1.423 + aDest += 4;
1.424 + }
1.425 + }
1.426 + }
1.427 +
1.428 + if (aSrce != srcePtrLimit || aDest != destPtrLimit)
1.429 + return DecompressionError;
1.430 +
1.431 + return NoError;
1.432 + }
1.433 +
1.434 +/** The function decodes 32-bit compressed pixel buffer into the 32-bit buffer, where top bytes are alpha channel*/
1.435 +int EpocLoader::ExpandThirtyTwoABitRLEData(char* aDest,int aDestSizeInBytes,char* aSrce,int aSrceSizeInBytes)
1.436 + {
1.437 + char* srcePtrLimit = aSrce + aSrceSizeInBytes;
1.438 + char* destPtrLimit = aDest + aDestSizeInBytes;
1.439 +
1.440 + while (aSrce < srcePtrLimit && aDest < destPtrLimit)
1.441 + {
1.442 + int runLength = *aSrce++;
1.443 +
1.444 + if (runLength >= 0)
1.445 + {
1.446 + char component1 = *aSrce++;
1.447 + char component2 = *aSrce++;
1.448 + char component3 = *aSrce++;
1.449 + char component4 = *aSrce++;
1.450 + for (runLength++; runLength > 0; runLength--)
1.451 + {
1.452 + *aDest++ = component1;
1.453 + *aDest++ = component2;
1.454 + *aDest++ = component3;
1.455 + *aDest++ = component4;
1.456 + }
1.457 + }
1.458 + else
1.459 + {
1.460 + runLength = -runLength;
1.461 + memcpy(aDest,aSrce,4*runLength);
1.462 + aSrce += 4*runLength;
1.463 + aDest += 4*runLength;
1.464 + }
1.465 + }
1.466 +
1.467 + if (aSrce != srcePtrLimit || aDest != destPtrLimit)
1.468 + return DecompressionError;
1.469 +
1.470 + return NoError;
1.471 + }
1.472 +
1.473 +TRgb EpocLoader::GetPixel(int aXCoord,int aYCoord)
1.474 + {
1.475 + unsigned char col;
1.476 + aXCoord%=iPbmHeader.iWidthInPixels;
1.477 + aYCoord%=iPbmHeader.iHeightInPixels;
1.478 + int byteWidth = BitmapUtils::ByteWidth(iPbmHeader.iWidthInPixels,iPbmHeader.iBitsPerPixel);
1.479 + int yOffset = aYCoord * byteWidth;
1.480 +
1.481 + switch(iPbmHeader.iBitsPerPixel)
1.482 + {
1.483 + case 1:
1.484 + col = iPbmBits[yOffset + (aXCoord / 8)];
1.485 + col >>= (aXCoord&7);
1.486 + return TRgb::Gray2(col & 1);
1.487 + case 2:
1.488 + col = iPbmBits[yOffset + (aXCoord / 4)];
1.489 + col = (unsigned char)(col>>(2*(aXCoord%4)));
1.490 + return TRgb::Gray4(col & 3);
1.491 + case 4:
1.492 + col = iPbmBits[yOffset + (aXCoord / 2)];
1.493 + if (aXCoord & 1)
1.494 + col >>= 4;
1.495 + col &= 0xf;
1.496 + if (iPbmHeader.iColor==EColorBitmap)
1.497 + return TRgb::Color16(col);
1.498 + else
1.499 + return TRgb::Gray16(col);
1.500 + case 8:
1.501 + col=iPbmBits[yOffset + aXCoord];
1.502 + if (iPbmHeader.iColor==EColorBitmap)
1.503 + return TRgb::Color256(col);
1.504 + else
1.505 + return TRgb::Gray256(col);
1.506 + case 12:
1.507 + case 16:
1.508 + {
1.509 + unsigned short* shortPtr = (unsigned short*)&iPbmBits[yOffset + (aXCoord * 2)];
1.510 + if (iPbmHeader.iBitsPerPixel == 12)
1.511 + return TRgb::Color4K(*shortPtr);
1.512 + else
1.513 + return TRgb::Color64K(*shortPtr);
1.514 + }
1.515 + case 24:
1.516 + {
1.517 + char* pixelPtr = iPbmBits + yOffset + (aXCoord * 3);
1.518 + TRgb pixelColor;
1.519 + pixelColor.iBlue = *pixelPtr++;
1.520 + pixelColor.iGreen = *pixelPtr++;
1.521 + pixelColor.iRed = *pixelPtr;
1.522 + return pixelColor;
1.523 + }
1.524 + case 32:
1.525 + {
1.526 + char* pixelPtr = iPbmBits + yOffset + (aXCoord * 4);
1.527 + TRgb pixelColor;
1.528 + pixelColor.iBlue = *pixelPtr++;
1.529 + pixelColor.iGreen = *pixelPtr++;
1.530 + pixelColor.iRed = *pixelPtr++;
1.531 + pixelColor.iSpare = *pixelPtr;
1.532 + return pixelColor;
1.533 + }
1.534 + default:
1.535 + return TRgb(0);
1.536 + }
1.537 + }
1.538 +
1.539 +unsigned char EpocLoader::GetAlpha(int aXCoord,int aYCoord)
1.540 + {
1.541 + aXCoord%=iPbmHeader.iWidthInPixels;
1.542 + aYCoord%=iPbmHeader.iHeightInPixels;
1.543 + int byteWidth = BitmapUtils::ByteWidth(iPbmHeader.iWidthInPixels,iPbmHeader.iBitsPerPixel);
1.544 + int yOffset = aYCoord * byteWidth;
1.545 +
1.546 + assert(iPbmHeader.iBitsPerPixel == 32); // this method must only be called for 32bpp bitmaps!
1.547 +
1.548 + char* pixelPtr = iPbmBits + yOffset + (aXCoord * 4);
1.549 + pixelPtr += 3; // skip over RGB
1.550 + unsigned char col = *pixelPtr;
1.551 + return col;
1.552 + }
1.553 +
1.554 +int EpocLoader::SaveBitmap(char* aFileName)
1.555 + {
1.556 + TBitmapFileHeader fileheader;
1.557 + TBitmapInfoHeader bmpHeader;
1.558 + char* bmpBits;
1.559 +
1.560 + bmpHeader.biSize = sizeof(TBitmapInfoHeader);
1.561 + bmpHeader.biWidth = iPbmHeader.iWidthInPixels;
1.562 + bmpHeader.biHeight = iPbmHeader.iHeightInPixels;
1.563 + bmpHeader.biPlanes = 1;
1.564 + bmpHeader.biBitCount = 24;
1.565 + bmpHeader.biCompression = 0;
1.566 + bmpHeader.biSizeImage = 0;
1.567 + bmpHeader.biXPelsPerMeter = 0;
1.568 + bmpHeader.biYPelsPerMeter = 0;
1.569 + bmpHeader.biClrUsed = 0;
1.570 + bmpHeader.biClrImportant = 0;
1.571 +
1.572 + long byteWidth = ((bmpHeader.biWidth * 3) + 3) & ~3;
1.573 + long destlength = bmpHeader.biHeight * byteWidth;
1.574 +
1.575 + fileheader.bfType = 'B'+('M'<<8);
1.576 + fileheader.bfSize = sizeof(TBitmapFileHeader)+sizeof(TBitmapInfoHeader)+destlength;
1.577 + fileheader.bfReserved1 = 0;
1.578 + fileheader.bfReserved2 = 0;
1.579 + fileheader.bfOffBits = sizeof(TBitmapFileHeader)+sizeof(TBitmapInfoHeader);
1.580 +
1.581 + bmpBits = new char[destlength];
1.582 + if (bmpBits == NULL)
1.583 + return NoMemory;
1.584 + memset(bmpBits,0xff,destlength);
1.585 +
1.586 + for(long y=0;y<bmpHeader.biHeight;y++)
1.587 + {
1.588 + char* dest=&bmpBits[(bmpHeader.biHeight-y-1)*byteWidth];
1.589 + for(long x=0;x<bmpHeader.biWidth;x++)
1.590 + {
1.591 + TRgb pixel=GetPixel(x,y);
1.592 + *dest++=pixel.iBlue;
1.593 + *dest++=pixel.iGreen;
1.594 + *dest++=pixel.iRed;
1.595 + }
1.596 + }
1.597 +
1.598 + fstream file(aFileName, ios::out | ios::binary);
1.599 +
1.600 + if (file.is_open()==0)
1.601 + {
1.602 + delete bmpBits;
1.603 + return DestFile;
1.604 + }
1.605 +
1.606 + file.write((char *)&fileheader,sizeof(TBitmapFileHeader));
1.607 + file.write((char *)&bmpHeader,sizeof(TBitmapInfoHeader));
1.608 + file.write((char *)bmpBits,destlength);
1.609 + file.close();
1.610 +
1.611 + delete bmpBits;
1.612 +
1.613 + if (iPbmHeader.iColor == EColorBitmapAlpha || iPbmHeader.iColor ==EColorBitmapAlphaPM)
1.614 + SaveAlpha(aFileName);
1.615 + return NoError;
1.616 + }
1.617 +
1.618 +
1.619 +int EpocLoader::SaveAlpha(char* aFileName)
1.620 + {
1.621 + TBitmapFileHeader fileheader;
1.622 + TBitmapInfoHeader alphaHeader;
1.623 +
1.624 + alphaHeader.biSize = sizeof(TBitmapInfoHeader);
1.625 + alphaHeader.biWidth = iPbmHeader.iWidthInPixels;
1.626 + alphaHeader.biHeight = iPbmHeader.iHeightInPixels;
1.627 + alphaHeader.biPlanes = 1;
1.628 + alphaHeader.biBitCount = 8;
1.629 + alphaHeader.biCompression = 0;
1.630 + alphaHeader.biSizeImage = 0;
1.631 + alphaHeader.biXPelsPerMeter = 0;
1.632 + alphaHeader.biYPelsPerMeter = 0;
1.633 + alphaHeader.biClrUsed = 256;
1.634 + alphaHeader.biClrImportant = 0;
1.635 +
1.636 + const long paletteSize = 1024;
1.637 +
1.638 + // ensure bytes-per-scanline is a large enough multiple of 4
1.639 + long byteWidth = (alphaHeader.biWidth + 3) & ~3;
1.640 + long destlength = alphaHeader.biHeight * byteWidth;
1.641 + alphaHeader.biSizeImage = destlength;
1.642 +
1.643 + fileheader.bfType = 'B'+('M'<<8);
1.644 + fileheader.bfSize = sizeof(TBitmapFileHeader)+sizeof(TBitmapInfoHeader)+paletteSize+destlength;
1.645 + fileheader.bfReserved1 = 0;
1.646 + fileheader.bfReserved2 = 0;
1.647 + fileheader.bfOffBits = sizeof(TBitmapFileHeader)+sizeof(TBitmapInfoHeader)+paletteSize;
1.648 +
1.649 + // we need to output the grayscale palette before the actual alpha data as this is an 8bpp BMP
1.650 + char* palette = new char[paletteSize]; // 256 colors x 4bytes/color (r,g,b,unused)
1.651 + if (palette == NULL)
1.652 + return NoMemory;
1.653 + memset(palette,0,paletteSize);
1.654 + char* pEnt = palette;
1.655 + for (long p=0;p<256;++p)
1.656 + {
1.657 + unsigned char col = (unsigned char)p;
1.658 + *pEnt++=col;
1.659 + *pEnt++=col;
1.660 + *pEnt++=col;
1.661 + pEnt++;
1.662 + }
1.663 +
1.664 + char* alphaBits = new char[destlength];
1.665 + if (alphaBits == NULL)
1.666 + {
1.667 + delete [] palette;
1.668 + return NoMemory;
1.669 + }
1.670 + memset(alphaBits,0,destlength);
1.671 +
1.672 + for(long y=0;y<alphaHeader.biHeight;y++)
1.673 + {
1.674 + char* dest=&alphaBits[(alphaHeader.biHeight-y-1)*byteWidth];
1.675 + for(long x=0;x<alphaHeader.biWidth;x++)
1.676 + {
1.677 + unsigned char alphaVal=GetAlpha(x,y);
1.678 + *dest++=alphaVal;
1.679 + }
1.680 + }
1.681 +
1.682 + int fileNameLen = strlen(aFileName);
1.683 + char* alphaFileName = new char[fileNameLen + 7];// -alpha suffix is 6 chars, plus NUL termination
1.684 + if (alphaFileName == NULL)
1.685 + {
1.686 + delete [] palette;
1.687 + delete [] alphaBits;
1.688 + return NoMemory;
1.689 + }
1.690 + int dotPos = -1;
1.691 + for (int i = 0; i < fileNameLen; ++i)
1.692 + if (aFileName[i]=='.')
1.693 + dotPos=i;
1.694 + int prefixLen = (dotPos>=0?dotPos:fileNameLen);
1.695 + memcpy(alphaFileName,aFileName,prefixLen);
1.696 + const char* suffix = "-alpha";
1.697 + memcpy(alphaFileName+prefixLen,suffix,6);
1.698 + if (dotPos>=0)
1.699 + memcpy(alphaFileName+prefixLen+6,aFileName+dotPos,fileNameLen-dotPos);
1.700 + *(alphaFileName + fileNameLen + 6) = '\0';
1.701 + fstream file(alphaFileName, ios::out | ios::binary);
1.702 + if (file.is_open()==0)
1.703 + {
1.704 + delete [] alphaFileName;
1.705 + delete [] alphaBits;
1.706 + delete [] palette;
1.707 + return DestFile;
1.708 + }
1.709 +
1.710 + file.write((char *)&fileheader,sizeof(TBitmapFileHeader));
1.711 + file.write((char *)&alphaHeader,sizeof(TBitmapInfoHeader));
1.712 + file.write((char *)palette,paletteSize);
1.713 + file.write((char *)alphaBits,destlength);
1.714 + file.close();
1.715 +
1.716 + delete [] alphaFileName;
1.717 + delete [] alphaBits;
1.718 + delete [] palette;
1.719 + return NoError;
1.720 + }
1.721 +
1.722 +int EpocLoader::DupBitmap(SEpocBitmapHeader*& aPbm)
1.723 + {
1.724 + char* newPbm = new char[iPbmHeader.iBitmapSize];
1.725 + if (newPbm == NULL)
1.726 + return NoMemory;
1.727 + memcpy(newPbm, &iPbmHeader, sizeof(SEpocBitmapHeader));
1.728 + memcpy(newPbm+sizeof(SEpocBitmapHeader), iPbmBits, iPbmHeader.iBitmapSize - sizeof(SEpocBitmapHeader));
1.729 + aPbm = (SEpocBitmapHeader*)newPbm;
1.730 + return NoError;
1.731 + }
1.732 +
1.733 +/**
1.734 +Validates an mbm files bitmap information to ensure that the length of each bitmaps matches their location
1.735 +within the file. Also checks that the number of bitmap offsets available matches the number of bitmaps.
1.736 +
1.737 +@param aFilename The mbm file to be validated
1.738 +@return An error code from the Errors enumeration
1.739 + */
1.740 +int EpocLoader::ValidateEpocBitmap(char* aFilename)
1.741 + {
1.742 +#if defined(__MSVCDOTNET__) || defined(__TOOLS2__)
1.743 + fstream file(aFilename, ios::in | ios::binary);
1.744 +#else //!__MSVCDOTNET__
1.745 + fstream file(aFilename, ios::in | ios::binary | ios::nocreate);
1.746 +#endif //__MSVCDOTNET__
1.747 + if (file.is_open()==0)
1.748 + return Files;
1.749 +
1.750 + long int wordbuffer;
1.751 + int isRomFormat;
1.752 +
1.753 + file.read((char *)&wordbuffer,4);
1.754 + if (file.gcount()!=4)
1.755 + return SourceFile;
1.756 + if (wordbuffer==KMultiBitmapRomImageUid)
1.757 + {
1.758 + //Validation not implemented for ROM only files.
1.759 + }
1.760 + else
1.761 + {
1.762 + if (wordbuffer!=KWriteOnceFileStoreUid)
1.763 + return SourceFile;
1.764 + isRomFormat=0;
1.765 + file.read((char *)&wordbuffer,4);
1.766 + if (file.gcount()!=4 || wordbuffer!=KMultiBitmapFileImageUid)
1.767 + return SourceFile;
1.768 + file.read((char *)&wordbuffer,4);
1.769 + if (file.gcount()!=4 || wordbuffer!=0)
1.770 + return SourceFile;
1.771 + file.read((char *)&wordbuffer,4);
1.772 + if (file.gcount()!=4 || wordbuffer!=KMultiBitmapFileImageChecksum)
1.773 + return SourceFile;
1.774 + file.read((char *)&wordbuffer,4);
1.775 + if (file.gcount()!=4)
1.776 + return SourceFile;
1.777 + //The 5th byte is a pointer to the footer containing the number of bitmaps
1.778 + //and the offset from the beginning of the file each begins at.
1.779 + int footerPos = wordbuffer;
1.780 +
1.781 + file.seekg(footerPos,ios::beg);
1.782 +
1.783 + file.read((char *)&wordbuffer, 4);
1.784 + if (file.gcount()!=4)
1.785 + return SourceFile;
1.786 +
1.787 + int numBitmaps = wordbuffer;
1.788 +
1.789 + int bmpOffset[numBitmaps+1];
1.790 +
1.791 + for (int i = 0; i < numBitmaps; i++)
1.792 + {
1.793 + file.read((char *)&wordbuffer,4);
1.794 + if (file.gcount()!=4)
1.795 + return SourceFile;
1.796 + bmpOffset[i] = wordbuffer;
1.797 + }
1.798 +
1.799 + //The byte at the end of the last bitmap is the location of the footer
1.800 + bmpOffset[numBitmaps] = footerPos;
1.801 +
1.802 + //Check length of a bitmap is reflected in the difference between the offsets
1.803 + for (int i = 0 ; i < numBitmaps ; i++)
1.804 + {
1.805 + file.seekg(bmpOffset[i]);
1.806 +
1.807 + //Read the length of this bitmap
1.808 + file.read((char *)&wordbuffer,4);
1.809 + if (file.gcount()!=4)
1.810 + return SourceFile;
1.811 +
1.812 + //Validate length against offsets of this & next bmp
1.813 + if (bmpOffset[i+1]-wordbuffer != bmpOffset[i])
1.814 + return SourceFile;
1.815 + }
1.816 + }
1.817 + return NoError;
1.818 + }