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.
21 extern TRgb* color256Palette;
22 extern char* color256InversePalette;
24 static inline bool CanCopy16bppData(const char* aDest, const char* aDestEnd, int aDataSize)
26 const char* aDestNew = aDest + (aDataSize / 128) * (256 + 1) + //(2 * 128) bytes data + 1 leading byte
27 (aDataSize % 128 ? (aDataSize % 128) * 2 + 1 : 0);
28 return aDestNew <= aDestEnd;
31 static inline bool CanWrite16bppValue(const char* aDest, const char* aDestEnd, int aDataSize)
33 const char* aDestNew = aDest + (aDataSize / 128) * (2 + 1) + //2 bytes data + 1 leading byte
34 (aDataSize % 128 ? 2 + 1 : 0);
35 return aDestNew <= aDestEnd;
38 static inline bool CanCopy8bppData(const char* aDest, const char* aDestEnd, int aDataSize)
40 const char* aDestNew = aDest + (aDataSize / 128) * (128 + 1) + //128 bytes data + 1 leading byte
41 (aDataSize % 128 ? (aDataSize % 128) + 1 : 0);
42 return aDestNew <= aDestEnd;
45 static inline bool CanWrite8bppValue(const char* aDest, const char* aDestEnd, int aDataSize)
47 const char* aDestNew = aDest + (aDataSize / 128) * (1 + 1) + //1 byte data + 1 leading byte
48 (aDataSize % 128 ? 1 + 1 : 0);
49 return aDestNew <= aDestEnd;
52 static inline bool CanCopy24bppData(const char* aDest, const char* aDestEnd, int aDataSize)
54 const char* aDestNew = aDest + (aDataSize / 128) * (384 + 1) + //(128 * 3) bytes data + 1 leading byte
55 (aDataSize % 128 ? (aDataSize % 128) * 3 + 1 : 0);
56 return aDestNew <= aDestEnd;
59 static inline bool CanWrite24bppValue(const char* aDest, const char* aDestEnd, int aDataSize)
61 const char* aDestNew = aDest + (aDataSize / 128) * (3 + 1) + //3 bytes data + 1 leading byte
62 (aDataSize % 128 ? 3 + 1 : 0);
63 return aDestNew <= aDestEnd;
66 static inline bool CanCopy32bppData(const char* aDest, const char* aDestEnd, int aDataSize)
68 const char* aDestNew = aDest + (aDataSize / 128) * (512 + 1) + //(128 * 4) bytes data + 1 leading byte
69 (aDataSize % 128 ? (aDataSize % 128) * 4 + 1 : 0);
70 return aDestNew <= aDestEnd;
73 static inline bool CanWrite32bppValue(const char* aDest, const char* aDestEnd, int aDataSize)
75 const char* aDestNew = aDest + (aDataSize / 128) * (4 + 1) + //4 bytes data + 1 leading byte
76 (aDataSize % 128 ? 4 + 1 : 0);
77 return aDestNew <= aDestEnd;
80 BitmapCompiler::BitmapCompiler(char* aSourcefilenames[],int aNumSources):
81 iSourcefilenames(aSourcefilenames),
83 iNumSources(aNumSources)
86 BitmapCompiler::~BitmapCompiler()
90 for(int count=0;count<iNumSources;count++)
91 delete iPbmSources[count];
92 delete [] iPbmSources;
96 int BitmapCompiler::Compile(TStoreType aSt,int aCompress,char* aDestfilename,char* aHeaderFilename,char* aPaletteFilename)
98 int ret = LoadPalette(aPaletteFilename);
102 iDestFile.open(aDestfilename, ios::out | ios::binary);
103 if(iDestFile.is_open()==0)
108 if (iNumSources==1 && ((iSourcefilenames[0][0]==OPTCHAR || iSourcefilenames[0][0]==ALTERNATE_OPTCHAR)) && iSourcefilenames[0][1]=='m')
110 // Source is an existing MBM file, presumably because we want to convert
111 // a filestore into a ROM image.
114 cout << "Header file generation is not permitted with /m or -m\n";
115 aHeaderFilename = NULL;
117 iSourcefilenames[0]+=2;
120 ret = pl.EpocBitmapCount(iSourcefilenames[0], iNumSources, romFormat);
122 ret = LoadPbmFile(iSourcefilenames[0]);
126 // Read the usual list of [OPT]bmp_n source bitmaps.
127 ret = LoadSourcefiles();
132 ret = CreateHeader(aHeaderFilename);
136 if (aSt == ERomStore || aSt == ECompressedRomStore)
137 ret = RomImage(aSt == ECompressedRomStore);
139 ret = FileImage(aCompress);
143 int BitmapCompiler::AllocatePbmSourcesArray()
145 iPbmSources = new SEpocBitmapHeader*[iNumSources];
146 if(iPbmSources == NULL)
149 memset(iPbmSources,0xff,sizeof(SEpocBitmapHeader*)*iNumSources);
151 for(count=0;count<iNumSources;count++)
152 iPbmSources[count]=NULL;
156 int BitmapCompiler::LoadSourcefiles()
158 int ret = AllocatePbmSourcesArray();
161 for(int count=0;count<iNumSources;count++)
163 bool useAlpha = false;
165 TBitmapColor color = EMonochromeBitmap;
166 BitmapLoader bmpload;
167 if(iSourcefilenames[count][0]==OPTCHAR || iSourcefilenames[count][0]==ALTERNATE_OPTCHAR)
169 iSourcefilenames[count]++;
170 if(iSourcefilenames[count][0]=='c')
172 color = EColorBitmap;
173 iSourcefilenames[count]++;
175 bpp=iSourcefilenames[count][0]-'0';
176 iSourcefilenames[count]++;
177 int next=iSourcefilenames[count][0]-'0';
178 if(next==2 || next==6 || next==4)
181 iSourcefilenames[count]++;
183 if(iSourcefilenames[count][0]=='a')
186 iSourcefilenames[count]++;
187 if (color!=EColorBitmap || bpp!=32)
190 if (color == EColorBitmap && bpp!=4 && bpp!=8 && bpp!=12 && bpp!=16 && bpp!=24 && bpp!=32)
192 if (color == EMonochromeBitmap && bpp!=1 && bpp!=2 && bpp!=4 && bpp!=8)
195 if (color == EColorBitmap && useAlpha)
196 color = EColorBitmapAlpha;
197 ret = bmpload.LoadBitmap(iSourcefilenames[count],bpp,color,iPbmSources[count]);
204 int BitmapCompiler::LoadPbmFile(char* aPbmFilename)
206 int ret = AllocatePbmSourcesArray();
209 for(int count=0;count<iNumSources;count++)
212 ret = pl.LoadEpocBitmap(aPbmFilename,count);
214 ret = pl.DupBitmap(iPbmSources[count]);
221 int BitmapCompiler::RomImage(bool aCompress)
227 for(; count<iNumSources; count++)
229 int ret = CompressBitmap(iPbmSources[count]);
235 int ret = WriteRomheader();
239 for(count=0;count<iNumSources;count++)
241 ret = WriteRombitmap(iPbmSources[count]);
248 int BitmapCompiler::FileImage(int aCompress)
253 for(;count<iNumSources;count++)
255 int ret = CompressBitmap(iPbmSources[count]);
260 int ret = WriteFileheader();
263 for (count=0;count<iNumSources;count++)
265 ret = WriteFilebitmap(iPbmSources[count]);
269 ret = WriteHeadStream();
273 int BitmapCompiler::WriteRomheader()
275 iDestFile.write((char*)&KMultiBitmapRomImageUid,4);
276 iDestFile.write((char*)&iNumSources,4);
277 int byteswritten=8+(iNumSources*4);
278 for (int count=0;count<iNumSources;count++)
280 iDestFile.write((char*)&byteswritten,4);
281 byteswritten+=(((iPbmSources[count]->iBitmapSize)+3)/4*4)-sizeof(SEpocBitmapHeader)+sizeof(Bitmap);
286 int BitmapCompiler::WriteRombitmap(SEpocBitmapHeader* aPbm)
288 if (aPbm->iPaletteEntries != 0)
289 return PaletteSupportNotImplemented;
291 int bitmapsize = aPbm->iBitmapSize-sizeof(SEpocBitmapHeader)+sizeof(Bitmap);
292 bitmapsize = ((bitmapsize+3)/4)*4;
294 char* buffer = new char[bitmapsize];
297 memset(buffer,0xff,bitmapsize);
299 Bitmap* bmp = (Bitmap*)buffer;
300 bmp->iUid=KCBitwiseBitmapUid;
301 TBitmapColor color = aPbm->iColor;
303 switch(aPbm->iBitsPerPixel)
304 { // for corresponding enums, see TDisplayMode in <gdi.h>
312 if (color == EMonochromeBitmap)
318 if (color == EMonochromeBitmap)
333 if (color == EColorBitmapAlpha)
334 bmp->iDispMode=12; // Color16MA
336 bmp->iDispMode=11; // Color16MU
345 bmp->iHeader = *aPbm;
346 bmp->iByteWidth = BitmapUtils::ByteWidth(bmp->iHeader.iWidthInPixels,bmp->iHeader.iBitsPerPixel);
347 bmp->iDataOffset = sizeof(Bitmap);
349 CopyTail(buffer + sizeof(Bitmap), aPbm, aPbm->iBitmapSize, sizeof(SEpocBitmapHeader));
350 iDestFile.write(buffer,bitmapsize);
357 int BitmapCompiler::WriteFileheader()
360 iDestFile.write((char*)&KWriteOnceFileStoreUid,4);
361 iDestFile.write((char*)&KMultiBitmapFileImageUid,4);
362 iDestFile.write((char*)&zero,4);
363 iDestFile.write((char*)&KMultiBitmapFileImageChecksum,4);
365 for(int count=0;count<iNumSources;count++)
367 byteswritten+=iPbmSources[count]->iBitmapSize;
370 iDestFile.write((char*)&byteswritten,4);
374 int BitmapCompiler::WriteHeadStream()
376 iDestFile.write((char*)&iNumSources,4);
378 for(int count=0;count<iNumSources;count++)
380 iDestFile.write((char*)&byteswritten,4);
381 byteswritten+=iPbmSources[count]->iBitmapSize;
386 int BitmapCompiler::WriteFilebitmap(SEpocBitmapHeader* aPbm)
388 if (aPbm->iPaletteEntries != 0)
389 return PaletteSupportNotImplemented;
391 int dataSize = aPbm->iBitmapSize - sizeof(SEpocBitmapHeader);
393 iDestFile.write((char*)(aPbm),sizeof(SEpocBitmapHeader));
395 iDestFile.write(((char*)(aPbm)) + sizeof(SEpocBitmapHeader),dataSize);
400 void BitmapCompiler::WritePalette()
402 for (int index = 0; index < 256; index++)
404 iDestFile.write((char*)(&iPalette[index]),3);
408 int BitmapCompiler::LoadPalette(char* aPaletteFilename)
410 if (!aPaletteFilename)
417 color256Palette = iPalette;
418 color256InversePalette = iInversePalette;
420 struct stat fileinfo;
421 if (stat(aPaletteFilename,&fileinfo)==-1)
424 int filesize = fileinfo.st_size;
428 #if defined(__MSVCDOTNET__) || defined(__TOOLS2__)
429 fstream paletteFile(aPaletteFilename, ios::in | ios::binary);
431 fstream paletteFile(aPaletteFilename, ios::in | ios::binary | ios::nocreate);
434 if(!paletteFile.is_open())
437 char* paletteData = new char[filesize];
441 memset(paletteData,0,filesize);
442 paletteFile.read(paletteData,filesize);
445 char* dataPtr = (char*)paletteData;
446 char* dataPtrLimit = dataPtr + filesize;
448 for (int index = 0; index < 256; index++)
451 int ret = ReadHexString(hexBuf,dataPtr,dataPtrLimit);
458 int red = HexToInt(hexBuf[8],hexBuf[9]);
459 int green = HexToInt(hexBuf[6],hexBuf[7]);
460 int blue = HexToInt(hexBuf[4],hexBuf[5]);
462 iPalette[index] = TRgb(red,green,blue);
466 CalculateInversePalette();
470 void BitmapCompiler::CalculateInversePalette()
472 for (int index = 0; index < 4096; index++)
474 TRgb color = TRgb((index & 0x00f) * 17,((index & 0x0f0) >> 4) * 17,((index & 0xf00) >> 8) * 17);
477 int distance = iPalette[0].Difference(color);
479 for (int paletteIndex = 0; paletteIndex < 256; paletteIndex++)
481 TRgb paletteColor = iPalette[paletteIndex];
483 if (paletteColor == color)
485 nearest = paletteIndex;
489 int paletteDistance = paletteColor.Difference(color);
490 if (paletteDistance < distance)
492 nearest = paletteIndex;
493 distance = paletteDistance;
497 iInversePalette[index] = (char)nearest;
498 TRgb palColor = iPalette[nearest];
503 int BitmapCompiler::CreateHeader(char* aHeaderFilename)
505 if (!aHeaderFilename)
508 fstream hf(aHeaderFilename,ios::out);
512 char* basename = strrchr(aHeaderFilename,'\\');
514 basename = aHeaderFilename;
516 basename++; // skip directory separator
518 char unadornedFile[256];
519 strcpy(unadornedFile, UnadornedName(aHeaderFilename));
521 hf << "// " << basename << "\n";
522 hf << "// Generated by BitmapCompiler\n";
523 hf << "// Copyright (c) 1998-2009 Nokia Corporation and/or its subsidiary(-ies).\n";
525 hf << "enum TMbm" << unadornedFile << "\n";
528 for (int count=0;count<iNumSources;count++)
530 hf << "\tEMbm" << unadornedFile << UnadornedName(iSourcefilenames[count]);
531 if(count<iNumSources-1)
540 char* BitmapCompiler::UnadornedName(char* aName)
542 int len = strlen(aName);
544 char* foundDir = strrchr(aName,'\\');
545 char* foundUrl = strrchr(aName,'/');
546 char* foundExt = strrchr(aName,'.');
548 char* foundPath = (foundDir > foundUrl) ? foundDir : foundUrl;
549 char* firstchar = (foundPath > 0) ? foundPath + 1 : aName;
550 char* lastchar = (foundExt > firstchar) ? foundExt : aName+len;
552 static char result[256];
553 if (lastchar-firstchar > 255)
555 strcpy(result,"NameTooLong");
559 result[0] = (char)toupper(*firstchar);
561 for (i=1; firstchar<lastchar; i++,firstchar++)
563 result[i] = (char)tolower(*firstchar);
569 int BitmapCompiler::CompressBitmap(SEpocBitmapHeader*& aPbm)
571 int originalsize = aPbm->iBitmapSize-sizeof(SEpocBitmapHeader);
572 char* newbits=new char[originalsize+sizeof(SEpocBitmapHeader)];
576 memset(newbits,0xff,sizeof(SEpocBitmapHeader)+originalsize);
577 memcpy(newbits,aPbm,sizeof(SEpocBitmapHeader));
578 char* newbitsptr=newbits+sizeof(SEpocBitmapHeader);
579 char* oldbits=((char*)(aPbm))+sizeof(SEpocBitmapHeader);
581 TBitmapfileCompression compression = ENoBitmapCompression;
582 int ret = NoCompression;
584 if (aPbm->iBitsPerPixel <= 8)
586 compression = EByteRLECompression;
587 ret = CompressByteData(newbitsptr,oldbits,originalsize);
589 else if (aPbm->iBitsPerPixel == 12)
591 compression = ETwelveBitRLECompression;
592 ret = CompressTwelveBitData(newbitsptr,oldbits,originalsize);
594 else if (aPbm->iBitsPerPixel == 16)
596 compression = ESixteenBitRLECompression;
597 ret = CompressSixteenBitData(newbitsptr,oldbits,originalsize);
599 else if (aPbm->iBitsPerPixel == 24)
601 compression = ETwentyFourBitRLECompression;
602 ret = CompressTwentyFourBitData(newbitsptr,oldbits,originalsize);
604 else if (aPbm->iBitsPerPixel == 32 && (aPbm->iColor == EColorBitmap))
606 compression = EThirtyTwoUBitRLECompression;
607 ret = CompressThirtyTwoUBitData(newbitsptr,oldbits,originalsize);
609 else if (aPbm->iBitsPerPixel == 32 && (aPbm->iColor == EColorBitmapAlpha))
611 compression = EThirtyTwoABitRLECompression;
612 ret = CompressThirtyTwoABitData(newbitsptr,oldbits,originalsize);
624 aPbm = (SEpocBitmapHeader*)newbits;
625 aPbm->iBitmapSize = newbitsptr-(newbits+sizeof(SEpocBitmapHeader))+sizeof(SEpocBitmapHeader);
626 aPbm->iCompression = compression;
630 int BitmapCompiler::CompressByteData(char*& aDest,char* aSrce,int aSize)
632 const char* destEnd = aDest + aSize;
633 char* bytepointer=aSrce;
634 char* limitpointer=bytepointer+aSize;
635 int margin = (aSize>>6);
636 char* limitmargin = limitpointer - ((margin > 4) ? margin : 4);
637 char* cutoffpointer=aDest+(aSize>>1)+(aSize>>2);
639 char* oldbytepointer=NULL;
640 while(bytepointer<limitmargin)
642 char value=*bytepointer;
643 if(*(bytepointer+1)==value && *(bytepointer+2)==value)
645 oldbytepointer=bytepointer;
647 while ( ( bytepointer < limitmargin ) && ( *bytepointer == value ) )
649 ret = WriteCompressedByteValues(aDest,value,bytepointer-oldbytepointer, destEnd);
654 oldbytepointer=bytepointer;
655 while((bytepointer<limitmargin) && (*(bytepointer+1)!=value || *(bytepointer+2)!=value))
660 ret = WriteCompressedByteData(aDest,oldbytepointer,bytepointer-oldbytepointer, destEnd);
663 if(aDest>cutoffpointer)
664 return NoCompression;
667 int remaining = limitpointer-bytepointer;
670 if (aDest + remaining > cutoffpointer)
671 return NoCompression;
672 ret=WriteCompressedByteData(aDest,bytepointer,remaining, destEnd);
678 int BitmapCompiler::WriteCompressedByteData(char*& aDest,char* aData,int aLength, const char* aDestEnd)
680 if(!CanCopy8bppData(aDest, aDestEnd, aLength))
681 return NoCompression;
685 for(int count=0;count<128;count++)
689 if(aLength>128 || aLength<1) return CompressionError;
690 *aDest++=char(-aLength);
691 for(int count=0;count<aLength;count++)
696 int BitmapCompiler::WriteCompressedByteValues(char*& aDest,char aValue,int aLength, const char* aDestEnd)
699 return CompressionError;
701 if(!CanWrite8bppValue(aDest, aDestEnd, aLength))
702 return NoCompression;
704 while (aLength > 128)
711 *aDest++ = char(aLength-1);
717 int BitmapCompiler::CompressTwelveBitData(char*& aDest,char* aSrce,int aSizeInBytes)
719 unsigned short* srcePtr = (unsigned short*)aSrce;
720 unsigned short* srceLimitPtr = srcePtr + (aSizeInBytes / 2);
721 unsigned short* dest = (unsigned short*)aDest;
723 while (srcePtr < srceLimitPtr)
725 unsigned short* runStartPtr = srcePtr;
726 unsigned short value = *srcePtr;
729 while (srcePtr < srceLimitPtr && *srcePtr == value);
730 WriteCompressedTwelveBitData(dest,*runStartPtr,srcePtr - runStartPtr); // Ignore error return as 12bpp compression never fails
737 int BitmapCompiler::WriteCompressedTwelveBitData(unsigned short*& aDest,unsigned short aData,int aLength)
739 // The run length l is stored as l-1 in the top 4 bits of each 16-bit word (between 1 and 16)
741 unsigned short maxLengthData = (unsigned short)(aData | 0xf000);
745 *aDest++ = maxLengthData;
750 *aDest++ = (unsigned short)(aData | ((aLength - 1) << 12));
755 int BitmapCompiler::CompressSixteenBitData(char*& aDest,char* aSrce,int aSizeInBytes)
757 char* destEnd = aDest + aSizeInBytes;
758 unsigned short* srcePtr = (unsigned short*)aSrce;
759 unsigned short* srceLimitPtr = srcePtr + (aSizeInBytes / 2);
760 unsigned short* srceLimitPtrMinusOne = srceLimitPtr - 1;
761 char* destCompressedLimitPtr = aDest + (aSizeInBytes * 7 / 8);
764 while (srcePtr < srceLimitPtrMinusOne)
766 unsigned short value = *srcePtr;
767 unsigned short* runStartPtr = srcePtr++;
769 if(*srcePtr == value)
773 while ( ( srcePtr < srceLimitPtr ) && ( *srcePtr == value ) );
775 ret = WriteCompressedSixteenBitValues(aDest,value,srcePtr-runStartPtr, destEnd);
782 while (srcePtr < srceLimitPtrMinusOne && *(srcePtr + 1) != value)
788 ret = WriteCompressedSixteenBitData(aDest,runStartPtr,srcePtr - runStartPtr, destEnd);
792 if (aDest > destCompressedLimitPtr)
793 return NoCompression;
796 if (srcePtr < srceLimitPtr)
798 ret = WriteCompressedSixteenBitData(aDest,srcePtr,srceLimitPtr - srcePtr, destEnd);
803 if (aDest > destCompressedLimitPtr)
804 return NoCompression;
808 int BitmapCompiler::WriteCompressedSixteenBitData(char*& aDest,unsigned short* aData,
809 int aLength, const char* aDestEnd)
812 return CompressionError;
814 if(!CanCopy16bppData(aDest, aDestEnd, aLength))
815 return NoCompression;
817 char* srcePtr = (char*)aData;
819 while (aLength > 128)
822 memcpy(aDest,srcePtr,256);
828 *aDest++ = char(-aLength);
830 int remainingBytes = aLength * 2;
831 memcpy(aDest,srcePtr,remainingBytes);
832 aDest += remainingBytes;
837 int BitmapCompiler::WriteCompressedSixteenBitValues(char*& aDest,unsigned short aValue,
838 int aLength, const char* aDestEnd)
841 return CompressionError;
843 if(!CanWrite16bppValue(aDest, aDestEnd, aLength))
844 return NoCompression;
846 char lowByte = char(aValue);
847 char highByte = char(aValue >> 8);
849 while (aLength > 128)
857 *aDest++ = char(aLength-1);
864 int BitmapCompiler::CompressTwentyFourBitData(char*& aDest,char* aSrce,int aSizeInBytes)
866 const char* destEnd = aDest + aSizeInBytes;
867 char* srceLimitPtr = aSrce + aSizeInBytes;
868 char* srceLimitPtrMinusThree = srceLimitPtr - 3; // three bytes == one pixel
869 char* destCompressedLimitPtr = aDest + (aSizeInBytes * 7 / 8);
872 while (aSrce < srceLimitPtrMinusThree)
874 char* runStartPtr = aSrce;
875 char component1 = *aSrce++;
876 char component2 = *aSrce++;
877 char component3 = *aSrce++;
879 if (TrueColorPointerCompare(aSrce,component1,component2,component3))
883 while (aSrce < srceLimitPtr && TrueColorPointerCompare(aSrce,component1,component2,component3));
885 ret = WriteCompressedTwentyFourBitValues(aDest,component1,component2,component3,(aSrce - runStartPtr) / 3, destEnd);
895 component1 = *aSrce++;
896 component2 = *aSrce++;
897 component3 = *aSrce++;
898 more = (aSrce < srceLimitPtr);
899 eqRun = more && TrueColorPointerCompare(aSrce,component1,component2,component3);
901 while (more && !eqRun);
904 int pixelLength = (aSrce - runStartPtr) / 3;
905 ret = WriteCompressedTwentyFourBitData(aDest,runStartPtr,pixelLength,destEnd);
909 if (aDest > destCompressedLimitPtr)
910 return NoCompression;
913 if (aSrce < srceLimitPtr)
915 ret = WriteCompressedTwentyFourBitData(aDest,aSrce,(srceLimitPtr - aSrce) / 3, destEnd);
920 if (aDest > destCompressedLimitPtr)
921 return NoCompression;
925 int BitmapCompiler::WriteCompressedTwentyFourBitData(char*& aDest,char* aData,
926 int aLength, const char* aDestEnd)
929 return CompressionError;
931 if(!CanCopy24bppData(aDest, aDestEnd, aLength))
932 return NoCompression;
934 while (aLength > 128)
937 memcpy(aDest,aData,384);
943 *aDest++ = char(-aLength);
945 int remainingBytes = aLength * 3;
946 memcpy(aDest,aData,remainingBytes);
947 aDest += remainingBytes;
952 int BitmapCompiler::WriteCompressedTwentyFourBitValues(char*& aDest,char aComponent1,
953 char aComponent2,char aComponent3,
954 int aLength, const char* aDestEnd)
957 return CompressionError;
959 if(!CanWrite24bppValue(aDest, aDestEnd, aLength))
960 return NoCompression;
962 while (aLength > 128)
965 *aDest++ = aComponent1;
966 *aDest++ = aComponent2;
967 *aDest++ = aComponent3;
971 *aDest++ = char(aLength-1);
972 *aDest++ = aComponent1;
973 *aDest++ = aComponent2;
974 *aDest++ = aComponent3;
979 /** The function encodes 32-bit buffer into 24-bit stream by using RLE compression algorithm*/
980 int BitmapCompiler::CompressThirtyTwoUBitData(char*& aDest,char* aSrce,int aSizeInBytes)
982 const char* destEnd = aDest + aSizeInBytes;
983 char* srceLimitPtr = aSrce + aSizeInBytes;
984 char* srceLimitPtrMinusFour = srceLimitPtr - 4; // four bytes == one pixel
985 char* destCompressedLimitPtr = aDest + (aSizeInBytes * 7 / 8);
988 while (aSrce < srceLimitPtrMinusFour)
990 char* runStartPtr = aSrce;
991 char component1 = *aSrce++;
992 char component2 = *aSrce++;
993 char component3 = *aSrce++;
996 if (TrueColorPointerCompare(aSrce,component1,component2,component3))
1000 while (aSrce < srceLimitPtr && TrueColorPointerCompare(aSrce,component1,component2,component3));
1002 ret = WriteCompressedThirtyTwoUBitValues(aDest,component1,component2,component3,(aSrce - runStartPtr) / 4, destEnd);
1006 else //non repeating pixel buffer
1011 component1 = *aSrce++;
1012 component2 = *aSrce++;
1013 component3 = *aSrce++;
1014 aSrce++; //last byte is unused (no alpha component)
1015 more = (aSrce < srceLimitPtr);
1016 eqRun = more && TrueColorPointerCompare(aSrce,component1,component2,component3);
1017 } while (more && !eqRun);
1020 int pixelLength = (aSrce - runStartPtr) / 4;
1021 ret = WriteCompressedThirtyTwoUBitData(aDest,runStartPtr,pixelLength,destEnd);
1025 if (aDest > destCompressedLimitPtr)
1026 return NoCompression;
1029 if (aSrce < srceLimitPtr)
1031 ret = WriteCompressedThirtyTwoUBitData(aDest,aSrce,(srceLimitPtr - aSrce) / 4, destEnd);
1036 if (aDest > destCompressedLimitPtr)
1037 return NoCompression;
1041 /** The function copies non repeating 32-bit buffer to 24-bit compressed buffer*/
1042 int BitmapCompiler::WriteCompressedThirtyTwoUBitData(char*& aDest,char* aData,
1043 int aLength, const char* aDestEnd)
1046 return CompressionError;
1048 if(!CanCopy24bppData(aDest, aDestEnd, aLength))
1049 return NoCompression;
1051 while (aLength > 128)
1054 for(int ii = 0; ii < 128; ii++)
1056 memcpy(aDest,aData, 3);
1063 *aDest++ = char(-aLength);
1067 memcpy(aDest,aData, 3);
1075 /** The function copies repeating colour to the 24-bit compressed buffer */
1076 int BitmapCompiler::WriteCompressedThirtyTwoUBitValues(char*& aDest,char aComponent1,
1077 char aComponent2,char aComponent3,
1078 int aLength, const char* aDestEnd)
1081 return CompressionError;
1083 if(!CanWrite24bppValue(aDest, aDestEnd, aLength))
1084 return NoCompression;
1085 while (aLength > 128)
1088 *aDest++ = aComponent1;
1089 *aDest++ = aComponent2;
1090 *aDest++ = aComponent3;
1094 *aDest++ = char(aLength-1);
1095 *aDest++ = aComponent1;
1096 *aDest++ = aComponent2;
1097 *aDest++ = aComponent3;
1102 int BitmapCompiler::TrueColorPointerCompare(char* aColorPointer,char aComponent1,char aComponent2,char aComponent3)
1104 return (*aColorPointer == aComponent1 && *(aColorPointer + 1) == aComponent2 && *(aColorPointer + 2) == aComponent3);
1107 /** The function encodes 32-bit buffer into 32-bit stream by using RLE compression algorithm*/
1108 int BitmapCompiler::CompressThirtyTwoABitData(char*& aDest,char* aSrce,int aSizeInBytes)
1110 const char* destEnd = aDest + aSizeInBytes;
1111 char* srceLimitPtr = aSrce + aSizeInBytes;
1112 char* srceLimitPtrMinusFour = srceLimitPtr - 4; // four bytes == one pixel
1113 char* destCompressedLimitPtr = aDest + (aSizeInBytes * 7 / 8);
1116 while (aSrce < srceLimitPtrMinusFour)
1118 char* runStartPtr = aSrce;
1119 char component1 = *aSrce++;
1120 char component2 = *aSrce++;
1121 char component3 = *aSrce++;
1122 char component4 = *aSrce++;
1124 if (ColorAlphaPointerCompare(aSrce,component1,component2,component3,component4))
1128 while (aSrce < srceLimitPtr && ColorAlphaPointerCompare(aSrce,component1,component2,component3,component4));
1129 int eqPixelLength = (aSrce - runStartPtr) / 4;
1130 ret = WriteCompressedThirtyTwoABitValues(aDest,component1,component2,component3,component4,eqPixelLength,destEnd);
1134 else //non repeating pixel buffer
1139 component1 = *aSrce++;
1140 component2 = *aSrce++;
1141 component3 = *aSrce++;
1142 component4 = *aSrce++;
1143 more = (aSrce < srceLimitPtr);
1144 eqRun = more && ColorAlphaPointerCompare(aSrce,component1,component2,component3,component4);
1145 } while (more && !eqRun);
1148 int nePixelLength = (aSrce - runStartPtr) / 4;
1149 ret = WriteCompressedThirtyTwoABitData(aDest,runStartPtr,nePixelLength,destEnd);
1153 if (aDest > destCompressedLimitPtr)
1154 return NoCompression;
1157 if (aSrce < srceLimitPtr)
1159 ret = WriteCompressedThirtyTwoABitData(aDest,aSrce,(srceLimitPtr - aSrce) / 4, destEnd);
1164 if (aDest > destCompressedLimitPtr)
1165 return NoCompression;
1169 /** The function copies non repeating 32-bit buffer to 32-bit compressed buffer*/
1170 int BitmapCompiler::WriteCompressedThirtyTwoABitData(char*& aDest,char* aData,
1171 int aLength, const char* aDestEnd)
1174 return CompressionError;
1176 if(!CanCopy32bppData(aDest, aDestEnd, aLength))
1177 return NoCompression;
1179 while (aLength > 128)
1182 memcpy(aDest,aData,512);
1188 *aDest++ = char(-aLength);
1189 memcpy(aDest,aData, aLength*4);
1194 /** The function copies repeating pixels including alpha to a 32-bit compressed buffer */
1195 int BitmapCompiler::WriteCompressedThirtyTwoABitValues(char*& aDest,char aComponent1,
1196 char aComponent2,char aComponent3,
1197 char aComponent4,int aLength,
1198 const char* aDestEnd)
1202 return CompressionError;
1204 if(!CanWrite32bppValue(aDest, aDestEnd, aLength))
1205 return NoCompression;
1206 while (aLength > 128)
1209 *aDest++ = aComponent1;
1210 *aDest++ = aComponent2;
1211 *aDest++ = aComponent3;
1212 *aDest++ = aComponent4;
1216 *aDest++ = char(aLength-1);
1217 *aDest++ = aComponent1;
1218 *aDest++ = aComponent2;
1219 *aDest++ = aComponent3;
1220 *aDest++ = aComponent4;
1225 int BitmapCompiler::ColorAlphaPointerCompare(char* aColorPointer,char aComponent1,char aComponent2,char aComponent3,char aComponent4)
1227 return (*aColorPointer == aComponent1 && *(aColorPointer + 1) == aComponent2 && *(aColorPointer + 2) == aComponent3 && *(aColorPointer + 3) == aComponent4);
1230 int BitmapCompiler::ReadHexString(char aHexBuf[10],char*& aDataPtr,char* aDataPtrLimit)
1232 while (aDataPtr < aDataPtrLimit)
1234 if (aDataPtr[0] == '0')
1236 if (aDataPtr[1] == 'x' || aDataPtr[1] == 'X')
1238 memcpy(aHexBuf,aDataPtr,10);
1249 int BitmapCompiler::HexToInt(char aHighNibble,char aLowNibble)
1251 return (HexToInt(aHighNibble) << 4) + HexToInt(aLowNibble);
1254 int BitmapCompiler::HexToInt(char aNibble)
1258 if (aNibble >= '0' && aNibble <= '9')
1259 value = aNibble - '0';
1260 else if (aNibble >= 'a' && aNibble <= 'f')
1261 value = aNibble - 'a' + 10;
1262 else if (aNibble >= 'A' && aNibble <= 'F')
1263 value = aNibble - 'A' + 10;
1268 void BitmapCompiler::CopyTail(void* aDst, void* aSrc, int aFullSize, int aSkipped)
1270 memcpy(aDst, ((char*)aSrc)+aSkipped, aFullSize-aSkipped);