2001-11-03 07:10:51 +00:00
|
|
|
/* vim:set tw=80 expandtab softtabstop=4 ts=4 sw=4: */
|
2012-05-21 11:12:37 +00:00
|
|
|
/* This Source Code Form is subject to the terms of the Mozilla Public
|
|
|
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
|
|
* file, You can obtain one at http://mozilla.org/MPL/2.0/. */
|
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>
|
|
|
|
|
2011-10-02 19:01:38 +00:00
|
|
|
#include "EndianMacros.h"
|
2001-11-03 07:10:51 +00:00
|
|
|
#include "nsICODecoder.h"
|
|
|
|
|
|
|
|
#include "nsIInputStream.h"
|
|
|
|
#include "nsIComponentManager.h"
|
2010-08-14 04:09:49 +00:00
|
|
|
#include "RasterImage.h"
|
2001-11-03 07:10:51 +00:00
|
|
|
|
2005-01-12 20:16:07 +00:00
|
|
|
#include "nsIProperties.h"
|
|
|
|
#include "nsISupportsPrimitives.h"
|
2013-01-15 12:22:03 +00:00
|
|
|
#include <algorithm>
|
2005-01-12 20:16:07 +00:00
|
|
|
|
2010-08-23 02:30:46 +00:00
|
|
|
namespace mozilla {
|
2012-01-06 16:02:27 +00:00
|
|
|
namespace image {
|
2001-11-03 07:10:51 +00:00
|
|
|
|
|
|
|
#define ICONCOUNTOFFSET 4
|
|
|
|
#define DIRENTRYOFFSET 6
|
|
|
|
#define BITMAPINFOSIZE 40
|
|
|
|
#define PREFICONSIZE 16
|
|
|
|
|
2001-11-03 09:50:31 +00:00
|
|
|
// ----------------------------------------
|
|
|
|
// Actual Data Processing
|
|
|
|
// ----------------------------------------
|
|
|
|
|
2012-08-22 15:56:38 +00:00
|
|
|
uint32_t
|
2011-08-25 20:09:01 +00:00
|
|
|
nsICODecoder::CalcAlphaRowSize()
|
2002-09-27 10:22:53 +00:00
|
|
|
{
|
2007-07-17 21:08:46 +00:00
|
|
|
// Calculate rowsize in DWORD's and then return in # of bytes
|
2012-08-22 15:56:38 +00:00
|
|
|
uint32_t rowSize = (GetRealWidth() + 31) / 32; // + 31 to round up
|
2011-08-25 20:09:01 +00:00
|
|
|
return rowSize * 4; // Return rowSize in bytes
|
2002-09-27 10:22:53 +00:00
|
|
|
}
|
|
|
|
|
2011-08-25 20:09:01 +00:00
|
|
|
// Obtains the number of colors from the bits per pixel
|
2012-08-22 15:56:38 +00:00
|
|
|
uint16_t
|
2011-08-25 20:09:01 +00:00
|
|
|
nsICODecoder::GetNumColors()
|
|
|
|
{
|
2012-08-22 15:56:38 +00:00
|
|
|
uint16_t numColors = 0;
|
2011-08-25 20:09:01 +00:00
|
|
|
if (mBPP <= 8) {
|
|
|
|
switch (mBPP) {
|
|
|
|
case 1:
|
|
|
|
numColors = 2;
|
|
|
|
break;
|
|
|
|
case 4:
|
|
|
|
numColors = 16;
|
|
|
|
break;
|
|
|
|
case 8:
|
|
|
|
numColors = 256;
|
|
|
|
break;
|
|
|
|
default:
|
2012-08-22 15:56:38 +00:00
|
|
|
numColors = (uint16_t)-1;
|
2011-08-25 20:09:01 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return numColors;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-12-18 16:37:15 +00:00
|
|
|
nsICODecoder::nsICODecoder(RasterImage &aImage, imgDecoderObserver* aObserver)
|
2011-09-27 16:24:03 +00:00
|
|
|
: Decoder(aImage, aObserver)
|
2001-11-03 07:10:51 +00:00
|
|
|
{
|
2011-08-25 20:09:01 +00:00
|
|
|
mPos = mImageOffset = mCurrIcon = mNumIcons = mBPP = mRowBytes = 0;
|
2011-10-17 14:59:28 +00:00
|
|
|
mIsPNG = false;
|
2012-07-30 14:20:58 +00:00
|
|
|
mRow = nullptr;
|
2011-08-25 20:09:01 +00:00
|
|
|
mOldLine = mCurLine = 1; // Otherwise decoder will never start
|
2001-11-03 07:10:51 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
nsICODecoder::~nsICODecoder()
|
|
|
|
{
|
2010-08-23 02:30:46 +00:00
|
|
|
if (mRow) {
|
2011-08-25 20:09:01 +00:00
|
|
|
moz_free(mRow);
|
2010-08-23 02:30:46 +00:00
|
|
|
}
|
2001-11-03 07:10:51 +00:00
|
|
|
}
|
|
|
|
|
2010-09-12 15:22:30 +00:00
|
|
|
void
|
2010-08-23 02:30:46 +00:00
|
|
|
nsICODecoder::FinishInternal()
|
2001-11-03 07:10:51 +00:00
|
|
|
{
|
2010-09-12 15:22:31 +00:00
|
|
|
// We shouldn't be called in error cases
|
2010-09-12 15:22:31 +00:00
|
|
|
NS_ABORT_IF_FALSE(!HasError(), "Shouldn't call FinishInternal after error!");
|
2010-09-12 15:22:31 +00:00
|
|
|
|
2011-08-25 20:09:01 +00:00
|
|
|
// Finish the internally used decoder as well
|
|
|
|
if (mContainedDecoder) {
|
|
|
|
mContainedDecoder->FinishSharedDecoder();
|
|
|
|
mDecodeDone = mContainedDecoder->GetDecodeDone();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Returns a buffer filled with the bitmap file header in little endian:
|
|
|
|
// Signature 2 bytes 'BM'
|
|
|
|
// FileSize 4 bytes File size in bytes
|
|
|
|
// reserved 4 bytes unused (=0)
|
|
|
|
// DataOffset 4 bytes File offset to Raster Data
|
2011-10-17 14:59:28 +00:00
|
|
|
// Returns true if successful
|
2012-08-22 15:56:38 +00:00
|
|
|
bool nsICODecoder::FillBitmapFileHeaderBuffer(int8_t *bfh)
|
2011-08-25 20:09:01 +00:00
|
|
|
{
|
|
|
|
memset(bfh, 0, 14);
|
|
|
|
bfh[0] = 'B';
|
|
|
|
bfh[1] = 'M';
|
2012-08-22 15:56:38 +00:00
|
|
|
int32_t dataOffset = 0;
|
|
|
|
int32_t fileSize = 0;
|
2011-08-25 20:09:01 +00:00
|
|
|
dataOffset = BFH_LENGTH + BITMAPINFOSIZE;
|
|
|
|
|
|
|
|
// The color table is present only if BPP is <= 8
|
|
|
|
if (mDirEntry.mBitCount <= 8) {
|
2012-08-22 15:56:38 +00:00
|
|
|
uint16_t numColors = GetNumColors();
|
|
|
|
if (numColors == (uint16_t)-1) {
|
2011-10-17 14:59:28 +00:00
|
|
|
return false;
|
2011-08-25 20:09:01 +00:00
|
|
|
}
|
|
|
|
dataOffset += 4 * numColors;
|
2011-09-22 13:43:13 +00:00
|
|
|
fileSize = dataOffset + GetRealWidth() * GetRealHeight();
|
2011-08-25 20:09:01 +00:00
|
|
|
} else {
|
2011-09-22 13:43:13 +00:00
|
|
|
fileSize = dataOffset + (mDirEntry.mBitCount * GetRealWidth() *
|
|
|
|
GetRealHeight()) / 8;
|
2011-08-25 20:09:01 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
fileSize = NATIVE32_TO_LITTLE(fileSize);
|
|
|
|
memcpy(bfh + 2, &fileSize, sizeof(fileSize));
|
|
|
|
dataOffset = NATIVE32_TO_LITTLE(dataOffset);
|
|
|
|
memcpy(bfh + 10, &dataOffset, sizeof(dataOffset));
|
2011-10-17 14:59:28 +00:00
|
|
|
return true;
|
2011-08-25 20:09:01 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// A BMP inside of an ICO has *2 height because of the AND mask
|
|
|
|
// that follows the actual bitmap. The BMP shouldn't know about
|
|
|
|
// this difference though.
|
2011-11-04 13:56:17 +00:00
|
|
|
bool
|
2012-08-22 15:56:38 +00:00
|
|
|
nsICODecoder::FixBitmapHeight(int8_t *bih)
|
2011-08-25 20:09:01 +00:00
|
|
|
{
|
2011-11-04 13:56:17 +00:00
|
|
|
// Get the height from the BMP file information header
|
2012-08-22 15:56:38 +00:00
|
|
|
int32_t height;
|
2011-11-04 13:56:17 +00:00
|
|
|
memcpy(&height, bih + 8, sizeof(height));
|
|
|
|
height = LITTLE_TO_NATIVE32(height);
|
2012-07-21 17:57:35 +00:00
|
|
|
// BMPs can be stored inverted by having a negative height
|
|
|
|
height = abs(height);
|
2011-11-04 13:56:17 +00:00
|
|
|
|
|
|
|
// The bitmap height is by definition * 2 what it should be to account for
|
|
|
|
// the 'AND mask'. It is * 2 even if the `AND mask` is not present.
|
|
|
|
height /= 2;
|
|
|
|
|
|
|
|
if (height > 256) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// We should always trust the height from the bitmap itself instead of
|
|
|
|
// the ICO height. So fix the ICO height.
|
|
|
|
if (height == 256) {
|
|
|
|
mDirEntry.mHeight = 0;
|
|
|
|
} else {
|
2012-08-22 15:56:38 +00:00
|
|
|
mDirEntry.mHeight = (int8_t)height;
|
2011-11-04 13:56:17 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Fix the BMP height in the BIH so that the BMP decoder can work properly
|
2011-08-25 20:09:01 +00:00
|
|
|
height = NATIVE32_TO_LITTLE(height);
|
|
|
|
memcpy(bih + 8, &height, sizeof(height));
|
2011-11-04 13:56:17 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
// We should always trust the contained resource for the width
|
|
|
|
// information over our own information.
|
|
|
|
bool
|
2012-08-22 15:56:38 +00:00
|
|
|
nsICODecoder::FixBitmapWidth(int8_t *bih)
|
2011-11-04 13:56:17 +00:00
|
|
|
{
|
|
|
|
// Get the width from the BMP file information header
|
2012-08-22 15:56:38 +00:00
|
|
|
int32_t width;
|
2011-11-04 13:56:17 +00:00
|
|
|
memcpy(&width, bih + 4, sizeof(width));
|
|
|
|
width = LITTLE_TO_NATIVE32(width);
|
|
|
|
if (width > 256) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// We should always trust the width from the bitmap itself instead of
|
|
|
|
// the ICO width.
|
|
|
|
if (width == 256) {
|
|
|
|
mDirEntry.mWidth = 0;
|
|
|
|
} else {
|
2012-08-22 15:56:38 +00:00
|
|
|
mDirEntry.mWidth = (int8_t)width;
|
2011-11-04 13:56:17 +00:00
|
|
|
}
|
|
|
|
return true;
|
2011-08-25 20:09:01 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// The BMP information header's bits per pixel should be trusted
|
|
|
|
// more than what we have. Usually the ICO's BPP is set to 0
|
2012-08-22 15:56:38 +00:00
|
|
|
int32_t
|
|
|
|
nsICODecoder::ExtractBPPFromBitmap(int8_t *bih)
|
2011-08-25 20:09:01 +00:00
|
|
|
{
|
2012-08-22 15:56:38 +00:00
|
|
|
int32_t bitsPerPixel;
|
2011-08-25 20:09:01 +00:00
|
|
|
memcpy(&bitsPerPixel, bih + 14, sizeof(bitsPerPixel));
|
|
|
|
bitsPerPixel = LITTLE_TO_NATIVE32(bitsPerPixel);
|
|
|
|
return bitsPerPixel;
|
|
|
|
}
|
2010-08-23 02:30:46 +00:00
|
|
|
|
2012-08-22 15:56:38 +00:00
|
|
|
int32_t
|
|
|
|
nsICODecoder::ExtractBIHSizeFromBitmap(int8_t *bih)
|
2011-08-30 05:12:59 +00:00
|
|
|
{
|
2012-08-22 15:56:38 +00:00
|
|
|
int32_t headerSize;
|
2011-08-30 05:12:59 +00:00
|
|
|
memcpy(&headerSize, bih, sizeof(headerSize));
|
|
|
|
headerSize = LITTLE_TO_NATIVE32(headerSize);
|
|
|
|
return headerSize;
|
|
|
|
}
|
|
|
|
|
2011-08-25 20:09:01 +00:00
|
|
|
void
|
|
|
|
nsICODecoder::SetHotSpotIfCursor() {
|
|
|
|
if (!mIsCursor) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
nsCOMPtr<nsISupportsPRUint32> intwrapx =
|
2012-11-28 02:07:01 +00:00
|
|
|
do_CreateInstance("@mozilla.org/supports-PRUint32;1");
|
2011-08-25 20:09:01 +00:00
|
|
|
nsCOMPtr<nsISupportsPRUint32> intwrapy =
|
2012-11-28 02:07:01 +00:00
|
|
|
do_CreateInstance("@mozilla.org/supports-PRUint32;1");
|
2009-09-12 22:44:18 +00:00
|
|
|
|
2011-08-25 20:09:01 +00:00
|
|
|
if (intwrapx && intwrapy) {
|
|
|
|
intwrapx->SetData(mDirEntry.mXHotspot);
|
|
|
|
intwrapy->SetData(mDirEntry.mYHotspot);
|
2009-09-12 22:44:18 +00:00
|
|
|
|
2011-09-29 13:17:13 +00:00
|
|
|
mImage.Set("hotspotX", intwrapx);
|
|
|
|
mImage.Set("hotspotY", intwrapy);
|
2007-01-17 10:39:10 +00:00
|
|
|
}
|
2001-11-03 07:10:51 +00:00
|
|
|
}
|
|
|
|
|
2010-09-12 15:22:30 +00:00
|
|
|
void
|
2012-08-22 15:56:38 +00:00
|
|
|
nsICODecoder::WriteInternal(const char* aBuffer, uint32_t aCount)
|
2001-11-03 07:10:51 +00:00
|
|
|
{
|
2010-09-12 15:22:31 +00:00
|
|
|
NS_ABORT_IF_FALSE(!HasError(), "Shouldn't call WriteInternal after error!");
|
2001-11-03 07:10:51 +00:00
|
|
|
|
|
|
|
if (!aCount) // aCount=0 means EOF
|
2010-09-12 15:22:30 +00:00
|
|
|
return;
|
2001-11-03 07:10:51 +00:00
|
|
|
|
|
|
|
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)) {
|
2010-09-12 15:22:27 +00:00
|
|
|
PostDataError();
|
2010-09-12 15:22:30 +00:00
|
|
|
return;
|
2005-04-29 12:16:00 +00:00
|
|
|
}
|
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) {
|
2012-08-22 15:56:38 +00:00
|
|
|
mNumIcons = LITTLE_TO_NATIVE16(((uint16_t*)aBuffer)[0]);
|
2001-11-03 07:10:51 +00:00
|
|
|
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)
|
2010-09-12 15:22:30 +00:00
|
|
|
return; // Nothing to do.
|
2001-11-03 07:10:51 +00:00
|
|
|
|
2012-08-22 15:56:38 +00:00
|
|
|
uint16_t colorDepth = 0;
|
2011-08-25 20:09:01 +00:00
|
|
|
// Loop through each entry's dir entry
|
|
|
|
while (mCurrIcon < mNumIcons) {
|
|
|
|
if (mPos >= DIRENTRYOFFSET + (mCurrIcon * sizeof(mDirEntryArray)) &&
|
|
|
|
mPos < DIRENTRYOFFSET + ((mCurrIcon + 1) * sizeof(mDirEntryArray))) {
|
2012-08-22 15:56:38 +00:00
|
|
|
uint32_t toCopy = sizeof(mDirEntryArray) -
|
2011-08-25 20:09:01 +00:00
|
|
|
(mPos - DIRENTRYOFFSET - mCurrIcon * sizeof(mDirEntryArray));
|
|
|
|
if (toCopy > aCount) {
|
2001-11-03 07:10:51 +00:00
|
|
|
toCopy = aCount;
|
2011-08-25 20:09:01 +00:00
|
|
|
}
|
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)
|
2010-09-12 15:22:30 +00:00
|
|
|
return; // Need more data
|
2001-11-03 07:10:51 +00:00
|
|
|
|
2002-05-14 20:32:13 +00:00
|
|
|
IconDirEntry e;
|
2011-08-25 20:09:01 +00:00
|
|
|
if (mPos == (DIRENTRYOFFSET + ICODIRENTRYSIZE) +
|
|
|
|
(mCurrIcon * sizeof(mDirEntryArray))) {
|
2001-11-03 07:10:51 +00:00
|
|
|
mCurrIcon++;
|
2002-05-14 20:32:13 +00:00
|
|
|
ProcessDirEntry(e);
|
2011-09-22 13:43:13 +00:00
|
|
|
// We can't use GetRealWidth and GetRealHeight here because those operate
|
|
|
|
// on mDirEntry, here we are going through each item in the directory
|
|
|
|
if (((e.mWidth == 0 ? 256 : e.mWidth) == PREFICONSIZE &&
|
|
|
|
(e.mHeight == 0 ? 256 : e.mHeight) == PREFICONSIZE &&
|
|
|
|
(e.mBitCount >= colorDepth)) ||
|
2011-08-25 20:09:01 +00:00
|
|
|
(mCurrIcon == mNumIcons && mImageOffset == 0)) {
|
2002-05-14 20:32:13 +00:00
|
|
|
mImageOffset = e.mImageOffset;
|
2005-04-29 12:16:00 +00:00
|
|
|
|
2011-08-25 20:09:01 +00:00
|
|
|
// ensure mImageOffset is >= size of the direntry headers (bug #245631)
|
2012-08-22 15:56:38 +00:00
|
|
|
uint32_t minImageOffset = DIRENTRYOFFSET +
|
2011-08-25 20:09:01 +00:00
|
|
|
mNumIcons * sizeof(mDirEntryArray);
|
2009-10-16 02:54:44 +00:00
|
|
|
if (mImageOffset < minImageOffset) {
|
2010-09-12 15:22:27 +00:00
|
|
|
PostDataError();
|
2010-09-12 15:22:30 +00:00
|
|
|
return;
|
2009-10-16 02:54:44 +00:00
|
|
|
}
|
2005-04-29 12:16:00 +00:00
|
|
|
|
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
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2008-02-19 23:40:04 +00:00
|
|
|
if (mPos < mImageOffset) {
|
|
|
|
// Skip to (or at least towards) the desired image offset
|
2012-08-22 15:56:38 +00:00
|
|
|
uint32_t toSkip = mImageOffset - mPos;
|
2008-02-19 23:40:04 +00:00
|
|
|
if (toSkip > aCount)
|
|
|
|
toSkip = aCount;
|
|
|
|
|
|
|
|
mPos += toSkip;
|
|
|
|
aBuffer += toSkip;
|
|
|
|
aCount -= toSkip;
|
2001-11-03 07:10:51 +00:00
|
|
|
}
|
|
|
|
|
2011-08-25 20:09:01 +00:00
|
|
|
// If we are within the first PNGSIGNATURESIZE bytes of the image data,
|
|
|
|
// then we have either a BMP or a PNG. We use the first PNGSIGNATURESIZE
|
|
|
|
// bytes to determine which one we have.
|
|
|
|
if (mCurrIcon == mNumIcons && mPos >= mImageOffset &&
|
|
|
|
mPos < mImageOffset + PNGSIGNATURESIZE)
|
|
|
|
{
|
2012-08-22 15:56:38 +00:00
|
|
|
uint32_t toCopy = PNGSIGNATURESIZE - (mPos - mImageOffset);
|
2011-08-25 20:09:01 +00:00
|
|
|
if (toCopy > aCount) {
|
|
|
|
toCopy = aCount;
|
|
|
|
}
|
|
|
|
|
|
|
|
memcpy(mSignature + (mPos - mImageOffset), aBuffer, toCopy);
|
|
|
|
mPos += toCopy;
|
|
|
|
aCount -= toCopy;
|
|
|
|
aBuffer += toCopy;
|
|
|
|
|
|
|
|
mIsPNG = !memcmp(mSignature, nsPNGDecoder::pngSignatureBytes,
|
|
|
|
PNGSIGNATURESIZE);
|
|
|
|
if (mIsPNG) {
|
2011-09-27 16:24:03 +00:00
|
|
|
mContainedDecoder = new nsPNGDecoder(mImage, mObserver);
|
|
|
|
mContainedDecoder->InitSharedDecoder();
|
2012-05-08 12:38:43 +00:00
|
|
|
if (!WriteToContainedDecoder(mSignature, PNGSIGNATURESIZE)) {
|
2011-08-25 20:09:01 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// If we have a PNG, let the PNG decoder do all of the rest of the work
|
|
|
|
if (mIsPNG && mContainedDecoder && mPos >= mImageOffset + PNGSIGNATURESIZE) {
|
2012-05-08 12:38:43 +00:00
|
|
|
if (!WriteToContainedDecoder(aBuffer, aCount)) {
|
2011-08-25 20:09:01 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
mPos += aCount;
|
|
|
|
aBuffer += aCount;
|
|
|
|
aCount = 0;
|
|
|
|
|
|
|
|
// Raymond Chen says that 32bpp only are valid PNG ICOs
|
|
|
|
// http://blogs.msdn.com/b/oldnewthing/archive/2010/10/22/10079192.aspx
|
2011-11-05 10:48:26 +00:00
|
|
|
if (!static_cast<nsPNGDecoder*>(mContainedDecoder.get())->IsValidICO()) {
|
2011-08-25 20:09:01 +00:00
|
|
|
PostDataError();
|
|
|
|
}
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// We've processed all of the icon dir entries and are within the
|
|
|
|
// bitmap info size
|
|
|
|
if (!mIsPNG && mCurrIcon == mNumIcons && mPos >= mImageOffset &&
|
|
|
|
mPos >= mImageOffset + PNGSIGNATURESIZE &&
|
|
|
|
mPos < mImageOffset + BITMAPINFOSIZE) {
|
|
|
|
|
|
|
|
// As we were decoding, we did not know if we had a PNG signature or the
|
|
|
|
// start of a bitmap information header. At this point we know we had
|
|
|
|
// a bitmap information header and not a PNG signature, so fill the bitmap
|
|
|
|
// information header with the data it should already have.
|
|
|
|
memcpy(mBIHraw, mSignature, PNGSIGNATURESIZE);
|
|
|
|
|
2001-11-03 07:10:51 +00:00
|
|
|
// We've found the icon.
|
2012-08-22 15:56:38 +00:00
|
|
|
uint32_t toCopy = sizeof(mBIHraw) - (mPos - mImageOffset);
|
2001-11-03 07:10:51 +00:00
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
2011-08-25 20:09:01 +00:00
|
|
|
// If we have a BMP inside the ICO and we have read the BIH header
|
|
|
|
if (!mIsPNG && mPos == mImageOffset + BITMAPINFOSIZE) {
|
2011-08-30 05:12:59 +00:00
|
|
|
|
|
|
|
// Make sure we have a sane value for the bitmap information header
|
2012-08-22 15:56:38 +00:00
|
|
|
int32_t bihSize = ExtractBIHSizeFromBitmap(reinterpret_cast<int8_t*>(mBIHraw));
|
2011-08-30 05:12:59 +00:00
|
|
|
if (bihSize != BITMAPINFOSIZE) {
|
|
|
|
PostDataError();
|
|
|
|
return;
|
|
|
|
}
|
2011-08-25 20:09:01 +00:00
|
|
|
// We are extracting the BPP from the BIH header as it should be trusted
|
|
|
|
// over the one we have from the icon header
|
2012-08-22 15:56:38 +00:00
|
|
|
mBPP = ExtractBPPFromBitmap(reinterpret_cast<int8_t*>(mBIHraw));
|
2011-08-25 20:09:01 +00:00
|
|
|
|
|
|
|
// Init the bitmap decoder which will do most of the work for us
|
|
|
|
// It will do everything except the AND mask which isn't present in bitmaps
|
|
|
|
// bmpDecoder is for local scope ease, it will be freed by mContainedDecoder
|
2011-09-27 16:24:03 +00:00
|
|
|
nsBMPDecoder *bmpDecoder = new nsBMPDecoder(mImage, mObserver);
|
2011-08-25 20:09:01 +00:00
|
|
|
mContainedDecoder = bmpDecoder;
|
2011-10-17 14:59:28 +00:00
|
|
|
bmpDecoder->SetUseAlphaData(true);
|
2011-08-25 20:09:01 +00:00
|
|
|
mContainedDecoder->SetSizeDecode(IsSizeDecode());
|
2011-09-27 16:24:03 +00:00
|
|
|
mContainedDecoder->InitSharedDecoder();
|
2011-08-25 20:09:01 +00:00
|
|
|
|
|
|
|
// The ICO format when containing a BMP does not include the 14 byte
|
|
|
|
// bitmap file header. To use the code of the BMP decoder we need to
|
|
|
|
// generate this header ourselves and feed it to the BMP decoder.
|
2012-08-22 15:56:38 +00:00
|
|
|
int8_t bfhBuffer[BMPFILEHEADERSIZE];
|
2011-08-25 20:09:01 +00:00
|
|
|
if (!FillBitmapFileHeaderBuffer(bfhBuffer)) {
|
|
|
|
PostDataError();
|
|
|
|
return;
|
|
|
|
}
|
2012-05-08 12:38:43 +00:00
|
|
|
if (!WriteToContainedDecoder((const char*)bfhBuffer, sizeof(bfhBuffer))) {
|
2010-09-12 15:22:30 +00:00
|
|
|
return;
|
2001-11-03 07:10:51 +00:00
|
|
|
}
|
|
|
|
|
2011-08-25 20:09:01 +00:00
|
|
|
// Setup the cursor hot spot if one is present
|
|
|
|
SetHotSpotIfCursor();
|
2005-01-12 20:16:07 +00:00
|
|
|
|
2011-11-04 13:56:17 +00:00
|
|
|
// Fix the ICO height from the BIH.
|
|
|
|
// Fix the height on the BIH to be /2 so our BMP decoder will understand.
|
2012-08-22 15:56:38 +00:00
|
|
|
if (!FixBitmapHeight(reinterpret_cast<int8_t*>(mBIHraw))) {
|
2011-11-04 13:56:17 +00:00
|
|
|
PostDataError();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Fix the ICO width from the BIH.
|
2012-08-22 15:56:38 +00:00
|
|
|
if (!FixBitmapWidth(reinterpret_cast<int8_t*>(mBIHraw))) {
|
2011-11-04 13:56:17 +00:00
|
|
|
PostDataError();
|
|
|
|
return;
|
|
|
|
}
|
2005-01-12 20:16:07 +00:00
|
|
|
|
2011-08-25 20:09:01 +00:00
|
|
|
// Write out the BMP's bitmap info header
|
2012-05-08 12:38:43 +00:00
|
|
|
if (!WriteToContainedDecoder(mBIHraw, sizeof(mBIHraw))) {
|
2011-08-25 20:09:01 +00:00
|
|
|
return;
|
2005-01-12 20:16:07 +00:00
|
|
|
}
|
|
|
|
|
2011-08-30 05:12:59 +00:00
|
|
|
// We have the size. If we're doing a size decode, we got what
|
|
|
|
// we came for.
|
|
|
|
if (IsSizeDecode())
|
|
|
|
return;
|
|
|
|
|
2011-08-25 20:09:01 +00:00
|
|
|
// Sometimes the ICO BPP header field is not filled out
|
|
|
|
// so we should trust the contained resource over our own
|
|
|
|
// information.
|
|
|
|
mBPP = bmpDecoder->GetBitsPerPixel();
|
|
|
|
|
|
|
|
// Check to make sure we have valid color settings
|
2012-08-22 15:56:38 +00:00
|
|
|
uint16_t numColors = GetNumColors();
|
|
|
|
if (numColors == (uint16_t)-1) {
|
2011-08-25 20:09:01 +00:00
|
|
|
PostDataError();
|
2010-09-12 15:22:30 +00:00
|
|
|
return;
|
2009-10-16 02:54:44 +00:00
|
|
|
}
|
2011-08-25 20:09:01 +00:00
|
|
|
}
|
2009-10-16 02:54:44 +00:00
|
|
|
|
2011-08-25 20:09:01 +00:00
|
|
|
// If we have a BMP
|
|
|
|
if (!mIsPNG && mContainedDecoder && mPos >= mImageOffset + BITMAPINFOSIZE) {
|
2012-08-22 15:56:38 +00:00
|
|
|
uint16_t numColors = GetNumColors();
|
|
|
|
if (numColors == (uint16_t)-1) {
|
2011-08-25 20:09:01 +00:00
|
|
|
PostDataError();
|
2010-09-12 15:22:30 +00:00
|
|
|
return;
|
|
|
|
}
|
2011-08-25 20:09:01 +00:00
|
|
|
// Feed the actual image data (not including headers) into the BMP decoder
|
2012-10-12 16:11:20 +00:00
|
|
|
uint32_t bmpDataOffset = mDirEntry.mImageOffset + BITMAPINFOSIZE;
|
|
|
|
uint32_t bmpDataEnd = mDirEntry.mImageOffset + BITMAPINFOSIZE +
|
|
|
|
static_cast<nsBMPDecoder*>(mContainedDecoder.get())->GetCompressedImageSize() +
|
|
|
|
4 * numColors;
|
2011-08-25 20:09:01 +00:00
|
|
|
|
|
|
|
// If we are feeding in the core image data, but we have not yet
|
|
|
|
// reached the ICO's 'AND buffer mask'
|
|
|
|
if (mPos >= bmpDataOffset && mPos < bmpDataEnd) {
|
|
|
|
|
|
|
|
// Figure out how much data the BMP decoder wants
|
2012-08-22 15:56:38 +00:00
|
|
|
uint32_t toFeed = bmpDataEnd - mPos;
|
2011-08-25 20:09:01 +00:00
|
|
|
if (toFeed > aCount) {
|
|
|
|
toFeed = aCount;
|
|
|
|
}
|
2007-07-17 21:08:46 +00:00
|
|
|
|
2012-05-08 12:38:43 +00:00
|
|
|
if (!WriteToContainedDecoder(aBuffer, toFeed)) {
|
2011-08-25 20:09:01 +00:00
|
|
|
return;
|
2001-11-03 07:10:51 +00:00
|
|
|
}
|
|
|
|
|
2011-08-25 20:09:01 +00:00
|
|
|
mPos += toFeed;
|
|
|
|
aCount -= toFeed;
|
|
|
|
aBuffer += toFeed;
|
2001-11-07 03:27:58 +00:00
|
|
|
}
|
2011-08-25 20:09:01 +00:00
|
|
|
|
|
|
|
// If the bitmap is fully processed, treat any left over data as the ICO's
|
|
|
|
// 'AND buffer mask' which appears after the bitmap resource.
|
|
|
|
if (!mIsPNG && mPos >= bmpDataEnd) {
|
|
|
|
// There may be an optional AND bit mask after the data. This is
|
|
|
|
// only used if the alpha data is not already set. The alpha data
|
|
|
|
// is used for 32bpp bitmaps as per the comment in ICODecoder.h
|
|
|
|
// The alpha mask should be checked in all other cases.
|
2011-09-09 15:24:23 +00:00
|
|
|
if (static_cast<nsBMPDecoder*>(mContainedDecoder.get())->GetBitsPerPixel() != 32 ||
|
|
|
|
!static_cast<nsBMPDecoder*>(mContainedDecoder.get())->HasAlphaData()) {
|
2012-08-22 15:56:38 +00:00
|
|
|
uint32_t rowSize = ((GetRealWidth() + 31) / 32) * 4; // + 31 to round up
|
2011-08-25 20:09:01 +00:00
|
|
|
if (mPos == bmpDataEnd) {
|
|
|
|
mPos++;
|
|
|
|
mRowBytes = 0;
|
2011-09-22 13:43:13 +00:00
|
|
|
mCurLine = GetRealHeight();
|
2012-08-22 15:56:38 +00:00
|
|
|
mRow = (uint8_t*)moz_realloc(mRow, rowSize);
|
2011-08-25 20:09:01 +00:00
|
|
|
if (!mRow) {
|
|
|
|
PostDecoderError(NS_ERROR_OUT_OF_MEMORY);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
2001-11-03 07:10:51 +00:00
|
|
|
|
2011-08-25 20:09:01 +00:00
|
|
|
// Ensure memory has been allocated before decoding.
|
|
|
|
NS_ABORT_IF_FALSE(mRow, "mRow is null");
|
2011-09-29 13:17:13 +00:00
|
|
|
if (!mRow) {
|
2011-08-25 20:09:01 +00:00
|
|
|
PostDataError();
|
|
|
|
return;
|
|
|
|
}
|
2005-04-29 12:16:00 +00:00
|
|
|
|
2011-08-25 20:09:01 +00:00
|
|
|
while (mCurLine > 0 && aCount > 0) {
|
2013-01-15 12:22:03 +00:00
|
|
|
uint32_t toCopy = std::min(rowSize - mRowBytes, aCount);
|
2011-08-25 20:09:01 +00:00
|
|
|
if (toCopy) {
|
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;
|
2011-08-25 20:09:01 +00:00
|
|
|
}
|
|
|
|
if (rowSize == mRowBytes) {
|
2002-12-11 14:06:51 +00:00
|
|
|
mCurLine--;
|
|
|
|
mRowBytes = 0;
|
2001-11-05 08:38:21 +00:00
|
|
|
|
2012-08-22 15:56:38 +00:00
|
|
|
uint32_t* imageData =
|
2011-08-30 05:12:59 +00:00
|
|
|
static_cast<nsBMPDecoder*>(mContainedDecoder.get())->GetImageData();
|
|
|
|
if (!imageData) {
|
|
|
|
PostDataError();
|
|
|
|
return;
|
|
|
|
}
|
2012-08-22 15:56:38 +00:00
|
|
|
uint32_t* decoded = imageData + mCurLine * GetRealWidth();
|
|
|
|
uint32_t* decoded_end = decoded + GetRealWidth();
|
|
|
|
uint8_t* p = mRow, *p_end = mRow + rowSize;
|
2011-08-25 20:09:01 +00:00
|
|
|
while (p < p_end) {
|
2012-08-22 15:56:38 +00:00
|
|
|
uint8_t idx = *p++;
|
|
|
|
for (uint8_t bit = 0x80; bit && decoded<decoded_end; bit >>= 1) {
|
2011-08-25 20:09:01 +00:00
|
|
|
// Clear pixel completely for transparency.
|
|
|
|
if (idx & bit) {
|
|
|
|
*decoded = 0;
|
|
|
|
}
|
|
|
|
decoded++;
|
|
|
|
}
|
|
|
|
}
|
2007-07-17 21:08:46 +00:00
|
|
|
}
|
2001-11-03 07:10:51 +00:00
|
|
|
}
|
2007-07-17 21:08:46 +00:00
|
|
|
}
|
|
|
|
}
|
2001-11-03 07:10:51 +00:00
|
|
|
}
|
2012-05-08 12:38:43 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
bool
|
2012-08-22 15:56:38 +00:00
|
|
|
nsICODecoder::WriteToContainedDecoder(const char* aBuffer, uint32_t aCount)
|
2012-05-08 12:38:43 +00:00
|
|
|
{
|
|
|
|
mContainedDecoder->Write(aBuffer, aCount);
|
|
|
|
if (mContainedDecoder->HasDataError()) {
|
|
|
|
mDataError = mContainedDecoder->HasDataError();
|
|
|
|
}
|
|
|
|
if (mContainedDecoder->HasDecoderError()) {
|
|
|
|
PostDecoderError(mContainedDecoder->GetDecoderError());
|
|
|
|
}
|
|
|
|
return !HasError();
|
2001-11-03 07:10:51 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
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));
|
2011-08-25 20:09:01 +00:00
|
|
|
memcpy(&aTarget.mHeight, mDirEntryArray + 1, sizeof(aTarget.mHeight));
|
|
|
|
memcpy(&aTarget.mColorCount, mDirEntryArray + 2, sizeof(aTarget.mColorCount));
|
|
|
|
memcpy(&aTarget.mReserved, mDirEntryArray + 3, sizeof(aTarget.mReserved));
|
|
|
|
memcpy(&aTarget.mPlanes, mDirEntryArray + 4, sizeof(aTarget.mPlanes));
|
2002-05-14 20:32:13 +00:00
|
|
|
aTarget.mPlanes = LITTLE_TO_NATIVE16(aTarget.mPlanes);
|
2011-08-25 20:09:01 +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);
|
2011-08-25 20:09:01 +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);
|
2011-08-25 20:09:01 +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
|
|
|
}
|
|
|
|
|
2012-01-06 16:02:27 +00:00
|
|
|
} // namespace image
|
2010-08-23 02:30:46 +00:00
|
|
|
} // namespace mozilla
|