1.1 --- /dev/null Thu Jan 01 00:00:00 1970 +0000
1.2 +++ b/os/graphics/graphicstools/gdi_tools/bmconv/MAINFUNC.CPP Fri Jun 15 03:10:57 2012 +0200
1.3 @@ -0,0 +1,532 @@
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 +#if defined(__MSVCDOTNET__) || defined(__TOOLS2__)
1.20 +#include <ostream>
1.21 +#include <iostream>
1.22 +using namespace std;
1.23 +#else //!__MSVCDOTNET__ && !__TOOLS2__
1.24 +#include <ostream.h>
1.25 +#endif //__MSVCDOTNET__
1.26 +
1.27 +#include "TOOLSVER.H"
1.28 +#include "BMCONV.H"
1.29 +#include <sys/types.h>
1.30 +#include <sys/stat.h>
1.31 +#include <stdio.h>
1.32 +
1.33 +
1.34 +/**
1.35 +Returns an informative error message, the result of the program actions performed.
1.36 +@return Informative error string
1.37 +@param aErrorNumber The error returned from the actions performed
1.38 +@param aDestfile The multiple bitmap store file name
1.39 +@param aDestCreated True if the multiple bitmap store has been created/modified
1.40 +*/
1.41 +
1.42 +char* ErrorMessage(int aErrorNumber, char* aDestfile=NULL, bool aDestCreated=false)
1.43 + {
1.44 + // Remove the multiple bitmap store if it has been created/modified during an fstream session and there has been an error
1.45 + if(aDestfile && (aErrorNumber != NoError) && (aDestCreated == true))
1.46 + {
1.47 + remove(aDestfile);
1.48 + }
1.49 +
1.50 + switch(aErrorNumber)
1.51 + {
1.52 + case NoError:
1.53 + return "Success.";
1.54 + case NoMemory:
1.55 + return "Out of memory.";
1.56 + case Arg:
1.57 + return "Bad argument.";
1.58 + case Files:
1.59 + return "File does not exist";
1.60 + case SourceFile:
1.61 + return "Bad source file(s).";
1.62 + case DestFile:
1.63 + return "Bad destination file(s).";
1.64 + case CommandFile:
1.65 + return "Bad command file.";
1.66 + case OutOfRange:
1.67 + return "Number of sources/targets mismatch.";
1.68 + case TooManyArgs:
1.69 + return "Too many arguments.";
1.70 + case UnknownCompression:
1.71 + return "Unknown source compression type.";
1.72 + case CompressionError:
1.73 + return "Compression error.";
1.74 + case DecompressionError:
1.75 + return "Decompression error.";
1.76 + case Bpp:
1.77 + return "Invalid bitmap mode specified.";
1.78 + case PaletteFile:
1.79 + return "Bad palette file.";
1.80 + case PaletteSupportNotImplemented:
1.81 + return "Palettes not supported";
1.82 + case AlphaFiles:
1.83 + return "Alpha bitmap file does not exist";
1.84 + case AlphaDimensions:
1.85 + return "Alpha channel bitmap's dimensions don't match pixel bitmap's dimensions.";
1.86 + case AlphaBpp:
1.87 + return "Alpha channel bitmap must be 8bpp.";
1.88 + default:
1.89 + return "Unknown error!";
1.90 + };
1.91 + }
1.92 +
1.93 +void Header()
1.94 + {
1.95 + cout << "\n";
1.96 + cout << "\n";
1.97 + cout << "BMCONV version "<< version << ".\n";
1.98 + }
1.99 +
1.100 +void Report(int aError)
1.101 + {
1.102 + Header();
1.103 + cout << ErrorMessage(aError) << "\n";
1.104 + }
1.105 +
1.106 +/**
1.107 +Compiliation information to print to the user at the end of the program.
1.108 +@param aQuiet Flag if the user selected quiet output mode
1.109 +@param aError The error returned from the actions performed
1.110 +@param aType The multiple bitmap store type created
1.111 +@param aDestfile The multiple bitmap store file name
1.112 +@param aBitmapFiles The array of bitmaps used
1.113 +@param aNumFiles The amount of bitmaps used
1.114 +@param aDestCreated True if the multiple bitmap store has been created/modified
1.115 +*/
1.116 +
1.117 +void CompilationReport(int aQuiet,int aError,TStoreType aType,char* aDestfile,char** aBitmapFiles,int aNumFiles, bool aDestCreated)
1.118 + {
1.119 + if(!aQuiet || aError)
1.120 + {
1.121 + Header();
1.122 + cout << "Compiling...\n";
1.123 + if(aType!=ENoStore)
1.124 + cout << "Multiple bitmap store type: ";
1.125 + if(aType==EFileStore)
1.126 + cout << "File store" << "\n";
1.127 + else if(aType==ERomStore)
1.128 + cout << "ROM image store" << "\n";
1.129 + else if(aType==ECompressedRomStore)
1.130 + cout << "Compressed ROM image store" << "\n";
1.131 + if(aDestfile!=NULL)
1.132 + cout << "Epoc file: " << aDestfile << "\n\n";
1.133 + for(int count=0;count<aNumFiles;count++)
1.134 + {
1.135 + cout << "Bitmap file " << count+1 << " : ";
1.136 + cout << aBitmapFiles[count] << "\n";
1.137 + }
1.138 + cout << ErrorMessage(aError, aDestfile, aDestCreated) << "\n";
1.139 + }
1.140 + }
1.141 +
1.142 +void DecompilationReport(int aError,char* aDestfile,char** aBitmapFiles,int aNumFiles)
1.143 + {
1.144 + Header();
1.145 + cout << "Decompiling...\n";
1.146 + if(aDestfile!=NULL)
1.147 + cout << "Epoc file: " << aDestfile << "\n\n";
1.148 + for(int count=0;count<aNumFiles;count++)
1.149 + {
1.150 + cout << "Bitmap file " << count+1 << " : ";
1.151 + cout << aBitmapFiles[count] << "\n";
1.152 + }
1.153 + cout << ErrorMessage(aError) << "\n";
1.154 + }
1.155 +
1.156 +void Usage()
1.157 + {
1.158 + cout << "\n";
1.159 + cout << "BMCONV version "<< version << ".\n";
1.160 + cout << "Symbian OS multiple bitmap file/rom store conversion program.\n";
1.161 + cout << "Copyright (c) 2007 Nokia Corporation and/or its subsidiary(-ies).";
1.162 + cout << "\n";
1.163 + cout << "\n";
1.164 + cout << "Usage:\n";
1.165 + cout << "BMCONV [-r|-s|-n] [-hfilename] [-q] [-pfilename] epocfile [OPT]bmp_1 ... [OPT]bmp_n\n";
1.166 + cout << "BMCONV [-r|-s|-n] [-q] [-pfilename] epocfile -mepocfile2\n";
1.167 + cout << "BMCONV -u epocfile bmp_1 [... bmp_n]\n";
1.168 + cout << "BMCONV -v epocfile\n";
1.169 + cout << "BMCONV commandfile\n";
1.170 + cout << "\n";
1.171 + cout << " -r specifies a ROM image destination file,\n";
1.172 + cout << " -s specifies a compressed ROM image file,\n";
1.173 + cout << " -n disables bitmap File Store compression,\n";
1.174 + cout << " the default is a compressed File Store file.\n\n";
1.175 + cout << " -q specifies quiet mode - only errors are reported.\n\n";
1.176 + cout << " -hfilename specifies the filename for the automatic\n";
1.177 + cout << " generation of a header file for inclusion into code.\n\n";
1.178 + cout << " -pfilename gives the filename of a palette file containing 256 hex\n";
1.179 + cout << " numbers (0x00BBGGRR) specifying the palette for 8bpp colour bitmaps.\n";
1.180 + cout << " (Omission results in the use of a default palette.)\n\n";
1.181 + cout << " OPT may be one of -1, -2, -4, -8, -c4, -c8, -c12, -c16, -c24 -c32 -c32a\n";
1.182 + cout << " specifying bits per pixel and grey-scale-colour, or -mepocfile2\n";
1.183 + cout << " to specify an existing multiple bitmap file. default is -2.\n\n";
1.184 + cout << " To avoid ambiguity when specifying -c32 with a bitmap file whose name\n";
1.185 + cout << " begins with an 'a', use a relative or direct directory reference\n";
1.186 + cout << " e.g. -c32.\\abitmap.bmp or -c32c:\\abitmap.bmp\n";
1.187 + cout << " Directory names must not include spaces.\n\n";
1.188 + cout << " -c32a specifies use of an alpha channel in a 32bpp bitmap. Alpha data\n";
1.189 + cout << " is supplied in a separate 8bpp bmp file with identical dimensions to\n";
1.190 + cout << " the pixel data. This file must be named as bmp_n with the suffix '-alpha'\n";
1.191 + cout << " e.g. if bmp_1 is 'my.bmp' then the file 'my-alpha.bmp' is required in the\n";
1.192 + cout << " same directory. The alpha file does not need to be specified.\n\n";
1.193 + cout << " epocfile specifies the epoc multi-bitmap file name.\n";
1.194 + cout << " bmp_n specifies the nth bitmap file name.\n\n";
1.195 + cout << " -u decompiles epocfile to bmp_1,...,bmp_n.\n";
1.196 + cout << " If an alpha channel is present then a further, 8bpp file is output for \n";
1.197 + cout << " the alpha data, named with an '-alpha' suffix as described above.\n\n";
1.198 + cout << " -v displays a summary of the bitmaps in epocfile\n";
1.199 + cout << " otherwise bmp_1,...,bmp_n are compiled to epocfile\n\n";
1.200 + cout << " commandfile specifies a file containing the commandline\n";
1.201 + cout << " with commands separated by spaces or newlines.\n\n";
1.202 + cout << " When bmconv is used on Windows, options may start with '/' or '-'\n";
1.203 +
1.204 + }
1.205 +
1.206 +int IsWhiteSpace(char aCharacter)
1.207 + {
1.208 + return(aCharacter==' ' || aCharacter=='\n' || aCharacter=='\r' || aCharacter==0x1a);
1.209 + }
1.210 +
1.211 +int ProcessCommandFile(char* aCommandFileName,char** aArgPtrs,int& aNumArgs)
1.212 + {
1.213 + struct stat fileinfo;
1.214 + if (stat(aCommandFileName,&fileinfo)==-1)
1.215 + return CommandFile;
1.216 +
1.217 + int filesize=fileinfo.st_size;
1.218 + if (filesize==0)
1.219 + return NoError;
1.220 +#if defined(__MSVCDOTNET__) || defined(__TOOLS2__)
1.221 + fstream commandfile(aCommandFileName, ios::in | ios::binary);
1.222 +#else //!__MSVCDOTNET__
1.223 + fstream commandfile(aCommandFileName, ios::in | ios::binary | ios::nocreate);
1.224 +#endif //__MSVCDOTNET__
1.225 + if(!commandfile.is_open())
1.226 + return CommandFile;
1.227 +
1.228 + char* commandData=new char[filesize+1];
1.229 + if(commandData==NULL)
1.230 + return NoMemory;
1.231 +
1.232 + memset(commandData,0,filesize+1);
1.233 + commandfile.read(commandData,filesize);
1.234 + commandData[filesize]='\0';
1.235 +
1.236 + char* commandptr = (char*)commandData;
1.237 + char* commandptrLimit = (char*)(commandData + filesize);
1.238 + while (commandptr < commandptrLimit)
1.239 + {
1.240 + if(*commandptr=='/' && *(commandptr+1)=='/')
1.241 + while(*commandptr!='\n' && *commandptr!='\r' && commandptr < commandptrLimit)
1.242 + *commandptr++=' ';
1.243 + else if (*commandptr==0x1a)
1.244 + *commandptr++=' ';
1.245 + commandptr++;
1.246 + }
1.247 +
1.248 + commandptr = (char*)commandData;
1.249 + while (commandptr < commandptrLimit)
1.250 + {
1.251 + while(IsWhiteSpace(*commandptr) && commandptr < commandptrLimit)
1.252 + *commandptr++='\0';
1.253 + if (commandptr == commandptrLimit)
1.254 + break;
1.255 + aArgPtrs[aNumArgs]=commandptr;
1.256 + while(!IsWhiteSpace(*commandptr) && commandptr < commandptrLimit)
1.257 + commandptr++;
1.258 + if (commandptr == commandptrLimit)
1.259 + break;
1.260 + aNumArgs++;
1.261 + }
1.262 +
1.263 + commandfile.close();
1.264 + return NoError;
1.265 + }
1.266 +
1.267 +int Decompile(int aArgc,int aNumArgs,char** aArgPtrs)
1.268 + {
1.269 + int ret=OutOfRange;
1.270 + char* destfilename=aArgPtrs[1];
1.271 +
1.272 + if(aArgc>=4 || aArgc==2)
1.273 + {
1.274 + for(int count=2;count<aNumArgs;count++)
1.275 + {
1.276 + EpocLoader pl;
1.277 + ret=pl.LoadEpocBitmap(destfilename,count-2);
1.278 + if(!ret) ret=pl.SaveBitmap(aArgPtrs[count]);
1.279 + if(ret) break;
1.280 + }
1.281 + DecompilationReport(ret,destfilename,&aArgPtrs[2],aNumArgs-2);
1.282 + }
1.283 + else
1.284 + DecompilationReport(ret,NULL,NULL,0);
1.285 +
1.286 + return ret;
1.287 + }
1.288 +
1.289 +int Compile(int aNumArgs,int aArgArraySize, char** aArgPtrs)
1.290 + {
1.291 + TStoreType storeType = EFileStore;
1.292 + int compression = 1;
1.293 + int quiet = 0;
1.294 + char* headerfilename = NULL;
1.295 + char* palettefilename = NULL;
1.296 + char* destfilename = NULL;
1.297 + int ret = OutOfRange;
1.298 + bool aDestCreated = false;
1.299 +
1.300 + for(int argnum=0;argnum<aNumArgs;argnum++)
1.301 + {
1.302 + if(aArgPtrs[argnum] && (aArgPtrs[argnum][0] == OPTCHAR || aArgPtrs[argnum][0]==ALTERNATE_OPTCHAR))
1.303 + {
1.304 + if(aArgPtrs[argnum][1]=='r' || aArgPtrs[argnum][1]=='R')
1.305 + {
1.306 + if(storeType==ECompressedRomStore)
1.307 + {
1.308 + ret=TooManyArgs;
1.309 + CompilationReport(quiet,ret,storeType,NULL,NULL,0,aDestCreated);
1.310 + return ret;
1.311 + }
1.312 + storeType=ERomStore;
1.313 + aArgPtrs[argnum] = NULL;
1.314 + }
1.315 + else if(aArgPtrs[argnum][1]=='s' || aArgPtrs[argnum][1]=='S')
1.316 + {
1.317 + if(storeType==ERomStore)
1.318 + {
1.319 + ret=TooManyArgs;
1.320 + CompilationReport(quiet,ret,storeType,NULL,NULL,0,aDestCreated);
1.321 + return ret;
1.322 + }
1.323 + storeType=ECompressedRomStore;
1.324 + aArgPtrs[argnum] = NULL;
1.325 + }
1.326 + else if(aArgPtrs[argnum][1]=='n' || aArgPtrs[argnum][1]=='N')
1.327 + {
1.328 + compression=0;
1.329 + aArgPtrs[argnum] = NULL;
1.330 + }
1.331 + else if(aArgPtrs[argnum][1]=='h' || aArgPtrs[argnum][1]=='H')
1.332 + {
1.333 + headerfilename = &aArgPtrs[argnum][2];
1.334 + aArgPtrs[argnum] = NULL;
1.335 + }
1.336 + else if(aArgPtrs[argnum][1]=='q' || aArgPtrs[argnum][1]=='Q')
1.337 + {
1.338 + quiet=1;
1.339 + aArgPtrs[argnum] = NULL;
1.340 + }
1.341 + else if(aArgPtrs[argnum][1]=='p' || aArgPtrs[argnum][1]=='P')
1.342 + {
1.343 + palettefilename = &aArgPtrs[argnum][2];
1.344 + aArgPtrs[argnum] = NULL;
1.345 + }
1.346 + }
1.347 + else
1.348 + break; // the RNHQP arguments must precede the output filename
1.349 + }
1.350 +
1.351 + int firstsource=0;
1.352 + while(firstsource<aArgArraySize && aArgPtrs[firstsource]==NULL)
1.353 + firstsource++;
1.354 + if(firstsource==aArgArraySize) firstsource=0;
1.355 + destfilename=aArgPtrs[firstsource];
1.356 + firstsource++;
1.357 + int numsources=firstsource;
1.358 + while(numsources<aArgArraySize && aArgPtrs[numsources]!=NULL)
1.359 + numsources++;
1.360 + if(numsources==aArgArraySize) numsources=0;
1.361 + numsources-=firstsource;
1.362 +
1.363 + if (numsources > 0)
1.364 + {
1.365 + BitmapCompiler mp(&aArgPtrs[firstsource],numsources);
1.366 + ret = mp.Compile(storeType,compression,destfilename,headerfilename,palettefilename);
1.367 + aDestCreated = true; // The multiple bitmap store has been created/modified
1.368 + }
1.369 +
1.370 + CompilationReport(quiet,ret,storeType,destfilename,&aArgPtrs[firstsource],aNumArgs-firstsource,aDestCreated);
1.371 +
1.372 + return ret;
1.373 + }
1.374 +
1.375 +void GetInfo(char* aSourceFile)
1.376 + {
1.377 + Header();
1.378 +
1.379 + EpocLoader pl;
1.380 + int numSources=-1;
1.381 + int romFormat=0;
1.382 + int ret = pl.EpocBitmapCount(aSourceFile, numSources, romFormat);
1.383 + if (ret)
1.384 + {
1.385 + cout << "Problem reading number of bitmaps \n";
1.386 + cout << ErrorMessage(ret) << "\n";
1.387 + return;
1.388 + }
1.389 +
1.390 + cout << aSourceFile << " is a " << (romFormat? "ROM image":"File store")
1.391 + << " containing " << numSources << ((numSources==1)? " bitmap\n":" bitmaps\n");
1.392 +
1.393 + for (int count = 0;count<numSources;count++)
1.394 + {
1.395 + ret = pl.LoadEpocBitmap(aSourceFile,count);
1.396 + if (ret == OutOfRange)
1.397 + break;
1.398 + cout << "\n";
1.399 + if (ret)
1.400 + {
1.401 + cout << "Problem loading bitmap number " << count << "\n";
1.402 + cout << ErrorMessage(ret) << "\n";
1.403 + break;
1.404 + }
1.405 + else
1.406 + {
1.407 + SEpocBitmapHeader h = pl.Header();
1.408 + cout << "Bitmap " << count + 1 << " information:\n";
1.409 + cout << "Pixel size " << h.iWidthInPixels << " x " << h.iHeightInPixels << "\n";
1.410 + cout << "Twips size " << h.iWidthInTwips << " x " << h.iHeightInTwips << "\n";
1.411 + cout << h.iBitsPerPixel << " Bpp ";
1.412 + if (h.iColor == EColorBitmap)
1.413 + cout << "Colour";
1.414 + else if (h.iColor == EColorBitmapAlpha || h.iColor == EColorBitmapAlphaPM)
1.415 + cout << "Colour with alpha channel";
1.416 + else if(h.iColor == EMonochromeBitmap)
1.417 + cout << "Monochrome";
1.418 + else
1.419 + cout << "Unknown colour format";
1.420 + cout << "\n";
1.421 + if (h.iPaletteEntries > 0)
1.422 + cout << "Palette entries " << h.iPaletteEntries;
1.423 +
1.424 + int byteSize = BitmapUtils::ByteWidth(h.iWidthInPixels,h.iBitsPerPixel) * h.iHeightInPixels;
1.425 + int compressionRatio = 0;
1.426 + if (byteSize > 0)
1.427 + compressionRatio = (h.iBitmapSize - sizeof(SEpocBitmapHeader)) * 100 / byteSize;
1.428 +
1.429 + switch (h.iCompression)
1.430 + {
1.431 + case ENoBitmapCompression:
1.432 + cout << "No compression\n";
1.433 + break;
1.434 + case EByteRLECompression:
1.435 + cout << "Bytewise RLE compression " << compressionRatio << "%\n";
1.436 + break;
1.437 + case ETwelveBitRLECompression:
1.438 + cout << "12 bit RLE compression " << compressionRatio << "%\n";
1.439 + break;
1.440 + case ESixteenBitRLECompression:
1.441 + cout << "16 bit RLE compression " << compressionRatio << "%\n";
1.442 + break;
1.443 + case ETwentyFourBitRLECompression:
1.444 + cout << "24 bit RLE compression " << compressionRatio << "%\n";
1.445 + break;
1.446 + case EThirtyTwoUBitRLECompression:
1.447 + cout << "unsigned 32 bit RLE compression (no alpha channel) " << compressionRatio << "%\n";
1.448 + break;
1.449 + case EThirtyTwoABitRLECompression:
1.450 + cout << "unsigned 32 bit RLE compression (with alpha channel) " << compressionRatio << "%\n";
1.451 + break;
1.452 + // case ERLECompressionLast: // Added to supress unhandled switch warning
1.453 + default:
1.454 + break;
1.455 + }
1.456 + }
1.457 + }
1.458 +
1.459 + cout << "\n";
1.460 + }
1.461 +
1.462 +class TAutoPtr
1.463 + {
1.464 +public:
1.465 + TAutoPtr(char** aPtr) :
1.466 + iPtr(aPtr)
1.467 + {
1.468 + }
1.469 + ~TAutoPtr()
1.470 + {
1.471 + delete iPtr;
1.472 + }
1.473 +private:
1.474 + char** iPtr;
1.475 + };
1.476 +
1.477 +int main(int argc,char* argv[],char* [])
1.478 + {
1.479 + if (argc <= 1)
1.480 + {
1.481 + Usage();
1.482 + return 0;
1.483 + }
1.484 +
1.485 + int optMaxCnt = argc;
1.486 +
1.487 + if(argc==2) // The single argument must be a command file name
1.488 + {
1.489 + struct stat fileinfo;
1.490 + if (stat(argv[1],&fileinfo)==-1)
1.491 + {
1.492 + Report(CommandFile);
1.493 + return 0;
1.494 + }
1.495 + optMaxCnt = fileinfo.st_size;
1.496 + }
1.497 +
1.498 + char** argptrs = new char*[optMaxCnt];
1.499 + if(!argptrs)
1.500 + {
1.501 + Report(NoMemory);
1.502 + return 0;
1.503 + }
1.504 + TAutoPtr autoPtr(argptrs);
1.505 + memset(argptrs, 0, optMaxCnt * sizeof(char*));
1.506 +
1.507 + int numargs = 0;
1.508 + if(argc>2) // Explicit arguments are present
1.509 + {
1.510 + for(int count=0;count<argc-1;count++)
1.511 + argptrs[count]=argv[count+1];
1.512 + numargs = argc-1;
1.513 + }
1.514 + else // The single argument must be a command file name
1.515 + {
1.516 + int ret = ProcessCommandFile(argv[1],argptrs,numargs);
1.517 + if (ret)
1.518 + {
1.519 + Report(ret);
1.520 + return 0;
1.521 + }
1.522 + }
1.523 +
1.524 + if ((argptrs[0]!=NULL && (argptrs[0][0]==OPTCHAR || argptrs[0][0]==ALTERNATE_OPTCHAR)) && (argptrs[0][1]=='u' || argptrs[0][1]=='U')) {
1.525 + return Decompile(argc,numargs,argptrs); }
1.526 +
1.527 + if ((argptrs[0]!=NULL && (argptrs[0][0]==OPTCHAR || argptrs[0][0]==ALTERNATE_OPTCHAR)) && (argptrs[0][1]=='v' || argptrs[0][1]=='V'))
1.528 + {
1.529 + GetInfo(argptrs[1]);
1.530 + return 0;
1.531 + }
1.532 +
1.533 + return Compile(numargs,optMaxCnt,argptrs);
1.534 + }
1.535 +