os/graphics/graphicstools/gdi_tools/bmconv/PBMTOBM.CPP
changeset 0 bde4ae8d615e
     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 +	}