2001-11-03 07:10:51 +00:00
|
|
|
/* vim:set tw=80 expandtab softtabstop=4 ts=4 sw=4: */
|
2004-04-18 22:01:16 +00:00
|
|
|
/* ***** BEGIN LICENSE BLOCK *****
|
|
|
|
* Version: MPL 1.1/GPL 2.0/LGPL 2.1
|
|
|
|
*
|
|
|
|
* The contents of this file are subject to the Mozilla Public License Version
|
|
|
|
* 1.1 (the "License"); you may not use this file except in compliance with
|
|
|
|
* the License. You may obtain a copy of the License at
|
|
|
|
* http://www.mozilla.org/MPL/
|
|
|
|
*
|
|
|
|
* Software distributed under the License is distributed on an "AS IS" basis,
|
|
|
|
* WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License
|
|
|
|
* for the specific language governing rights and limitations under the
|
|
|
|
* License.
|
|
|
|
*
|
2001-11-03 07:10:51 +00:00
|
|
|
* The Original Code is the Mozilla ICO Decoder.
|
2004-04-18 22:01:16 +00:00
|
|
|
*
|
|
|
|
* The Initial Developer of the Original Code is
|
|
|
|
* Netscape.
|
|
|
|
* Portions created by the Initial Developer are Copyright (C) 2001
|
|
|
|
* the Initial Developer. All Rights Reserved.
|
|
|
|
*
|
2001-11-03 07:10:51 +00:00
|
|
|
* Contributor(s):
|
|
|
|
* David Hyatt <hyatt@netscape.com> (Original Author)
|
2003-07-23 18:39:11 +00:00
|
|
|
* Christian Biesinger <cbiesinger@web.de>
|
2001-11-03 07:10:51 +00:00
|
|
|
*
|
|
|
|
* Alternatively, the contents of this file may be used under the terms of
|
2004-04-18 22:01:16 +00:00
|
|
|
* either the GNU General Public License Version 2 or later (the "GPL"), or
|
|
|
|
* the GNU Lesser General Public License Version 2.1 or later (the "LGPL"),
|
2001-11-03 07:10:51 +00:00
|
|
|
* in which case the provisions of the GPL or the LGPL are applicable instead
|
|
|
|
* of those above. If you wish to allow use of your version of this file only
|
|
|
|
* under the terms of either the GPL or the LGPL, and not to allow others to
|
|
|
|
* use your version of this file under the terms of the MPL, indicate your
|
|
|
|
* decision by deleting the provisions above and replace them with the notice
|
2004-04-18 22:01:16 +00:00
|
|
|
* and other provisions required by the GPL or the LGPL. If you do not delete
|
2001-11-03 07:10:51 +00:00
|
|
|
* the provisions above, a recipient may use your version of this file under
|
|
|
|
* the terms of any one of the MPL, the GPL or the LGPL.
|
2004-04-18 22:01:16 +00:00
|
|
|
*
|
|
|
|
* ***** END LICENSE BLOCK ***** */
|
2001-11-03 07:10:51 +00:00
|
|
|
|
|
|
|
/* This is a Cross-Platform ICO Decoder, which should work everywhere, including
|
|
|
|
* Big-Endian machines like the PowerPC. */
|
|
|
|
|
2005-05-26 22:43:36 +00:00
|
|
|
#include <stdlib.h>
|
|
|
|
|
2001-11-03 07:10:51 +00:00
|
|
|
#include "nsICODecoder.h"
|
|
|
|
|
|
|
|
#include "nsIInputStream.h"
|
|
|
|
#include "nsIComponentManager.h"
|
|
|
|
#include "imgIContainerObserver.h"
|
|
|
|
|
|
|
|
#include "imgILoad.h"
|
2007-07-17 21:08:46 +00:00
|
|
|
#include "nsIInterfaceRequestor.h"
|
|
|
|
#include "nsIInterfaceRequestorUtils.h"
|
|
|
|
#include "nsIImage.h"
|
2001-11-03 07:10:51 +00:00
|
|
|
|
2005-01-12 20:16:07 +00:00
|
|
|
#include "nsIProperties.h"
|
|
|
|
#include "nsISupportsPrimitives.h"
|
|
|
|
|
2003-07-23 18:39:11 +00:00
|
|
|
#include "nsAutoPtr.h"
|
|
|
|
|
2001-11-03 07:10:51 +00:00
|
|
|
NS_IMPL_ISUPPORTS1(nsICODecoder, imgIDecoder)
|
|
|
|
|
|
|
|
#define ICONCOUNTOFFSET 4
|
|
|
|
#define DIRENTRYOFFSET 6
|
|
|
|
#define BITMAPINFOSIZE 40
|
|
|
|
#define PREFICONSIZE 16
|
|
|
|
|
2001-11-03 09:50:31 +00:00
|
|
|
// ----------------------------------------
|
|
|
|
// Actual Data Processing
|
|
|
|
// ----------------------------------------
|
|
|
|
|
2002-09-27 10:22:53 +00:00
|
|
|
PRUint32 nsICODecoder::CalcAlphaRowSize()
|
|
|
|
{
|
2007-07-17 21:08:46 +00:00
|
|
|
// Calculate rowsize in DWORD's and then return in # of bytes
|
|
|
|
PRUint32 rowSize = (mDirEntry.mWidth + 31) / 32; // +31 to round up
|
|
|
|
return rowSize * 4; // Return rowSize in bytes
|
2002-09-27 10:22:53 +00:00
|
|
|
}
|
|
|
|
|
2001-11-03 07:10:51 +00:00
|
|
|
nsICODecoder::nsICODecoder()
|
|
|
|
{
|
2002-12-11 14:06:51 +00:00
|
|
|
mPos = mNumColors = mRowBytes = mImageOffset = mCurrIcon = mNumIcons = 0;
|
|
|
|
mCurLine = 1; // Otherwise decoder will never start
|
2001-11-03 07:10:51 +00:00
|
|
|
mColors = nsnull;
|
|
|
|
mRow = nsnull;
|
2007-07-17 21:08:46 +00:00
|
|
|
mHaveAlphaData = mDecodingAndMask = PR_FALSE;
|
2001-11-03 07:10:51 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
nsICODecoder::~nsICODecoder()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP nsICODecoder::Init(imgILoad *aLoad)
|
|
|
|
{
|
|
|
|
mObserver = do_QueryInterface(aLoad);
|
|
|
|
|
|
|
|
mImage = do_CreateInstance("@mozilla.org/image/container;1");
|
|
|
|
if (!mImage)
|
|
|
|
return NS_ERROR_OUT_OF_MEMORY;
|
|
|
|
|
|
|
|
mFrame = do_CreateInstance("@mozilla.org/gfx/image/frame;2");
|
|
|
|
if (!mFrame)
|
|
|
|
return NS_ERROR_OUT_OF_MEMORY;
|
|
|
|
|
|
|
|
return aLoad->SetImage(mImage);
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP nsICODecoder::Close()
|
|
|
|
{
|
2007-07-17 21:08:46 +00:00
|
|
|
// Tell the image that it's data has been updated
|
|
|
|
// This should be a mFrame function, so that we don't have to query for interface...
|
|
|
|
nsIntRect r(0, 0, mDirEntry.mWidth, mDirEntry.mHeight);
|
|
|
|
nsCOMPtr<nsIImage> img(do_GetInterface(mFrame));
|
|
|
|
if (img)
|
|
|
|
img->ImageUpdated(nsnull, nsImageUpdateFlags_kBitsChanged, &r);
|
|
|
|
|
2008-02-13 10:54:39 +00:00
|
|
|
mImage->DecodingComplete();
|
|
|
|
|
2007-01-17 10:39:10 +00:00
|
|
|
if (mObserver) {
|
2007-07-17 21:08:46 +00:00
|
|
|
mObserver->OnDataAvailable(nsnull, mFrame, &r);
|
|
|
|
mObserver->OnStopFrame(nsnull, mFrame);
|
2007-01-17 10:39:10 +00:00
|
|
|
mObserver->OnStopContainer(nsnull, mImage);
|
|
|
|
mObserver->OnStopDecode(nsnull, NS_OK, nsnull);
|
|
|
|
mObserver = nsnull;
|
|
|
|
}
|
2001-11-03 07:10:51 +00:00
|
|
|
|
|
|
|
mImage = nsnull;
|
|
|
|
mFrame = nsnull;
|
|
|
|
|
|
|
|
mPos = 0;
|
|
|
|
|
|
|
|
delete[] mColors;
|
|
|
|
mColors = nsnull;
|
|
|
|
|
|
|
|
mCurLine = 0;
|
|
|
|
mRowBytes = 0;
|
|
|
|
mImageOffset = 0;
|
|
|
|
mCurrIcon = 0;
|
|
|
|
mNumIcons = 0;
|
|
|
|
|
2007-07-17 21:08:46 +00:00
|
|
|
if (mRow) {
|
|
|
|
free(mRow);
|
|
|
|
mRow = nsnull;
|
|
|
|
}
|
2001-11-03 07:10:51 +00:00
|
|
|
mDecodingAndMask = PR_FALSE;
|
|
|
|
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP nsICODecoder::Flush()
|
|
|
|
{
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
2002-01-24 14:36:40 +00:00
|
|
|
|
|
|
|
NS_METHOD nsICODecoder::ReadSegCb(nsIInputStream* aIn, void* aClosure,
|
|
|
|
const char* aFromRawSegment, PRUint32 aToOffset,
|
|
|
|
PRUint32 aCount, PRUint32 *aWriteCount) {
|
2007-07-08 07:08:04 +00:00
|
|
|
nsICODecoder *decoder = reinterpret_cast<nsICODecoder*>(aClosure);
|
2005-04-29 12:16:00 +00:00
|
|
|
*aWriteCount = aCount;
|
2007-07-17 21:08:46 +00:00
|
|
|
return decoder->ProcessData(aFromRawSegment, aCount);
|
2002-01-24 14:36:40 +00:00
|
|
|
}
|
|
|
|
|
2001-11-03 07:10:51 +00:00
|
|
|
NS_IMETHODIMP nsICODecoder::WriteFrom(nsIInputStream *aInStr, PRUint32 aCount, PRUint32 *aRetval)
|
|
|
|
{
|
2007-07-17 21:08:46 +00:00
|
|
|
return aInStr->ReadSegments(ReadSegCb, this, aCount, aRetval);
|
2001-11-03 07:10:51 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
nsresult nsICODecoder::ProcessData(const char* aBuffer, PRUint32 aCount) {
|
|
|
|
if (!aCount) // aCount=0 means EOF
|
|
|
|
return NS_OK;
|
|
|
|
|
|
|
|
while (aCount && (mPos < ICONCOUNTOFFSET)) { // Skip to the # of icons.
|
2005-04-29 12:16:00 +00:00
|
|
|
if (mPos == 2) { // if the third byte is 1: This is an icon, 2: a cursor
|
|
|
|
if ((*aBuffer != 1) && (*aBuffer != 2)) {
|
|
|
|
return NS_ERROR_FAILURE;
|
|
|
|
}
|
2005-01-12 20:16:07 +00:00
|
|
|
mIsCursor = (*aBuffer == 2);
|
2005-04-29 12:16:00 +00:00
|
|
|
}
|
2001-11-03 07:10:51 +00:00
|
|
|
mPos++; aBuffer++; aCount--;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (mPos == ICONCOUNTOFFSET && aCount >= 2) {
|
|
|
|
mNumIcons = LITTLE_TO_NATIVE16(((PRUint16*)aBuffer)[0]);
|
|
|
|
aBuffer += 2;
|
|
|
|
mPos += 2;
|
2001-11-07 03:27:58 +00:00
|
|
|
aCount -= 2;
|
2001-11-03 07:10:51 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (mNumIcons == 0)
|
|
|
|
return NS_OK; // Nothing to do.
|
|
|
|
|
2002-05-14 20:32:13 +00:00
|
|
|
PRUint16 colorDepth = 0;
|
2001-11-03 07:10:51 +00:00
|
|
|
while (mCurrIcon < mNumIcons) {
|
|
|
|
if (mPos >= DIRENTRYOFFSET + (mCurrIcon*sizeof(mDirEntryArray)) &&
|
|
|
|
mPos < DIRENTRYOFFSET + ((mCurrIcon+1)*sizeof(mDirEntryArray))) {
|
|
|
|
PRUint32 toCopy = sizeof(mDirEntryArray) - (mPos - DIRENTRYOFFSET - mCurrIcon*sizeof(mDirEntryArray));
|
|
|
|
if (toCopy > aCount)
|
|
|
|
toCopy = aCount;
|
2002-01-12 03:18:55 +00:00
|
|
|
memcpy(mDirEntryArray + sizeof(mDirEntryArray) - toCopy, aBuffer, toCopy);
|
2001-11-03 07:10:51 +00:00
|
|
|
mPos += toCopy;
|
|
|
|
aCount -= toCopy;
|
|
|
|
aBuffer += toCopy;
|
|
|
|
}
|
2001-12-18 13:27:52 +00:00
|
|
|
if (aCount == 0)
|
|
|
|
return NS_OK; // Need more data
|
2001-11-03 07:10:51 +00:00
|
|
|
|
2002-05-14 20:32:13 +00:00
|
|
|
IconDirEntry e;
|
2001-11-03 07:10:51 +00:00
|
|
|
if (mPos == 22+mCurrIcon*sizeof(mDirEntryArray)) {
|
|
|
|
mCurrIcon++;
|
2002-05-14 20:32:13 +00:00
|
|
|
ProcessDirEntry(e);
|
|
|
|
if ((e.mWidth == PREFICONSIZE && e.mHeight == PREFICONSIZE && e.mBitCount >= colorDepth)
|
|
|
|
|| (mCurrIcon == mNumIcons && mImageOffset == 0)) {
|
|
|
|
mImageOffset = e.mImageOffset;
|
2005-04-29 12:16:00 +00:00
|
|
|
|
|
|
|
// ensure mImageOffset is >= the size of the direntry headers (bug #245631)
|
|
|
|
PRUint32 minImageOffset = DIRENTRYOFFSET + mNumIcons*sizeof(mDirEntryArray);
|
|
|
|
if (mImageOffset < minImageOffset)
|
|
|
|
return NS_ERROR_FAILURE;
|
|
|
|
|
2002-05-14 20:32:13 +00:00
|
|
|
colorDepth = e.mBitCount;
|
|
|
|
memcpy(&mDirEntry, &e, sizeof(IconDirEntry));
|
2001-11-05 08:38:21 +00:00
|
|
|
}
|
2001-11-03 07:10:51 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
while (aCount && mPos < mImageOffset) { // Skip to our offset
|
|
|
|
mPos++; aBuffer++; aCount--;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (mCurrIcon == mNumIcons && mPos >= mImageOffset && mPos < mImageOffset + BITMAPINFOSIZE) {
|
|
|
|
// We've found the icon.
|
|
|
|
PRUint32 toCopy = sizeof(mBIHraw) - (mPos - mImageOffset);
|
|
|
|
if (toCopy > aCount)
|
|
|
|
toCopy = aCount;
|
|
|
|
|
2002-01-12 03:18:55 +00:00
|
|
|
memcpy(mBIHraw + (mPos - mImageOffset), aBuffer, toCopy);
|
2001-11-03 07:10:51 +00:00
|
|
|
mPos += toCopy;
|
|
|
|
aCount -= toCopy;
|
|
|
|
aBuffer += toCopy;
|
|
|
|
}
|
|
|
|
|
2005-10-11 00:21:51 +00:00
|
|
|
nsresult rv;
|
|
|
|
|
2001-11-03 07:10:51 +00:00
|
|
|
if (mPos == mImageOffset + BITMAPINFOSIZE) {
|
2005-10-11 00:21:51 +00:00
|
|
|
rv = mObserver->OnStartDecode(nsnull);
|
|
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
|
|
|
2001-11-03 07:10:51 +00:00
|
|
|
ProcessInfoHeader();
|
|
|
|
if (mBIH.bpp <= 8) {
|
|
|
|
switch (mBIH.bpp) {
|
|
|
|
case 1:
|
|
|
|
mNumColors = 2;
|
|
|
|
break;
|
|
|
|
case 4:
|
|
|
|
mNumColors = 16;
|
|
|
|
break;
|
|
|
|
case 8:
|
|
|
|
mNumColors = 256;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
return NS_ERROR_FAILURE;
|
|
|
|
}
|
|
|
|
|
|
|
|
mColors = new colorTable[mNumColors];
|
|
|
|
if (!mColors)
|
|
|
|
return NS_ERROR_OUT_OF_MEMORY;
|
|
|
|
}
|
|
|
|
|
2005-10-11 00:21:51 +00:00
|
|
|
rv = mImage->Init(mDirEntry.mWidth, mDirEntry.mHeight, mObserver);
|
2001-11-03 07:10:51 +00:00
|
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
2005-01-12 20:16:07 +00:00
|
|
|
|
|
|
|
if (mIsCursor) {
|
|
|
|
nsCOMPtr<nsIProperties> props(do_QueryInterface(mImage));
|
|
|
|
if (props) {
|
|
|
|
nsCOMPtr<nsISupportsPRUint32> intwrapx = do_CreateInstance("@mozilla.org/supports-PRUint32;1");
|
|
|
|
nsCOMPtr<nsISupportsPRUint32> intwrapy = do_CreateInstance("@mozilla.org/supports-PRUint32;1");
|
|
|
|
|
|
|
|
if (intwrapx && intwrapy) {
|
|
|
|
intwrapx->SetData(mDirEntry.mXHotspot);
|
|
|
|
intwrapy->SetData(mDirEntry.mYHotspot);
|
|
|
|
|
|
|
|
props->Set("hotspotX", intwrapx);
|
|
|
|
props->Set("hotspotY", intwrapy);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2003-02-26 05:51:32 +00:00
|
|
|
rv = mObserver->OnStartContainer(nsnull, mImage);
|
2001-11-03 07:10:51 +00:00
|
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
|
|
|
2002-12-11 14:06:51 +00:00
|
|
|
mCurLine = mDirEntry.mHeight;
|
2005-05-26 22:43:36 +00:00
|
|
|
mRow = (PRUint8*)malloc((mDirEntry.mWidth * mBIH.bpp)/8 + 4);
|
2001-11-03 07:10:51 +00:00
|
|
|
// +4 because the line is padded to a 4 bit boundary, but I don't want
|
|
|
|
// to make exact calculations here, that's unnecessary.
|
|
|
|
// Also, it compensates rounding error.
|
|
|
|
if (!mRow)
|
|
|
|
return NS_ERROR_OUT_OF_MEMORY;
|
|
|
|
|
2004-03-02 09:14:42 +00:00
|
|
|
rv = mFrame->Init(0, 0, mDirEntry.mWidth, mDirEntry.mHeight, GFXFORMATALPHA8, 24);
|
2001-11-03 07:10:51 +00:00
|
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
|
|
rv = mImage->AppendFrame(mFrame);
|
|
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
2003-02-26 05:51:32 +00:00
|
|
|
mObserver->OnStartFrame(nsnull, mFrame);
|
2001-11-03 07:10:51 +00:00
|
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
2007-07-17 21:08:46 +00:00
|
|
|
|
|
|
|
PRUint32 imageLength;
|
|
|
|
mFrame->GetImageData((PRUint8**)&mImageData, &imageLength);
|
2001-11-03 07:10:51 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (mColors && (mPos >= mImageOffset + BITMAPINFOSIZE) &&
|
|
|
|
(mPos < (mImageOffset + BITMAPINFOSIZE + mNumColors * 4))) {
|
|
|
|
// We will receive (mNumColors * 4) bytes of color data
|
|
|
|
PRUint32 colorBytes = mPos - (mImageOffset + 40); // Number of bytes already received
|
|
|
|
PRUint8 colorNum = colorBytes / 4; // Color which is currently received
|
|
|
|
PRUint8 at = colorBytes % 4;
|
|
|
|
while (aCount && (mPos < (mImageOffset + BITMAPINFOSIZE + mNumColors * 4))) {
|
|
|
|
switch (at) {
|
|
|
|
case 0:
|
|
|
|
mColors[colorNum].blue = *aBuffer;
|
|
|
|
break;
|
|
|
|
case 1:
|
|
|
|
mColors[colorNum].green = *aBuffer;
|
|
|
|
break;
|
|
|
|
case 2:
|
|
|
|
mColors[colorNum].red = *aBuffer;
|
|
|
|
break;
|
|
|
|
case 3:
|
|
|
|
colorNum++; // This is a padding byte
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
mPos++; aBuffer++; aCount--;
|
|
|
|
at = (at + 1) % 4;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!mDecodingAndMask && (mPos >= (mImageOffset + BITMAPINFOSIZE + mNumColors*4))) {
|
2001-11-07 03:27:58 +00:00
|
|
|
if (mPos == (mImageOffset + BITMAPINFOSIZE + mNumColors*4)) {
|
2003-07-23 18:39:11 +00:00
|
|
|
// Increment mPos to avoid reprocessing the info header.
|
2001-11-05 08:38:21 +00:00
|
|
|
mPos++;
|
2001-11-07 03:27:58 +00:00
|
|
|
}
|
2001-11-03 07:10:51 +00:00
|
|
|
|
2005-04-29 12:16:00 +00:00
|
|
|
// Ensure memory has been allocated before decoding. If we get this far
|
|
|
|
// without allocated memory, the file is most likely invalid.
|
|
|
|
NS_ASSERTION(mRow, "mRow is null");
|
2007-07-17 21:08:46 +00:00
|
|
|
NS_ASSERTION(mImageData, "mImageData is null");
|
|
|
|
if (!mRow || !mImageData)
|
2005-04-29 12:16:00 +00:00
|
|
|
return NS_ERROR_FAILURE;
|
|
|
|
|
2001-11-05 08:38:21 +00:00
|
|
|
PRUint32 rowSize = (mBIH.bpp * mDirEntry.mWidth + 7) / 8; // +7 to round up
|
2001-11-03 07:10:51 +00:00
|
|
|
if (rowSize % 4)
|
|
|
|
rowSize += (4 - (rowSize % 4)); // Pad to DWORD Boundary
|
|
|
|
PRUint32 toCopy;
|
|
|
|
do {
|
|
|
|
toCopy = rowSize - mRowBytes;
|
|
|
|
if (toCopy) {
|
|
|
|
if (toCopy > aCount)
|
|
|
|
toCopy = aCount;
|
2002-01-12 03:18:55 +00:00
|
|
|
memcpy(mRow + mRowBytes, aBuffer, toCopy);
|
2001-11-03 07:10:51 +00:00
|
|
|
aCount -= toCopy;
|
|
|
|
aBuffer += toCopy;
|
|
|
|
mRowBytes += toCopy;
|
|
|
|
}
|
2001-12-18 13:27:52 +00:00
|
|
|
if (rowSize == mRowBytes) {
|
2002-12-11 14:06:51 +00:00
|
|
|
mCurLine--;
|
2007-07-17 21:08:46 +00:00
|
|
|
PRUint32* d = mImageData + (mCurLine * mDirEntry.mWidth);
|
2001-11-03 07:10:51 +00:00
|
|
|
PRUint8* p = mRow;
|
2002-12-11 14:06:51 +00:00
|
|
|
PRUint32 lpos = mDirEntry.mWidth;
|
2001-11-03 07:10:51 +00:00
|
|
|
switch (mBIH.bpp) {
|
|
|
|
case 1:
|
2002-12-11 14:06:51 +00:00
|
|
|
while (lpos > 0) {
|
2001-11-03 07:10:51 +00:00
|
|
|
PRInt8 bit;
|
|
|
|
PRUint8 idx;
|
2002-12-11 14:06:51 +00:00
|
|
|
for (bit = 7; bit >= 0 && lpos > 0; bit--) {
|
2001-11-03 07:10:51 +00:00
|
|
|
idx = (*p >> bit) & 1;
|
2002-09-16 21:49:41 +00:00
|
|
|
SetPixel(d, idx, mColors);
|
2002-12-11 14:06:51 +00:00
|
|
|
--lpos;
|
2001-11-03 07:10:51 +00:00
|
|
|
}
|
|
|
|
++p;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case 4:
|
2002-12-11 14:06:51 +00:00
|
|
|
while (lpos > 0) {
|
|
|
|
Set4BitPixel(d, *p, lpos, mColors);
|
2001-11-03 07:10:51 +00:00
|
|
|
++p;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case 8:
|
2002-12-11 14:06:51 +00:00
|
|
|
while (lpos > 0) {
|
2002-09-16 21:49:41 +00:00
|
|
|
SetPixel(d, *p, mColors);
|
2002-12-11 14:06:51 +00:00
|
|
|
--lpos;
|
2001-11-03 07:10:51 +00:00
|
|
|
++p;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case 16:
|
2002-12-11 14:06:51 +00:00
|
|
|
while (lpos > 0) {
|
2001-11-03 07:10:51 +00:00
|
|
|
SetPixel(d,
|
|
|
|
(p[1] & 124) << 1,
|
|
|
|
((p[1] & 3) << 6) | ((p[0] & 224) >> 2),
|
|
|
|
(p[0] & 31) << 3);
|
|
|
|
|
2002-12-11 14:06:51 +00:00
|
|
|
--lpos;
|
2001-11-03 07:10:51 +00:00
|
|
|
p+=2;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case 24:
|
2002-12-11 14:06:51 +00:00
|
|
|
while (lpos > 0) {
|
2007-01-17 10:39:10 +00:00
|
|
|
SetPixel(d, p[2], p[1], p[0]);
|
|
|
|
p += 3;
|
|
|
|
--lpos;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case 32:
|
2007-07-17 21:08:46 +00:00
|
|
|
// We assume that 32bit doesn't have alpha data until we
|
|
|
|
// find a non-zero alpha byte. If we find such a byte,
|
|
|
|
// it means that all previous pixels are really clear (alphabyte=0).
|
|
|
|
// This working assumption prevents us having to premultiply afterwards.
|
2007-01-17 10:39:10 +00:00
|
|
|
while (lpos > 0) {
|
2007-07-17 21:08:46 +00:00
|
|
|
if (!mHaveAlphaData && p[3]) {
|
|
|
|
// Non-zero alpha byte detected! Clear previous pixels from current row to end
|
|
|
|
memset(mImageData + mCurLine * mDirEntry.mWidth, 0,
|
|
|
|
(mDirEntry.mHeight - mCurLine) * mDirEntry.mWidth * sizeof(PRUint32));
|
|
|
|
mHaveAlphaData = PR_TRUE;
|
|
|
|
}
|
|
|
|
SetPixel(d, p[2], p[1], p[0], mHaveAlphaData ? p[3] : 0xFF);
|
2007-01-17 10:39:10 +00:00
|
|
|
p += 4;
|
2002-12-11 14:06:51 +00:00
|
|
|
--lpos;
|
2001-11-03 07:10:51 +00:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
// This is probably the wrong place to check this...
|
|
|
|
return NS_ERROR_FAILURE;
|
|
|
|
}
|
|
|
|
|
2002-10-01 21:52:30 +00:00
|
|
|
if (mCurLine == 0)
|
2001-11-03 07:10:51 +00:00
|
|
|
mDecodingAndMask = PR_TRUE;
|
|
|
|
|
2002-12-11 14:06:51 +00:00
|
|
|
mRowBytes = 0;
|
2001-11-03 07:10:51 +00:00
|
|
|
}
|
|
|
|
} while (!mDecodingAndMask && aCount > 0);
|
2003-07-23 18:39:11 +00:00
|
|
|
|
2001-11-03 07:10:51 +00:00
|
|
|
}
|
|
|
|
|
2004-03-02 09:14:42 +00:00
|
|
|
if (mDecodingAndMask && !mHaveAlphaData) {
|
2002-09-27 10:22:53 +00:00
|
|
|
PRUint32 rowSize = CalcAlphaRowSize();
|
|
|
|
|
2001-11-05 08:38:21 +00:00
|
|
|
if (mPos == (1 + mImageOffset + BITMAPINFOSIZE + mNumColors*4)) {
|
|
|
|
mPos++;
|
|
|
|
mRowBytes = 0;
|
2002-12-11 14:06:51 +00:00
|
|
|
mCurLine = mDirEntry.mHeight;
|
2007-07-17 21:08:46 +00:00
|
|
|
mRow = (PRUint8*)realloc(mRow, rowSize);
|
2005-04-29 12:16:00 +00:00
|
|
|
if (!mRow)
|
|
|
|
return NS_ERROR_OUT_OF_MEMORY;
|
2001-11-05 08:38:21 +00:00
|
|
|
}
|
|
|
|
|
2005-04-29 12:16:00 +00:00
|
|
|
// Ensure memory has been allocated before decoding.
|
2006-04-15 17:31:26 +00:00
|
|
|
NS_ASSERTION(mRow, "mRow is null");
|
2007-07-17 21:08:46 +00:00
|
|
|
NS_ASSERTION(mImageData, "mImageData is null");
|
|
|
|
if (!mRow || !mImageData)
|
2006-04-15 17:31:26 +00:00
|
|
|
return NS_ERROR_FAILURE;
|
2005-04-29 12:16:00 +00:00
|
|
|
|
2007-07-17 21:08:46 +00:00
|
|
|
while (mCurLine > 0 && aCount > 0) {
|
|
|
|
PRUint32 toCopy = PR_MIN(rowSize - mRowBytes, aCount);
|
|
|
|
if (toCopy) {
|
|
|
|
memcpy(mRow + mRowBytes, aBuffer, toCopy);
|
|
|
|
aCount -= toCopy;
|
|
|
|
aBuffer += toCopy;
|
|
|
|
mRowBytes += toCopy;
|
|
|
|
}
|
|
|
|
if (rowSize == mRowBytes) {
|
|
|
|
mCurLine--;
|
|
|
|
mRowBytes = 0;
|
|
|
|
|
|
|
|
PRUint32* decoded = mImageData + mCurLine * mDirEntry.mWidth;
|
|
|
|
PRUint32* decoded_end = decoded + mDirEntry.mWidth;
|
|
|
|
PRUint8* p = mRow, *p_end = mRow + rowSize;
|
|
|
|
while (p < p_end) {
|
|
|
|
PRUint8 idx = *p++;
|
|
|
|
for (PRUint8 bit = 0x80; bit && decoded<decoded_end; bit >>= 1) {
|
|
|
|
// Clear pixel completely for transparency.
|
|
|
|
if (idx & bit) *decoded = 0;
|
|
|
|
decoded ++;
|
|
|
|
}
|
2001-11-03 07:10:51 +00:00
|
|
|
}
|
2007-07-17 21:08:46 +00:00
|
|
|
}
|
|
|
|
}
|
2001-11-03 07:10:51 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2002-05-14 20:32:13 +00:00
|
|
|
nsICODecoder::ProcessDirEntry(IconDirEntry& aTarget)
|
2001-11-03 07:10:51 +00:00
|
|
|
{
|
2002-05-14 20:32:13 +00:00
|
|
|
memset(&aTarget, 0, sizeof(aTarget));
|
2002-09-15 19:51:37 +00:00
|
|
|
memcpy(&aTarget.mWidth, mDirEntryArray, sizeof(aTarget.mWidth));
|
|
|
|
memcpy(&aTarget.mHeight, mDirEntryArray+1, sizeof(aTarget.mHeight));
|
|
|
|
memcpy(&aTarget.mColorCount, mDirEntryArray+2, sizeof(aTarget.mColorCount));
|
|
|
|
memcpy(&aTarget.mReserved, mDirEntryArray+3, sizeof(aTarget.mReserved));
|
2001-11-03 07:10:51 +00:00
|
|
|
|
2002-09-15 19:51:37 +00:00
|
|
|
memcpy(&aTarget.mPlanes, mDirEntryArray+4, sizeof(aTarget.mPlanes));
|
2002-05-14 20:32:13 +00:00
|
|
|
aTarget.mPlanes = LITTLE_TO_NATIVE16(aTarget.mPlanes);
|
2001-11-03 07:10:51 +00:00
|
|
|
|
2002-09-15 19:51:37 +00:00
|
|
|
memcpy(&aTarget.mBitCount, mDirEntryArray+6, sizeof(aTarget.mBitCount));
|
2002-05-14 20:32:13 +00:00
|
|
|
aTarget.mBitCount = LITTLE_TO_NATIVE16(aTarget.mBitCount);
|
2001-11-03 07:10:51 +00:00
|
|
|
|
2002-09-15 19:51:37 +00:00
|
|
|
memcpy(&aTarget.mBytesInRes, mDirEntryArray+8, sizeof(aTarget.mBytesInRes));
|
2002-05-14 20:32:13 +00:00
|
|
|
aTarget.mBytesInRes = LITTLE_TO_NATIVE32(aTarget.mBytesInRes);
|
2001-11-03 07:10:51 +00:00
|
|
|
|
2002-09-15 19:51:37 +00:00
|
|
|
memcpy(&aTarget.mImageOffset, mDirEntryArray+12, sizeof(aTarget.mImageOffset));
|
2002-05-14 20:32:13 +00:00
|
|
|
aTarget.mImageOffset = LITTLE_TO_NATIVE32(aTarget.mImageOffset);
|
2001-11-03 07:10:51 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void nsICODecoder::ProcessInfoHeader() {
|
2002-02-05 01:41:13 +00:00
|
|
|
memset(&mBIH, 0, sizeof(mBIH));
|
2001-11-03 07:10:51 +00:00
|
|
|
// Ignoring the size; it should always be 40 for icons, anyway
|
|
|
|
|
2002-09-15 19:51:37 +00:00
|
|
|
memcpy(&mBIH.width, mBIHraw + 4, sizeof(mBIH.width));
|
|
|
|
memcpy(&mBIH.height, mBIHraw + 8, sizeof(mBIH.height));
|
|
|
|
memcpy(&mBIH.planes, mBIHraw + 12, sizeof(mBIH.planes));
|
|
|
|
memcpy(&mBIH.bpp, mBIHraw + 14, sizeof(mBIH.bpp));
|
|
|
|
memcpy(&mBIH.compression, mBIHraw + 16, sizeof(mBIH.compression));
|
|
|
|
memcpy(&mBIH.image_size, mBIHraw + 20, sizeof(mBIH.image_size));
|
|
|
|
memcpy(&mBIH.xppm, mBIHraw + 24, sizeof(mBIH.xppm));
|
|
|
|
memcpy(&mBIH.yppm, mBIHraw + 28, sizeof(mBIH.yppm));
|
|
|
|
memcpy(&mBIH.colors, mBIHraw + 32, sizeof(mBIH.colors));
|
|
|
|
memcpy(&mBIH.important_colors, mBIHraw + 36, sizeof(mBIH.important_colors));
|
2001-11-03 07:10:51 +00:00
|
|
|
|
|
|
|
// Convert endianness
|
|
|
|
mBIH.width = LITTLE_TO_NATIVE32(mBIH.width);
|
|
|
|
mBIH.height = LITTLE_TO_NATIVE32(mBIH.height);
|
|
|
|
mBIH.planes = LITTLE_TO_NATIVE16(mBIH.planes);
|
|
|
|
mBIH.bpp = LITTLE_TO_NATIVE16(mBIH.bpp);
|
|
|
|
|
|
|
|
mBIH.compression = LITTLE_TO_NATIVE32(mBIH.compression);
|
|
|
|
mBIH.image_size = LITTLE_TO_NATIVE32(mBIH.image_size);
|
|
|
|
mBIH.xppm = LITTLE_TO_NATIVE32(mBIH.xppm);
|
|
|
|
mBIH.yppm = LITTLE_TO_NATIVE32(mBIH.yppm);
|
|
|
|
mBIH.colors = LITTLE_TO_NATIVE32(mBIH.colors);
|
|
|
|
mBIH.important_colors = LITTLE_TO_NATIVE32(mBIH.important_colors);
|
|
|
|
}
|