Update contrib.
1 // Copyright (c) 1997-2009 Nokia Corporation and/or its subsidiary(-ies).
2 // All rights reserved.
3 // This component and the accompanying materials are made available
4 // under the terms of "Eclipse Public License v1.0"
5 // which accompanies this distribution, and is available
6 // at the URL "http://www.eclipse.org/legal/epl-v10.html".
8 // Initial Contributors:
9 // Nokia Corporation - initial contribution.
19 EpocLoader::EpocLoader():
23 EpocLoader::~EpocLoader()
28 int EpocLoader::EpocBitmapCount(char* aFileName, int& aCount, int& isRomFormat)
30 #if defined(__MSVCDOTNET__) || defined(__TOOLS2__)
31 fstream file(aFileName, ios::in | ios::binary);
32 #else //!__MSVCDOTNET__
33 fstream file(aFileName, ios::in | ios::binary | ios::nocreate);
34 #endif //__MSVCDOTNET__
36 if (file.is_open()==0)
40 file.read((char *)&wordbuffer,4);
43 if (wordbuffer==KMultiBitmapRomImageUid)
45 // ROM format file - next 4 bytes are the number of bitmaps
50 if (wordbuffer!=KWriteOnceFileStoreUid)
53 file.read((char *)&wordbuffer,4);
54 if (file.gcount()!=4 || wordbuffer!=KMultiBitmapFileImageUid)
56 file.read((char *)&wordbuffer,4);
57 if (file.gcount()!=4 || wordbuffer!=0)
59 file.read((char *)&wordbuffer,4);
60 if (file.gcount()!=4 || wordbuffer!=KMultiBitmapFileImageChecksum)
62 file.read((char *)&wordbuffer,4);
65 file.seekg(wordbuffer,ios::beg);
67 file.read((char *)&wordbuffer,4);
74 int EpocLoader::LoadEpocBitmap(char* aFileName,int aIndex)
76 #if defined(__MSVCDOTNET__) || defined(__TOOLS2__)
77 fstream file(aFileName, ios::in | ios::binary);
78 #else //!__MSVCDOTNET__
79 fstream file(aFileName, ios::in | ios::binary | ios::nocreate);
80 #endif //__MSVCDOTNET__
82 if (file.is_open()==0)
86 file.read((char *)&wordbuffer,4);
91 if (wordbuffer==KMultiBitmapRomImageUid)
92 return LoadRom(aFileName,aIndex);
94 return LoadFile(aFileName,aIndex);
97 int EpocLoader::LoadFile(char* aFileName,int aIndex)
99 #if defined(__MSVCDOTNET__) || defined(__TOOLS2__)
100 fstream file(aFileName, ios::in | ios::binary);
101 #else //!__MSVCDOTNET__
102 fstream file(aFileName, ios::in | ios::binary | ios::nocreate);
103 #endif //__MSVCDOTNET__
105 if (file.is_open()==0)
109 file.read((char *)&wordbuffer,4);
110 if (file.gcount()!=4 || wordbuffer!=KWriteOnceFileStoreUid)
112 file.read((char *)&wordbuffer,4);
113 if (file.gcount()!=4 || wordbuffer!=KMultiBitmapFileImageUid)
115 file.read((char *)&wordbuffer,4);
116 if (file.gcount()!=4 || wordbuffer!=0)
118 file.read((char *)&wordbuffer,4);
119 if (file.gcount()!=4 || wordbuffer!=KMultiBitmapFileImageChecksum)
121 file.read((char *)&wordbuffer,4);
122 if (file.gcount()!=4)
124 file.seekg(wordbuffer,ios::beg);
125 file.read((char *)&wordbuffer,4);
126 if (file.gcount()!=4)
129 int numsources=wordbuffer;
130 if (aIndex >= numsources)
132 file.seekg(aIndex*4,ios::cur);
133 file.read((char *)&wordbuffer,4);
134 if (file.gcount()!=4)
136 file.seekg(wordbuffer,ios::beg);
137 int ret = DoLoadFile(file);
142 int EpocLoader::LoadRom(char* aFileName,int aIndex)
144 #if defined(__MSVCDOTNET__) || defined(__TOOLS2__)
145 fstream file(aFileName, ios::in | ios::binary);
146 #else //!__MSVCDOTNET__
147 fstream file(aFileName, ios::in | ios::binary | ios::nocreate);
148 #endif //__MSVCDOTNET__
150 if (file.is_open()==0)
154 file.read((char *)&wordbuffer,4);
155 if (file.gcount()!=4)
157 if (wordbuffer!=KMultiBitmapRomImageUid)
159 file.read((char *)&wordbuffer,4);
160 if (file.gcount()!=4)
162 if (aIndex>=wordbuffer)
164 file.seekg(aIndex*4,ios::cur);
165 file.read((char *)&wordbuffer,4);
166 if (file.gcount()!=4)
168 file.seekg(wordbuffer,ios::beg);
169 int ret=DoLoadRom(file);
174 int EpocLoader::DoLoadFile(fstream& aFile)
176 long size = sizeof(SEpocBitmapHeader);
177 aFile.read((char *)&iPbmHeader,size);
178 if (aFile.gcount()!=size)
180 iOriginalPbmHeader = iPbmHeader;
181 size=iPbmHeader.iBitmapSize-iPbmHeader.iStructSize;
183 iPbmBits=new char[size];
186 memset(iPbmBits,0xff,size);
187 aFile.read(iPbmBits,size);
189 if (aFile.gcount() != size)
192 if (iPbmHeader.iCompression != ENoBitmapCompression)
194 return Decompress(size);
200 int EpocLoader::DoLoadRom(fstream& aFile)
203 long size=sizeof(Bitmap);
204 aFile.read((char*)&bmp,size);
205 if (aFile.gcount() != size)
208 size = bmp.iHeader.iBitmapSize - sizeof(SEpocBitmapHeader);
209 iPbmBits = new char[size];
210 if (iPbmBits == NULL)
212 memset(iPbmBits,0xff,size);
214 aFile.read(iPbmBits,size);
215 if (aFile.gcount() != size)
217 iPbmHeader = bmp.iHeader;
218 iOriginalPbmHeader = iPbmHeader;
220 if (iPbmHeader.iCompression != ENoBitmapCompression)
222 return Decompress(size);
228 int EpocLoader::Decompress(int aSize)
230 int byteWidth = BitmapUtils::ByteWidth(iPbmHeader.iWidthInPixels,iPbmHeader.iBitsPerPixel);
231 int expandedsize = byteWidth * iPbmHeader.iHeightInPixels;
232 char* newbits = new char[expandedsize];
235 memset(newbits,0xff,expandedsize);
237 switch (iPbmHeader.iCompression)
239 case EByteRLECompression:
240 ret = ExpandByteRLEData(newbits,expandedsize,iPbmBits,aSize);
242 case ETwelveBitRLECompression:
243 ret = ExpandTwelveBitRLEData(newbits,expandedsize,iPbmBits,aSize);
245 case ESixteenBitRLECompression:
246 ret = ExpandSixteenBitRLEData(newbits,expandedsize,iPbmBits,aSize);
248 case ETwentyFourBitRLECompression:
249 ret = ExpandTwentyFourBitRLEData(newbits,expandedsize,iPbmBits,aSize);
251 case EThirtyTwoUBitRLECompression:
252 ret = ExpandThirtyTwoUBitRLEData(newbits,expandedsize,iPbmBits,aSize);
254 case EThirtyTwoABitRLECompression:
255 ret = ExpandThirtyTwoABitRLEData(newbits,expandedsize,iPbmBits,aSize);
258 ret = UnknownCompression;
263 iPbmHeader.iCompression = ENoBitmapCompression;
264 iPbmHeader.iBitmapSize += expandedsize-aSize;
268 int EpocLoader::ExpandByteRLEData(char* aDest,int aDestSize,char* aSrce,int aSrceSize)
270 char* srcelimit=aSrce+aSrceSize;
271 char* destlimit=aDest+aDestSize;
272 while(aSrce<srcelimit && aDest<destlimit)
277 int runLength=-count;
278 memcpy(aDest,aSrce,runLength);
292 if (aSrce!=srcelimit || aDest!=destlimit)
293 return DecompressionError;
297 int EpocLoader::ExpandTwelveBitRLEData(char* aDest,int aDestSizeInBytes,char* aSrce,int aSrceSizeInBytes)
299 unsigned short* srcePtr = (unsigned short*)aSrce;
300 unsigned short* destPtr = (unsigned short*)aDest;
301 unsigned short* srcePtrLimit = srcePtr + (aSrceSizeInBytes / 2);
302 unsigned short* destPtrLimit = destPtr + (aDestSizeInBytes / 2);
304 while (srcePtr < srcePtrLimit && destPtr < destPtrLimit)
306 unsigned short value = *srcePtr++;
307 int runLength = (value >> 12) + 1;
310 for (;runLength > 0;runLength--)
314 if (srcePtr != srcePtrLimit || destPtr != destPtrLimit)
315 return DecompressionError;
320 int EpocLoader::ExpandSixteenBitRLEData(char* aDest,int aDestSizeInBytes,char* aSrce,int aSrceSizeInBytes)
322 char* srcePtrLimit = aSrce + aSrceSizeInBytes;
323 unsigned short* destPtr = (unsigned short*)aDest;
324 unsigned short* destPtrLimit = destPtr + (aDestSizeInBytes / 2);
326 while (aSrce < srcePtrLimit && destPtr < destPtrLimit)
328 int runLength = *aSrce++;
332 unsigned short value = *((unsigned short*)(aSrce));
334 for (runLength++; runLength > 0; runLength--)
339 runLength = -runLength;
340 int byteLength = runLength * 2;
341 memcpy(destPtr,aSrce,byteLength);
343 destPtr += runLength;
347 if (aSrce != srcePtrLimit || destPtr != destPtrLimit)
348 return DecompressionError;
353 int EpocLoader::ExpandTwentyFourBitRLEData(char* aDest,int aDestSizeInBytes,char* aSrce,int aSrceSizeInBytes)
355 char* srcePtrLimit = aSrce + aSrceSizeInBytes;
356 char* destPtrLimit = aDest + aDestSizeInBytes;
358 while (aSrce < srcePtrLimit && aDest < destPtrLimit)
360 int runLength = *aSrce++;
364 char component1 = *aSrce++;
365 char component2 = *aSrce++;
366 char component3 = *aSrce++;
367 for (runLength++; runLength > 0; runLength--)
369 *aDest++ = component1;
370 *aDest++ = component2;
371 *aDest++ = component3;
376 runLength = -runLength;
377 int byteLength = runLength * 3;
378 memcpy(aDest,aSrce,byteLength);
384 if (aSrce != srcePtrLimit || aDest != destPtrLimit)
385 return DecompressionError;
390 /** The function decodes 24-bit compressed pixel buffer into the 32-bit buffer, where top bytes are unused*/
391 int EpocLoader::ExpandThirtyTwoUBitRLEData(char* aDest,int aDestSizeInBytes,char* aSrce,int aSrceSizeInBytes)
393 char* srcePtrLimit = aSrce + aSrceSizeInBytes;
394 char* destPtrLimit = aDest + aDestSizeInBytes;
396 while (aSrce < srcePtrLimit && aDest < destPtrLimit)
398 int runLength = *aSrce++;
402 char component1 = *aSrce++;
403 char component2 = *aSrce++;
404 char component3 = *aSrce++;
405 for (runLength++; runLength > 0; runLength--)
407 *aDest++ = component1;
408 *aDest++ = component2;
409 *aDest++ = component3;
415 runLength = -runLength;
416 for(int ii = 0; ii < runLength; ii++)
418 memcpy(aDest,aSrce,3);
425 if (aSrce != srcePtrLimit || aDest != destPtrLimit)
426 return DecompressionError;
431 /** The function decodes 32-bit compressed pixel buffer into the 32-bit buffer, where top bytes are alpha channel*/
432 int EpocLoader::ExpandThirtyTwoABitRLEData(char* aDest,int aDestSizeInBytes,char* aSrce,int aSrceSizeInBytes)
434 char* srcePtrLimit = aSrce + aSrceSizeInBytes;
435 char* destPtrLimit = aDest + aDestSizeInBytes;
437 while (aSrce < srcePtrLimit && aDest < destPtrLimit)
439 int runLength = *aSrce++;
443 char component1 = *aSrce++;
444 char component2 = *aSrce++;
445 char component3 = *aSrce++;
446 char component4 = *aSrce++;
447 for (runLength++; runLength > 0; runLength--)
449 *aDest++ = component1;
450 *aDest++ = component2;
451 *aDest++ = component3;
452 *aDest++ = component4;
457 runLength = -runLength;
458 memcpy(aDest,aSrce,4*runLength);
459 aSrce += 4*runLength;
460 aDest += 4*runLength;
464 if (aSrce != srcePtrLimit || aDest != destPtrLimit)
465 return DecompressionError;
470 TRgb EpocLoader::GetPixel(int aXCoord,int aYCoord)
473 aXCoord%=iPbmHeader.iWidthInPixels;
474 aYCoord%=iPbmHeader.iHeightInPixels;
475 int byteWidth = BitmapUtils::ByteWidth(iPbmHeader.iWidthInPixels,iPbmHeader.iBitsPerPixel);
476 int yOffset = aYCoord * byteWidth;
478 switch(iPbmHeader.iBitsPerPixel)
481 col = iPbmBits[yOffset + (aXCoord / 8)];
483 return TRgb::Gray2(col & 1);
485 col = iPbmBits[yOffset + (aXCoord / 4)];
486 col = (unsigned char)(col>>(2*(aXCoord%4)));
487 return TRgb::Gray4(col & 3);
489 col = iPbmBits[yOffset + (aXCoord / 2)];
493 if (iPbmHeader.iColor==EColorBitmap)
494 return TRgb::Color16(col);
496 return TRgb::Gray16(col);
498 col=iPbmBits[yOffset + aXCoord];
499 if (iPbmHeader.iColor==EColorBitmap)
500 return TRgb::Color256(col);
502 return TRgb::Gray256(col);
506 unsigned short* shortPtr = (unsigned short*)&iPbmBits[yOffset + (aXCoord * 2)];
507 if (iPbmHeader.iBitsPerPixel == 12)
508 return TRgb::Color4K(*shortPtr);
510 return TRgb::Color64K(*shortPtr);
514 char* pixelPtr = iPbmBits + yOffset + (aXCoord * 3);
516 pixelColor.iBlue = *pixelPtr++;
517 pixelColor.iGreen = *pixelPtr++;
518 pixelColor.iRed = *pixelPtr;
523 char* pixelPtr = iPbmBits + yOffset + (aXCoord * 4);
525 pixelColor.iBlue = *pixelPtr++;
526 pixelColor.iGreen = *pixelPtr++;
527 pixelColor.iRed = *pixelPtr++;
528 pixelColor.iSpare = *pixelPtr;
536 unsigned char EpocLoader::GetAlpha(int aXCoord,int aYCoord)
538 aXCoord%=iPbmHeader.iWidthInPixels;
539 aYCoord%=iPbmHeader.iHeightInPixels;
540 int byteWidth = BitmapUtils::ByteWidth(iPbmHeader.iWidthInPixels,iPbmHeader.iBitsPerPixel);
541 int yOffset = aYCoord * byteWidth;
543 assert(iPbmHeader.iBitsPerPixel == 32); // this method must only be called for 32bpp bitmaps!
545 char* pixelPtr = iPbmBits + yOffset + (aXCoord * 4);
546 pixelPtr += 3; // skip over RGB
547 unsigned char col = *pixelPtr;
551 int EpocLoader::SaveBitmap(char* aFileName)
553 TBitmapFileHeader fileheader;
554 TBitmapInfoHeader bmpHeader;
557 bmpHeader.biSize = sizeof(TBitmapInfoHeader);
558 bmpHeader.biWidth = iPbmHeader.iWidthInPixels;
559 bmpHeader.biHeight = iPbmHeader.iHeightInPixels;
560 bmpHeader.biPlanes = 1;
561 bmpHeader.biBitCount = 24;
562 bmpHeader.biCompression = 0;
563 bmpHeader.biSizeImage = 0;
564 bmpHeader.biXPelsPerMeter = 0;
565 bmpHeader.biYPelsPerMeter = 0;
566 bmpHeader.biClrUsed = 0;
567 bmpHeader.biClrImportant = 0;
569 long byteWidth = ((bmpHeader.biWidth * 3) + 3) & ~3;
570 long destlength = bmpHeader.biHeight * byteWidth;
572 fileheader.bfType = 'B'+('M'<<8);
573 fileheader.bfSize = sizeof(TBitmapFileHeader)+sizeof(TBitmapInfoHeader)+destlength;
574 fileheader.bfReserved1 = 0;
575 fileheader.bfReserved2 = 0;
576 fileheader.bfOffBits = sizeof(TBitmapFileHeader)+sizeof(TBitmapInfoHeader);
578 bmpBits = new char[destlength];
581 memset(bmpBits,0xff,destlength);
583 for(long y=0;y<bmpHeader.biHeight;y++)
585 char* dest=&bmpBits[(bmpHeader.biHeight-y-1)*byteWidth];
586 for(long x=0;x<bmpHeader.biWidth;x++)
588 TRgb pixel=GetPixel(x,y);
590 *dest++=pixel.iGreen;
595 fstream file(aFileName, ios::out | ios::binary);
597 if (file.is_open()==0)
603 file.write((char *)&fileheader,sizeof(TBitmapFileHeader));
604 file.write((char *)&bmpHeader,sizeof(TBitmapInfoHeader));
605 file.write((char *)bmpBits,destlength);
610 if (iPbmHeader.iColor == EColorBitmapAlpha || iPbmHeader.iColor ==EColorBitmapAlphaPM)
611 SaveAlpha(aFileName);
616 int EpocLoader::SaveAlpha(char* aFileName)
618 TBitmapFileHeader fileheader;
619 TBitmapInfoHeader alphaHeader;
621 alphaHeader.biSize = sizeof(TBitmapInfoHeader);
622 alphaHeader.biWidth = iPbmHeader.iWidthInPixels;
623 alphaHeader.biHeight = iPbmHeader.iHeightInPixels;
624 alphaHeader.biPlanes = 1;
625 alphaHeader.biBitCount = 8;
626 alphaHeader.biCompression = 0;
627 alphaHeader.biSizeImage = 0;
628 alphaHeader.biXPelsPerMeter = 0;
629 alphaHeader.biYPelsPerMeter = 0;
630 alphaHeader.biClrUsed = 256;
631 alphaHeader.biClrImportant = 0;
633 const long paletteSize = 1024;
635 // ensure bytes-per-scanline is a large enough multiple of 4
636 long byteWidth = (alphaHeader.biWidth + 3) & ~3;
637 long destlength = alphaHeader.biHeight * byteWidth;
638 alphaHeader.biSizeImage = destlength;
640 fileheader.bfType = 'B'+('M'<<8);
641 fileheader.bfSize = sizeof(TBitmapFileHeader)+sizeof(TBitmapInfoHeader)+paletteSize+destlength;
642 fileheader.bfReserved1 = 0;
643 fileheader.bfReserved2 = 0;
644 fileheader.bfOffBits = sizeof(TBitmapFileHeader)+sizeof(TBitmapInfoHeader)+paletteSize;
646 // we need to output the grayscale palette before the actual alpha data as this is an 8bpp BMP
647 char* palette = new char[paletteSize]; // 256 colors x 4bytes/color (r,g,b,unused)
650 memset(palette,0,paletteSize);
651 char* pEnt = palette;
652 for (long p=0;p<256;++p)
654 unsigned char col = (unsigned char)p;
661 char* alphaBits = new char[destlength];
662 if (alphaBits == NULL)
667 memset(alphaBits,0,destlength);
669 for(long y=0;y<alphaHeader.biHeight;y++)
671 char* dest=&alphaBits[(alphaHeader.biHeight-y-1)*byteWidth];
672 for(long x=0;x<alphaHeader.biWidth;x++)
674 unsigned char alphaVal=GetAlpha(x,y);
679 int fileNameLen = strlen(aFileName);
680 char* alphaFileName = new char[fileNameLen + 7];// -alpha suffix is 6 chars, plus NUL termination
681 if (alphaFileName == NULL)
688 for (int i = 0; i < fileNameLen; ++i)
689 if (aFileName[i]=='.')
691 int prefixLen = (dotPos>=0?dotPos:fileNameLen);
692 memcpy(alphaFileName,aFileName,prefixLen);
693 const char* suffix = "-alpha";
694 memcpy(alphaFileName+prefixLen,suffix,6);
696 memcpy(alphaFileName+prefixLen+6,aFileName+dotPos,fileNameLen-dotPos);
697 *(alphaFileName + fileNameLen + 6) = '\0';
698 fstream file(alphaFileName, ios::out | ios::binary);
699 if (file.is_open()==0)
701 delete [] alphaFileName;
707 file.write((char *)&fileheader,sizeof(TBitmapFileHeader));
708 file.write((char *)&alphaHeader,sizeof(TBitmapInfoHeader));
709 file.write((char *)palette,paletteSize);
710 file.write((char *)alphaBits,destlength);
713 delete [] alphaFileName;
719 int EpocLoader::DupBitmap(SEpocBitmapHeader*& aPbm)
721 char* newPbm = new char[iPbmHeader.iBitmapSize];
724 memcpy(newPbm, &iPbmHeader, sizeof(SEpocBitmapHeader));
725 memcpy(newPbm+sizeof(SEpocBitmapHeader), iPbmBits, iPbmHeader.iBitmapSize - sizeof(SEpocBitmapHeader));
726 aPbm = (SEpocBitmapHeader*)newPbm;
731 Validates an mbm files bitmap information to ensure that the length of each bitmaps matches their location
732 within the file. Also checks that the number of bitmap offsets available matches the number of bitmaps.
734 @param aFilename The mbm file to be validated
735 @return An error code from the Errors enumeration
737 int EpocLoader::ValidateEpocBitmap(char* aFilename)
739 #if defined(__MSVCDOTNET__) || defined(__TOOLS2__)
740 fstream file(aFilename, ios::in | ios::binary);
741 #else //!__MSVCDOTNET__
742 fstream file(aFilename, ios::in | ios::binary | ios::nocreate);
743 #endif //__MSVCDOTNET__
744 if (file.is_open()==0)
750 file.read((char *)&wordbuffer,4);
751 if (file.gcount()!=4)
753 if (wordbuffer==KMultiBitmapRomImageUid)
755 //Validation not implemented for ROM only files.
759 if (wordbuffer!=KWriteOnceFileStoreUid)
762 file.read((char *)&wordbuffer,4);
763 if (file.gcount()!=4 || wordbuffer!=KMultiBitmapFileImageUid)
765 file.read((char *)&wordbuffer,4);
766 if (file.gcount()!=4 || wordbuffer!=0)
768 file.read((char *)&wordbuffer,4);
769 if (file.gcount()!=4 || wordbuffer!=KMultiBitmapFileImageChecksum)
771 file.read((char *)&wordbuffer,4);
772 if (file.gcount()!=4)
774 //The 5th byte is a pointer to the footer containing the number of bitmaps
775 //and the offset from the beginning of the file each begins at.
776 int footerPos = wordbuffer;
778 file.seekg(footerPos,ios::beg);
780 file.read((char *)&wordbuffer, 4);
781 if (file.gcount()!=4)
784 int numBitmaps = wordbuffer;
786 int bmpOffset[numBitmaps+1];
788 for (int i = 0; i < numBitmaps; i++)
790 file.read((char *)&wordbuffer,4);
791 if (file.gcount()!=4)
793 bmpOffset[i] = wordbuffer;
796 //The byte at the end of the last bitmap is the location of the footer
797 bmpOffset[numBitmaps] = footerPos;
799 //Check length of a bitmap is reflected in the difference between the offsets
800 for (int i = 0 ; i < numBitmaps ; i++)
802 file.seekg(bmpOffset[i]);
804 //Read the length of this bitmap
805 file.read((char *)&wordbuffer,4);
806 if (file.gcount()!=4)
809 //Validate length against offsets of this & next bmp
810 if (bmpOffset[i+1]-wordbuffer != bmpOffset[i])