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.
16 #if defined(__MSVCDOTNET__) || defined(__TOOLS2__)
20 #else //!__MSVCDOTNET__ && !__TOOLS2__
22 #endif //__MSVCDOTNET__
26 #include <sys/types.h>
32 Returns an informative error message, the result of the program actions performed.
33 @return Informative error string
34 @param aErrorNumber The error returned from the actions performed
35 @param aDestfile The multiple bitmap store file name
36 @param aDestCreated True if the multiple bitmap store has been created/modified
39 char* ErrorMessage(int aErrorNumber, char* aDestfile=NULL, bool aDestCreated=false)
41 // Remove the multiple bitmap store if it has been created/modified during an fstream session and there has been an error
42 if(aDestfile && (aErrorNumber != NoError) && (aDestCreated == true))
52 return "Out of memory.";
54 return "Bad argument.";
56 return "File does not exist";
58 return "Bad source file(s).";
60 return "Bad destination file(s).";
62 return "Bad command file.";
64 return "Number of sources/targets mismatch.";
66 return "Too many arguments.";
67 case UnknownCompression:
68 return "Unknown source compression type.";
69 case CompressionError:
70 return "Compression error.";
71 case DecompressionError:
72 return "Decompression error.";
74 return "Invalid bitmap mode specified.";
76 return "Bad palette file.";
77 case PaletteSupportNotImplemented:
78 return "Palettes not supported";
80 return "Alpha bitmap file does not exist";
82 return "Alpha channel bitmap's dimensions don't match pixel bitmap's dimensions.";
84 return "Alpha channel bitmap must be 8bpp.";
86 return "Unknown error!";
94 cout << "BMCONV version "<< version << ".\n";
97 void Report(int aError)
100 cout << ErrorMessage(aError) << "\n";
104 Compiliation information to print to the user at the end of the program.
105 @param aQuiet Flag if the user selected quiet output mode
106 @param aError The error returned from the actions performed
107 @param aType The multiple bitmap store type created
108 @param aDestfile The multiple bitmap store file name
109 @param aBitmapFiles The array of bitmaps used
110 @param aNumFiles The amount of bitmaps used
111 @param aDestCreated True if the multiple bitmap store has been created/modified
114 void CompilationReport(int aQuiet,int aError,TStoreType aType,char* aDestfile,char** aBitmapFiles,int aNumFiles, bool aDestCreated)
116 if(!aQuiet || aError)
119 cout << "Compiling...\n";
121 cout << "Multiple bitmap store type: ";
122 if(aType==EFileStore)
123 cout << "File store" << "\n";
124 else if(aType==ERomStore)
125 cout << "ROM image store" << "\n";
126 else if(aType==ECompressedRomStore)
127 cout << "Compressed ROM image store" << "\n";
129 cout << "Epoc file: " << aDestfile << "\n\n";
130 for(int count=0;count<aNumFiles;count++)
132 cout << "Bitmap file " << count+1 << " : ";
133 cout << aBitmapFiles[count] << "\n";
135 cout << ErrorMessage(aError, aDestfile, aDestCreated) << "\n";
139 void DecompilationReport(int aError,char* aDestfile,char** aBitmapFiles,int aNumFiles)
142 cout << "Decompiling...\n";
144 cout << "Epoc file: " << aDestfile << "\n\n";
145 for(int count=0;count<aNumFiles;count++)
147 cout << "Bitmap file " << count+1 << " : ";
148 cout << aBitmapFiles[count] << "\n";
150 cout << ErrorMessage(aError) << "\n";
156 cout << "BMCONV version "<< version << ".\n";
157 cout << "Symbian OS multiple bitmap file/rom store conversion program.\n";
158 cout << "Copyright (c) 2007 Nokia Corporation and/or its subsidiary(-ies).";
162 cout << "BMCONV [-r|-s|-n] [-hfilename] [-q] [-pfilename] epocfile [OPT]bmp_1 ... [OPT]bmp_n\n";
163 cout << "BMCONV [-r|-s|-n] [-q] [-pfilename] epocfile -mepocfile2\n";
164 cout << "BMCONV -u epocfile bmp_1 [... bmp_n]\n";
165 cout << "BMCONV -v epocfile\n";
166 cout << "BMCONV commandfile\n";
168 cout << " -r specifies a ROM image destination file,\n";
169 cout << " -s specifies a compressed ROM image file,\n";
170 cout << " -n disables bitmap File Store compression,\n";
171 cout << " the default is a compressed File Store file.\n\n";
172 cout << " -q specifies quiet mode - only errors are reported.\n\n";
173 cout << " -hfilename specifies the filename for the automatic\n";
174 cout << " generation of a header file for inclusion into code.\n\n";
175 cout << " -pfilename gives the filename of a palette file containing 256 hex\n";
176 cout << " numbers (0x00BBGGRR) specifying the palette for 8bpp colour bitmaps.\n";
177 cout << " (Omission results in the use of a default palette.)\n\n";
178 cout << " OPT may be one of -1, -2, -4, -8, -c4, -c8, -c12, -c16, -c24 -c32 -c32a\n";
179 cout << " specifying bits per pixel and grey-scale-colour, or -mepocfile2\n";
180 cout << " to specify an existing multiple bitmap file. default is -2.\n\n";
181 cout << " To avoid ambiguity when specifying -c32 with a bitmap file whose name\n";
182 cout << " begins with an 'a', use a relative or direct directory reference\n";
183 cout << " e.g. -c32.\\abitmap.bmp or -c32c:\\abitmap.bmp\n";
184 cout << " Directory names must not include spaces.\n\n";
185 cout << " -c32a specifies use of an alpha channel in a 32bpp bitmap. Alpha data\n";
186 cout << " is supplied in a separate 8bpp bmp file with identical dimensions to\n";
187 cout << " the pixel data. This file must be named as bmp_n with the suffix '-alpha'\n";
188 cout << " e.g. if bmp_1 is 'my.bmp' then the file 'my-alpha.bmp' is required in the\n";
189 cout << " same directory. The alpha file does not need to be specified.\n\n";
190 cout << " epocfile specifies the epoc multi-bitmap file name.\n";
191 cout << " bmp_n specifies the nth bitmap file name.\n\n";
192 cout << " -u decompiles epocfile to bmp_1,...,bmp_n.\n";
193 cout << " If an alpha channel is present then a further, 8bpp file is output for \n";
194 cout << " the alpha data, named with an '-alpha' suffix as described above.\n\n";
195 cout << " -v displays a summary of the bitmaps in epocfile\n";
196 cout << " otherwise bmp_1,...,bmp_n are compiled to epocfile\n\n";
197 cout << " commandfile specifies a file containing the commandline\n";
198 cout << " with commands separated by spaces or newlines.\n\n";
199 cout << " When bmconv is used on Windows, options may start with '/' or '-'\n";
203 int IsWhiteSpace(char aCharacter)
205 return(aCharacter==' ' || aCharacter=='\n' || aCharacter=='\r' || aCharacter==0x1a);
208 int ProcessCommandFile(char* aCommandFileName,char** aArgPtrs,int& aNumArgs)
210 struct stat fileinfo;
211 if (stat(aCommandFileName,&fileinfo)==-1)
214 int filesize=fileinfo.st_size;
217 #if defined(__MSVCDOTNET__) || defined(__TOOLS2__)
218 fstream commandfile(aCommandFileName, ios::in | ios::binary);
219 #else //!__MSVCDOTNET__
220 fstream commandfile(aCommandFileName, ios::in | ios::binary | ios::nocreate);
221 #endif //__MSVCDOTNET__
222 if(!commandfile.is_open())
225 char* commandData=new char[filesize+1];
226 if(commandData==NULL)
229 memset(commandData,0,filesize+1);
230 commandfile.read(commandData,filesize);
231 commandData[filesize]='\0';
233 char* commandptr = (char*)commandData;
234 char* commandptrLimit = (char*)(commandData + filesize);
235 while (commandptr < commandptrLimit)
237 if(*commandptr=='/' && *(commandptr+1)=='/')
238 while(*commandptr!='\n' && *commandptr!='\r' && commandptr < commandptrLimit)
240 else if (*commandptr==0x1a)
245 commandptr = (char*)commandData;
246 while (commandptr < commandptrLimit)
248 while(IsWhiteSpace(*commandptr) && commandptr < commandptrLimit)
250 if (commandptr == commandptrLimit)
252 aArgPtrs[aNumArgs]=commandptr;
253 while(!IsWhiteSpace(*commandptr) && commandptr < commandptrLimit)
255 if (commandptr == commandptrLimit)
264 int Decompile(int aArgc,int aNumArgs,char** aArgPtrs)
267 char* destfilename=aArgPtrs[1];
269 if(aArgc>=4 || aArgc==2)
271 for(int count=2;count<aNumArgs;count++)
274 ret=pl.LoadEpocBitmap(destfilename,count-2);
275 if(!ret) ret=pl.SaveBitmap(aArgPtrs[count]);
278 DecompilationReport(ret,destfilename,&aArgPtrs[2],aNumArgs-2);
281 DecompilationReport(ret,NULL,NULL,0);
286 int Compile(int aNumArgs,int aArgArraySize, char** aArgPtrs)
288 TStoreType storeType = EFileStore;
291 char* headerfilename = NULL;
292 char* palettefilename = NULL;
293 char* destfilename = NULL;
294 int ret = OutOfRange;
295 bool aDestCreated = false;
297 for(int argnum=0;argnum<aNumArgs;argnum++)
299 if(aArgPtrs[argnum] && (aArgPtrs[argnum][0] == OPTCHAR || aArgPtrs[argnum][0]==ALTERNATE_OPTCHAR))
301 if(aArgPtrs[argnum][1]=='r' || aArgPtrs[argnum][1]=='R')
303 if(storeType==ECompressedRomStore)
306 CompilationReport(quiet,ret,storeType,NULL,NULL,0,aDestCreated);
310 aArgPtrs[argnum] = NULL;
312 else if(aArgPtrs[argnum][1]=='s' || aArgPtrs[argnum][1]=='S')
314 if(storeType==ERomStore)
317 CompilationReport(quiet,ret,storeType,NULL,NULL,0,aDestCreated);
320 storeType=ECompressedRomStore;
321 aArgPtrs[argnum] = NULL;
323 else if(aArgPtrs[argnum][1]=='n' || aArgPtrs[argnum][1]=='N')
326 aArgPtrs[argnum] = NULL;
328 else if(aArgPtrs[argnum][1]=='h' || aArgPtrs[argnum][1]=='H')
330 headerfilename = &aArgPtrs[argnum][2];
331 aArgPtrs[argnum] = NULL;
333 else if(aArgPtrs[argnum][1]=='q' || aArgPtrs[argnum][1]=='Q')
336 aArgPtrs[argnum] = NULL;
338 else if(aArgPtrs[argnum][1]=='p' || aArgPtrs[argnum][1]=='P')
340 palettefilename = &aArgPtrs[argnum][2];
341 aArgPtrs[argnum] = NULL;
345 break; // the RNHQP arguments must precede the output filename
349 while(firstsource<aArgArraySize && aArgPtrs[firstsource]==NULL)
351 if(firstsource==aArgArraySize) firstsource=0;
352 destfilename=aArgPtrs[firstsource];
354 int numsources=firstsource;
355 while(numsources<aArgArraySize && aArgPtrs[numsources]!=NULL)
357 if(numsources==aArgArraySize) numsources=0;
358 numsources-=firstsource;
362 BitmapCompiler mp(&aArgPtrs[firstsource],numsources);
363 ret = mp.Compile(storeType,compression,destfilename,headerfilename,palettefilename);
364 aDestCreated = true; // The multiple bitmap store has been created/modified
367 CompilationReport(quiet,ret,storeType,destfilename,&aArgPtrs[firstsource],aNumArgs-firstsource,aDestCreated);
372 void GetInfo(char* aSourceFile)
379 int ret = pl.EpocBitmapCount(aSourceFile, numSources, romFormat);
382 cout << "Problem reading number of bitmaps \n";
383 cout << ErrorMessage(ret) << "\n";
387 cout << aSourceFile << " is a " << (romFormat? "ROM image":"File store")
388 << " containing " << numSources << ((numSources==1)? " bitmap\n":" bitmaps\n");
390 for (int count = 0;count<numSources;count++)
392 ret = pl.LoadEpocBitmap(aSourceFile,count);
393 if (ret == OutOfRange)
398 cout << "Problem loading bitmap number " << count << "\n";
399 cout << ErrorMessage(ret) << "\n";
404 SEpocBitmapHeader h = pl.Header();
405 cout << "Bitmap " << count + 1 << " information:\n";
406 cout << "Pixel size " << h.iWidthInPixels << " x " << h.iHeightInPixels << "\n";
407 cout << "Twips size " << h.iWidthInTwips << " x " << h.iHeightInTwips << "\n";
408 cout << h.iBitsPerPixel << " Bpp ";
409 if (h.iColor == EColorBitmap)
411 else if (h.iColor == EColorBitmapAlpha || h.iColor == EColorBitmapAlphaPM)
412 cout << "Colour with alpha channel";
413 else if(h.iColor == EMonochromeBitmap)
414 cout << "Monochrome";
416 cout << "Unknown colour format";
418 if (h.iPaletteEntries > 0)
419 cout << "Palette entries " << h.iPaletteEntries;
421 int byteSize = BitmapUtils::ByteWidth(h.iWidthInPixels,h.iBitsPerPixel) * h.iHeightInPixels;
422 int compressionRatio = 0;
424 compressionRatio = (h.iBitmapSize - sizeof(SEpocBitmapHeader)) * 100 / byteSize;
426 switch (h.iCompression)
428 case ENoBitmapCompression:
429 cout << "No compression\n";
431 case EByteRLECompression:
432 cout << "Bytewise RLE compression " << compressionRatio << "%\n";
434 case ETwelveBitRLECompression:
435 cout << "12 bit RLE compression " << compressionRatio << "%\n";
437 case ESixteenBitRLECompression:
438 cout << "16 bit RLE compression " << compressionRatio << "%\n";
440 case ETwentyFourBitRLECompression:
441 cout << "24 bit RLE compression " << compressionRatio << "%\n";
443 case EThirtyTwoUBitRLECompression:
444 cout << "unsigned 32 bit RLE compression (no alpha channel) " << compressionRatio << "%\n";
446 case EThirtyTwoABitRLECompression:
447 cout << "unsigned 32 bit RLE compression (with alpha channel) " << compressionRatio << "%\n";
449 // case ERLECompressionLast: // Added to supress unhandled switch warning
462 TAutoPtr(char** aPtr) :
474 int main(int argc,char* argv[],char* [])
482 int optMaxCnt = argc;
484 if(argc==2) // The single argument must be a command file name
486 struct stat fileinfo;
487 if (stat(argv[1],&fileinfo)==-1)
492 optMaxCnt = fileinfo.st_size;
495 char** argptrs = new char*[optMaxCnt];
501 TAutoPtr autoPtr(argptrs);
502 memset(argptrs, 0, optMaxCnt * sizeof(char*));
505 if(argc>2) // Explicit arguments are present
507 for(int count=0;count<argc-1;count++)
508 argptrs[count]=argv[count+1];
511 else // The single argument must be a command file name
513 int ret = ProcessCommandFile(argv[1],argptrs,numargs);
521 if ((argptrs[0]!=NULL && (argptrs[0][0]==OPTCHAR || argptrs[0][0]==ALTERNATE_OPTCHAR)) && (argptrs[0][1]=='u' || argptrs[0][1]=='U')) {
522 return Decompile(argc,numargs,argptrs); }
524 if ((argptrs[0]!=NULL && (argptrs[0][0]==OPTCHAR || argptrs[0][0]==ALTERNATE_OPTCHAR)) && (argptrs[0][1]=='v' || argptrs[0][1]=='V'))
530 return Compile(numargs,optMaxCnt,argptrs);